/* ********************************************************************* *\ Copyright (C) 2013 Intel Corporation. All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: - Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. - Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. - Neither the name of Intel Corporation nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission. THIS SOFTWARE IS PROVIDED BY INTEL CORPORATION "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL INTEL CORPORATION BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. \* ********************************************************************* */ #ifdef USE_QSV #include "hb.h" #include "nal_units.h" #include "qsv_common.h" #include "qsv_memory.h" #include "h264_common.h" #include "h265_common.h" /* * The frame info struct remembers information about each frame across calls to * the encoder. Since frames are uniquely identified by their timestamp, we use * some bits of the timestamp as an index. The LSB is chosen so that two * successive frames will have different values in the bits over any plausible * range of frame rates (starting with bit 8 allows any frame rate slower than * 352fps). The MSB determines the size of the array. It is chosen so that two * frames can't use the same slot during the encoder's max frame delay so that, * up to some minimum frame rate, frames are guaranteed to map to different * slots (an MSB of 17 which is 2^(17-8+1) = 1024 slots guarantees no collisions * down to a rate of 0.7 fps). */ #define FRAME_INFO_MAX2 (8) // 2^8 = 256; 90000/256 = 352 frames/sec #define FRAME_INFO_MIN2 (17) // 2^17 = 128K; 90000/131072 = 0.7 frames/sec #define FRAME_INFO_SIZE (1 << (FRAME_INFO_MIN2 - FRAME_INFO_MAX2 + 1)) #define FRAME_INFO_MASK (FRAME_INFO_SIZE - 1) int encqsvInit (hb_work_object_t*, hb_job_t*); int encqsvWork (hb_work_object_t*, hb_buffer_t**, hb_buffer_t**); void encqsvClose(hb_work_object_t*); hb_work_object_t hb_encqsv = { WORK_ENCQSV, "Quick Sync Video encoder (Intel Media SDK)", encqsvInit, encqsvWork, encqsvClose }; struct hb_work_private_s { hb_job_t * job; uint32_t frames_in; uint32_t frames_out; int64_t last_start; hb_qsv_param_t param; hb_qsv_space enc_space; hb_qsv_info_t * qsv_info; hb_chapter_queue_t * chapter_queue; #define BFRM_DELAY_MAX 16 int * init_delay; int bfrm_delay; int64_t init_pts[BFRM_DELAY_MAX + 1]; hb_list_t * list_dts; int64_t frame_duration[FRAME_INFO_SIZE]; int async_depth; int max_async_depth; // if encode-only, system memory used int is_sys_mem; mfxSession mfx_session; struct SwsContext * sws_context_to_nv12; // whether to expect input from VPP or from QSV decode int is_vpp_present; // whether the encoder is initialized int init_done; hb_list_t * delayed_processing; hb_buffer_list_t encoded_frames; hb_list_t * loaded_plugins; }; static void hb_qsv_add_new_dts(hb_list_t *list, int64_t new_dts) { if (list != NULL) { int64_t *item = malloc(sizeof(int64_t)); if (item != NULL) { *item = new_dts; hb_list_add(list, item); } } } static int64_t hb_qsv_pop_next_dts(hb_list_t *list) { int64_t next_dts = INT64_MIN; if (list != NULL && hb_list_count(list) > 0) { int64_t *item = hb_list_item(list, 0); if (item != NULL) { next_dts = *item; hb_list_rem(list, item); free(item); } } return next_dts; } static void save_frame_duration(hb_work_private_t *pv, hb_buffer_t *buf) { int i = (buf->s.start >> FRAME_INFO_MAX2) & FRAME_INFO_MASK; pv->frame_duration[i] = buf->s.stop - buf->s.start; } static int64_t get_frame_duration(hb_work_private_t *pv, hb_buffer_t *buf) { int i = (buf->s.start >> FRAME_INFO_MAX2) & FRAME_INFO_MASK; return pv->frame_duration[i]; } static void qsv_handle_breftype(hb_work_private_t *pv) { /* * If B-pyramid is not possible (not supported, incompatible profile, etc.) * we don't need to adjust any settings, just sanitize it to off and return. */ if (!(pv->qsv_info->capabilities & HB_QSV_CAP_B_REF_PYRAMID)) { /* B-pyramid support not implemented */ goto unsupported; } else if (pv->param.videoParam->mfx.GopPicSize && pv->param.videoParam->mfx.GopPicSize <= 3) { /* GOP size must be at least 4 for B-pyramid */ goto unsupported; } else if (pv->param.videoParam->mfx.GopRefDist && pv->param.videoParam->mfx.GopRefDist <= 2) { /* We need 2 consecutive B-frames for B-pyramid (GopRefDist >= 3) */ goto unsupported; } else if (pv->param.videoParam->mfx.CodecId == MFX_CODEC_AVC) { switch (pv->param.videoParam->mfx.CodecProfile) { case MFX_PROFILE_AVC_BASELINE: case MFX_PROFILE_AVC_CONSTRAINED_HIGH: case MFX_PROFILE_AVC_CONSTRAINED_BASELINE: goto unsupported; // B-frames not allowed by profile default: break; } } else if (pv->param.videoParam->mfx.CodecId == MFX_CODEC_HEVC) { switch (pv->param.videoParam->mfx.CodecProfile) { case MFX_PROFILE_HEVC_MAINSP: goto unsupported; // B-frames not allowed by profile default: break; } } /* Handle B-pyramid auto (on for CQP, off otherwise) */ if (pv->param.gop.b_pyramid < 0) { pv->param.gop.b_pyramid = pv->param.videoParam->mfx.RateControlMethod == MFX_RATECONTROL_CQP; } if (pv->qsv_info->capabilities & HB_QSV_CAP_OPTION2_BREFTYPE) { /* B-pyramid can be controlled directly */ if (pv->param.gop.b_pyramid) { pv->param.codingOption2.BRefType = MFX_B_REF_PYRAMID; } else { pv->param.codingOption2.BRefType = MFX_B_REF_OFF; } } else { /* * We can't control B-pyramid directly, do it indirectly by * adjusting GopRefDist, GopPicSize and NumRefFrame instead. */ pv->param.codingOption2.BRefType = MFX_B_REF_UNKNOWN; /* * pyramid_ref_dist is the closest B-pyramid compatible * value (multiple of 2, >= 4) to the requested GopRefDist. */ int pyramid_ref_dist = 4; while (pv->param.videoParam->mfx.GopRefDist > pyramid_ref_dist) { pyramid_ref_dist *= 2; } if (pv->param.gop.b_pyramid) { /* GopRefDist must be B-pyramid compatible */ pv->param.videoParam->mfx.GopRefDist = pyramid_ref_dist; /* * GopPicSize must be a multiple of GopRefDist. * * Note: GopPicSize == 0 should always result in a value * that doesn't cause Media SDK to disable B-pyramid. */ if (pv->param.videoParam->mfx.GopPicSize) { pv->param.videoParam->mfx.GopPicSize = FFALIGN(pv->param.videoParam->mfx.GopPicSize, pv->param.videoParam->mfx.GopRefDist); } /* * NumRefFrame must be greater than 3 and than half of GopRefDist. * Otherwise, Media SDK may sometimes decide to disable B-pyramid * (whereas sometimes it will simply sanitize NumRefFrame instead). * * Note: Media SDK handles the NumRefFrame == 0 case for us. */ if (pv->param.videoParam->mfx.NumRefFrame) { pv->param.videoParam->mfx.NumRefFrame = FFMAX(pv->param.videoParam->mfx.NumRefFrame, pv->param.videoParam->mfx.GopRefDist / 2); pv->param.videoParam->mfx.NumRefFrame = FFMAX(pv->param.videoParam->mfx.NumRefFrame, 3); } } else if (pv->param.videoParam->mfx.GopRefDist == 0 || pv->param.videoParam->mfx.GopRefDist == pyramid_ref_dist) { /* * GopRefDist is either B-pyramid compatible or unknown (and thus * potentially compatible), so adjust it to force-disable B-pyramid. */ pv->param.videoParam->mfx.GopRefDist = pyramid_ref_dist - 1; } } return; unsupported: pv->param.gop.b_pyramid = 0; pv->param.codingOption2.BRefType = MFX_B_REF_OFF; } static int qsv_hevc_make_header(hb_work_object_t *w, mfxSession session) { size_t len; int ret = 0; uint8_t *buf, *end; mfxBitstream bitstream; hb_buffer_t *bitstream_buf; mfxStatus status; mfxSyncPoint syncPoint; mfxFrameSurface1 frameSurface1; hb_work_private_t *pv = w->private_data; memset(&bitstream, 0, sizeof(mfxBitstream)); memset(&syncPoint, 0, sizeof(mfxSyncPoint)); memset(&frameSurface1, 0, sizeof(mfxFrameSurface1)); /* The bitstream buffer should be able to hold any encoded frame */ bitstream_buf = hb_video_buffer_init(pv->job->width, pv->job->height); if (bitstream_buf == NULL) { hb_log("qsv_hevc_make_header: hb_buffer_init failed"); ret = -1; goto end; } /* need more space for 10bits */ if (pv->param.videoParam->mfx.FrameInfo.FourCC == MFX_FOURCC_P010) { hb_buffer_realloc(bitstream_buf,bitstream_buf->size*2); } bitstream.Data = bitstream_buf->data; bitstream.MaxLength = bitstream_buf->alloc; /* We only need to encode one frame, so we only need one surface */ mfxU16 Height = pv->param.videoParam->mfx.FrameInfo.Height; mfxU16 Width = pv->param.videoParam->mfx.FrameInfo.Width; frameSurface1.Info = pv->param.videoParam->mfx.FrameInfo; frameSurface1.Data.Y = av_mallocz(Width * Height * 3 / 2); frameSurface1.Data.VU = frameSurface1.Data.Y + Width * Height; frameSurface1.Data.Pitch = Width; /* Encode a single blank frame */ do { status = MFXVideoENCODE_EncodeFrameAsync(session, NULL, &frameSurface1, &bitstream, &syncPoint); if (status == MFX_ERR_MORE_DATA) { break; // more input needed, but we don't have any } if (status < MFX_ERR_NONE) { hb_log("qsv_hevc_make_header: MFXVideoENCODE_EncodeFrameAsync failed (%d)", status); ret = -1; goto end; } if (syncPoint) { break; // we have output } if (status == MFX_WRN_DEVICE_BUSY) { hb_qsv_sleep(1); } } while (status >= MFX_ERR_NONE); /* If we don't have any output yet, flush the encoder */ if (!syncPoint) { do { status = MFXVideoENCODE_EncodeFrameAsync(session, NULL, NULL, &bitstream, &syncPoint); if (status == MFX_ERR_MORE_DATA) { break; // done flushing } if (status < MFX_ERR_NONE) { hb_log("qsv_hevc_make_header: MFXVideoENCODE_EncodeFrameAsync failed (%d)", status); ret = -1; goto end; } if (syncPoint) { break; // we have output } if (status == MFX_WRN_DEVICE_BUSY) { hb_qsv_sleep(1); } } while (status >= MFX_ERR_NONE); } /* Still no data at this point, we can't proceed */ if (!syncPoint) { hb_log("qsv_hevc_make_header: no sync point"); ret = -1; goto end; } do { status = MFXVideoCORE_SyncOperation(session, syncPoint, 100); } while (status == MFX_WRN_IN_EXECUTION); if (status != MFX_ERR_NONE) { hb_log("qsv_hevc_make_header: MFXVideoCORE_SyncOperation failed (%d)", status); ret = -1; goto end; } if (!bitstream.DataLength) { hb_log("qsv_hevc_make_header: no bitstream data"); ret = -1; goto end; } /* Include any parameter sets and SEI NAL units in the headers. */ len = bitstream.DataLength; buf = bitstream.Data + bitstream.DataOffset; end = bitstream.Data + bitstream.DataOffset + bitstream.DataLength; w->config->h265.headers_length = 0; while ((buf = hb_annexb_find_next_nalu(buf, &len)) != NULL) { switch ((buf[0] >> 1) & 0x3f) { case 32: // VPS_NUT case 33: // SPS_NUT case 34: // PPS_NUT case 39: // PREFIX_SEI_NUT case 40: // SUFFIX_SEI_NUT break; default: len = end - buf; continue; } size_t size = hb_nal_unit_write_annexb(NULL, buf, len) + w->config->h265.headers_length; if (sizeof(w->config->h265.headers) < size) { /* Will never happen in practice */ hb_log("qsv_hevc_make_header: header too large (size: %lu, max: %lu)", size, sizeof(w->config->h265.headers)); } w->config->h265.headers_length += hb_nal_unit_write_annexb(w->config->h265.headers + w->config->h265.headers_length, buf, len); len = end - buf; } end: hb_buffer_close(&bitstream_buf); av_free(frameSurface1.Data.Y); return ret; } int qsv_enc_init(hb_work_private_t *pv) { hb_qsv_context *qsv = pv->job->qsv.ctx; hb_job_t *job = pv->job; mfxVersion version; mfxStatus sts; mfxIMPL impl; int i; if (pv->init_done) { return 0; } if (qsv == NULL) { if (!pv->is_sys_mem) { hb_error("qsv_enc_init: decode enabled but no context!"); return 3; } job->qsv.ctx = qsv = av_mallocz(sizeof(hb_qsv_context)); } hb_qsv_space *qsv_encode = qsv->enc_space; if (qsv_encode == NULL) { // if only for encode if (pv->is_sys_mem) { // no need to use additional sync as encode only -> single thread hb_qsv_add_context_usage(qsv, 0); // re-use the session from encqsvInit qsv->mfx_session = pv->mfx_session; } qsv->enc_space = qsv_encode = &pv->enc_space; } if (!pv->is_sys_mem) { if (!pv->is_vpp_present && job->list_filter != NULL) { for (i = 0; i < hb_list_count(job->list_filter); i++) { hb_filter_object_t *filter = hb_list_item(job->list_filter, i); if (filter->id == HB_FILTER_QSV_PRE || filter->id == HB_FILTER_QSV_POST || filter->id == HB_FILTER_QSV) { pv->is_vpp_present = 1; break; } } } if (pv->is_vpp_present) { if (qsv->vpp_space == NULL) { return 2; } for (i = 0; i < hb_qsv_list_count(qsv->vpp_space); i++) { hb_qsv_space *vpp = hb_qsv_list_item(qsv->vpp_space, i); if (!vpp->is_init_done) { return 2; } } } hb_qsv_space *dec_space = qsv->dec_space; if (dec_space == NULL || !dec_space->is_init_done) { return 2; } } else { if (pv->param.videoParam->mfx.CodecProfile == MFX_PROFILE_HEVC_MAIN10) { pv->sws_context_to_nv12 = hb_sws_get_context( job->width, job->height, AV_PIX_FMT_YUV420P, job->width, job->height, AV_PIX_FMT_P010LE, SWS_LANCZOS|SWS_ACCURATE_RND, SWS_CS_DEFAULT); } else { pv->sws_context_to_nv12 = hb_sws_get_context( job->width, job->height, AV_PIX_FMT_YUV420P, job->width, job->height, AV_PIX_FMT_NV12, SWS_LANCZOS|SWS_ACCURATE_RND, SWS_CS_DEFAULT); } } // allocate tasks qsv_encode->p_buf_max_size = HB_QSV_BUF_SIZE_DEFAULT; qsv_encode->tasks = hb_qsv_list_init(HAVE_THREADS); for (i = 0; i < pv->max_async_depth; i++) { hb_qsv_task *task = av_mallocz(sizeof(hb_qsv_task)); task->bs = av_mallocz(sizeof(mfxBitstream)); task->bs->Data = av_mallocz(sizeof(uint8_t) * qsv_encode->p_buf_max_size); task->bs->MaxLength = qsv_encode->p_buf_max_size; task->bs->DataLength = 0; task->bs->DataOffset = 0; hb_qsv_list_add(qsv_encode->tasks, task); } // plugins should be loaded before querying for surface allocation if (pv->loaded_plugins == NULL) { if (MFXQueryVersion(qsv->mfx_session, &version) != MFX_ERR_NONE) { hb_error("qsv_enc_init: MFXQueryVersion failed"); *job->done_error = HB_ERROR_INIT; *job->die = 1; return -1; } pv->loaded_plugins = hb_qsv_load_plugins(pv->qsv_info, qsv->mfx_session, version); if (pv->loaded_plugins == NULL) { hb_error("qsv_enc_init: hb_qsv_load_plugins failed"); *job->done_error = HB_ERROR_INIT; *job->die = 1; return -1; } } // setup surface allocation pv->param.videoParam->IOPattern = (pv->is_sys_mem ? MFX_IOPATTERN_IN_SYSTEM_MEMORY : MFX_IOPATTERN_IN_OPAQUE_MEMORY); memset(&qsv_encode->request, 0, sizeof(mfxFrameAllocRequest) * 2); sts = MFXVideoENCODE_QueryIOSurf(qsv->mfx_session, pv->param.videoParam, &qsv_encode->request[0]); if (sts < MFX_ERR_NONE) // ignore warnings { hb_error("qsv_enc_init: MFXVideoENCODE_QueryIOSurf failed (%d)", sts); *job->done_error = HB_ERROR_INIT; *job->die = 1; return -1; } // allocate surfaces if (pv->is_sys_mem) { qsv_encode->surface_num = FFMIN(qsv_encode->request[0].NumFrameSuggested + pv->max_async_depth, HB_QSV_SURFACE_NUM); if (qsv_encode->surface_num <= 0) { qsv_encode->surface_num = HB_QSV_SURFACE_NUM; } /* should have 15bpp/AV_PIX_FMT_YUV420P10LE (almost x2) instead of 12bpp/AV_PIX_FMT_NV12 */ int bpp12 = (pv->param.videoParam->mfx.CodecProfile == MFX_PROFILE_HEVC_MAIN10) ? 6 : 3; for (i = 0; i < qsv_encode->surface_num; i++) { mfxFrameSurface1 *surface = av_mallocz(sizeof(mfxFrameSurface1)); mfxFrameInfo info = pv->param.videoParam->mfx.FrameInfo; surface->Info = info; surface->Data.Pitch = info.Width * (bpp12 == 6 ? 2 : 1); surface->Data.Y = av_mallocz(info.Width * info.Height * (bpp12 / 2.0)); surface->Data.VU = surface->Data.Y + info.Width * info.Height * (bpp12 == 6 ? 2 : 1); qsv_encode->p_surfaces[i] = surface; } } else { hb_qsv_space *in_space = qsv->dec_space; if (pv->is_vpp_present) { // we get our input from VPP instead in_space = hb_qsv_list_item(qsv->vpp_space, hb_qsv_list_count(qsv->vpp_space) - 1); } // introduced in API 1.3 memset(&qsv_encode->ext_opaque_alloc, 0, sizeof(mfxExtOpaqueSurfaceAlloc)); qsv_encode->ext_opaque_alloc.Header.BufferId = MFX_EXTBUFF_OPAQUE_SURFACE_ALLOCATION; qsv_encode->ext_opaque_alloc.Header.BufferSz = sizeof(mfxExtOpaqueSurfaceAlloc); qsv_encode->ext_opaque_alloc.In.Surfaces = in_space->p_surfaces; qsv_encode->ext_opaque_alloc.In.NumSurface = in_space->surface_num; qsv_encode->ext_opaque_alloc.In.Type = qsv_encode->request[0].Type; pv->param.videoParam->ExtParam[pv->param.videoParam->NumExtParam++] = (mfxExtBuffer*)&qsv_encode->ext_opaque_alloc; } // allocate sync points qsv_encode->sync_num = (qsv_encode->surface_num ? FFMIN(qsv_encode->surface_num, HB_QSV_SYNC_NUM) : HB_QSV_SYNC_NUM); for (i = 0; i < qsv_encode->sync_num; i++) { qsv_encode->p_syncp[i] = av_mallocz(sizeof(hb_qsv_sync)); HB_QSV_CHECK_POINTER(qsv_encode->p_syncp[i], MFX_ERR_MEMORY_ALLOC); qsv_encode->p_syncp[i]->p_sync = av_mallocz(sizeof(mfxSyncPoint)); HB_QSV_CHECK_POINTER(qsv_encode->p_syncp[i]->p_sync, MFX_ERR_MEMORY_ALLOC); } // initialize the encoder sts = MFXVideoENCODE_Init(qsv->mfx_session, pv->param.videoParam); if (sts < MFX_ERR_NONE) // ignore warnings { hb_error("qsv_enc_init: MFXVideoENCODE_Init failed (%d)", sts); *job->done_error = HB_ERROR_INIT; *job->die = 1; return -1; } qsv_encode->is_init_done = 1; // query and log actual implementation details if ((MFXQueryIMPL (qsv->mfx_session, &impl) == MFX_ERR_NONE) && (MFXQueryVersion(qsv->mfx_session, &version) == MFX_ERR_NONE)) { hb_log("qsv_enc_init: using '%s' implementation, API: %"PRIu16".%"PRIu16"", hb_qsv_impl_get_name(impl), version.Major, version.Minor); } else { hb_log("qsv_enc_init: MFXQueryIMPL/MFXQueryVersion failure"); } pv->init_done = 1; return 0; } /*********************************************************************** * encqsvInit *********************************************************************** * **********************************************************************/ int encqsvInit(hb_work_object_t *w, hb_job_t *job) { hb_work_private_t *pv = calloc(1, sizeof(hb_work_private_t)); w->private_data = pv; pv->is_sys_mem = 1; // TODO: re-implement QSV VPP filtering support pv->job = job; pv->qsv_info = hb_qsv_info_get(job->vcodec); pv->delayed_processing = hb_list_init(); pv->last_start = INT64_MIN; hb_buffer_list_clear(&pv->encoded_frames); pv->chapter_queue = hb_chapter_queue_init(); // default encoding parameters if (hb_qsv_param_default_preset(&pv->param, &pv->enc_space.m_mfxVideoParam, pv->qsv_info, job->encoder_preset)) { hb_error("encqsvInit: hb_qsv_param_default_preset failed"); return -1; } // set AsyncDepth to match that of decode and VPP pv->param.videoParam->AsyncDepth = job->qsv.async_depth; // set and enable colorimetry (video signal information) switch (job->color_matrix_code) { case 5: // custom pv->param.videoSignalInfo.ColourPrimaries = job->color_prim; pv->param.videoSignalInfo.TransferCharacteristics = job->color_transfer; pv->param.videoSignalInfo.MatrixCoefficients = job->color_matrix; break; case 4: // ITU BT.2020 UHD content pv->param.videoSignalInfo.ColourPrimaries = HB_COLR_PRI_BT2020; pv->param.videoSignalInfo.TransferCharacteristics = HB_COLR_TRA_BT709; pv->param.videoSignalInfo.MatrixCoefficients = HB_COLR_MAT_BT2020_NCL; break; case 3: // ITU BT.709 HD content pv->param.videoSignalInfo.ColourPrimaries = HB_COLR_PRI_BT709; pv->param.videoSignalInfo.TransferCharacteristics = HB_COLR_TRA_BT709; pv->param.videoSignalInfo.MatrixCoefficients = HB_COLR_MAT_BT709; break; case 2: // ITU BT.601 DVD or SD TV content (PAL) pv->param.videoSignalInfo.ColourPrimaries = HB_COLR_PRI_EBUTECH; pv->param.videoSignalInfo.TransferCharacteristics = HB_COLR_TRA_BT709; pv->param.videoSignalInfo.MatrixCoefficients = HB_COLR_MAT_SMPTE170M; break; case 1: // ITU BT.601 DVD or SD TV content (NTSC) pv->param.videoSignalInfo.ColourPrimaries = HB_COLR_PRI_SMPTEC; pv->param.videoSignalInfo.TransferCharacteristics = HB_COLR_TRA_BT709; pv->param.videoSignalInfo.MatrixCoefficients = HB_COLR_MAT_SMPTE170M; break; default: // detected during scan pv->param.videoSignalInfo.ColourPrimaries = job->title->color_prim; pv->param.videoSignalInfo.TransferCharacteristics = job->title->color_transfer; pv->param.videoSignalInfo.MatrixCoefficients = job->title->color_matrix; break; } pv->param.videoSignalInfo.ColourDescriptionPresent = 1; // parse user-specified encoder options, if present if (job->encoder_options != NULL && *job->encoder_options) { hb_dict_t *options_list; options_list = hb_encopts_to_dict(job->encoder_options, job->vcodec); hb_dict_iter_t iter; for (iter = hb_dict_iter_init(options_list); iter != HB_DICT_ITER_DONE; iter = hb_dict_iter_next(options_list, iter)) { const char *key = hb_dict_iter_key(iter); hb_value_t *value = hb_dict_iter_value(iter); char *str = hb_value_get_string_xform(value); switch (hb_qsv_param_parse(&pv->param, pv->qsv_info, key, str)) { case HB_QSV_PARAM_OK: break; case HB_QSV_PARAM_BAD_NAME: hb_log("encqsvInit: hb_qsv_param_parse: bad key %s", key); break; case HB_QSV_PARAM_BAD_VALUE: hb_log("encqsvInit: hb_qsv_param_parse: bad value %s for key %s", str, key); break; case HB_QSV_PARAM_UNSUPPORTED: hb_log("encqsvInit: hb_qsv_param_parse: unsupported option %s", key); break; case HB_QSV_PARAM_ERROR: default: hb_log("encqsvInit: hb_qsv_param_parse: unknown error"); break; } free(str); } hb_dict_free(&options_list); } // reload colorimetry in case values were set in encoder_options if (pv->param.videoSignalInfo.ColourDescriptionPresent) { job->color_matrix_code = 4; job->color_prim = pv->param.videoSignalInfo.ColourPrimaries; job->color_transfer = pv->param.videoSignalInfo.TransferCharacteristics; job->color_matrix = pv->param.videoSignalInfo.MatrixCoefficients; } else { job->color_matrix_code = 0; job->color_prim = HB_COLR_PRI_UNDEF; job->color_transfer = HB_COLR_TRA_UNDEF; job->color_matrix = HB_COLR_MAT_UNDEF; } // sanitize values that may exceed the Media SDK variable size hb_rational_t par; hb_limit_rational(&par.num, &par.den, job->par.num, job->par.den, UINT16_MAX); // some encoding parameters are used by filters to configure their output switch (pv->qsv_info->codec_id) { case MFX_CODEC_HEVC: job->qsv.enc_info.align_width = HB_QSV_ALIGN32(job->width); job->qsv.enc_info.align_height = HB_QSV_ALIGN32(job->height); break; case MFX_CODEC_AVC: default: job->qsv.enc_info.align_width = HB_QSV_ALIGN16(job->width); job->qsv.enc_info.align_height = HB_QSV_ALIGN16(job->height); break; } if (pv->param.videoParam->mfx.FrameInfo.PicStruct != MFX_PICSTRUCT_PROGRESSIVE) { // additional alignment may be required switch (pv->qsv_info->codec_id) { case MFX_CODEC_AVC: job->qsv.enc_info.align_height = HB_QSV_ALIGN32(job->qsv.enc_info.align_height); break; default: break; } } job->qsv.enc_info.pic_struct = pv->param.videoParam->mfx.FrameInfo.PicStruct; job->qsv.enc_info.is_init_done = 1; // set codec, profile/level and FrameInfo pv->param.videoParam->mfx.CodecId = pv->qsv_info->codec_id; pv->param.videoParam->mfx.CodecLevel = MFX_LEVEL_UNKNOWN; pv->param.videoParam->mfx.CodecProfile = MFX_PROFILE_UNKNOWN; pv->param.videoParam->mfx.FrameInfo.FourCC = MFX_FOURCC_NV12; pv->param.videoParam->mfx.FrameInfo.ChromaFormat = MFX_CHROMAFORMAT_YUV420; pv->param.videoParam->mfx.FrameInfo.FrameRateExtN = job->vrate.num; pv->param.videoParam->mfx.FrameInfo.FrameRateExtD = job->vrate.den; pv->param.videoParam->mfx.FrameInfo.AspectRatioW = par.num; pv->param.videoParam->mfx.FrameInfo.AspectRatioH = par.den; pv->param.videoParam->mfx.FrameInfo.CropX = 0; pv->param.videoParam->mfx.FrameInfo.CropY = 0; pv->param.videoParam->mfx.FrameInfo.CropW = job->width; pv->param.videoParam->mfx.FrameInfo.CropH = job->height; pv->param.videoParam->mfx.FrameInfo.PicStruct = job->qsv.enc_info.pic_struct; pv->param.videoParam->mfx.FrameInfo.Width = job->qsv.enc_info.align_width; pv->param.videoParam->mfx.FrameInfo.Height = job->qsv.enc_info.align_height; // parse user-specified codec profile and level if (hb_qsv_profile_parse(&pv->param, pv->qsv_info, job->encoder_profile)) { hb_error("encqsvInit: bad profile %s", job->encoder_profile); return -1; } if (hb_qsv_level_parse(&pv->param, pv->qsv_info, job->encoder_level)) { hb_error("encqsvInit: bad level %s", job->encoder_level); return -1; } if (pv->param.videoParam->mfx.CodecProfile == MFX_PROFILE_HEVC_MAIN10) { pv->param.videoParam->mfx.FrameInfo.FourCC = MFX_FOURCC_P010; pv->param.videoParam->mfx.FrameInfo.BitDepthLuma = 10; pv->param.videoParam->mfx.FrameInfo.BitDepthChroma = 10; pv->param.videoParam->mfx.FrameInfo.Shift = 0; } // interlaced encoding is not always possible if (pv->param.videoParam->mfx.CodecId == MFX_CODEC_AVC && pv->param.videoParam->mfx.FrameInfo.PicStruct != MFX_PICSTRUCT_PROGRESSIVE) { if (pv->param.videoParam->mfx.CodecProfile == MFX_PROFILE_AVC_CONSTRAINED_BASELINE || pv->param.videoParam->mfx.CodecProfile == MFX_PROFILE_AVC_BASELINE || pv->param.videoParam->mfx.CodecProfile == MFX_PROFILE_AVC_PROGRESSIVE_HIGH) { hb_error("encqsvInit: profile %s doesn't support interlaced encoding", hb_qsv_profile_name(MFX_CODEC_AVC, pv->param.videoParam->mfx.CodecProfile)); return -1; } if ((pv->param.videoParam->mfx.CodecLevel >= MFX_LEVEL_AVC_1b && pv->param.videoParam->mfx.CodecLevel <= MFX_LEVEL_AVC_2) || (pv->param.videoParam->mfx.CodecLevel >= MFX_LEVEL_AVC_42)) { hb_error("encqsvInit: level %s doesn't support interlaced encoding", hb_qsv_level_name(MFX_CODEC_AVC, pv->param.videoParam->mfx.CodecLevel)); return -1; } } // sanitize ICQ if (!(pv->qsv_info->capabilities & HB_QSV_CAP_RATECONTROL_ICQ)) { // ICQ not supported pv->param.rc.icq = 0; } else { pv->param.rc.icq = pv->param.rc.icq && job->vquality > HB_INVALID_VIDEO_QUALITY; } // sanitize lookahead if (!(pv->qsv_info->capabilities & HB_QSV_CAP_RATECONTROL_LA)) { // lookahead not supported pv->param.rc.lookahead = 0; } else if ((pv->param.rc.lookahead) && (pv->qsv_info->capabilities & HB_QSV_CAP_RATECONTROL_LAi) == 0 && (pv->param.videoParam->mfx.FrameInfo.PicStruct != MFX_PICSTRUCT_PROGRESSIVE)) { // lookahead enabled but we can't use it hb_log("encqsvInit: LookAhead not used (LookAhead is progressive-only)"); pv->param.rc.lookahead = 0; } else { pv->param.rc.lookahead = pv->param.rc.lookahead && (pv->param.rc.icq || job->vquality <= HB_INVALID_VIDEO_QUALITY); } // set VBV here (this will be overridden for CQP and ignored for LA) // only set BufferSizeInKB, InitialDelayInKB and MaxKbps if we have // them - otheriwse Media SDK will pick values for us automatically if (pv->param.rc.vbv_buffer_size > 0) { if (pv->param.rc.vbv_buffer_init > 1.0) { pv->param.videoParam->mfx.InitialDelayInKB = (pv->param.rc.vbv_buffer_init / 8); } else if (pv->param.rc.vbv_buffer_init > 0.0) { pv->param.videoParam->mfx.InitialDelayInKB = (pv->param.rc.vbv_buffer_size * pv->param.rc.vbv_buffer_init / 8); } pv->param.videoParam->mfx.BufferSizeInKB = (pv->param.rc.vbv_buffer_size / 8); } if (pv->param.rc.vbv_max_bitrate > 0) { pv->param.videoParam->mfx.MaxKbps = pv->param.rc.vbv_max_bitrate; } // set rate control paremeters if (job->vquality > HB_INVALID_VIDEO_QUALITY) { if (pv->param.rc.icq) { // introduced in API 1.8 if (pv->param.rc.lookahead) { pv->param.videoParam->mfx.RateControlMethod = MFX_RATECONTROL_LA_ICQ; } else { pv->param.videoParam->mfx.RateControlMethod = MFX_RATECONTROL_ICQ; } pv->param.videoParam->mfx.ICQQuality = HB_QSV_CLIP3(1, 51, job->vquality); } else { // introduced in API 1.1 pv->param.videoParam->mfx.RateControlMethod = MFX_RATECONTROL_CQP; pv->param.videoParam->mfx.QPI = HB_QSV_CLIP3(0, 51, job->vquality + pv->param.rc.cqp_offsets[0]); pv->param.videoParam->mfx.QPP = HB_QSV_CLIP3(0, 51, job->vquality + pv->param.rc.cqp_offsets[1]); pv->param.videoParam->mfx.QPB = HB_QSV_CLIP3(0, 51, job->vquality + pv->param.rc.cqp_offsets[2]); // CQP + ExtBRC can cause bad output pv->param.codingOption2.ExtBRC = MFX_CODINGOPTION_OFF; } } else if (job->vbitrate > 0) { if (pv->param.rc.lookahead) { // introduced in API 1.7 pv->param.videoParam->mfx.RateControlMethod = MFX_RATECONTROL_LA; pv->param.videoParam->mfx.TargetKbps = job->vbitrate; // ignored, but some drivers will change AsyncDepth because of it pv->param.codingOption2.ExtBRC = MFX_CODINGOPTION_OFF; } else { // introduced in API 1.0 if (job->vbitrate == pv->param.rc.vbv_max_bitrate) { pv->param.videoParam->mfx.RateControlMethod = MFX_RATECONTROL_CBR; } else { pv->param.videoParam->mfx.RateControlMethod = MFX_RATECONTROL_VBR; } pv->param.videoParam->mfx.TargetKbps = job->vbitrate; } } else { hb_error("encqsvInit: invalid rate control (%f, %d)", job->vquality, job->vbitrate); return -1; } // if VBV is enabled but ignored, log it if (pv->param.rc.vbv_max_bitrate > 0 || pv->param.rc.vbv_buffer_size > 0) { switch (pv->param.videoParam->mfx.RateControlMethod) { case MFX_RATECONTROL_LA: case MFX_RATECONTROL_LA_ICQ: hb_log("encqsvInit: LookAhead enabled, ignoring VBV"); break; case MFX_RATECONTROL_ICQ: hb_log("encqsvInit: ICQ rate control, ignoring VBV"); break; default: break; } } // set the GOP structure if (pv->param.gop.gop_ref_dist < 0) { if (pv->param.videoParam->mfx.RateControlMethod == MFX_RATECONTROL_CQP) { pv->param.gop.gop_ref_dist = 4; } else { pv->param.gop.gop_ref_dist = 3; } } pv->param.videoParam->mfx.GopRefDist = pv->param.gop.gop_ref_dist; // set the keyframe interval if (pv->param.gop.gop_pic_size < 0) { int rate = (double)job->orig_vrate.num / job->orig_vrate.den + 0.5; if (pv->param.videoParam->mfx.RateControlMethod == MFX_RATECONTROL_CQP) { // ensure B-pyramid is enabled for CQP on Haswell pv->param.gop.gop_pic_size = 32; } else { // set the keyframe interval based on the framerate pv->param.gop.gop_pic_size = rate; } } pv->param.videoParam->mfx.GopPicSize = pv->param.gop.gop_pic_size; // sanitize some settings that affect memory consumption if (!pv->is_sys_mem) { // limit these to avoid running out of resources (causes hang) pv->param.videoParam->mfx.GopRefDist = FFMIN(pv->param.videoParam->mfx.GopRefDist, pv->param.rc.lookahead ? 8 : 16); pv->param.codingOption2.LookAheadDepth = FFMIN(pv->param.codingOption2.LookAheadDepth, pv->param.rc.lookahead ? (48 - pv->param.videoParam->mfx.GopRefDist - 3 * !pv->param.videoParam->mfx.GopRefDist) : 0); } else { // encode-only is a bit less sensitive to memory issues pv->param.videoParam->mfx.GopRefDist = FFMIN(pv->param.videoParam->mfx.GopRefDist, 16); pv->param.codingOption2.LookAheadDepth = FFMIN(pv->param.codingOption2.LookAheadDepth, pv->param.rc.lookahead ? 100 : 0); } if (pv->param.rc.lookahead) { // LookAheadDepth 10 will cause a hang with some driver versions pv->param.codingOption2.LookAheadDepth = FFMAX(pv->param.codingOption2.LookAheadDepth, 11); } /* * We may need to adjust GopRefDist, GopPicSize and * NumRefFrame to enable or disable B-pyramid, so do it last. */ qsv_handle_breftype(pv); /* * init a dummy encode-only session to get the SPS/PPS * and the final output settings sanitized by Media SDK * this is fine since the actual encode will use the same * values for all parameters relevant to the output bitstream */ mfxStatus err; mfxVersion version; mfxVideoParam videoParam; mfxExtBuffer *extParamArray[3]; mfxSession session = (mfxSession)0; mfxExtCodingOption option1_buf, *option1 = &option1_buf; mfxExtCodingOption2 option2_buf, *option2 = &option2_buf; mfxExtCodingOptionSPSPPS sps_pps_buf, *sps_pps = &sps_pps_buf; version.Major = HB_QSV_MINVERSION_MAJOR; version.Minor = HB_QSV_MINVERSION_MINOR; err = MFXInit(pv->qsv_info->implementation, &version, &session); if (err != MFX_ERR_NONE) { hb_error("encqsvInit: MFXInit failed (%d)", err); return -1; } /* Query the API version for hb_qsv_load_plugins */ err = MFXQueryVersion(session, &version); if (err != MFX_ERR_NONE) { hb_error("encqsvInit: MFXQueryVersion failed (%d)", err); MFXClose(session); return -1; } /* Load required MFX plug-ins */ pv->loaded_plugins = hb_qsv_load_plugins(pv->qsv_info, session, version); if (pv->loaded_plugins == NULL) { hb_error("encqsvInit: hb_qsv_load_plugins failed"); MFXClose(session); return -1; } err = MFXVideoENCODE_Init(session, pv->param.videoParam); // workaround for the early 15.33.x driver, should be removed later #define HB_DRIVER_FIX_33 #ifdef HB_DRIVER_FIX_33 int la_workaround = 0; if (err < MFX_ERR_NONE && pv->param.videoParam->mfx.RateControlMethod == MFX_RATECONTROL_LA) { pv->param.videoParam->mfx.RateControlMethod = MFX_RATECONTROL_CBR; err = MFXVideoENCODE_Init(session, pv->param.videoParam); la_workaround = 1; } #endif if (err < MFX_ERR_NONE) // ignore warnings { hb_error("encqsvInit: MFXVideoENCODE_Init failed (%d)", err); hb_qsv_unload_plugins(&pv->loaded_plugins, session, version); MFXClose(session); return -1; } memset(&videoParam, 0, sizeof(mfxVideoParam)); videoParam.ExtParam = extParamArray; videoParam.NumExtParam = 0; // introduced in API 1.3 memset(sps_pps, 0, sizeof(mfxExtCodingOptionSPSPPS)); sps_pps->Header.BufferId = MFX_EXTBUFF_CODING_OPTION_SPSPPS; sps_pps->Header.BufferSz = sizeof(mfxExtCodingOptionSPSPPS); sps_pps->SPSId = 0; sps_pps->SPSBuffer = w->config->h264.sps; sps_pps->SPSBufSize = sizeof(w->config->h264.sps); sps_pps->PPSId = 0; sps_pps->PPSBuffer = w->config->h264.pps; sps_pps->PPSBufSize = sizeof(w->config->h264.pps); if (pv->param.videoParam->mfx.CodecId == MFX_CODEC_AVC) { videoParam.ExtParam[videoParam.NumExtParam++] = (mfxExtBuffer*)sps_pps; } // introduced in API 1.0 memset(option1, 0, sizeof(mfxExtCodingOption)); option1->Header.BufferId = MFX_EXTBUFF_CODING_OPTION; option1->Header.BufferSz = sizeof(mfxExtCodingOption); if (pv->qsv_info->capabilities & HB_QSV_CAP_OPTION1) { videoParam.ExtParam[videoParam.NumExtParam++] = (mfxExtBuffer*)option1; } // introduced in API 1.6 memset(option2, 0, sizeof(mfxExtCodingOption2)); option2->Header.BufferId = MFX_EXTBUFF_CODING_OPTION2; option2->Header.BufferSz = sizeof(mfxExtCodingOption2); if (pv->qsv_info->capabilities & HB_QSV_CAP_OPTION2) { videoParam.ExtParam[videoParam.NumExtParam++] = (mfxExtBuffer*)option2; } err = MFXVideoENCODE_GetVideoParam(session, &videoParam); if (err != MFX_ERR_NONE) { hb_error("encqsvInit: MFXVideoENCODE_GetVideoParam failed (%d)", err); hb_qsv_unload_plugins(&pv->loaded_plugins, session, version); MFXClose(session); return -1; } /* We have the final encoding parameters, now get the headers for muxing */ if (videoParam.mfx.CodecId == MFX_CODEC_AVC) { // remove 4-byte Annex B NAL unit prefix (0x00 0x00 0x00 0x01) w->config->h264.sps_length = sps_pps->SPSBufSize - 4; memmove(w->config->h264.sps, w->config->h264.sps + 4, w->config->h264.sps_length); w->config->h264.pps_length = sps_pps->PPSBufSize - 4; memmove(w->config->h264.pps, w->config->h264.pps + 4, w->config->h264.pps_length); } else if (videoParam.mfx.CodecId == MFX_CODEC_HEVC) { if (qsv_hevc_make_header(w, session) < 0) { hb_error("encqsvInit: qsv_hevc_make_header failed"); hb_qsv_unload_plugins(&pv->loaded_plugins, session, version); MFXVideoENCODE_Close(session); MFXClose(session); return -1; } } /* We don't need this encode session once we have the header */ MFXVideoENCODE_Close(session); #ifdef HB_DRIVER_FIX_33 if (la_workaround) { videoParam.mfx.RateControlMethod = pv->param.videoParam->mfx.RateControlMethod = MFX_RATECONTROL_LA; option2->LookAheadDepth = pv->param.codingOption2.LookAheadDepth; hb_log("encqsvInit: using LookAhead workaround (\"early 33 fix\")"); } #endif // when using system memory, we re-use this same session if (pv->is_sys_mem) { pv->mfx_session = session; } else { hb_qsv_unload_plugins(&pv->loaded_plugins, session, version); MFXClose(session); } /* B-frame related setup */ if (videoParam.mfx.GopRefDist > 1) { /* the muxer needs to know to the init_delay */ switch (videoParam.mfx.CodecId) { case MFX_CODEC_AVC: case MFX_CODEC_HEVC: pv->init_delay = &w->config->init_delay; break; default: // unreachable break; } /* let the muxer know that it should expect B-frames */ job->areBframes = 1; /* holds the PTS sequence in display order, used to generate DTS */ pv->list_dts = hb_list_init(); } // log code path and main output settings hb_log("encqsvInit: using %s path", pv->is_sys_mem ? "encode-only" : "full QSV"); hb_log("encqsvInit: %s %s profile @ level %s", hb_qsv_codec_name (videoParam.mfx.CodecId), hb_qsv_profile_name(videoParam.mfx.CodecId, videoParam.mfx.CodecProfile), hb_qsv_level_name (videoParam.mfx.CodecId, videoParam.mfx.CodecLevel)); hb_log("encqsvInit: TargetUsage %"PRIu16" AsyncDepth %"PRIu16"", videoParam.mfx.TargetUsage, videoParam.AsyncDepth); hb_log("encqsvInit: GopRefDist %"PRIu16" GopPicSize %"PRIu16" NumRefFrame %"PRIu16"", videoParam.mfx.GopRefDist, videoParam.mfx.GopPicSize, videoParam.mfx.NumRefFrame); if (pv->qsv_info->capabilities & HB_QSV_CAP_B_REF_PYRAMID) { hb_log("encqsvInit: BFramesMax %d BRefType %s", videoParam.mfx.GopRefDist > 1 ? videoParam.mfx.GopRefDist - 1 : 0, pv->param.gop.b_pyramid ? "pyramid" : "off"); } else { hb_log("encqsvInit: BFramesMax %d", videoParam.mfx.GopRefDist > 1 ? videoParam.mfx.GopRefDist - 1 : 0); } if (pv->qsv_info->capabilities & HB_QSV_CAP_OPTION2_IB_ADAPT) { if (option2->AdaptiveI != MFX_CODINGOPTION_OFF || option2->AdaptiveB != MFX_CODINGOPTION_OFF) { if (videoParam.mfx.GopRefDist > 1) { hb_log("encqsvInit: AdaptiveI %s AdaptiveB %s", hb_qsv_codingoption_get_name(option2->AdaptiveI), hb_qsv_codingoption_get_name(option2->AdaptiveB)); } else { hb_log("encqsvInit: AdaptiveI %s", hb_qsv_codingoption_get_name(option2->AdaptiveI)); } } } if (videoParam.mfx.RateControlMethod == MFX_RATECONTROL_CQP) { char qpi[7], qpp[9], qpb[9]; snprintf(qpi, sizeof(qpi), "QPI %"PRIu16"", videoParam.mfx.QPI); snprintf(qpp, sizeof(qpp), " QPP %"PRIu16"", videoParam.mfx.QPP); snprintf(qpb, sizeof(qpb), " QPB %"PRIu16"", videoParam.mfx.QPB); hb_log("encqsvInit: RateControlMethod CQP with %s%s%s", qpi, videoParam.mfx.GopPicSize > 1 ? qpp : "", videoParam.mfx.GopRefDist > 1 ? qpb : ""); } else { switch (videoParam.mfx.RateControlMethod) { case MFX_RATECONTROL_LA: hb_log("encqsvInit: RateControlMethod LA TargetKbps %"PRIu16" LookAheadDepth %"PRIu16"", videoParam.mfx.TargetKbps, option2->LookAheadDepth); break; case MFX_RATECONTROL_LA_ICQ: hb_log("encqsvInit: RateControlMethod LA_ICQ ICQQuality %"PRIu16" LookAheadDepth %"PRIu16"", videoParam.mfx.ICQQuality, option2->LookAheadDepth); break; case MFX_RATECONTROL_ICQ: hb_log("encqsvInit: RateControlMethod ICQ ICQQuality %"PRIu16"", videoParam.mfx.ICQQuality); break; case MFX_RATECONTROL_CBR: case MFX_RATECONTROL_VBR: hb_log("encqsvInit: RateControlMethod %s TargetKbps %"PRIu16" MaxKbps %"PRIu16" BufferSizeInKB %"PRIu16" InitialDelayInKB %"PRIu16"", videoParam.mfx.RateControlMethod == MFX_RATECONTROL_CBR ? "CBR" : "VBR", videoParam.mfx.TargetKbps, videoParam.mfx.MaxKbps, videoParam.mfx.BufferSizeInKB, videoParam.mfx.InitialDelayInKB); break; default: hb_log("encqsvInit: invalid rate control method %"PRIu16"", videoParam.mfx.RateControlMethod); return -1; } } if ((pv->qsv_info->capabilities & HB_QSV_CAP_OPTION2_LA_DOWNS) && (videoParam.mfx.RateControlMethod == MFX_RATECONTROL_LA || videoParam.mfx.RateControlMethod == MFX_RATECONTROL_LA_ICQ)) { switch (option2->LookAheadDS) { case MFX_LOOKAHEAD_DS_UNKNOWN: hb_log("encqsvInit: LookAheadDS unknown (auto)"); break; case MFX_LOOKAHEAD_DS_OFF: // default break; case MFX_LOOKAHEAD_DS_2x: hb_log("encqsvInit: LookAheadDS 2x"); break; case MFX_LOOKAHEAD_DS_4x: hb_log("encqsvInit: LookAheadDS 4x"); break; default: hb_log("encqsvInit: invalid LookAheadDS value 0x%"PRIx16"", option2->LookAheadDS); break; } } switch (videoParam.mfx.FrameInfo.PicStruct) { case MFX_PICSTRUCT_PROGRESSIVE: // default break; case MFX_PICSTRUCT_FIELD_TFF: hb_log("encqsvInit: PicStruct top field first"); break; case MFX_PICSTRUCT_FIELD_BFF: hb_log("encqsvInit: PicStruct bottom field first"); break; default: hb_error("encqsvInit: invalid PicStruct value 0x%"PRIx16"", videoParam.mfx.FrameInfo.PicStruct); return -1; } if (pv->qsv_info->capabilities & HB_QSV_CAP_OPTION1) { if (videoParam.mfx.CodecId == MFX_CODEC_AVC) { if (option1->CAVLC != MFX_CODINGOPTION_OFF) { hb_log("encqsvInit: CAVLC %s", hb_qsv_codingoption_get_name(option1->CAVLC)); } } } if (pv->qsv_info->capabilities & HB_QSV_CAP_OPTION2_EXTBRC) { if (option2->ExtBRC != MFX_CODINGOPTION_OFF) { hb_log("encqsvInit: ExtBRC %s", hb_qsv_codingoption_get_name(option2->ExtBRC)); } } if (pv->qsv_info->capabilities & HB_QSV_CAP_OPTION2_MBBRC) { if (option2->MBBRC != MFX_CODINGOPTION_ON) { hb_log("encqsvInit: MBBRC %s", hb_qsv_codingoption_get_name(option2->MBBRC)); } } if (pv->qsv_info->capabilities & HB_QSV_CAP_OPTION2_TRELLIS) { switch (option2->Trellis) { case MFX_TRELLIS_OFF: // default break; case MFX_TRELLIS_UNKNOWN: hb_log("encqsvInit: Trellis unknown (auto)"); break; default: hb_log("encqsvInit: Trellis on (%s%s%s)", (option2->Trellis & MFX_TRELLIS_I) ? "I" : "", (option2->Trellis & MFX_TRELLIS_P) && (videoParam.mfx.GopPicSize > 1) ? "P" : "", (option2->Trellis & MFX_TRELLIS_B) && (videoParam.mfx.GopRefDist > 1) ? "B" : ""); break; } } // AsyncDepth has now been set and/or modified by Media SDK // fall back to default if zero pv->max_async_depth = videoParam.AsyncDepth ? videoParam.AsyncDepth : HB_QSV_ASYNC_DEPTH_DEFAULT; pv->async_depth = 0; return 0; } void encqsvClose(hb_work_object_t *w) { hb_work_private_t *pv = w->private_data; mfxVersion version; int i; if (pv != NULL && pv->job != NULL && pv->job->qsv.ctx != NULL && pv->job->qsv.ctx->is_context_active) { hb_qsv_context *qsv_ctx = pv->job->qsv.ctx; hb_qsv_space *qsv_enc_space = pv->job->qsv.ctx->enc_space; if (qsv_ctx != NULL) { /* Unload MFX plug-ins */ if (MFXQueryVersion(qsv_ctx->mfx_session, &version) == MFX_ERR_NONE) { hb_qsv_unload_plugins(&pv->loaded_plugins, qsv_ctx->mfx_session, version); } /* QSV context cleanup and MFXClose */ hb_qsv_context_clean(qsv_ctx); if (qsv_enc_space != NULL) { if (qsv_enc_space->is_init_done) { for (i = hb_qsv_list_count(qsv_enc_space->tasks); i >= 1; i--) { hb_qsv_task *task = hb_qsv_list_item(qsv_enc_space->tasks, i - 1); if (task != NULL) { if (task->bs != NULL) { av_freep(&task->bs->Data); } hb_qsv_list_rem(qsv_enc_space->tasks, task); av_freep(&task->bs); av_freep(&task); } } hb_qsv_list_close(&qsv_enc_space->tasks); for (i = 0; i < qsv_enc_space->surface_num; i++) { if (pv->is_sys_mem) { av_freep(&qsv_enc_space->p_surfaces[i]->Data.Y); } av_freep(&qsv_enc_space->p_surfaces[i]); } qsv_enc_space->surface_num = 0; for (i = 0; i < qsv_enc_space->sync_num; i++) { av_freep(&qsv_enc_space->p_syncp[i]->p_sync); av_freep(&qsv_enc_space->p_syncp[i]); } qsv_enc_space->sync_num = 0; } qsv_enc_space->is_init_done = 0; } if (pv->is_sys_mem) { av_freep(&qsv_ctx); } } } if (pv != NULL) { hb_chapter_queue_close(&pv->chapter_queue); if (pv->delayed_processing != NULL) { /* the list is already empty */ hb_list_close(&pv->delayed_processing); } if (pv->sws_context_to_nv12 != NULL) { sws_freeContext(pv->sws_context_to_nv12); } if (pv->list_dts != NULL) { int64_t *item; while ((item = hb_list_item(pv->list_dts, 0)) != NULL) { hb_list_rem(pv->list_dts, item); free(item); } hb_list_close(&pv->list_dts); } hb_buffer_list_close(&pv->encoded_frames); } free(pv); w->private_data = NULL; } static void compute_init_delay(hb_work_private_t *pv, mfxBitstream *bs) { if (pv->init_delay == NULL) { return; // not needed or already set } /* * In the MP4 container, DT(0) = STTS(0) = 0. * * Which gives us: * CT(0) = CTTS(0) + STTS(0) = CTTS(0) = PTS(0) - DTS(0) * When DTS(0) < PTS(0), we then have: * CT(0) > 0 for video, but not audio (breaks A/V sync). * * This is typically solved by writing an edit list shifting * video samples by the initial delay, PTS(0) - DTS(0). * * See: * ISO/IEC 14496-12:2008(E), ISO base media file format * - 8.6.1.2 Decoding Time to Sample Box */ if (pv->qsv_info->capabilities & HB_QSV_CAP_MSDK_API_1_6) { /* compute init_delay (in ticks) based on the DTS provided by MSDK. */ int64_t init_delay = bs->TimeStamp - bs->DecodeTimeStamp; /* * we also need to know the delay in frames to generate DTS. * * compute it based on the init_delay and average frame duration, * and account for potential rounding errors due to the timebase. */ double avg_frame_dur = ((double)pv->job->vrate.den / (double)pv->job->vrate.num * 90000.); pv->bfrm_delay = (init_delay + (avg_frame_dur / 2)) / avg_frame_dur; if (pv->bfrm_delay < 1 || pv->bfrm_delay > BFRM_DELAY_MAX) { hb_log("compute_init_delay: " "invalid delay %d (PTS: %"PRIu64", DTS: %"PRId64")", pv->bfrm_delay, bs->TimeStamp, bs->DecodeTimeStamp); /* we have B-frames, the frame delay should be at least 1 */ if (pv->bfrm_delay < 1) { mfxStatus sts; mfxVideoParam videoParam; mfxSession session = pv->job->qsv.ctx->mfx_session; memset(&videoParam, 0, sizeof(mfxVideoParam)); sts = MFXVideoENCODE_GetVideoParam(session, &videoParam); if (sts != MFX_ERR_NONE) { hb_log("compute_init_delay: " "MFXVideoENCODE_GetVideoParam failed (%d)", sts); pv->bfrm_delay = 1; } else { /* usually too large, but should cover all cases */ pv->bfrm_delay = FFMIN(pv->frames_in - 1, videoParam.mfx.GopRefDist - 1); } } pv->bfrm_delay = FFMIN(BFRM_DELAY_MAX, pv->bfrm_delay); } pv->init_delay[0] = pv->init_pts[pv->bfrm_delay] - pv->init_pts[0]; } else { /* * we can't get the DTS from MSDK, so we need to generate our own. * * B-pyramid not possible here, so the delay in frames is always 1. */ pv->bfrm_delay = 1; pv->init_delay[0] = pv->init_pts[1] - pv->init_pts[0]; } /* This can come in handy */ hb_deep_log(2, "compute_init_delay: %d (%d frames)", pv->init_delay[0], pv->bfrm_delay); /* The delay only needs to be set once. */ pv->init_delay = NULL; } static void qsv_bitstream_slurp(hb_work_private_t *pv, mfxBitstream *bs) { hb_buffer_t *buf; if (pv->param.videoParam->mfx.CodecId == MFX_CODEC_AVC) { /* * We provided the muxer with the parameter sets in an MP4-compatible * format (ISO/IEC 14496-15). We need to convert the bitstream to the * same format to match the extradata. */ if ((buf = hb_nal_bitstream_annexb_to_mp4(bs->Data + bs->DataOffset, bs->DataLength)) == NULL) { hb_error("encqsv: hb_nal_bitstream_annexb_to_mp4 failed"); goto fail; } } else { /* Both extradata and bitstream are in Annex B format. */ if ((buf = hb_buffer_init(bs->DataLength)) == NULL) { hb_error("encqsv: hb_buffer_init failed"); goto fail; } memcpy(buf->data, bs->Data + bs->DataOffset, bs->DataLength); } bs->DataLength = bs->DataOffset = 0; bs->MaxLength = pv->job->qsv.ctx->enc_space->p_buf_max_size; buf->s.frametype = hb_qsv_frametype_xlat(bs->FrameType, &buf->s.flags); if (pv->param.videoParam->mfx.CodecId == MFX_CODEC_HEVC) { size_t len = buf->size; uint8_t *pos = buf->data; uint8_t *end = pos + len; while ((pos = hb_annexb_find_next_nalu(pos, &len)) != NULL) { if (HB_HEVC_NALU_KEYFRAME((pos[0] >> 1) & 0x3f)) { buf->s.flags |= HB_FLAG_FRAMETYPE_KEY; break; } len = end - pos; continue; } } buf->s.start = buf->s.renderOffset = bs->TimeStamp; buf->s.stop = buf->s.start + get_frame_duration(pv, buf); buf->s.duration = buf->s.stop - buf->s.start; /* compute the init_delay before setting the DTS */ compute_init_delay(pv, bs); /* * Generate VFR-compatible output DTS based on input PTS. * * Depends on the B-frame delay: * * 0: ipts0, ipts1, ipts2... * 1: ipts0 - ipts1, ipts1 - ipts1, ipts1, ipts2... * 2: ipts0 - ipts2, ipts1 - ipts2, ipts2 - ipts2, ipts1... * ...and so on. */ if (pv->bfrm_delay) { if (pv->frames_out <= pv->bfrm_delay) { buf->s.renderOffset = (pv->init_pts[pv->frames_out] - pv->init_pts[pv->bfrm_delay]); } else { buf->s.renderOffset = hb_qsv_pop_next_dts(pv->list_dts); } } /* check if B-pyramid is used even though it's disabled */ if ((pv->param.gop.b_pyramid == 0) && (bs->FrameType & MFX_FRAMETYPE_B) && (bs->FrameType & MFX_FRAMETYPE_REF)) { hb_log("encqsv: BPyramid off not respected (delay: %d)", pv->bfrm_delay); /* don't pollute the log unnecessarily */ pv->param.gop.b_pyramid = 1; } /* check for PTS < DTS */ if (buf->s.start < buf->s.renderOffset) { hb_log("encqsv: PTS %"PRId64" < DTS %"PRId64" for frame %d with type '%s'", buf->s.start, buf->s.renderOffset, pv->frames_out + 1, hb_qsv_frametype_name(bs->FrameType)); } /* * If we have a chapter marker pending and this frame's PTS * is at or after the marker's PTS, use it as the chapter start. */ if (buf->s.flags & HB_FLAG_FRAMETYPE_KEY) { hb_chapter_dequeue(pv->chapter_queue, buf); } hb_buffer_list_append(&pv->encoded_frames, buf); pv->frames_out++; return; fail: *pv->job->done_error = HB_ERROR_UNKNOWN; *pv->job->die = 1; } static int qsv_enc_work(hb_work_private_t *pv, hb_qsv_list *qsv_atom, mfxFrameSurface1 *surface) { mfxStatus sts; hb_qsv_context *qsv_ctx = pv->job->qsv.ctx; hb_qsv_space *qsv_enc_space = pv->job->qsv.ctx->enc_space; do { int sync_idx = hb_qsv_get_free_sync(qsv_enc_space, qsv_ctx); if (sync_idx == -1) { hb_error("encqsv: hb_qsv_get_free_sync failed"); return -1; } hb_qsv_task *task = hb_qsv_list_item(qsv_enc_space->tasks, pv->async_depth); do { sts = MFXVideoENCODE_EncodeFrameAsync(qsv_ctx->mfx_session, NULL, surface, task->bs, qsv_enc_space->p_syncp[sync_idx]->p_sync); if (sts == MFX_ERR_MORE_DATA || (sts >= MFX_ERR_NONE && sts != MFX_WRN_DEVICE_BUSY)) { if (surface != NULL && !pv->is_sys_mem) { ff_qsv_atomic_dec(&surface->Data.Locked); } } if (sts == MFX_ERR_MORE_DATA) { if (qsv_atom != NULL) { hb_list_add(pv->delayed_processing, qsv_atom); } ff_qsv_atomic_dec(&qsv_enc_space->p_syncp[sync_idx]->in_use); break; } else if (sts < MFX_ERR_NONE) { hb_error("encqsv: MFXVideoENCODE_EncodeFrameAsync failed (%d)", sts); return -1; } else if (sts == MFX_WRN_DEVICE_BUSY) { hb_qsv_sleep(10); // device is busy, wait then repeat the call continue; } else { hb_qsv_stage *new_stage = hb_qsv_stage_init(); new_stage->type = HB_QSV_ENCODE; new_stage->in.p_surface = surface; new_stage->out.sync = qsv_enc_space->p_syncp[sync_idx]; new_stage->out.p_bs = task->bs; task->stage = new_stage; pv->async_depth++; if (qsv_atom != NULL) { hb_qsv_add_stagee(&qsv_atom, new_stage, HAVE_THREADS); } else { /* encode-only or flushing */ hb_qsv_list *new_qsv_atom = hb_qsv_list_init(HAVE_THREADS); hb_qsv_add_stagee(&new_qsv_atom, new_stage, HAVE_THREADS); hb_qsv_list_add (qsv_ctx->pipes, new_qsv_atom); } int i = hb_list_count(pv->delayed_processing); while (--i >= 0) { hb_qsv_list *item = hb_list_item(pv->delayed_processing, i); if (item != NULL) { hb_list_rem(pv->delayed_processing, item); hb_qsv_flush_stages(qsv_ctx->pipes, &item); } } break; } ff_qsv_atomic_dec(&qsv_enc_space->p_syncp[sync_idx]->in_use); break; } while (sts >= MFX_ERR_NONE); do { if (pv->async_depth == 0) break; /* we've done enough asynchronous operations or we're flushing */ if (pv->async_depth >= pv->max_async_depth || surface == NULL) { hb_qsv_task *task = hb_qsv_list_item(qsv_enc_space->tasks, 0); pv->async_depth--; /* perform a sync operation to get the output bitstream */ hb_qsv_wait_on_sync(qsv_ctx, task->stage); if (task->bs->DataLength > 0) { hb_qsv_list *pipe = hb_qsv_pipe_by_stage(qsv_ctx->pipes, task->stage); hb_qsv_flush_stages(qsv_ctx->pipes, &pipe); /* get the encoded frame from the bitstream */ qsv_bitstream_slurp(pv, task->bs); /* shift for fifo */ if (pv->async_depth) { hb_qsv_list_rem(qsv_enc_space->tasks, task); hb_qsv_list_add(qsv_enc_space->tasks, task); } task->stage = NULL; } } } while (surface == NULL); } while (surface == NULL && sts != MFX_ERR_MORE_DATA); return 0; } int encqsvWork(hb_work_object_t *w, hb_buffer_t **buf_in, hb_buffer_t **buf_out) { hb_work_private_t *pv = w->private_data; hb_buffer_t *in = *buf_in; hb_job_t *job = pv->job; while (qsv_enc_init(pv) >= 2) { hb_qsv_sleep(1); // encoding not initialized, wait and repeat the call } if (*job->die) { goto fail; // unrecoverable error, don't attempt to encode } /* * EOF on input. Flush the decoder, then send the * EOF downstream to let the muxer know we're done. */ if (in->s.flags & HB_BUF_FLAG_EOF) { qsv_enc_work(pv, NULL, NULL); hb_buffer_list_append(&pv->encoded_frames, in); *buf_out = hb_buffer_list_clear(&pv->encoded_frames); *buf_in = NULL; // don't let 'work_loop' close this buffer return HB_WORK_DONE; } mfxFrameSurface1 *surface = NULL; hb_qsv_list *qsv_atom = NULL; hb_qsv_context *qsv_ctx = job->qsv.ctx; hb_qsv_space *qsv_enc_space = job->qsv.ctx->enc_space; if (pv->is_sys_mem) { mfxFrameInfo *info = &pv->param.videoParam->mfx.FrameInfo; int surface_index = hb_qsv_get_free_surface(qsv_enc_space, qsv_ctx, info, QSV_PART_ANY); if (surface_index == -1) { hb_error("encqsv: hb_qsv_get_free_surface failed"); goto fail; } surface = qsv_enc_space->p_surfaces[surface_index]; qsv_yuv420_to_nv12(pv->sws_context_to_nv12, surface, in); } else { qsv_atom = in->qsv_details.qsv_atom; surface = hb_qsv_get_last_stage(qsv_atom)->out.p_surface; // At this point, enc_qsv takes ownership of the QSV resources // in the 'in' buffer. in->qsv_details.qsv_atom = NULL; /* * QSV decoding fills the QSV context's dts_seq list, we need to * pop this surface's DTS so dts_seq doesn't grow unnecessarily. */ hb_qsv_dts_pop(qsv_ctx); } /* * Debugging code to check that the upstream modules have generated * a continuous, self-consistent frame stream. */ if (pv->last_start > in->s.start) { hb_log("encqsv: input continuity error, " "last start %"PRId64" start %"PRId64"", pv->last_start, in->s.start); } pv->last_start = in->s.start; /* for DTS generation */ if (pv->frames_in <= BFRM_DELAY_MAX) { pv->init_pts[pv->frames_in] = in->s.start; } if (pv->frames_in) { hb_qsv_add_new_dts(pv->list_dts, in->s.start); } pv->frames_in++; /* * Chapters have to start with a keyframe, so request one here. * * Using an mfxEncodeCtrl structure to force key frame generation is not * possible when using a lookahead and frame reordering, so instead do * the following before encoding the frame attached to the chapter: * * - flush the encoder to encode and retrieve any buffered frames * * - do a hard reset (MFXVideoENCODE_Close, then Init) of * the encoder to make sure the next frame is a keyframe * * The hard reset ensures encoding resumes with a clean state, avoiding * miscellaneous hard-to-disagnose issues that may occur when resuming * an encode after flushing the encoder or using MFXVideoENCODE_Reset. */ if (in->s.new_chap > 0 && job->chapter_markers) { mfxStatus sts; if (qsv_enc_work(pv, NULL, NULL) < 0) { goto fail; } sts = MFXVideoENCODE_Close(qsv_ctx->mfx_session); if (sts != MFX_ERR_NONE) { hb_error("encqsv: MFXVideoENCODE_Close failed (%d)", sts); goto fail; } sts = MFXVideoENCODE_Init(qsv_ctx->mfx_session, pv->param.videoParam); if (sts < MFX_ERR_NONE) { hb_error("encqsv: MFXVideoENCODE_Init failed (%d)", sts); goto fail; } hb_chapter_enqueue(pv->chapter_queue, in); } /* * If interlaced encoding is requested during encoder initialization, * but the input mfxFrameSurface1 is flagged as progressive here, * the output bitstream will be progressive (according to MediaInfo). * * Assume the user knows what he's doing (say he is e.g. encoding a * progressive-flagged source using interlaced compression - he may * well have a good reason to do so; mis-flagged sources do exist). */ surface->Info.PicStruct = pv->param.videoParam->mfx.FrameInfo.PicStruct; surface->Data.TimeStamp = in->s.start; save_frame_duration(pv, in); /* * Now that the input surface is setup, we can encode it. */ if (qsv_enc_work(pv, qsv_atom, surface) < 0) { goto fail; } *buf_out = hb_buffer_list_clear(&pv->encoded_frames); return HB_WORK_OK; fail: if (*job->done_error == HB_ERROR_NONE) { *job->done_error = HB_ERROR_UNKNOWN; } *job->die = 1; *buf_out = NULL; return HB_WORK_ERROR; } #endif // USE_QSV