summaryrefslogtreecommitdiffstats
path: root/libhb/enc_qsv.c
diff options
context:
space:
mode:
Diffstat (limited to 'libhb/enc_qsv.c')
-rw-r--r--libhb/enc_qsv.c53
1 files changed, 42 insertions, 11 deletions
diff --git a/libhb/enc_qsv.c b/libhb/enc_qsv.c
index b6e80687e..a419767ee 100644
--- a/libhb/enc_qsv.c
+++ b/libhb/enc_qsv.c
@@ -302,8 +302,14 @@ static int qsv_hevc_make_header(hb_work_object_t *w, mfxSession session)
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->size;
+ 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;
@@ -521,13 +527,26 @@ int qsv_enc_init(hb_work_private_t *pv)
}
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);
+ 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
@@ -589,14 +608,17 @@ int qsv_enc_init(hb_work_private_t *pv)
{
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;
- surface->Data.Y = av_mallocz(info.Width * info.Height * 3 / 2);
- surface->Data.VU = surface->Data.Y + info.Width * info.Height;
+ 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;
}
}
@@ -847,12 +869,21 @@ int encqsvInit(hb_work_object_t *w, hb_job_t *job)
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)