123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843 |
- /*
- * Copyright (C)2019 Teluu Inc. (http://www.teluu.com)
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation; either version 2 of the License, or
- * (at your option) any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with this program; if not, write to the Free Software
- * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
- */
- #include <pjmedia-codec/vpx.h>
- #include <pjmedia/vid_codec_util.h>
- #include <pjmedia/errno.h>
- #include <pj/log.h>
- #include <pj/math.h>
- #if defined(PJMEDIA_HAS_VPX_CODEC) && \
- PJMEDIA_HAS_VPX_CODEC != 0 && \
- defined(PJMEDIA_HAS_VIDEO) && (PJMEDIA_HAS_VIDEO != 0)
- #ifdef _MSC_VER
- # pragma comment( lib, "vpx.lib")
- #endif
- #include <pjmedia-codec/vpx_packetizer.h>
- /* VPX */
- #include <vpx/vpx_encoder.h>
- #include <vpx/vpx_decoder.h>
- #include <vpx/vp8cx.h>
- #include <vpx/vp8dx.h>
- /*
- * Constants
- */
- #define THIS_FILE "vpx.c"
- #if (defined(PJ_DARWINOS) && PJ_DARWINOS != 0 && TARGET_OS_IPHONE) || \
- defined(__ANDROID__)
- # define DEFAULT_WIDTH 352
- # define DEFAULT_HEIGHT 288
- #else
- # define DEFAULT_WIDTH 720
- # define DEFAULT_HEIGHT 480
- #endif
- #define DEFAULT_FPS 15
- #define DEFAULT_AVG_BITRATE 200000
- #define DEFAULT_MAX_BITRATE 200000
- #define MAX_RX_RES 1200
- /* VPX VP8 default PT */
- #define VPX_VP8_PT PJMEDIA_RTP_PT_VP8
- /* VPX VP9 default PT */
- #define VPX_VP9_PT PJMEDIA_RTP_PT_VP9
- /*
- * Factory operations.
- */
- static pj_status_t vpx_test_alloc(pjmedia_vid_codec_factory *factory,
- const pjmedia_vid_codec_info *info );
- static pj_status_t vpx_default_attr(pjmedia_vid_codec_factory *factory,
- const pjmedia_vid_codec_info *info,
- pjmedia_vid_codec_param *attr );
- static pj_status_t vpx_enum_info(pjmedia_vid_codec_factory *factory,
- unsigned *count,
- pjmedia_vid_codec_info codecs[]);
- static pj_status_t vpx_alloc_codec(pjmedia_vid_codec_factory *factory,
- const pjmedia_vid_codec_info *info,
- pjmedia_vid_codec **p_codec);
- static pj_status_t vpx_dealloc_codec(pjmedia_vid_codec_factory *factory,
- pjmedia_vid_codec *codec );
- /*
- * Codec operations
- */
- static pj_status_t vpx_codec_init(pjmedia_vid_codec *codec,
- pj_pool_t *pool );
- static pj_status_t vpx_codec_open(pjmedia_vid_codec *codec,
- pjmedia_vid_codec_param *param );
- static pj_status_t vpx_codec_close(pjmedia_vid_codec *codec);
- static pj_status_t vpx_codec_modify(pjmedia_vid_codec *codec,
- const pjmedia_vid_codec_param *param);
- static pj_status_t vpx_codec_get_param(pjmedia_vid_codec *codec,
- pjmedia_vid_codec_param *param);
- static pj_status_t vpx_codec_encode_begin(pjmedia_vid_codec *codec,
- const pjmedia_vid_encode_opt *opt,
- const pjmedia_frame *input,
- unsigned out_size,
- pjmedia_frame *output,
- pj_bool_t *has_more);
- static pj_status_t vpx_codec_encode_more(pjmedia_vid_codec *codec,
- unsigned out_size,
- pjmedia_frame *output,
- pj_bool_t *has_more);
- static pj_status_t vpx_codec_decode_(pjmedia_vid_codec *codec,
- pj_size_t count,
- pjmedia_frame packets[],
- unsigned out_size,
- pjmedia_frame *output);
- /* Definition for VPX codecs operations. */
- static pjmedia_vid_codec_op vpx_codec_op =
- {
- &vpx_codec_init,
- &vpx_codec_open,
- &vpx_codec_close,
- &vpx_codec_modify,
- &vpx_codec_get_param,
- &vpx_codec_encode_begin,
- &vpx_codec_encode_more,
- &vpx_codec_decode_,
- NULL
- };
- /* Definition for VPX codecs factory operations. */
- static pjmedia_vid_codec_factory_op vpx_factory_op =
- {
- &vpx_test_alloc,
- &vpx_default_attr,
- &vpx_enum_info,
- &vpx_alloc_codec,
- &vpx_dealloc_codec
- };
- static struct vpx_factory
- {
- pjmedia_vid_codec_factory base;
- pjmedia_vid_codec_mgr *mgr;
- pj_pool_factory *pf;
- pj_pool_t *pool;
- } vpx_factory;
- typedef struct vpx_codec_data
- {
- pj_pool_t *pool;
- pjmedia_vid_codec_param *prm;
- pj_bool_t whole;
- pjmedia_vpx_packetizer *pktz;
- /* Encoder */
- vpx_codec_iface_t *(*enc_if)();
- vpx_codec_ctx_t enc;
- vpx_codec_iter_t enc_iter;
- unsigned enc_input_size;
- pj_uint8_t *enc_frame_whole;
- unsigned enc_frame_size;
- unsigned enc_processed;
- pj_bool_t enc_frame_is_keyframe;
- pj_timestamp ets;
- /* Decoder */
- vpx_codec_iface_t *(*dec_if)();
- vpx_codec_ctx_t dec;
- pj_uint8_t *dec_buf;
- unsigned dec_buf_size;
- } vpx_codec_data;
- PJ_DEF(pj_status_t) pjmedia_codec_vpx_vid_init(pjmedia_vid_codec_mgr *mgr,
- pj_pool_factory *pf)
- {
- pj_status_t status;
- if (vpx_factory.pool != NULL) {
- /* Already initialized. */
- return PJ_SUCCESS;
- }
- if (!mgr) mgr = pjmedia_vid_codec_mgr_instance();
- PJ_ASSERT_RETURN(mgr, PJ_EINVAL);
- /* Create VPX codec factory. */
- vpx_factory.base.op = &vpx_factory_op;
- vpx_factory.base.factory_data = NULL;
- vpx_factory.mgr = mgr;
- vpx_factory.pf = pf;
- vpx_factory.pool = pj_pool_create(pf, "vpxfactory", 256, 256, NULL);
- if (!vpx_factory.pool)
- return PJ_ENOMEM;
- /* Register codec factory to codec manager. */
- status = pjmedia_vid_codec_mgr_register_factory(mgr,
- &vpx_factory.base);
- if (status != PJ_SUCCESS)
- goto on_error;
- PJ_LOG(4,(THIS_FILE, "VPX codec initialized"));
- /* Done. */
- return PJ_SUCCESS;
- on_error:
- pj_pool_release(vpx_factory.pool);
- vpx_factory.pool = NULL;
- return status;
- }
- /*
- * Unregister VPX codecs factory from pjmedia endpoint.
- */
- PJ_DEF(pj_status_t) pjmedia_codec_vpx_vid_deinit(void)
- {
- pj_status_t status = PJ_SUCCESS;
- if (vpx_factory.pool == NULL) {
- /* Already deinitialized */
- return PJ_SUCCESS;
- }
- /* Unregister VPX codecs factory. */
- status = pjmedia_vid_codec_mgr_unregister_factory(vpx_factory.mgr,
- &vpx_factory.base);
- /* Destroy pool. */
- pj_pool_release(vpx_factory.pool);
- vpx_factory.pool = NULL;
- return status;
- }
- static pj_status_t vpx_test_alloc(pjmedia_vid_codec_factory *factory,
- const pjmedia_vid_codec_info *info )
- {
- PJ_ASSERT_RETURN(factory == &vpx_factory.base, PJ_EINVAL);
- if (((info->fmt_id == PJMEDIA_FORMAT_VP8) && (info->pt == VPX_VP8_PT)) ||
- ((info->fmt_id == PJMEDIA_FORMAT_VP9) && (info->pt == VPX_VP9_PT)))
- {
- return PJ_SUCCESS;
- }
- return PJMEDIA_CODEC_EUNSUP;
- }
- static pj_status_t vpx_default_attr(pjmedia_vid_codec_factory *factory,
- const pjmedia_vid_codec_info *info,
- pjmedia_vid_codec_param *attr )
- {
- PJ_ASSERT_RETURN(factory == &vpx_factory.base, PJ_EINVAL);
- PJ_ASSERT_RETURN(info && attr, PJ_EINVAL);
- pj_bzero(attr, sizeof(pjmedia_vid_codec_param));
- attr->dir = PJMEDIA_DIR_ENCODING_DECODING;
- attr->packing = PJMEDIA_VID_PACKING_PACKETS;
- /* Encoded format */
- pjmedia_format_init_video(&attr->enc_fmt, info->fmt_id,
- DEFAULT_WIDTH, DEFAULT_HEIGHT,
- DEFAULT_FPS, 1);
- /* Decoded format */
- pjmedia_format_init_video(&attr->dec_fmt, PJMEDIA_FORMAT_I420,
- DEFAULT_WIDTH, DEFAULT_HEIGHT,
- DEFAULT_FPS, 1);
- /* Decoding fmtp */
- /* If the implementation is willing to receive media, both parameters
- * MUST be provided.
- */
- attr->dec_fmtp.cnt = 2;
- attr->dec_fmtp.param[0].name = pj_str((char*)"max-fr");
- attr->dec_fmtp.param[0].val = pj_str((char*)"30");
- attr->dec_fmtp.param[1].name = pj_str((char*)" max-fs");
- attr->dec_fmtp.param[1].val = pj_str((char*)"580");
- /* Bitrate */
- attr->enc_fmt.det.vid.avg_bps = DEFAULT_AVG_BITRATE;
- attr->enc_fmt.det.vid.max_bps = DEFAULT_MAX_BITRATE;
- /* Encoding MTU */
- attr->enc_mtu = PJMEDIA_MAX_VID_PAYLOAD_SIZE;
- return PJ_SUCCESS;
- }
- static pj_status_t vpx_enum_info(pjmedia_vid_codec_factory *factory,
- unsigned *count,
- pjmedia_vid_codec_info info[])
- {
- unsigned i = 0;
- PJ_ASSERT_RETURN(info && *count > 0, PJ_EINVAL);
- PJ_ASSERT_RETURN(factory == &vpx_factory.base, PJ_EINVAL);
- #if PJMEDIA_HAS_VPX_CODEC_VP8
- info[i].fmt_id = PJMEDIA_FORMAT_VP8;
- info[i].pt = VPX_VP8_PT;
- info[i].encoding_name = pj_str((char*)"VP8");
- info[i].encoding_desc = pj_str((char*)"VPX VP8 codec");
- i++;
- #endif
- #if PJMEDIA_HAS_VPX_CODEC_VP9
- if (i + 1 < *count) {
- info[i].fmt_id = PJMEDIA_FORMAT_VP9;
- info[i].pt = VPX_VP9_PT;
- info[i].encoding_name = pj_str((char*)"VP9");
- info[i].encoding_desc = pj_str((char*)"VPX VP9 codec");
- i++;
- }
- #endif
- *count = i;
- for (i = 0; i < *count; i++) {
- info[i].clock_rate = 90000;
- info[i].dir = PJMEDIA_DIR_ENCODING_DECODING;
- info[i].dec_fmt_id_cnt = 1;
- info[i].dec_fmt_id[0] = PJMEDIA_FORMAT_I420;
- info[i].packings = PJMEDIA_VID_PACKING_PACKETS;
- info[i].fps_cnt = 3;
- info[i].fps[0].num = 15;
- info[i].fps[0].denum = 1;
- info[i].fps[1].num = 25;
- info[i].fps[1].denum = 1;
- info[i].fps[2].num = 30;
- info[i].fps[2].denum = 1;
- }
- return PJ_SUCCESS;
- }
- static pj_status_t vpx_alloc_codec(pjmedia_vid_codec_factory *factory,
- const pjmedia_vid_codec_info *info,
- pjmedia_vid_codec **p_codec)
- {
- pj_pool_t *pool;
- pjmedia_vid_codec *codec;
- vpx_codec_data *vpx_data;
- PJ_ASSERT_RETURN(factory == &vpx_factory.base && info && p_codec,
- PJ_EINVAL);
- *p_codec = NULL;
- pool = pj_pool_create(vpx_factory.pf, "vpx%p", 512, 512, NULL);
- if (!pool)
- return PJ_ENOMEM;
- /* codec instance */
- codec = PJ_POOL_ZALLOC_T(pool, pjmedia_vid_codec);
- codec->factory = factory;
- codec->op = &vpx_codec_op;
- /* codec data */
- vpx_data = PJ_POOL_ZALLOC_T(pool, vpx_codec_data);
- vpx_data->pool = pool;
- codec->codec_data = vpx_data;
- /* encoder and decoder interfaces */
- if (info->fmt_id == PJMEDIA_FORMAT_VP8) {
- vpx_data->enc_if = &vpx_codec_vp8_cx;
- vpx_data->dec_if = &vpx_codec_vp8_dx;
- } else if (info->fmt_id == PJMEDIA_FORMAT_VP9) {
- vpx_data->enc_if = &vpx_codec_vp9_cx;
- vpx_data->dec_if = &vpx_codec_vp9_dx;
- }
- *p_codec = codec;
- return PJ_SUCCESS;
- }
- static pj_status_t vpx_dealloc_codec(pjmedia_vid_codec_factory *factory,
- pjmedia_vid_codec *codec )
- {
- vpx_codec_data *vpx_data;
- PJ_ASSERT_RETURN(codec, PJ_EINVAL);
- PJ_UNUSED_ARG(factory);
- vpx_data = (vpx_codec_data*) codec->codec_data;
- vpx_data->enc_if = NULL;
- vpx_data->dec_if = NULL;
- pj_pool_release(vpx_data->pool);
- return PJ_SUCCESS;
- }
- static pj_status_t vpx_codec_init(pjmedia_vid_codec *codec,
- pj_pool_t *pool )
- {
- PJ_ASSERT_RETURN(codec && pool, PJ_EINVAL);
- PJ_UNUSED_ARG(codec);
- PJ_UNUSED_ARG(pool);
- return PJ_SUCCESS;
- }
- static pj_status_t vpx_codec_open(pjmedia_vid_codec *codec,
- pjmedia_vid_codec_param *codec_param )
- {
- vpx_codec_data *vpx_data;
- pjmedia_vid_codec_param *param;
- pjmedia_vid_codec_vpx_fmtp vpx_fmtp;
- vpx_codec_enc_cfg_t cfg;
- vpx_codec_err_t res;
- pjmedia_vpx_packetizer_cfg pktz_cfg;
- unsigned max_res = MAX_RX_RES;
- pj_status_t status;
- PJ_ASSERT_RETURN(codec && codec_param, PJ_EINVAL);
- PJ_LOG(5,(THIS_FILE, "Opening codec.."));
- vpx_data = (vpx_codec_data*) codec->codec_data;
- vpx_data->prm = pjmedia_vid_codec_param_clone(vpx_data->pool,
- codec_param);
- param = vpx_data->prm;
- vpx_data->whole = (param->packing == PJMEDIA_VID_PACKING_WHOLE);
- /* Apply SDP fmtp to format in codec param */
- if (!param->ignore_fmtp) {
- status = pjmedia_vid_codec_vpx_apply_fmtp(param);
- if (status != PJ_SUCCESS)
- return status;
- }
- /*
- * Encoder
- */
- /* Init encoder parameters */
- res = vpx_codec_enc_config_default(vpx_data->enc_if(), &cfg, 0);
- if (res) {
- PJ_LOG(3, (THIS_FILE, "Failed to get encoder default config"));
- return PJMEDIA_CODEC_EFAILED;
- }
-
- cfg.g_w = vpx_data->prm->enc_fmt.det.vid.size.w;
- cfg.g_h = vpx_data->prm->enc_fmt.det.vid.size.h;
- /* timebase is the inverse of fps */
- cfg.g_timebase.num = vpx_data->prm->enc_fmt.det.vid.fps.denum;
- cfg.g_timebase.den = vpx_data->prm->enc_fmt.det.vid.fps.num;
- /* bitrate in KBps */
- cfg.rc_target_bitrate = vpx_data->prm->enc_fmt.det.vid.avg_bps / 1000;
-
- cfg.g_pass = VPX_RC_ONE_PASS;
- cfg.rc_end_usage = VPX_CBR;
- cfg.g_threads = 4;
- cfg.g_lag_in_frames = 0;
- cfg.g_error_resilient = 0;
- cfg.rc_undershoot_pct = 95;
- cfg.rc_min_quantizer = 4;
- cfg.rc_max_quantizer = 56;
- cfg.rc_buf_initial_sz = 400;
- cfg.rc_buf_optimal_sz = 500;
- cfg.rc_buf_sz = 600;
- /* kf max distance is 60s. */
- cfg.kf_max_dist = 60 * vpx_data->prm->enc_fmt.det.vid.fps.num/
- vpx_data->prm->enc_fmt.det.vid.fps.denum;
- cfg.rc_resize_allowed = 0;
- cfg.rc_dropframe_thresh = 25;
- vpx_data->enc_input_size = cfg.g_w * cfg.g_h * 3 >> 1;
- /* Initialize encoder */
- res = vpx_codec_enc_init(&vpx_data->enc, vpx_data->enc_if(), &cfg, 0);
- if (res) {
- PJ_LOG(3, (THIS_FILE, "Failed to initialize encoder"));
- return PJMEDIA_CODEC_EFAILED;
- }
- /* Values greater than 0 will increase encoder speed at the expense of
- * quality.
- * Valid range for VP8: -16..16
- * Valid range for VP9: -9..9
- */
- vpx_codec_control(&vpx_data->enc, VP8E_SET_CPUUSED, 9);
- /*
- * Decoder
- */
- res = vpx_codec_dec_init(&vpx_data->dec, vpx_data->dec_if(), NULL, 0);
- if (res) {
- PJ_LOG(3, (THIS_FILE, "Failed to initialize decoder"));
- return PJMEDIA_CODEC_EFAILED;
- }
- /* Parse local fmtp */
- status = pjmedia_vid_codec_vpx_parse_fmtp(¶m->dec_fmtp, &vpx_fmtp);
- if (status != PJ_SUCCESS)
- return status;
- if (vpx_fmtp.max_fs > 0) {
- max_res = ((int)pj_isqrt(vpx_fmtp.max_fs * 8)) * 16;
- }
- vpx_data->dec_buf_size = (max_res * max_res * 3 >> 1) + (max_res);
- vpx_data->dec_buf = (pj_uint8_t*)pj_pool_alloc(vpx_data->pool,
- vpx_data->dec_buf_size);
- /* Need to update param back after values are negotiated */
- pj_memcpy(codec_param, param, sizeof(*codec_param));
- pj_bzero(&pktz_cfg, sizeof(pktz_cfg));
- pktz_cfg.mtu = param->enc_mtu;
- pktz_cfg.fmt_id = param->enc_fmt.id;
- status = pjmedia_vpx_packetizer_create(vpx_data->pool, &pktz_cfg,
- &vpx_data->pktz);
- if (status != PJ_SUCCESS)
- return status;
- return PJ_SUCCESS;
- }
- static pj_status_t vpx_codec_close(pjmedia_vid_codec *codec)
- {
- struct vpx_codec_data *vpx_data;
- PJ_ASSERT_RETURN(codec, PJ_EINVAL);
- vpx_data = (vpx_codec_data*) codec->codec_data;
- vpx_codec_destroy(&vpx_data->enc);
- vpx_codec_destroy(&vpx_data->dec);
- return PJ_SUCCESS;
- }
- static pj_status_t vpx_codec_modify(pjmedia_vid_codec *codec,
- const pjmedia_vid_codec_param *param)
- {
- PJ_ASSERT_RETURN(codec && param, PJ_EINVAL);
- PJ_UNUSED_ARG(codec);
- PJ_UNUSED_ARG(param);
- return PJ_EINVALIDOP;
- }
- static pj_status_t vpx_codec_get_param(pjmedia_vid_codec *codec,
- pjmedia_vid_codec_param *param)
- {
- struct vpx_codec_data *vpx_data;
- PJ_ASSERT_RETURN(codec && param, PJ_EINVAL);
- vpx_data = (vpx_codec_data*) codec->codec_data;
- pj_memcpy(param, vpx_data->prm, sizeof(*param));
- return PJ_SUCCESS;
- }
- static pj_status_t vpx_codec_encode_begin(pjmedia_vid_codec *codec,
- const pjmedia_vid_encode_opt *opt,
- const pjmedia_frame *input,
- unsigned out_size,
- pjmedia_frame *output,
- pj_bool_t *has_more)
- {
- struct vpx_codec_data *vpx_data;
- vpx_image_t img;
- vpx_enc_frame_flags_t flags = 0;
- vpx_codec_err_t res;
- PJ_ASSERT_RETURN(codec && input && out_size && output && has_more,
- PJ_EINVAL);
- vpx_data = (vpx_codec_data*) codec->codec_data;
- PJ_ASSERT_RETURN(input->size == vpx_data->enc_input_size,
- PJMEDIA_CODEC_EFRMINLEN);
- vpx_img_wrap(&img, VPX_IMG_FMT_I420,
- vpx_data->prm->enc_fmt.det.vid.size.w,
- vpx_data->prm->enc_fmt.det.vid.size.h,
- 1, (unsigned char *)input->buf);
- if (opt && opt->force_keyframe) {
- flags |= VPX_EFLAG_FORCE_KF;
- }
- vpx_data->ets = input->timestamp;
- vpx_data->enc_frame_size = vpx_data->enc_processed = 0;
- vpx_data->enc_iter = NULL;
- res = vpx_codec_encode(&vpx_data->enc, &img, vpx_data->ets.u64, 1,
- flags, VPX_DL_REALTIME);
- if (res) {
- PJ_LOG(4, (THIS_FILE, "Failed to encode frame %s",
- vpx_codec_error(&vpx_data->enc)));
- return PJMEDIA_CODEC_EFAILED;
- }
- do {
- const vpx_codec_cx_pkt_t *pkt;
- pkt = vpx_codec_get_cx_data(&vpx_data->enc, &vpx_data->enc_iter);
- if (!pkt) break;
- if (pkt->kind == VPX_CODEC_CX_FRAME_PKT) {
- /* We have a valid frame packet */
- vpx_data->enc_frame_whole = pkt->data.frame.buf;
- vpx_data->enc_frame_size = pkt->data.frame.sz;
- vpx_data->enc_processed = 0;
- if (pkt->data.frame.flags & VPX_FRAME_IS_KEY)
- vpx_data->enc_frame_is_keyframe = PJ_TRUE;
- else
- vpx_data->enc_frame_is_keyframe = PJ_FALSE;
-
- break;
- }
- } while (1);
- if (vpx_data->enc_frame_size == 0) {
- *has_more = PJ_FALSE;
- output->size = 0;
- output->type = PJMEDIA_FRAME_TYPE_NONE;
- if (vpx_data->enc.err) {
- PJ_LOG(4, (THIS_FILE, "Failed to get encoded frame %s",
- vpx_codec_error(&vpx_data->enc)));
- } else {
- return PJ_SUCCESS;
- }
- }
-
- if (vpx_data->whole) {
- *has_more = PJ_FALSE;
- if (vpx_data->enc_frame_size > out_size)
- return PJMEDIA_CODEC_EFRMTOOSHORT;
- output->timestamp = vpx_data->ets;
- output->type = PJMEDIA_FRAME_TYPE_VIDEO;
- output->size = vpx_data->enc_frame_size;
- output->bit_info = 0;
- if (vpx_data->enc_frame_is_keyframe) {
- output->bit_info |= PJMEDIA_VID_FRM_KEYFRAME;
- }
- pj_memcpy(output->buf, vpx_data->enc_frame_whole, output->size);
-
- return PJ_SUCCESS;
- }
- return vpx_codec_encode_more(codec, out_size, output, has_more);
- }
- static pj_status_t vpx_codec_encode_more(pjmedia_vid_codec *codec,
- unsigned out_size,
- pjmedia_frame *output,
- pj_bool_t *has_more)
- {
- struct vpx_codec_data *vpx_data;
- pj_status_t status = PJ_SUCCESS;
- PJ_ASSERT_RETURN(codec && out_size && output && has_more,
- PJ_EINVAL);
- vpx_data = (vpx_codec_data*) codec->codec_data;
-
- if (vpx_data->enc_processed < vpx_data->enc_frame_size) {
- unsigned payload_desc_size = 1;
- pj_size_t payload_len = out_size;
- pj_uint8_t *p = (pj_uint8_t *)output->buf;
- status = pjmedia_vpx_packetize(vpx_data->pktz,
- vpx_data->enc_frame_size,
- &vpx_data->enc_processed,
- vpx_data->enc_frame_is_keyframe,
- &p,
- &payload_len);
- if (status != PJ_SUCCESS) {
- return status;
- }
- pj_memcpy(p + payload_desc_size,
- (vpx_data->enc_frame_whole + vpx_data->enc_processed),
- payload_len);
- output->size = payload_len + payload_desc_size;
- output->timestamp = vpx_data->ets;
- output->type = PJMEDIA_FRAME_TYPE_VIDEO;
- output->bit_info = 0;
- if (vpx_data->enc_frame_is_keyframe) {
- output->bit_info |= PJMEDIA_VID_FRM_KEYFRAME;
- }
- vpx_data->enc_processed += payload_len;
- *has_more = (vpx_data->enc_processed < vpx_data->enc_frame_size);
- }
- return status;
- }
- static pj_status_t vpx_codec_decode_(pjmedia_vid_codec *codec,
- pj_size_t count,
- pjmedia_frame packets[],
- unsigned out_size,
- pjmedia_frame *output)
- {
- struct vpx_codec_data *vpx_data;
- pj_bool_t has_frame = PJ_FALSE;
- unsigned i, whole_len = 0;
- vpx_codec_iter_t iter = NULL;
- vpx_image_t *img = NULL;
- vpx_codec_err_t res;
- unsigned pos = 0;
- int plane;
- PJ_ASSERT_RETURN(codec && count && packets && out_size && output,
- PJ_EINVAL);
- PJ_ASSERT_RETURN(output->buf, PJ_EINVAL);
- vpx_data = (vpx_codec_data*) codec->codec_data;
- /*
- * Step 1: unpacketize the packets/frames
- */
- whole_len = 0;
- if (vpx_data->whole) {
- for (i = 0; i < count; ++i) {
- if (whole_len + packets[i].size > vpx_data->dec_buf_size) {
- PJ_LOG(4,(THIS_FILE, "Decoding buffer overflow [1]"));
- return PJMEDIA_CODEC_EFRMTOOSHORT;
- }
- pj_memcpy( vpx_data->dec_buf + whole_len,
- (pj_uint8_t*)packets[i].buf,
- packets[i].size);
- whole_len += packets[i].size;
- }
- } else {
- for (i = 0; i < count; ++i) {
- unsigned desc_len;
- unsigned packet_size = packets[i].size;
- pj_status_t status;
- status = pjmedia_vpx_unpacketize(vpx_data->pktz, packets[i].buf,
- packet_size, &desc_len);
- if (status != PJ_SUCCESS) {
- PJ_LOG(4,(THIS_FILE, "Unpacketize error"));
- return status;
- }
- packet_size -= desc_len;
- if (whole_len + packet_size > vpx_data->dec_buf_size) {
- PJ_LOG(4,(THIS_FILE, "Decoding buffer overflow [2]"));
- return PJMEDIA_CODEC_EFRMTOOSHORT;
- }
- pj_memcpy(vpx_data->dec_buf + whole_len,
- (char *)packets[i].buf + desc_len, packet_size);
- whole_len += packet_size;
- }
- }
- /* Decode */
- res = vpx_codec_decode(&vpx_data->dec, vpx_data->dec_buf, whole_len,
- 0, VPX_DL_REALTIME);
- if (res) {
- PJ_LOG(4, (THIS_FILE, "Failed to decode frame %s",
- vpx_codec_error(&vpx_data->dec)));
- goto on_return;
- }
- img = vpx_codec_get_frame(&vpx_data->dec, &iter);
- if (!img) {
- PJ_LOG(4, (THIS_FILE, "Failed to get decoded frame %s",
- vpx_codec_error(&vpx_data->dec)));
- goto on_return;
- }
- has_frame = PJ_TRUE;
- /* Detect format change */
- if (img->d_w != vpx_data->prm->dec_fmt.det.vid.size.w ||
- img->d_h != vpx_data->prm->dec_fmt.det.vid.size.h)
- {
- pjmedia_event event;
- PJ_LOG(4,(THIS_FILE, "Frame size changed: %dx%d --> %dx%d",
- vpx_data->prm->dec_fmt.det.vid.size.w,
- vpx_data->prm->dec_fmt.det.vid.size.h,
- img->d_w, img->d_h));
- vpx_data->prm->dec_fmt.det.vid.size.w = img->d_w;
- vpx_data->prm->dec_fmt.det.vid.size.h = img->d_h;
- /* Broadcast format changed event */
- pjmedia_event_init(&event, PJMEDIA_EVENT_FMT_CHANGED,
- &packets[0].timestamp, codec);
- event.data.fmt_changed.dir = PJMEDIA_DIR_DECODING;
- pjmedia_format_copy(&event.data.fmt_changed.new_fmt,
- &vpx_data->prm->dec_fmt);
- pjmedia_event_publish(NULL, codec, &event,
- PJMEDIA_EVENT_PUBLISH_DEFAULT);
- }
- if (img->d_w * img->d_h * 3/2 > output->size)
- return PJMEDIA_CODEC_EFRMTOOSHORT;
- output->type = PJMEDIA_FRAME_TYPE_VIDEO;
- output->timestamp = packets[0].timestamp;
- for (plane = 0; plane < 3; ++plane) {
- const unsigned char *buf = img->planes[plane];
- const int stride = img->stride[plane];
- const int w = (plane? img->d_w / 2: img->d_w);
- const int h = (plane? img->d_h / 2: img->d_h);
- int y;
- for (y = 0; y < h; ++y) {
- pj_memcpy((char *)output->buf + pos, buf, w);
- pos += w;
- buf += stride;
- }
- }
- output->size = pos;
-
- on_return:
- if (!has_frame) {
- pjmedia_event event;
- /* Broadcast missing keyframe event */
- pjmedia_event_init(&event, PJMEDIA_EVENT_KEYFRAME_MISSING,
- &packets[0].timestamp, codec);
- pjmedia_event_publish(NULL, codec, &event,
- PJMEDIA_EVENT_PUBLISH_DEFAULT);
- PJ_LOG(4,(THIS_FILE, "Decode couldn't produce picture, "
- "input nframes=%lu, concatenated size=%d bytes",
- (unsigned long)count, whole_len));
- output->type = PJMEDIA_FRAME_TYPE_NONE;
- output->size = 0;
- output->timestamp = packets[0].timestamp;
- }
- return PJ_SUCCESS;
- }
- #endif /* PJMEDIA_HAS_VPX_CODEC */
|