39 #define AMF_AV_FRAME_REF L"av_frame_ref"
40 #define PTS_PROP L"PtsProp"
52 #define FFMPEG_AMF_WRITER_ID L"ffmpeg_amf"
92 if(!
ctx->pts_property_name)
97 if(!
ctx->av_frame_property_name)
123 AMF_RETURN_IF_FALSE(
ctx, amf_device_ctx->version >= AMF_MAKE_FULL_VERSION(1, 4, 32, 0),
AVERROR_UNKNOWN,
"10-bit encoder is not supported by AMD GPU drivers versions lower than 23.30.\n");
130 res = amf_device_ctx->factory->pVtbl->CreateComponent(amf_device_ctx->factory, amf_device_ctx->context,
codec_id, &
ctx->encoder);
133 ctx->submitted_frame = 0;
134 ctx->encoded_frame = 0;
145 ctx->encoder->pVtbl->Terminate(
ctx->encoder);
146 ctx->encoder->pVtbl->Release(
ctx->encoder);
153 if (
ctx->output_list) {
173 uint8_t *dst_data[4] = {0};
174 int dst_linesize[4] = {0};
178 planes = (int)surface->pVtbl->GetPlanesCount(surface);
182 plane = surface->pVtbl->GetPlaneAt(surface,
i);
183 dst_data[
i] = plane->pVtbl->GetNative(plane);
184 dst_linesize[
i] = plane->pVtbl->GetHPitch(plane);
197 AMFVariantStruct var = {0};
210 buffer->pVtbl->GetProperty(
buffer, AMF_VIDEO_ENCODER_OUTPUT_DATA_TYPE, &var);
212 pict_type = var.int64Value == AMF_VIDEO_ENCODER_OUTPUT_DATA_TYPE_IDR ?
AV_PICTURE_TYPE_I :
218 if ((
buffer->pVtbl->GetProperty(
buffer, AMF_VIDEO_ENCODER_STATISTIC_AVERAGE_QP, &var)) == AMF_OK) {
219 average_qp =
FFMAX((
int)var.int64Value, -1);
223 buffer->pVtbl->GetProperty(
buffer, AMF_VIDEO_ENCODER_HEVC_OUTPUT_DATA_TYPE, &var);
225 pict_type = var.int64Value == AMF_VIDEO_ENCODER_HEVC_OUTPUT_DATA_TYPE_IDR ?
AV_PICTURE_TYPE_I :
226 var.int64Value == AMF_VIDEO_ENCODER_HEVC_OUTPUT_DATA_TYPE_I ?
AV_PICTURE_TYPE_I :
227 var.int64Value == AMF_VIDEO_ENCODER_HEVC_OUTPUT_DATA_TYPE_P ?
AV_PICTURE_TYPE_P : 0;
230 if ((
buffer->pVtbl->GetProperty(
buffer, AMF_VIDEO_ENCODER_HEVC_STATISTIC_AVERAGE_QP, &var)) == AMF_OK) {
231 average_qp =
FFMAX((
int)var.int64Value, -1);
235 buffer->pVtbl->GetProperty(
buffer, AMF_VIDEO_ENCODER_AV1_OUTPUT_FRAME_TYPE, &var);
237 pict_type = var.int64Value == AMF_VIDEO_ENCODER_AV1_OUTPUT_FRAME_TYPE_KEY ?
AV_PICTURE_TYPE_I :
238 var.int64Value == AMF_VIDEO_ENCODER_AV1_OUTPUT_FRAME_TYPE_INTRA_ONLY ?
AV_PICTURE_TYPE_I :
239 var.int64Value == AMF_VIDEO_ENCODER_AV1_OUTPUT_FRAME_TYPE_INTER ?
AV_PICTURE_TYPE_P : 0;
242 if ((
buffer->pVtbl->GetProperty(
buffer, AMF_VIDEO_ENCODER_AV1_STATISTIC_AVERAGE_Q_INDEX, &var)) == AMF_OK) {
243 average_qp =
FFMAX((
int)var.int64Value, -1);
244 if (average_qp >= 0) {
245 average_qp = (average_qp > 244) ? (average_qp <= 249 ? 62 : 63) : (average_qp + 3) >> 2;
253 if (average_qp >= 0) {
259 pkt->
pts = var.int64Value;
265 if ((
ctx->max_b_frames > 0 || ((
ctx->pa_adaptive_mini_gop == 1) ?
true :
false)) &&
ctx->dts_delay == 0) {
270 "timestamp_list is empty while max_b_frames = %d\n", avctx->
max_b_frames);
275 ctx->dts_delay = timestamp_last - timestamp;
291 if (!
ctx->timestamp_list) {
295 if (!
ctx->output_list)
300 ctx->hwsurfaces_in_queue = 0;
329 if (
ctx->pa_lookahead_buffer_depth >=
ctx->hwsurfaces_in_queue_max) {
331 "async_depth (%d) too small for lookahead (%d), increasing to (%d)\n",
332 ctx->hwsurfaces_in_queue_max,
333 ctx->pa_lookahead_buffer_depth,
334 ctx->pa_lookahead_buffer_depth + 1);
335 ctx->hwsurfaces_in_queue_max =
ctx->pa_lookahead_buffer_depth + 1;
349 AMFVariantStruct var;
350 res = AMFVariantInit(&var);
352 AMFGuid guid_AMFInterface = IID_AMFInterface();
353 AMFInterface *amf_interface;
354 res =
val->pVtbl->QueryInterface(
val, &guid_AMFInterface, (
void**)&amf_interface);
357 res = AMFVariantAssignInterface(&var, amf_interface);
358 amf_interface->pVtbl->Release(amf_interface);
361 res =
object->pVtbl->SetProperty(
object,
name, var);
363 AMFVariantClear(&var);
370 AMF_RESULT res = AMF_FAIL;
375 AMF_ASSIGN_PROPERTY_INT64(res, surface,
ctx->av_frame_property_name,
data);
382 AMFVariantStruct var = {0};
383 AMF_RESULT res =
buffer->pVtbl->GetProperty(
buffer,
ctx->av_frame_property_name, &var);
384 if(res == AMF_OK && var.int64Value){
404 switch (
frame->format) {
408 static const GUID AMFTextureArrayIndexGUID = { 0x28115527, 0xe7c3, 0x4b66, { 0x99, 0xd3, 0x4f, 0x2a, 0xe6, 0xb4, 0x7f, 0xaf } };
409 ID3D11Texture2D *texture = (ID3D11Texture2D*)
frame->data[0];
413 texture->lpVtbl->SetPrivateData(texture, &AMFTextureArrayIndexGUID,
sizeof(
index), &
index);
414 res = amf_device_ctx->context->pVtbl->CreateSurfaceFromDX11Native(amf_device_ctx->context, texture, &surface,
NULL);
423 IDirect3DSurface9 *texture = (IDirect3DSurface9 *)
frame->data[3];
424 res = amf_device_ctx->context->pVtbl->CreateSurfaceFromDX9Native(amf_device_ctx->context, texture, &surface,
NULL);
432 surface = (AMFSurface*)
frame->data[0];
433 surface->pVtbl->Acquire(surface);
439 res = amf_device_ctx->context->pVtbl->AllocSurface(amf_device_ctx->context, AMF_MEMORY_HOST,
ctx->format, avctx->
width, avctx->
height, &surface);
447 ctx->hwsurfaces_in_queue++;
449 surface->pVtbl->SetCrop(surface, 0, 0,
frame->width,
frame->height);
453 AMFBuffer * hdrmeta_buffer =
NULL;
454 res = amf_device_ctx->context->pVtbl->AllocBuffer(amf_device_ctx->context, AMF_MEMORY_HOST,
sizeof(AMFHDRMetadata), &hdrmeta_buffer);
456 AMFHDRMetadata * hdrmeta = (AMFHDRMetadata*)hdrmeta_buffer->pVtbl->GetNative(hdrmeta_buffer);
460 AMF_ASSIGN_PROPERTY_INTERFACE(res,
ctx->encoder, AMF_VIDEO_ENCODER_INPUT_HDR_METADATA, hdrmeta_buffer);
break;
462 AMF_ASSIGN_PROPERTY_INTERFACE(res,
ctx->encoder, AMF_VIDEO_ENCODER_HEVC_INPUT_HDR_METADATA, hdrmeta_buffer);
break;
464 AMF_ASSIGN_PROPERTY_INTERFACE(res,
ctx->encoder, AMF_VIDEO_ENCODER_AV1_INPUT_HDR_METADATA, hdrmeta_buffer);
break;
469 hdrmeta_buffer->pVtbl->Release(hdrmeta_buffer);
472 surface->pVtbl->SetPts(surface,
frame->pts);
474 AMF_ASSIGN_PROPERTY_INT64(res, surface,
ctx->pts_property_name,
frame->pts);
478 AMF_ASSIGN_PROPERTY_BOOL(res, surface, AMF_VIDEO_ENCODER_STATISTICS_FEEDBACK, 1);
479 AMF_ASSIGN_PROPERTY_INT64(res, surface, AMF_VIDEO_ENCODER_INSERT_AUD, !!
ctx->aud);
480 switch (
frame->pict_type) {
482 if (
ctx->forced_idr) {
483 AMF_ASSIGN_PROPERTY_INT64(res, surface, AMF_VIDEO_ENCODER_INSERT_SPS, 1);
484 AMF_ASSIGN_PROPERTY_INT64(res, surface, AMF_VIDEO_ENCODER_INSERT_PPS, 1);
485 AMF_ASSIGN_PROPERTY_INT64(res, surface, AMF_VIDEO_ENCODER_FORCE_PICTURE_TYPE, AMF_VIDEO_ENCODER_PICTURE_TYPE_IDR);
487 AMF_ASSIGN_PROPERTY_INT64(res, surface, AMF_VIDEO_ENCODER_FORCE_PICTURE_TYPE, AMF_VIDEO_ENCODER_PICTURE_TYPE_I);
491 AMF_ASSIGN_PROPERTY_INT64(res, surface, AMF_VIDEO_ENCODER_FORCE_PICTURE_TYPE, AMF_VIDEO_ENCODER_PICTURE_TYPE_P);
494 AMF_ASSIGN_PROPERTY_INT64(res, surface, AMF_VIDEO_ENCODER_FORCE_PICTURE_TYPE, AMF_VIDEO_ENCODER_PICTURE_TYPE_B);
499 AMF_ASSIGN_PROPERTY_BOOL(res, surface, AMF_VIDEO_ENCODER_HEVC_STATISTICS_FEEDBACK, 1);
500 AMF_ASSIGN_PROPERTY_INT64(res, surface, AMF_VIDEO_ENCODER_HEVC_INSERT_AUD, !!
ctx->aud);
501 switch (
frame->pict_type) {
503 if (
ctx->forced_idr) {
504 AMF_ASSIGN_PROPERTY_INT64(res, surface, AMF_VIDEO_ENCODER_HEVC_INSERT_HEADER, 1);
505 AMF_ASSIGN_PROPERTY_INT64(res, surface, AMF_VIDEO_ENCODER_HEVC_FORCE_PICTURE_TYPE, AMF_VIDEO_ENCODER_HEVC_PICTURE_TYPE_IDR);
507 AMF_ASSIGN_PROPERTY_INT64(res, surface, AMF_VIDEO_ENCODER_HEVC_FORCE_PICTURE_TYPE, AMF_VIDEO_ENCODER_HEVC_PICTURE_TYPE_I);
511 AMF_ASSIGN_PROPERTY_INT64(res, surface, AMF_VIDEO_ENCODER_HEVC_FORCE_PICTURE_TYPE, AMF_VIDEO_ENCODER_HEVC_PICTURE_TYPE_P);
516 AMF_ASSIGN_PROPERTY_BOOL(res, surface, AMF_VIDEO_ENCODER_AV1_STATISTICS_FEEDBACK, 1);
518 if (
ctx->forced_idr) {
519 AMF_ASSIGN_PROPERTY_INT64(res, surface, AMF_VIDEO_ENCODER_AV1_FORCE_INSERT_SEQUENCE_HEADER, 1);
520 AMF_ASSIGN_PROPERTY_INT64(res, surface, AMF_VIDEO_ENCODER_AV1_FORCE_FRAME_TYPE, AMF_VIDEO_ENCODER_AV1_FORCE_FRAME_TYPE_KEY);
522 AMF_ASSIGN_PROPERTY_INT64(res, surface, AMF_VIDEO_ENCODER_AV1_FORCE_FRAME_TYPE, AMF_VIDEO_ENCODER_AV1_FORCE_FRAME_TYPE_INTRA_ONLY);
530 res =
ctx->encoder->pVtbl->SubmitInput(
ctx->encoder, (AMFData*)surface);
531 if (res == AMF_INPUT_FULL) {
533 *surface_resubmit = surface;
535 surface->pVtbl->Release(surface);
537 ctx->submitted_frame++;
541 if(
ctx->submitted_frame <=
ctx->encoded_frame + output_delay)
554 if (amf_device_ctx->lock)
555 amf_device_ctx->lock(amf_device_ctx->lock_ctx);
557 if (amf_device_ctx->unlock)
558 amf_device_ctx->unlock(amf_device_ctx->lock_ctx);
566 AMF_RESULT
ret =
ctx->encoder->pVtbl->QueryOutput(
ctx->encoder, &
data);
569 AMFGuid guid = IID_AMFBuffer();
573 ctx->hwsurfaces_in_queue--;
574 ctx->encoded_frame++;
582 AMFSurface *surface =
NULL;
585 AMF_RESULT res_query;
609 if(
ctx->submitted_frame <=
ctx->encoded_frame + output_delay)
615 if (!
frame->buf[0]) {
617 if(!
ctx->delayed_drain) {
618 res =
ctx->encoder->pVtbl->Drain(
ctx->encoder);
619 if (res == AMF_INPUT_FULL) {
620 ctx->delayed_drain = 1;
650 if (
ctx->delayed_drain) {
651 res =
ctx->encoder->pVtbl->Drain(
ctx->encoder);
652 if (res != AMF_INPUT_FULL) {
653 ctx->delayed_drain = 0;
657 av_log(avctx,
AV_LOG_WARNING,
"Data acquired but delayed drain submission got AMF_INPUT_FULL- should not happen\n");
660 }
else if (
ctx->delayed_drain || (
ctx->eof && res_query != AMF_EOF) || (
ctx->hwsurfaces_in_queue >=
ctx->hwsurfaces_in_queue_max) || surface) {
664 if (!
ctx->query_timeout_supported || avpkt->
data || avpkt->
buf) {
668 }
while (block_and_wait);
670 if (res_query == AMF_EOF) {
678 res =
ctx->encoder->pVtbl->SubmitInput(
ctx->encoder, (AMFData*)surface);
679 if (res != AMF_INPUT_FULL)
682 if (!
ctx->query_timeout_supported)
693 }
while(res == AMF_INPUT_FULL);
695 surface->pVtbl->Release(surface);
696 if (res == AMF_INPUT_FULL) {
697 av_log(avctx,
AV_LOG_WARNING,
"Data acquired but delayed SubmitInput returned AMF_INPUT_FULL- should not happen\n");
703 ctx->submitted_frame++;