2014-11-11 03:38:56 +00:00
|
|
|
/*
|
|
|
|
* FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application
|
2015-02-26 00:58:04 +00:00
|
|
|
* Copyright (C) 2005-2015, Anthony Minessale II <anthm@freeswitch.org>
|
2014-11-11 03:38:56 +00:00
|
|
|
*
|
|
|
|
* Version: MPL 1.1
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
|
|
* the License. You may obtain a copy of the License at
|
|
|
|
* http://www.mozilla.org/MPL/
|
|
|
|
*
|
|
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
|
|
* for the specific language governing rights and limitations under the
|
|
|
|
* License.
|
|
|
|
*
|
|
|
|
* The Original Code is FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application
|
|
|
|
*
|
|
|
|
* The Initial Developer of the Original Code is
|
|
|
|
* Seven Du <dujinfang@gmail.com>
|
|
|
|
* Portions created by the Initial Developer are Copyright (C)
|
|
|
|
* the Initial Developer. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
|
|
|
*
|
|
|
|
* Anthony Minessale II <anthm@freeswitch.org>
|
|
|
|
* Seven Du <dujinfang@gmail.com>
|
|
|
|
* Sam Russell <sam.h.russell@gmail.com>
|
|
|
|
*
|
|
|
|
* mod_vpx.c -- VP8/9 Video Codec, with transcoding
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <switch.h>
|
|
|
|
#include <vpx/vpx_encoder.h>
|
|
|
|
#include <vpx/vpx_decoder.h>
|
|
|
|
#include <vpx/vp8cx.h>
|
|
|
|
#include <vpx/vp8dx.h>
|
|
|
|
#include <vpx/vp8.h>
|
|
|
|
|
2014-11-20 14:40:27 +00:00
|
|
|
#define SLICE_SIZE SWITCH_DEFAULT_VIDEO_SIZE
|
2014-11-20 00:01:32 +00:00
|
|
|
#define KEY_FRAME_MIN_FREQ 1000000
|
2014-11-11 03:38:56 +00:00
|
|
|
|
2015-03-12 02:26:14 +00:00
|
|
|
#define IS_VP8_KEY_FRAME(byte) (((byte) & 0x01) ^ 0x01)
|
|
|
|
#define IS_VP9_KEY_FRAME(byte) ((byte) & 0x01)
|
|
|
|
#define IS_VP9_START_PKT(byte) ((byte) & 0x02)
|
|
|
|
|
2014-11-11 03:38:56 +00:00
|
|
|
SWITCH_MODULE_LOAD_FUNCTION(mod_vpx_load);
|
|
|
|
SWITCH_MODULE_DEFINITION(mod_vpx, mod_vpx_load, NULL, NULL);
|
|
|
|
|
|
|
|
struct vpx_context {
|
|
|
|
switch_codec_t *codec;
|
2015-03-11 23:33:20 +00:00
|
|
|
int is_vp9;
|
2015-03-12 16:17:35 +00:00
|
|
|
int lossless;
|
2015-03-11 23:33:20 +00:00
|
|
|
vpx_codec_iface_t *encoder_interface;
|
|
|
|
vpx_codec_iface_t *decoder_interface;
|
2014-11-11 03:38:56 +00:00
|
|
|
unsigned int flags;
|
2014-11-15 01:01:56 +00:00
|
|
|
switch_codec_settings_t codec_settings;
|
|
|
|
unsigned int bandwidth;
|
2014-11-11 03:38:56 +00:00
|
|
|
vpx_codec_enc_cfg_t config;
|
2014-11-20 00:01:32 +00:00
|
|
|
switch_time_t last_key_frame;
|
2014-11-11 03:38:56 +00:00
|
|
|
|
|
|
|
vpx_codec_ctx_t encoder;
|
2014-11-15 01:01:56 +00:00
|
|
|
uint8_t encoder_init;
|
2014-11-11 03:38:56 +00:00
|
|
|
vpx_image_t *pic;
|
|
|
|
switch_bool_t force_key_frame;
|
|
|
|
int fps;
|
|
|
|
int format;
|
|
|
|
int intra_period;
|
|
|
|
int num;
|
|
|
|
int partition_index;
|
|
|
|
const vpx_codec_cx_pkt_t *pkt;
|
|
|
|
vpx_codec_iter_t iter;
|
2015-01-29 22:37:29 +00:00
|
|
|
uint32_t last_ts;
|
2015-03-13 15:41:47 +00:00
|
|
|
switch_time_t last_ms;
|
2014-11-11 03:38:56 +00:00
|
|
|
vpx_codec_ctx_t decoder;
|
2014-11-15 01:01:56 +00:00
|
|
|
uint8_t decoder_init;
|
2014-11-11 03:38:56 +00:00
|
|
|
switch_buffer_t *vpx_packet_buffer;
|
|
|
|
int got_key_frame;
|
|
|
|
switch_size_t last_received_timestamp;
|
|
|
|
switch_bool_t last_received_complete_picture;
|
|
|
|
int need_key_frame;
|
2015-02-13 20:10:42 +00:00
|
|
|
int need_encoder_reset;
|
|
|
|
int need_decoder_reset;
|
2015-02-06 22:13:32 +00:00
|
|
|
int32_t change_bandwidth;
|
2015-02-03 04:58:39 +00:00
|
|
|
uint64_t framecount;
|
2015-03-11 23:33:20 +00:00
|
|
|
switch_memory_pool_t *pool;
|
|
|
|
switch_buffer_t *pbuffer;
|
2015-03-12 16:17:35 +00:00
|
|
|
switch_time_t start_time;
|
2014-11-11 03:38:56 +00:00
|
|
|
};
|
|
|
|
typedef struct vpx_context vpx_context_t;
|
|
|
|
|
|
|
|
|
2015-02-17 21:55:23 +00:00
|
|
|
static switch_status_t init_decoder(switch_codec_t *codec)
|
|
|
|
{
|
|
|
|
vpx_context_t *context = (vpx_context_t *)codec->private_info;
|
2015-03-11 23:33:20 +00:00
|
|
|
vpx_codec_dec_cfg_t cfg = {0, 0, 0};
|
|
|
|
vpx_codec_flags_t dec_flags = 0;
|
|
|
|
|
2015-02-17 21:55:23 +00:00
|
|
|
if (context->flags & SWITCH_CODEC_FLAG_DECODE && !context->decoder_init) {
|
|
|
|
vp8_postproc_cfg_t ppcfg;
|
|
|
|
|
|
|
|
//if (context->decoder_init) {
|
|
|
|
// vpx_codec_destroy(&context->decoder);
|
|
|
|
// context->decoder_init = 0;
|
|
|
|
//}
|
|
|
|
|
2015-03-11 23:33:20 +00:00
|
|
|
cfg.threads = switch_core_cpu_count();
|
|
|
|
|
|
|
|
if (!context->is_vp9) { // vp8 only
|
|
|
|
dec_flags = VPX_CODEC_USE_POSTPROC;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (vpx_codec_dec_init(&context->decoder, context->decoder_interface, &cfg, dec_flags) != VPX_CODEC_OK) {
|
2015-02-17 21:55:23 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Codec init error: [%d:%s]\n", context->encoder.err, context->encoder.err_detail);
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
context->decoder_init = 1;
|
|
|
|
|
|
|
|
// the types of post processing to be done, should be combination of "vp8_postproc_level"
|
|
|
|
ppcfg.post_proc_flag = VP8_DEMACROBLOCK | VP8_DEBLOCK;
|
|
|
|
// the strength of deblocking, valid range [0, 16]
|
|
|
|
ppcfg.deblocking_level = 3;
|
|
|
|
// Set deblocking settings
|
|
|
|
vpx_codec_control(&context->decoder, VP8_SET_POSTPROC, &ppcfg);
|
|
|
|
|
|
|
|
switch_buffer_create_dynamic(&context->vpx_packet_buffer, 512, 512, 1024000);
|
|
|
|
}
|
|
|
|
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static switch_status_t init_encoder(switch_codec_t *codec)
|
2014-11-15 01:01:56 +00:00
|
|
|
{
|
|
|
|
vpx_context_t *context = (vpx_context_t *)codec->private_info;
|
|
|
|
vpx_codec_enc_cfg_t *config = &context->config;
|
2014-11-20 00:01:32 +00:00
|
|
|
int token_parts = 1;
|
2014-11-19 20:48:14 +00:00
|
|
|
int cpus = switch_core_cpu_count();
|
2014-11-11 03:38:56 +00:00
|
|
|
|
2014-11-18 22:39:32 +00:00
|
|
|
if (!context->codec_settings.video.width) {
|
|
|
|
context->codec_settings.video.width = 1280;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!context->codec_settings.video.height) {
|
|
|
|
context->codec_settings.video.height = 720;
|
|
|
|
}
|
|
|
|
|
2015-02-06 22:13:32 +00:00
|
|
|
if (context->codec_settings.video.bandwidth == -1) {
|
|
|
|
context->codec_settings.video.bandwidth = 0;
|
|
|
|
}
|
|
|
|
|
2014-11-18 22:39:32 +00:00
|
|
|
if (context->codec_settings.video.bandwidth) {
|
2015-02-20 02:32:53 +00:00
|
|
|
context->bandwidth = context->codec_settings.video.bandwidth * 4;
|
2014-11-18 22:39:32 +00:00
|
|
|
} else {
|
2015-02-20 02:32:53 +00:00
|
|
|
context->bandwidth = ((context->codec_settings.video.width * context->codec_settings.video.height) / 900) * 4;
|
2014-11-18 22:39:32 +00:00
|
|
|
}
|
|
|
|
|
2015-02-20 02:32:53 +00:00
|
|
|
if (context->bandwidth > 40960) {
|
|
|
|
context->bandwidth = 40960;
|
2014-11-18 22:39:32 +00:00
|
|
|
}
|
|
|
|
|
2015-03-03 17:44:38 +00:00
|
|
|
context->pkt = NULL;
|
|
|
|
|
2014-11-20 00:01:32 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(codec->session), SWITCH_LOG_NOTICE,
|
|
|
|
"VPX reset encoder picture from %dx%d to %dx%d %u BW\n",
|
|
|
|
config->g_w, config->g_h, context->codec_settings.video.width, context->codec_settings.video.height, context->bandwidth);
|
|
|
|
|
2015-03-12 16:17:35 +00:00
|
|
|
context->start_time = switch_micro_time_now();
|
|
|
|
|
|
|
|
config->g_timebase.num = 1;
|
|
|
|
config->g_timebase.den = 1000;
|
|
|
|
config->g_pass = VPX_RC_ONE_PASS;
|
|
|
|
config->g_w = context->codec_settings.video.width;
|
|
|
|
config->g_h = context->codec_settings.video.height;
|
|
|
|
config->rc_target_bitrate = context->bandwidth;
|
|
|
|
config->g_lag_in_frames = 0;
|
|
|
|
config->kf_max_dist = 2000;
|
|
|
|
config->g_threads = (cpus > 1) ? 2 : 1;
|
|
|
|
|
2015-03-11 23:33:20 +00:00
|
|
|
if (context->is_vp9) {
|
2015-03-12 16:17:35 +00:00
|
|
|
//config->rc_dropframe_thresh = 2;
|
2015-03-11 23:33:20 +00:00
|
|
|
token_parts = (cpus > 1) ? 3 : 0;
|
|
|
|
|
2015-03-12 16:17:35 +00:00
|
|
|
if (context->lossless) {
|
|
|
|
config->rc_min_quantizer = 0;
|
|
|
|
config->rc_max_quantizer = 0;
|
|
|
|
} else {
|
|
|
|
config->rc_min_quantizer = 0;
|
|
|
|
config->rc_max_quantizer = 63;
|
|
|
|
}
|
2014-11-15 01:01:56 +00:00
|
|
|
|
2015-03-11 23:33:20 +00:00
|
|
|
} else {
|
|
|
|
|
|
|
|
// settings
|
2015-03-12 16:17:35 +00:00
|
|
|
config->g_profile = 2;
|
2015-03-11 23:33:20 +00:00
|
|
|
config->g_error_resilient = VPX_ERROR_RESILIENT_PARTITIONS;
|
|
|
|
token_parts = (cpus > 1) ? 3 : 0;
|
|
|
|
|
|
|
|
// rate control settings
|
|
|
|
config->rc_dropframe_thresh = 0;
|
|
|
|
config->rc_end_usage = VPX_CBR;
|
2015-03-12 16:17:35 +00:00
|
|
|
//config->g_pass = VPX_RC_ONE_PASS;
|
2015-03-11 23:33:20 +00:00
|
|
|
config->kf_mode = VPX_KF_AUTO;
|
|
|
|
config->kf_max_dist = 1000;
|
|
|
|
|
|
|
|
//config->kf_mode = VPX_KF_DISABLED;
|
|
|
|
config->rc_resize_allowed = 1;
|
2015-03-12 16:17:35 +00:00
|
|
|
//config->rc_min_quantizer = 0;
|
|
|
|
//config->rc_max_quantizer = 63;
|
2015-03-11 23:33:20 +00:00
|
|
|
config->rc_min_quantizer = 0;
|
|
|
|
config->rc_max_quantizer = 63;
|
|
|
|
//Rate control adaptation undershoot control.
|
|
|
|
// This value, expressed as a percentage of the target bitrate,
|
|
|
|
// controls the maximum allowed adaptation speed of the codec.
|
|
|
|
// This factor controls the maximum amount of bits that can be
|
|
|
|
// subtracted from the target bitrate in order to compensate for
|
|
|
|
// prior overshoot.
|
|
|
|
// Valid values in the range 0-1000.
|
|
|
|
config->rc_undershoot_pct = 100;
|
|
|
|
//Rate control adaptation overshoot control.
|
|
|
|
// This value, expressed as a percentage of the target bitrate,
|
|
|
|
// controls the maximum allowed adaptation speed of the codec.
|
|
|
|
// This factor controls the maximum amount of bits that can be
|
|
|
|
// added to the target bitrate in order to compensate for prior
|
|
|
|
// undershoot.
|
|
|
|
// Valid values in the range 0-1000.
|
|
|
|
config->rc_overshoot_pct = 15;
|
|
|
|
//Decoder Buffer Size.
|
|
|
|
// This value indicates the amount of data that may be buffered
|
|
|
|
// by the decoding application. Note that this value is expressed
|
|
|
|
// in units of time (milliseconds). For example, a value of 5000
|
|
|
|
// indicates that the client will buffer (at least) 5000ms worth
|
|
|
|
// of encoded data. Use the target bitrate (rc_target_bitrate) to
|
|
|
|
// convert to bits/bytes, if necessary.
|
|
|
|
config->rc_buf_sz = 5000;
|
|
|
|
//Decoder Buffer Initial Size.
|
|
|
|
// This value indicates the amount of data that will be buffered
|
|
|
|
// by the decoding application prior to beginning playback.
|
|
|
|
// This value is expressed in units of time (milliseconds).
|
|
|
|
// Use the target bitrate (rc_target_bitrate) to convert to
|
|
|
|
// bits/bytes, if necessary.
|
|
|
|
config->rc_buf_initial_sz = 1000;
|
|
|
|
//Decoder Buffer Optimal Size.
|
|
|
|
// This value indicates the amount of data that the encoder should
|
|
|
|
// try to maintain in the decoder's buffer. This value is expressed
|
|
|
|
// in units of time (milliseconds).
|
|
|
|
// Use the target bitrate (rc_target_bitrate) to convert to
|
|
|
|
// bits/bytes, if necessary.
|
|
|
|
config->rc_buf_optimal_sz = 1000;
|
|
|
|
}
|
2014-11-15 01:01:56 +00:00
|
|
|
|
2014-11-20 00:01:32 +00:00
|
|
|
if (context->encoder_init) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_INFO, "VPX ENCODER RESET\n");
|
|
|
|
if (vpx_codec_enc_config_set(&context->encoder, config) != VPX_CODEC_OK) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Codec init error: [%d:%s]\n", context->encoder.err, context->encoder.err_detail);
|
2014-11-15 01:01:56 +00:00
|
|
|
}
|
2014-11-20 00:01:32 +00:00
|
|
|
} else if (context->flags & SWITCH_CODEC_FLAG_ENCODE) {
|
2014-11-11 03:38:56 +00:00
|
|
|
|
2015-03-11 23:33:20 +00:00
|
|
|
if (vpx_codec_enc_init(&context->encoder, context->encoder_interface, config, 0 & VPX_CODEC_USE_OUTPUT_PARTITION) != VPX_CODEC_OK) {
|
2014-11-15 01:01:56 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Codec init error: [%d:%s]\n", context->encoder.err, context->encoder.err_detail);
|
2014-11-11 03:38:56 +00:00
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
2015-03-12 16:17:35 +00:00
|
|
|
|
2014-11-15 01:01:56 +00:00
|
|
|
context->encoder_init = 1;
|
|
|
|
|
2015-03-11 23:33:20 +00:00
|
|
|
if (context->is_vp9) {
|
2015-03-12 16:17:35 +00:00
|
|
|
if (context->lossless) {
|
|
|
|
vpx_codec_control(&context->encoder, VP9E_SET_LOSSLESS, 1);
|
|
|
|
vpx_codec_control(&context->encoder, VP8E_SET_CPUUSED, -6);
|
|
|
|
} else {
|
|
|
|
vpx_codec_control(&context->encoder, VP8E_SET_CPUUSED, -8);
|
|
|
|
}
|
|
|
|
|
2015-03-11 23:33:20 +00:00
|
|
|
vpx_codec_control(&context->encoder, VP8E_SET_STATIC_THRESHOLD, 100);
|
|
|
|
vpx_codec_control(&context->encoder, VP8E_SET_TOKEN_PARTITIONS, token_parts);
|
2015-03-12 16:17:35 +00:00
|
|
|
vpx_codec_control(&context->encoder, VP9E_SET_TUNE_CONTENT, VP9E_CONTENT_SCREEN);
|
2014-11-19 20:48:14 +00:00
|
|
|
|
2015-03-11 23:33:20 +00:00
|
|
|
} else {
|
|
|
|
// The static threshold imposes a change threshold on blocks below which they will be skipped by the encoder.
|
|
|
|
vpx_codec_control(&context->encoder, VP8E_SET_STATIC_THRESHOLD, 100);
|
|
|
|
//Set cpu usage, a bit lower than normal (-6) but higher than android (-12)
|
|
|
|
vpx_codec_control(&context->encoder, VP8E_SET_CPUUSED, -6);
|
|
|
|
vpx_codec_control(&context->encoder, VP8E_SET_TOKEN_PARTITIONS, token_parts);
|
|
|
|
|
|
|
|
// Enable noise reduction
|
|
|
|
vpx_codec_control(&context->encoder, VP8E_SET_NOISE_SENSITIVITY, 1);
|
|
|
|
//Set max data rate for Intra frames.
|
|
|
|
// This value controls additional clamping on the maximum size of a keyframe.
|
|
|
|
// It is expressed as a percentage of the average per-frame bitrate, with the
|
|
|
|
// special (and default) value 0 meaning unlimited, or no additional clamping
|
|
|
|
// beyond the codec's built-in algorithm.
|
|
|
|
// For example, to allocate no more than 4.5 frames worth of bitrate to a keyframe, set this to 450.
|
|
|
|
//vpx_codec_control(&context->encoder, VP8E_SET_MAX_INTRA_BITRATE_PCT, 0);
|
|
|
|
}
|
2014-11-11 03:38:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2014-11-15 01:01:56 +00:00
|
|
|
static switch_status_t switch_vpx_init(switch_codec_t *codec, switch_codec_flag_t flags, const switch_codec_settings_t *codec_settings)
|
|
|
|
{
|
|
|
|
vpx_context_t *context = NULL;
|
|
|
|
int encoding, decoding;
|
|
|
|
|
|
|
|
encoding = (flags & SWITCH_CODEC_FLAG_ENCODE);
|
|
|
|
decoding = (flags & SWITCH_CODEC_FLAG_DECODE);
|
|
|
|
|
|
|
|
if (!(encoding || decoding) || ((context = switch_core_alloc(codec->memory_pool, sizeof(*context))) == 0)) {
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
memset(context, 0, sizeof(*context));
|
|
|
|
context->flags = flags;
|
|
|
|
codec->private_info = context;
|
2015-03-11 23:33:20 +00:00
|
|
|
context->pool = codec->memory_pool;
|
2014-11-15 01:01:56 +00:00
|
|
|
|
|
|
|
if (codec_settings) {
|
|
|
|
context->codec_settings = *codec_settings;
|
|
|
|
}
|
|
|
|
|
2015-03-11 23:33:20 +00:00
|
|
|
if (!strcmp(codec->implementation->iananame, "VP9")) {
|
|
|
|
context->is_vp9 = 1;
|
|
|
|
context->encoder_interface = vpx_codec_vp9_cx();
|
|
|
|
context->decoder_interface = vpx_codec_vp9_dx();
|
|
|
|
} else {
|
|
|
|
context->encoder_interface = vpx_codec_vp8_cx();
|
|
|
|
context->decoder_interface = vpx_codec_vp8_dx();
|
|
|
|
}
|
|
|
|
|
2014-11-15 01:01:56 +00:00
|
|
|
if (codec->fmtp_in) {
|
|
|
|
codec->fmtp_out = switch_core_strdup(codec->memory_pool, codec->fmtp_in);
|
|
|
|
}
|
|
|
|
|
2015-03-11 23:33:20 +00:00
|
|
|
if (vpx_codec_enc_config_default(context->encoder_interface, &context->config, 0) != VPX_CODEC_OK) {
|
2014-11-15 01:01:56 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Encoder config Error\n");
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
|
|
|
|
2015-02-17 21:55:23 +00:00
|
|
|
context->codec_settings.video.width = 320;
|
|
|
|
context->codec_settings.video.height = 240;
|
2014-11-20 00:01:32 +00:00
|
|
|
|
2014-11-15 01:01:56 +00:00
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2014-11-11 03:38:56 +00:00
|
|
|
/* http://tools.ietf.org/html/draft-ietf-payload-vp8-10
|
|
|
|
|
|
|
|
The first octets after the RTP header are the VP8 payload descriptor, with the following structure.
|
|
|
|
|
|
|
|
0 1 2 3 4 5 6 7
|
|
|
|
+-+-+-+-+-+-+-+-+
|
|
|
|
|X|R|N|S|R| PID | (REQUIRED)
|
|
|
|
+-+-+-+-+-+-+-+-+
|
|
|
|
X: |I|L|T|K| RSV | (OPTIONAL)
|
|
|
|
+-+-+-+-+-+-+-+-+
|
|
|
|
I: |M| PictureID | (OPTIONAL)
|
|
|
|
+-+-+-+-+-+-+-+-+
|
|
|
|
L: | TL0PICIDX | (OPTIONAL)
|
|
|
|
+-+-+-+-+-+-+-+-+
|
|
|
|
T/K:|TID|Y| KEYIDX | (OPTIONAL)
|
|
|
|
+-+-+-+-+-+-+-+-+
|
|
|
|
|
|
|
|
|
|
|
|
VP8 Payload Header
|
|
|
|
|
|
|
|
0 1 2 3 4 5 6 7
|
|
|
|
+-+-+-+-+-+-+-+-+
|
|
|
|
|Size0|H| VER |P|
|
|
|
|
+-+-+-+-+-+-+-+-+
|
|
|
|
| Size1 |
|
|
|
|
+-+-+-+-+-+-+-+-+
|
|
|
|
| Size2 |
|
|
|
|
+-+-+-+-+-+-+-+-+
|
|
|
|
| Bytes 4..N of |
|
|
|
|
| VP8 payload |
|
|
|
|
: :
|
|
|
|
+-+-+-+-+-+-+-+-+
|
|
|
|
| OPTIONAL RTP |
|
|
|
|
| padding |
|
|
|
|
: :
|
|
|
|
+-+-+-+-+-+-+-+-+
|
|
|
|
*/
|
|
|
|
|
2015-03-11 23:33:20 +00:00
|
|
|
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
#pragma pack(push, r1, 1)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if SWITCH_BYTE_ORDER == __BIG_ENDIAN
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
unsigned extended:1;
|
|
|
|
unsigned reserved1:1;
|
|
|
|
unsigned non_referenced:1;
|
|
|
|
unsigned start:1;
|
|
|
|
unsigned reserved2:1;
|
|
|
|
unsigned pid:3;
|
|
|
|
} vp8_payload_descriptor_t;
|
|
|
|
|
|
|
|
#ifdef WHAT_THEY_FUCKING_SAY
|
|
|
|
typedef struct {
|
|
|
|
unsigned have_pid:1;
|
|
|
|
unsigned have_layer_ind:1;
|
|
|
|
unsigned have_ref_ind:1;
|
|
|
|
unsigned start:1;
|
|
|
|
unsigned end:1;
|
|
|
|
unsigned have_ss:1;
|
|
|
|
unsigned have_su:1;
|
|
|
|
unsigned zero:1;
|
|
|
|
} vp9_payload_descriptor_t;
|
|
|
|
|
|
|
|
#else
|
|
|
|
typedef struct {
|
|
|
|
unsigned dunno:6;
|
|
|
|
unsigned start:1;
|
|
|
|
unsigned key:1;
|
|
|
|
} vp9_payload_descriptor_t;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#else /* ELSE LITTLE */
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
unsigned pid:3;
|
|
|
|
unsigned reserved2:1;
|
|
|
|
unsigned start:1;
|
|
|
|
unsigned non_referenced:1;
|
|
|
|
unsigned reserved1:1;
|
|
|
|
unsigned extended:1;
|
|
|
|
} vp8_payload_descriptor_t;
|
|
|
|
|
|
|
|
#ifdef WHAT_THEY_FUCKING_SAY
|
|
|
|
typedef struct {
|
|
|
|
unsigned zero:1;
|
|
|
|
unsigned have_su:1;
|
|
|
|
unsigned have_ss:1;
|
|
|
|
unsigned end:1;
|
|
|
|
unsigned start:1;
|
|
|
|
unsigned have_ref_ind:1;
|
|
|
|
unsigned have_layer_ind:1;
|
|
|
|
unsigned have_pid:1;
|
|
|
|
} vp9_payload_descriptor_t;
|
|
|
|
#else
|
|
|
|
typedef struct {
|
|
|
|
unsigned key:1;
|
|
|
|
unsigned start:1;
|
|
|
|
unsigned dunno:6;
|
|
|
|
} vp9_payload_descriptor_t;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
typedef union {
|
|
|
|
vp8_payload_descriptor_t vp8;
|
|
|
|
vp9_payload_descriptor_t vp9;
|
|
|
|
} vpx_payload_descriptor_t;
|
|
|
|
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
#pragma pack(pop, r1)
|
|
|
|
#endif
|
|
|
|
|
2014-11-18 22:39:32 +00:00
|
|
|
static switch_status_t consume_partition(vpx_context_t *context, switch_frame_t *frame)
|
2014-11-11 03:38:56 +00:00
|
|
|
{
|
2015-03-11 23:33:20 +00:00
|
|
|
vpx_payload_descriptor_t *payload_descriptor;
|
|
|
|
uint8_t *body;
|
|
|
|
uint32_t hdrlen = 0, payload_size = 0, packet_size = 0, start = 0, key = 0;
|
|
|
|
switch_size_t remaining_bytes = 0;
|
|
|
|
|
2014-11-20 22:30:19 +00:00
|
|
|
if (!context->pkt) {
|
2015-03-11 23:33:20 +00:00
|
|
|
if ((context->pkt = vpx_codec_get_cx_data(&context->encoder, &context->iter))) {
|
|
|
|
start = 1;
|
|
|
|
if (!context->pbuffer) {
|
|
|
|
switch_buffer_create_partition(context->pool, &context->pbuffer, context->pkt->data.frame.buf, context->pkt->data.frame.sz);
|
|
|
|
} else {
|
|
|
|
switch_buffer_set_partition_data(context->pbuffer, context->pkt->data.frame.buf, context->pkt->data.frame.sz);
|
|
|
|
}
|
|
|
|
}
|
2014-11-20 22:30:19 +00:00
|
|
|
}
|
2014-11-11 03:38:56 +00:00
|
|
|
|
2015-03-11 23:33:20 +00:00
|
|
|
if (context->pbuffer) {
|
|
|
|
remaining_bytes = switch_buffer_inuse(context->pbuffer);
|
|
|
|
}
|
2014-11-11 03:38:56 +00:00
|
|
|
|
2015-03-11 23:33:20 +00:00
|
|
|
if (!context->pkt || context->pkt->kind != VPX_CODEC_CX_FRAME_PKT || !remaining_bytes) {
|
2014-11-18 22:39:32 +00:00
|
|
|
frame->datalen = 0;
|
|
|
|
frame->m = 1;
|
2014-11-11 03:38:56 +00:00
|
|
|
context->pkt = NULL;
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2015-03-11 23:33:20 +00:00
|
|
|
key = (context->pkt->data.frame.flags & VPX_FRAME_IS_KEY);
|
|
|
|
|
|
|
|
/* reset header */
|
|
|
|
*(uint8_t *)frame->data = 0;
|
|
|
|
payload_descriptor = (vpx_payload_descriptor_t *) frame->data;
|
|
|
|
|
|
|
|
// if !extended
|
|
|
|
hdrlen = 1;
|
|
|
|
body = ((uint8_t *)frame->data) + hdrlen;
|
|
|
|
packet_size = SLICE_SIZE;
|
|
|
|
payload_size = packet_size - hdrlen;
|
|
|
|
// else add extended TBD
|
|
|
|
|
|
|
|
frame->datalen = hdrlen;
|
|
|
|
|
|
|
|
if (context->is_vp9) {
|
|
|
|
payload_descriptor->vp9.start = start;
|
|
|
|
payload_descriptor->vp9.key = key;
|
|
|
|
} else {
|
|
|
|
payload_descriptor->vp8.start = start;
|
|
|
|
}
|
2014-11-11 03:38:56 +00:00
|
|
|
|
2015-03-11 23:33:20 +00:00
|
|
|
if (remaining_bytes <= payload_size) {
|
|
|
|
switch_buffer_read(context->pbuffer, body, remaining_bytes);
|
2014-11-11 03:38:56 +00:00
|
|
|
context->pkt = NULL;
|
2015-03-11 23:33:20 +00:00
|
|
|
frame->datalen += remaining_bytes;
|
|
|
|
frame->m = 1;
|
2014-11-11 03:38:56 +00:00
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
} else {
|
2015-03-11 23:33:20 +00:00
|
|
|
switch_buffer_read(context->pbuffer, body, payload_size);
|
|
|
|
frame->datalen += payload_size;
|
|
|
|
frame->m = 0;
|
|
|
|
return SWITCH_STATUS_MORE_DATA;
|
2014-11-11 03:38:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-17 21:55:23 +00:00
|
|
|
static void reset_codec_encoder(switch_codec_t *codec)
|
|
|
|
{
|
|
|
|
vpx_context_t *context = (vpx_context_t *)codec->private_info;
|
|
|
|
|
|
|
|
if (context->encoder_init) {
|
|
|
|
vpx_codec_destroy(&context->encoder);
|
|
|
|
}
|
2015-03-13 15:41:47 +00:00
|
|
|
context->last_ts = 0;
|
|
|
|
context->last_ms = 0;
|
2015-02-17 21:55:23 +00:00
|
|
|
context->framecount = 0;
|
|
|
|
context->encoder_init = 0;
|
2015-03-03 17:44:38 +00:00
|
|
|
context->pkt = NULL;
|
2015-02-17 21:55:23 +00:00
|
|
|
init_encoder(codec);
|
|
|
|
}
|
|
|
|
|
2014-11-18 22:39:32 +00:00
|
|
|
static switch_status_t switch_vpx_encode(switch_codec_t *codec, switch_frame_t *frame)
|
2014-11-11 03:38:56 +00:00
|
|
|
{
|
|
|
|
vpx_context_t *context = (vpx_context_t *)codec->private_info;
|
|
|
|
int width = 0;
|
|
|
|
int height = 0;
|
2015-03-13 15:41:47 +00:00
|
|
|
uint32_t dur;
|
|
|
|
int64_t pts;
|
2014-11-11 03:38:56 +00:00
|
|
|
vpx_enc_frame_flags_t vpx_flags = 0;
|
2015-03-13 15:41:47 +00:00
|
|
|
switch_time_t now;
|
2014-11-11 03:38:56 +00:00
|
|
|
|
2014-11-18 22:39:32 +00:00
|
|
|
if (frame->flags & SFF_SAME_IMAGE) {
|
|
|
|
return consume_partition(context, frame);
|
2014-11-11 03:38:56 +00:00
|
|
|
}
|
|
|
|
|
2015-02-13 20:10:42 +00:00
|
|
|
if (context->need_encoder_reset != 0) {
|
2015-02-17 21:55:23 +00:00
|
|
|
reset_codec_encoder(codec);
|
2015-02-13 20:10:42 +00:00
|
|
|
context->need_encoder_reset = 0;
|
|
|
|
}
|
|
|
|
|
2014-11-18 22:39:32 +00:00
|
|
|
if (frame->img->d_h > 1) {
|
|
|
|
width = frame->img->d_w;
|
|
|
|
height = frame->img->d_h;
|
|
|
|
} else {
|
|
|
|
width = frame->img->w;
|
|
|
|
height = frame->img->h;
|
|
|
|
}
|
2014-11-11 03:38:56 +00:00
|
|
|
|
|
|
|
if (context->config.g_w != width || context->config.g_h != height) {
|
2014-11-15 01:01:56 +00:00
|
|
|
context->codec_settings.video.width = width;
|
|
|
|
context->codec_settings.video.height = height;
|
2015-02-17 21:55:23 +00:00
|
|
|
reset_codec_encoder(codec);
|
2014-11-18 22:39:32 +00:00
|
|
|
frame->flags |= SFF_PICTURE_RESET;
|
2014-11-15 01:01:56 +00:00
|
|
|
context->need_key_frame = 1;
|
2014-11-11 03:38:56 +00:00
|
|
|
}
|
|
|
|
|
2014-11-18 22:39:32 +00:00
|
|
|
|
|
|
|
if (!context->encoder_init) {
|
2015-02-17 21:55:23 +00:00
|
|
|
init_encoder(codec);
|
2014-11-18 22:39:32 +00:00
|
|
|
}
|
|
|
|
|
2015-02-06 22:13:32 +00:00
|
|
|
if (context->change_bandwidth) {
|
|
|
|
context->codec_settings.video.bandwidth = context->change_bandwidth;
|
|
|
|
context->change_bandwidth = 0;
|
2015-02-17 21:55:23 +00:00
|
|
|
init_encoder(codec);
|
2015-02-06 22:13:32 +00:00
|
|
|
}
|
|
|
|
|
2015-03-13 15:41:47 +00:00
|
|
|
now = switch_time_now();
|
|
|
|
|
2014-11-18 22:39:32 +00:00
|
|
|
if (context->need_key_frame != 0) {
|
2014-11-11 03:38:56 +00:00
|
|
|
// force generate a key frame
|
2014-11-20 00:01:32 +00:00
|
|
|
|
2015-03-12 16:17:35 +00:00
|
|
|
if (!context->last_key_frame || (now - context->last_key_frame) > KEY_FRAME_MIN_FREQ) {
|
2014-11-20 00:01:32 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "VPX KEYFRAME GENERATED\n");
|
|
|
|
vpx_flags |= VPX_EFLAG_FORCE_KF;
|
|
|
|
context->need_key_frame = 0;
|
|
|
|
context->last_key_frame = now;
|
|
|
|
}
|
2014-11-11 03:38:56 +00:00
|
|
|
}
|
|
|
|
|
2015-02-03 04:58:39 +00:00
|
|
|
context->framecount++;
|
|
|
|
|
2015-03-13 15:41:47 +00:00
|
|
|
pts = (now - context->start_time) / 1000;
|
|
|
|
|
|
|
|
dur = context->last_ms ? (now - context->last_ms) / 1000 : pts;
|
|
|
|
|
2015-03-12 16:17:35 +00:00
|
|
|
if (vpx_codec_encode(&context->encoder,
|
|
|
|
(vpx_image_t *) frame->img,
|
2015-03-13 15:41:47 +00:00
|
|
|
pts,
|
|
|
|
dur,
|
2015-03-12 16:17:35 +00:00
|
|
|
vpx_flags,
|
|
|
|
VPX_DL_REALTIME) != VPX_CODEC_OK) {
|
2015-03-11 23:33:20 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "VPX encode error %d:%s\n",
|
2014-11-11 03:38:56 +00:00
|
|
|
context->encoder.err, context->encoder.err_detail);
|
2014-11-19 20:48:14 +00:00
|
|
|
|
2014-11-18 22:39:32 +00:00
|
|
|
frame->datalen = 0;
|
2014-11-11 03:38:56 +00:00
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
context->iter = NULL;
|
2015-01-29 22:37:29 +00:00
|
|
|
context->last_ts = frame->timestamp;
|
2015-03-13 15:41:47 +00:00
|
|
|
context->last_ms = now;
|
2015-02-03 04:58:39 +00:00
|
|
|
|
2014-11-18 22:39:32 +00:00
|
|
|
return consume_partition(context, frame);
|
2014-11-11 03:38:56 +00:00
|
|
|
}
|
|
|
|
|
2015-03-11 23:33:20 +00:00
|
|
|
static switch_status_t buffer_vp8_packets(vpx_context_t *context, switch_frame_t *frame)
|
2014-11-11 03:38:56 +00:00
|
|
|
{
|
|
|
|
uint8_t *data = frame->data;
|
|
|
|
uint8_t S;
|
|
|
|
uint8_t DES;
|
|
|
|
uint8_t PID;
|
|
|
|
int len;
|
|
|
|
|
|
|
|
DES = *data;
|
|
|
|
data++;
|
|
|
|
S = DES & 0x10;
|
|
|
|
PID = DES & 0x07;
|
|
|
|
|
|
|
|
if (DES & 0x80) { // X
|
|
|
|
uint8_t X = *data;
|
|
|
|
data++;
|
|
|
|
if (X & 0x80) { // I
|
|
|
|
uint8_t M = (*data) & 0x80;
|
|
|
|
data++;
|
|
|
|
if (M) data++;
|
|
|
|
}
|
|
|
|
if (X & 0x40) data++; // L
|
|
|
|
if (X & 0x30) data++; // T/K
|
|
|
|
}
|
|
|
|
|
|
|
|
len = frame->datalen - (data - (uint8_t *)frame->data);
|
2014-11-18 22:39:32 +00:00
|
|
|
|
2014-11-11 03:38:56 +00:00
|
|
|
if (len <= 0) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Invalid packet %d\n", len);
|
2014-11-18 22:39:32 +00:00
|
|
|
return SWITCH_STATUS_RESTART;
|
2014-11-11 03:38:56 +00:00
|
|
|
}
|
2014-11-18 22:39:32 +00:00
|
|
|
|
2014-11-11 03:38:56 +00:00
|
|
|
if (S && (PID == 0)) {
|
2015-03-12 02:26:14 +00:00
|
|
|
int is_keyframe = IS_VP8_KEY_FRAME(*data);
|
2014-11-11 03:38:56 +00:00
|
|
|
|
2015-01-24 07:54:38 +00:00
|
|
|
if (is_keyframe && context->got_key_frame <= 0) {
|
2014-11-18 22:39:32 +00:00
|
|
|
context->got_key_frame = 1;
|
2014-11-11 03:38:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-24 07:54:38 +00:00
|
|
|
if (context->got_key_frame <= 0) {
|
|
|
|
if ((context->got_key_frame-- % 200) == 0) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_WARNING, "Waiting for key frame\n");
|
|
|
|
}
|
2014-11-18 22:39:32 +00:00
|
|
|
return SWITCH_STATUS_RESTART;
|
2014-11-11 03:38:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
switch_buffer_write(context->vpx_packet_buffer, data, len);
|
2014-11-18 22:39:32 +00:00
|
|
|
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
|
2014-11-11 03:38:56 +00:00
|
|
|
}
|
|
|
|
|
2015-03-11 23:33:20 +00:00
|
|
|
static switch_status_t buffer_vp9_packets(vpx_context_t *context, switch_frame_t *frame)
|
|
|
|
{
|
2015-03-12 02:26:14 +00:00
|
|
|
uint8_t *data = (uint8_t *)frame->data;
|
|
|
|
uint8_t *vp9 = (uint8_t *)frame->data;
|
2015-03-11 23:33:20 +00:00
|
|
|
int len = 0;
|
|
|
|
|
2015-03-13 01:03:52 +00:00
|
|
|
// switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "%02x %02x %02x %02x %d %d\n", *data, *(data+1), *(data+2), *(data+3), frame->m, frame->datalen);
|
2015-03-11 23:33:20 +00:00
|
|
|
|
2015-03-12 02:26:14 +00:00
|
|
|
if (switch_buffer_inuse(context->vpx_packet_buffer)) { // middle packet
|
|
|
|
if (IS_VP9_START_PKT(*vp9)) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_WARNING, "got invalid vp9 packet, packet loss? resetting buffer\n");
|
|
|
|
switch_buffer_zero(context->vpx_packet_buffer);
|
|
|
|
}
|
|
|
|
} else { // start packet
|
|
|
|
if (!IS_VP9_START_PKT(*vp9)) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_WARNING, "got invalid vp9 packet, packet loss? waiting for a start packet\n");
|
2015-03-13 01:03:52 +00:00
|
|
|
goto end;
|
2015-03-12 02:26:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-11 23:33:20 +00:00
|
|
|
vp9 = data + 1;
|
|
|
|
|
|
|
|
len = frame->datalen - (vp9 - data);
|
|
|
|
switch_buffer_write(context->vpx_packet_buffer, vp9, len);
|
|
|
|
|
2015-03-13 01:03:52 +00:00
|
|
|
end:
|
2015-03-11 23:33:20 +00:00
|
|
|
// switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "buffered %d bytes, buffer size: %" SWITCH_SIZE_T_FMT "\n", len, switch_buffer_inuse(context->vpx_packet_buffer));
|
|
|
|
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2014-11-18 22:39:32 +00:00
|
|
|
static switch_status_t switch_vpx_decode(switch_codec_t *codec, switch_frame_t *frame)
|
2014-11-11 03:38:56 +00:00
|
|
|
{
|
|
|
|
vpx_context_t *context = (vpx_context_t *)codec->private_info;
|
|
|
|
switch_size_t len;
|
2014-11-18 22:39:32 +00:00
|
|
|
vpx_codec_ctx_t *decoder = NULL;
|
|
|
|
switch_status_t status = SWITCH_STATUS_SUCCESS;
|
2015-03-12 02:26:14 +00:00
|
|
|
int is_keyframe = 0;
|
2015-02-13 20:10:42 +00:00
|
|
|
|
2015-03-12 02:26:14 +00:00
|
|
|
if (context->is_vp9) {
|
|
|
|
is_keyframe = IS_VP9_KEY_FRAME(*(unsigned char *)frame->data);
|
|
|
|
} else { // vp8
|
|
|
|
is_keyframe = IS_VP8_KEY_FRAME(*(unsigned char *)frame->data);
|
|
|
|
}
|
2015-03-11 23:33:20 +00:00
|
|
|
|
2015-02-13 20:10:42 +00:00
|
|
|
if (context->need_decoder_reset != 0) {
|
|
|
|
vpx_codec_destroy(&context->decoder);
|
|
|
|
context->decoder_init = 0;
|
2015-02-17 21:55:23 +00:00
|
|
|
init_decoder(codec);
|
2015-02-13 20:10:42 +00:00
|
|
|
context->need_decoder_reset = 0;
|
|
|
|
}
|
2015-01-26 16:48:59 +00:00
|
|
|
|
2014-11-18 22:39:32 +00:00
|
|
|
if (!context->decoder_init) {
|
2015-02-17 21:55:23 +00:00
|
|
|
init_decoder(codec);
|
2014-11-18 22:39:32 +00:00
|
|
|
}
|
2014-11-11 03:38:56 +00:00
|
|
|
|
2014-11-18 22:39:32 +00:00
|
|
|
if (!context->decoder_init) {
|
2014-11-11 03:38:56 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_WARNING, "VPX decoder is not initialized!\n");
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
|
|
|
|
2014-11-18 22:39:32 +00:00
|
|
|
decoder = &context->decoder;
|
|
|
|
|
2015-03-11 23:33:20 +00:00
|
|
|
// switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_INFO, "len: %d ts: %u mark:%d\n", frame->datalen, frame->timestamp, frame->m);
|
2014-11-11 03:38:56 +00:00
|
|
|
|
2015-01-26 16:48:59 +00:00
|
|
|
if (!is_keyframe && context->last_received_timestamp && context->last_received_timestamp != frame->timestamp &&
|
2014-11-11 03:38:56 +00:00
|
|
|
(!frame->m) && (!context->last_received_complete_picture)) {
|
|
|
|
// possible packet loss
|
2015-01-24 07:54:38 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_WARNING, "Reset\n");
|
2015-01-26 16:48:59 +00:00
|
|
|
context->need_key_frame = 1;
|
2015-01-29 22:37:29 +00:00
|
|
|
context->last_ts = 0;
|
2014-11-24 16:18:19 +00:00
|
|
|
switch_goto_status(SWITCH_STATUS_RESTART, end);
|
2014-11-11 03:38:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
context->last_received_timestamp = frame->timestamp;
|
|
|
|
context->last_received_complete_picture = frame->m ? SWITCH_TRUE : SWITCH_FALSE;
|
|
|
|
|
2015-03-11 23:33:20 +00:00
|
|
|
status = context->is_vp9 ? buffer_vp9_packets(context, frame) : buffer_vp8_packets(context, frame);
|
2014-11-11 03:38:56 +00:00
|
|
|
|
2014-11-20 00:01:32 +00:00
|
|
|
//printf("READ buf:%ld got_key:%d st:%d m:%d\n", switch_buffer_inuse(context->vpx_packet_buffer), context->got_key_frame, status, frame->m);
|
|
|
|
|
|
|
|
len = switch_buffer_inuse(context->vpx_packet_buffer);
|
|
|
|
|
2014-12-02 16:51:21 +00:00
|
|
|
//if (frame->m && (status != SWITCH_STATUS_SUCCESS || !len)) {
|
|
|
|
//switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "WTF????? %d %ld\n", status, len);
|
|
|
|
//}
|
2014-11-20 00:01:32 +00:00
|
|
|
|
2014-11-11 03:38:56 +00:00
|
|
|
|
2014-11-20 00:01:32 +00:00
|
|
|
if (status == SWITCH_STATUS_SUCCESS && frame->m && len) {
|
2014-11-11 03:38:56 +00:00
|
|
|
uint8_t *data;
|
|
|
|
vpx_codec_iter_t iter = NULL;
|
|
|
|
int corrupted = 0;
|
|
|
|
int err;
|
2014-11-18 22:39:32 +00:00
|
|
|
//int keyframe = 0;
|
|
|
|
|
2014-11-20 00:01:32 +00:00
|
|
|
//printf("WTF %d %ld\n", frame->m, len);
|
2014-11-11 03:38:56 +00:00
|
|
|
|
|
|
|
switch_buffer_peek_zerocopy(context->vpx_packet_buffer, (void *)&data);
|
2014-11-18 22:39:32 +00:00
|
|
|
//keyframe = (*data & 0x01) ? 0 : 1;
|
2014-11-11 03:38:56 +00:00
|
|
|
|
2014-11-18 22:39:32 +00:00
|
|
|
//switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "buffered: %" SWITCH_SIZE_T_FMT ", key: %d\n", len, keyframe);
|
2014-11-11 03:38:56 +00:00
|
|
|
|
2015-02-13 20:10:42 +00:00
|
|
|
|
2014-11-11 03:38:56 +00:00
|
|
|
err = vpx_codec_decode(decoder, data, (unsigned int)len, NULL, 0);
|
|
|
|
|
|
|
|
if (err != VPX_CODEC_OK) {
|
2015-03-11 23:33:20 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Error decoding %" SWITCH_SIZE_T_FMT " bytes, [%d:%s:%s]\n",
|
|
|
|
len, err, vpx_codec_error(decoder), vpx_codec_error_detail(decoder));
|
2014-11-24 16:18:19 +00:00
|
|
|
switch_goto_status(SWITCH_STATUS_RESTART, end);
|
2014-11-11 03:38:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (vpx_codec_control(decoder, VP8D_GET_FRAME_CORRUPTED, &corrupted) != VPX_CODEC_OK) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_WARNING, "VPX control error!\n");
|
2014-11-18 22:39:32 +00:00
|
|
|
switch_goto_status(SWITCH_STATUS_RESTART, end);
|
2014-11-11 03:38:56 +00:00
|
|
|
}
|
|
|
|
|
2014-11-18 22:39:32 +00:00
|
|
|
frame->img = (switch_image_t *) vpx_codec_get_frame(decoder, &iter);
|
2014-11-11 03:38:56 +00:00
|
|
|
|
2014-11-18 22:39:32 +00:00
|
|
|
if (!(frame->img) || corrupted) {
|
2014-12-19 06:39:50 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_WARNING, "VPX invalid packet\n");
|
2014-11-24 16:18:19 +00:00
|
|
|
switch_goto_status(SWITCH_STATUS_RESTART, end);
|
2014-11-11 03:38:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
switch_buffer_zero(context->vpx_packet_buffer);
|
|
|
|
}
|
|
|
|
|
2014-11-18 22:39:32 +00:00
|
|
|
end:
|
|
|
|
|
|
|
|
if (status == SWITCH_STATUS_RESTART) {
|
2015-01-24 07:54:38 +00:00
|
|
|
if (context->got_key_frame > 0) {
|
|
|
|
context->got_key_frame = 0;
|
|
|
|
}
|
2014-11-18 22:39:32 +00:00
|
|
|
switch_buffer_zero(context->vpx_packet_buffer);
|
|
|
|
}
|
|
|
|
|
2015-01-24 07:54:38 +00:00
|
|
|
if (!frame->img || status == SWITCH_STATUS_RESTART) {
|
2014-11-20 18:07:05 +00:00
|
|
|
//switch_set_flag(frame, SFF_USE_VIDEO_TIMESTAMP);
|
|
|
|
//} else {
|
2014-11-18 22:39:32 +00:00
|
|
|
status = SWITCH_STATUS_MORE_DATA;
|
|
|
|
}
|
2014-11-11 03:38:56 +00:00
|
|
|
|
2015-01-24 07:54:38 +00:00
|
|
|
if (context->got_key_frame <= 0) {
|
2014-12-03 00:39:53 +00:00
|
|
|
switch_set_flag(frame, SFF_WAIT_KEY_FRAME);
|
2014-11-18 22:39:32 +00:00
|
|
|
}
|
2014-11-11 03:38:56 +00:00
|
|
|
|
2014-12-02 16:51:21 +00:00
|
|
|
|
2014-11-18 22:39:32 +00:00
|
|
|
return status;
|
2014-11-11 03:38:56 +00:00
|
|
|
}
|
|
|
|
|
2014-11-13 03:30:39 +00:00
|
|
|
|
|
|
|
static switch_status_t switch_vpx_control(switch_codec_t *codec,
|
|
|
|
switch_codec_control_command_t cmd,
|
|
|
|
switch_codec_control_type_t ctype,
|
|
|
|
void *cmd_data,
|
|
|
|
switch_codec_control_type_t *rtype,
|
|
|
|
void **ret_data)
|
|
|
|
{
|
|
|
|
|
|
|
|
vpx_context_t *context = (vpx_context_t *)codec->private_info;
|
|
|
|
|
|
|
|
switch(cmd) {
|
2015-02-13 20:10:42 +00:00
|
|
|
case SCC_VIDEO_RESET:
|
|
|
|
{
|
|
|
|
int mask = *((int *) cmd_data);
|
|
|
|
if (mask & 1) {
|
|
|
|
context->need_encoder_reset = 1;
|
|
|
|
}
|
|
|
|
if (mask & 2) {
|
|
|
|
context->need_decoder_reset = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
2014-11-13 03:30:39 +00:00
|
|
|
case SCC_VIDEO_REFRESH:
|
|
|
|
context->need_key_frame = 1;
|
|
|
|
break;
|
2015-02-06 22:13:32 +00:00
|
|
|
case SCC_VIDEO_BANDWIDTH:
|
|
|
|
{
|
|
|
|
switch(ctype) {
|
|
|
|
case SCCT_INT:
|
|
|
|
context->change_bandwidth = *((int *) cmd_data);
|
|
|
|
break;
|
|
|
|
case SCCT_STRING:
|
|
|
|
{
|
|
|
|
char *bwv = (char *) cmd_data;
|
|
|
|
context->change_bandwidth = switch_parse_bandwidth_string(bwv);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
2014-11-13 03:30:39 +00:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-11-11 03:38:56 +00:00
|
|
|
static switch_status_t switch_vpx_destroy(switch_codec_t *codec)
|
|
|
|
{
|
|
|
|
vpx_context_t *context = (vpx_context_t *)codec->private_info;
|
|
|
|
|
|
|
|
if (context) {
|
|
|
|
if ((codec->flags & SWITCH_CODEC_FLAG_ENCODE)) {
|
2014-11-15 01:01:56 +00:00
|
|
|
vpx_codec_destroy(&context->encoder);
|
2014-11-11 03:38:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if ((codec->flags & SWITCH_CODEC_FLAG_DECODE)) {
|
|
|
|
vpx_codec_destroy(&context->decoder);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (context->pic) {
|
|
|
|
vpx_img_free(context->pic);
|
|
|
|
context->pic = NULL;
|
|
|
|
}
|
|
|
|
if (context->vpx_packet_buffer) {
|
|
|
|
switch_buffer_destroy(&context->vpx_packet_buffer);
|
|
|
|
context->vpx_packet_buffer = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_MODULE_LOAD_FUNCTION(mod_vpx_load)
|
|
|
|
{
|
|
|
|
switch_codec_interface_t *codec_interface;
|
|
|
|
|
|
|
|
/* connect my internal structure to the blank pointer passed to me */
|
|
|
|
*module_interface = switch_loadable_module_create_module_interface(pool, modname);
|
|
|
|
SWITCH_ADD_CODEC(codec_interface, "VP8 Video");
|
|
|
|
switch_core_codec_add_video_implementation(pool, codec_interface, 99, "VP8", NULL,
|
2014-11-13 03:30:39 +00:00
|
|
|
switch_vpx_init, switch_vpx_encode, switch_vpx_decode, switch_vpx_control, switch_vpx_destroy);
|
2015-03-11 23:33:20 +00:00
|
|
|
SWITCH_ADD_CODEC(codec_interface, "VP9 Video");
|
|
|
|
switch_core_codec_add_video_implementation(pool, codec_interface, 99, "VP9", NULL,
|
|
|
|
switch_vpx_init, switch_vpx_encode, switch_vpx_decode, switch_vpx_control, switch_vpx_destroy);
|
2014-11-11 03:38:56 +00:00
|
|
|
|
|
|
|
/* indicate that the module should continue to be loaded */
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* For Emacs:
|
|
|
|
* Local Variables:
|
|
|
|
* mode:c
|
|
|
|
* indent-tabs-mode:t
|
|
|
|
* tab-width:4
|
|
|
|
* c-basic-offset:4
|
|
|
|
* End:
|
|
|
|
* For VIM:
|
|
|
|
* vim:set softtabstop=4 shiftwidth=4 tabstop=4 noet:
|
|
|
|
*/
|