mirror of https://github.com/FFmpeg/FFmpeg.git
1667 lines
66 KiB
1667 lines
66 KiB
/* |
|
* This file is part of FFmpeg. |
|
* |
|
* FFmpeg is free software; you can redistribute it and/or |
|
* modify it under the terms of the GNU Lesser General Public |
|
* License as published by the Free Software Foundation; either |
|
* version 2.1 of the License, or (at your option) any later version. |
|
* |
|
* FFmpeg 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 |
|
* Lesser General Public License for more details. |
|
* |
|
* You should have received a copy of the GNU Lesser General Public |
|
* License along with FFmpeg; if not, write to the Free Software |
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA |
|
*/ |
|
|
|
#include "libavutil/internal.h" |
|
#include "libavutil/opt.h" |
|
#include "libavutil/mem.h" |
|
|
|
#include "cbs.h" |
|
#include "cbs_h264.h" |
|
#include "atsc_a53.h" |
|
|
|
#include "h264_levels.h" |
|
#include "h2645data.h" |
|
#include "codec_internal.h" |
|
#include "version.h" |
|
#include "hw_base_encode_h264.h" |
|
|
|
#include "vulkan_encode.h" |
|
|
|
enum UnitElems { |
|
UNIT_AUD = 1 << 0, |
|
UNIT_SEI_TIMING = 1 << 1, |
|
UNIT_SEI_IDENTIFIER = 1 << 2, |
|
UNIT_SEI_RECOVERY = 1 << 3, |
|
UNIT_SEI_A53_CC = 1 << 4, |
|
}; |
|
|
|
const FFVulkanEncodeDescriptor ff_vk_enc_h264_desc = { |
|
.codec_id = AV_CODEC_ID_H264, |
|
.encode_extension = FF_VK_EXT_VIDEO_ENCODE_H264, |
|
.encode_op = VK_VIDEO_CODEC_OPERATION_ENCODE_H264_BIT_KHR, |
|
.ext_props = { |
|
.extensionName = VK_STD_VULKAN_VIDEO_CODEC_H264_ENCODE_EXTENSION_NAME, |
|
.specVersion = VK_STD_VULKAN_VIDEO_CODEC_H264_ENCODE_SPEC_VERSION, |
|
}, |
|
}; |
|
|
|
/* Random (version 4) ISO 11578 UUID. */ |
|
static const uint8_t vulkan_encode_h264_sei_identifier_uuid[16] = { |
|
0x03, 0xfd, 0xf2, 0x0a, 0x5d, 0x4c, 0x05, 0x48, |
|
0x20, 0x98, 0xca, 0x6b, 0x0c, 0x95, 0x30, 0x1c, |
|
}; |
|
|
|
typedef struct VulkanEncodeH264Picture { |
|
int frame_num; |
|
int64_t last_idr_frame; |
|
uint16_t idr_pic_id; |
|
int primary_pic_type; |
|
int slice_type; |
|
int pic_order_cnt; |
|
|
|
enum UnitElems units_needed; |
|
|
|
VkVideoEncodeH264RateControlInfoKHR vkrc_info; |
|
VkVideoEncodeH264RateControlLayerInfoKHR vkrc_layer_info; |
|
VkVideoEncodeH264GopRemainingFrameInfoKHR vkrc_remaining; |
|
|
|
StdVideoEncodeH264WeightTable slice_wt; |
|
StdVideoEncodeH264SliceHeader slice_hdr; |
|
VkVideoEncodeH264NaluSliceInfoKHR vkslice; |
|
|
|
StdVideoEncodeH264PictureInfo h264pic_info; |
|
VkVideoEncodeH264PictureInfoKHR vkh264pic_info; |
|
|
|
StdVideoEncodeH264ReferenceInfo h264dpb_info; |
|
VkVideoEncodeH264DpbSlotInfoKHR vkh264dpb_info; |
|
|
|
StdVideoEncodeH264RefListModEntry mods[MAX_REFERENCE_LIST_NUM][H264_MAX_RPLM_COUNT]; |
|
StdVideoEncodeH264RefPicMarkingEntry mmco[H264_MAX_RPLM_COUNT]; |
|
StdVideoEncodeH264ReferenceListsInfo ref_list_info; |
|
} VulkanEncodeH264Picture; |
|
|
|
typedef struct VulkanEncodeH264Context { |
|
FFVulkanEncodeContext common; |
|
|
|
FFHWBaseEncodeH264 units; |
|
FFHWBaseEncodeH264Opts unit_opts; |
|
|
|
enum UnitElems unit_elems; |
|
|
|
uint8_t fixed_qp_p; |
|
uint8_t fixed_qp_b; |
|
|
|
VkVideoEncodeH264ProfileInfoKHR profile; |
|
|
|
VkVideoEncodeH264CapabilitiesKHR caps; |
|
VkVideoEncodeH264QualityLevelPropertiesKHR quality_props; |
|
|
|
CodedBitstreamContext *cbs; |
|
CodedBitstreamFragment current_access_unit; |
|
|
|
H264RawAUD raw_aud; |
|
|
|
SEIRawUserDataUnregistered sei_identifier; |
|
H264RawSEIPicTiming sei_pic_timing; |
|
H264RawSEIRecoveryPoint sei_recovery_point; |
|
SEIRawUserDataRegistered sei_a53cc; |
|
void *sei_a53cc_data; |
|
char *sei_identifier_string; |
|
} VulkanEncodeH264Context; |
|
|
|
static int init_pic_rc(AVCodecContext *avctx, FFHWBaseEncodePicture *pic, |
|
VkVideoEncodeRateControlInfoKHR *rc_info, |
|
VkVideoEncodeRateControlLayerInfoKHR *rc_layer) |
|
{ |
|
VulkanEncodeH264Context *enc = avctx->priv_data; |
|
FFVulkanEncodeContext *ctx = &enc->common; |
|
VulkanEncodeH264Picture *hp = pic->codec_priv; |
|
|
|
hp->vkrc_info = (VkVideoEncodeH264RateControlInfoKHR) { |
|
.sType = VK_STRUCTURE_TYPE_VIDEO_ENCODE_H264_RATE_CONTROL_INFO_KHR, |
|
.flags = VK_VIDEO_ENCODE_H264_RATE_CONTROL_REFERENCE_PATTERN_FLAT_BIT_KHR | |
|
VK_VIDEO_ENCODE_H264_RATE_CONTROL_REGULAR_GOP_BIT_KHR, |
|
.idrPeriod = ctx->base.gop_size, |
|
.gopFrameCount = ctx->base.gop_size, |
|
.consecutiveBFrameCount = FFMAX(ctx->base.b_per_p - 1, 0), |
|
.temporalLayerCount = 0, |
|
}; |
|
rc_info->pNext = &hp->vkrc_info; |
|
|
|
if (rc_info->rateControlMode > VK_VIDEO_ENCODE_RATE_CONTROL_MODE_DISABLED_BIT_KHR) { |
|
rc_info->virtualBufferSizeInMs = (enc->unit_opts.hrd_buffer_size * 1000LL) / avctx->bit_rate; |
|
rc_info->initialVirtualBufferSizeInMs = (enc->unit_opts.initial_buffer_fullness * 1000LL) / avctx->bit_rate; |
|
|
|
hp->vkrc_layer_info = (VkVideoEncodeH264RateControlLayerInfoKHR) { |
|
.sType = VK_STRUCTURE_TYPE_VIDEO_ENCODE_H264_RATE_CONTROL_LAYER_INFO_KHR, |
|
|
|
.useMinQp = avctx->qmin > 0, |
|
.minQp.qpI = avctx->qmin > 0 ? avctx->qmin : 0, |
|
.minQp.qpP = avctx->qmin > 0 ? avctx->qmin : 0, |
|
.minQp.qpB = avctx->qmin > 0 ? avctx->qmin : 0, |
|
|
|
.useMaxQp = avctx->qmax > 0, |
|
.maxQp.qpI = avctx->qmax > 0 ? avctx->qmax : 0, |
|
.maxQp.qpP = avctx->qmax > 0 ? avctx->qmax : 0, |
|
.maxQp.qpB = avctx->qmax > 0 ? avctx->qmax : 0, |
|
|
|
.useMaxFrameSize = 0, |
|
}; |
|
rc_layer->pNext = &hp->vkrc_layer_info; |
|
hp->vkrc_info.temporalLayerCount = 1; |
|
} |
|
|
|
return 0; |
|
} |
|
|
|
static int vk_enc_h264_update_pic_info(AVCodecContext *avctx, |
|
FFHWBaseEncodePicture *pic) |
|
{ |
|
VulkanEncodeH264Context *enc = avctx->priv_data; |
|
FFVulkanEncodeContext *ctx = &enc->common; |
|
VulkanEncodeH264Picture *hp = pic->codec_priv; |
|
FFHWBaseEncodePicture *prev = pic->prev; |
|
VulkanEncodeH264Picture *hprev = prev ? prev->codec_priv : NULL; |
|
|
|
if (pic->type == FF_HW_PICTURE_TYPE_IDR) { |
|
av_assert0(pic->display_order == pic->encode_order); |
|
|
|
hp->frame_num = 0; |
|
hp->last_idr_frame = pic->display_order; |
|
hp->idr_pic_id = hprev ? hprev->idr_pic_id + 1 : 0; |
|
|
|
hp->primary_pic_type = 0; |
|
hp->slice_type = STD_VIDEO_H264_SLICE_TYPE_I; |
|
} else { |
|
av_assert0(prev); |
|
|
|
hp->frame_num = hprev->frame_num + prev->is_reference; |
|
|
|
hp->last_idr_frame = hprev->last_idr_frame; |
|
hp->idr_pic_id = hprev->idr_pic_id; |
|
|
|
if (pic->type == FF_HW_PICTURE_TYPE_I) { |
|
hp->slice_type = STD_VIDEO_H264_SLICE_TYPE_I; |
|
hp->primary_pic_type = 0; |
|
} else if (pic->type == FF_HW_PICTURE_TYPE_P) { |
|
hp->slice_type = STD_VIDEO_H264_SLICE_TYPE_P; |
|
hp->primary_pic_type = 1; |
|
} else { |
|
hp->slice_type = STD_VIDEO_H264_SLICE_TYPE_B; |
|
hp->primary_pic_type = 2; |
|
} |
|
} |
|
|
|
hp->pic_order_cnt = pic->display_order - hp->last_idr_frame; |
|
if (enc->units.raw_sps.pic_order_cnt_type == 2) |
|
hp->pic_order_cnt *= 2; |
|
|
|
hp->units_needed = 0; |
|
|
|
if (enc->unit_elems & UNIT_SEI_IDENTIFIER && pic->encode_order == 0) |
|
hp->units_needed |= UNIT_SEI_IDENTIFIER; |
|
|
|
if (enc->unit_elems & UNIT_SEI_TIMING) { |
|
enc->sei_pic_timing = (H264RawSEIPicTiming) { |
|
.cpb_removal_delay = 2 * (pic->encode_order - hp->last_idr_frame), |
|
.dpb_output_delay = 2 * (pic->display_order - pic->encode_order + ctx->base.max_b_depth), |
|
}; |
|
|
|
hp->units_needed |= UNIT_SEI_TIMING; |
|
} |
|
|
|
if (enc->unit_elems & UNIT_SEI_RECOVERY && pic->type == FF_HW_PICTURE_TYPE_I) { |
|
enc->sei_recovery_point = (H264RawSEIRecoveryPoint) { |
|
.recovery_frame_cnt = 0, |
|
.exact_match_flag = 1, |
|
.broken_link_flag = ctx->base.b_per_p > 0, |
|
}; |
|
|
|
hp->units_needed |= UNIT_SEI_RECOVERY; |
|
} |
|
|
|
if (enc->unit_elems & UNIT_SEI_A53_CC) { |
|
int err; |
|
size_t sei_a53cc_len; |
|
av_freep(&enc->sei_a53cc_data); |
|
err = ff_alloc_a53_sei(pic->input_image, 0, &enc->sei_a53cc_data, &sei_a53cc_len); |
|
if (err < 0) |
|
return err; |
|
if (enc->sei_a53cc_data != NULL) { |
|
enc->sei_a53cc.itu_t_t35_country_code = 181; |
|
enc->sei_a53cc.data = (uint8_t *)enc->sei_a53cc_data + 1; |
|
enc->sei_a53cc.data_length = sei_a53cc_len - 1; |
|
|
|
hp->units_needed |= UNIT_SEI_A53_CC; |
|
} |
|
} |
|
|
|
return 0; |
|
} |
|
|
|
static void setup_slices(AVCodecContext *avctx, |
|
FFHWBaseEncodePicture *pic) |
|
{ |
|
VulkanEncodeH264Context *enc = avctx->priv_data; |
|
VulkanEncodeH264Picture *hp = pic->codec_priv; |
|
|
|
hp->slice_wt = (StdVideoEncodeH264WeightTable) { |
|
.flags = (StdVideoEncodeH264WeightTableFlags) { |
|
.luma_weight_l0_flag = 0, |
|
.chroma_weight_l0_flag = 0, |
|
.luma_weight_l1_flag = 0, |
|
.chroma_weight_l1_flag = 0, |
|
}, |
|
.luma_log2_weight_denom = 0, |
|
.chroma_log2_weight_denom = 0, |
|
.luma_weight_l0 = { 0 }, |
|
.luma_offset_l0 = { 0 }, |
|
.chroma_weight_l0 = { { 0 } }, |
|
.chroma_offset_l0 = { { 0 } }, |
|
.luma_weight_l1 = { 0 }, |
|
.luma_offset_l1 = { 0 }, |
|
.chroma_weight_l1 = { { 0 } }, |
|
.chroma_offset_l1 = { { 0 } }, |
|
}; |
|
|
|
hp->slice_hdr = (StdVideoEncodeH264SliceHeader) { |
|
.flags = (StdVideoEncodeH264SliceHeaderFlags) { |
|
.direct_spatial_mv_pred_flag = 1, |
|
/* The vk_samples code does this */ |
|
.num_ref_idx_active_override_flag = |
|
((enc->units.raw_pps.num_ref_idx_l0_default_active_minus1) && |
|
(pic->type == FF_HW_PICTURE_TYPE_B)) ? 1 : 0, |
|
}, |
|
.first_mb_in_slice = 1, |
|
.slice_type = hp->slice_type, |
|
.slice_alpha_c0_offset_div2 = 0, |
|
.slice_beta_offset_div2 = 0, |
|
.slice_qp_delta = 0, /* Filled in below */ |
|
/* Reserved */ |
|
.cabac_init_idc = 0, |
|
.disable_deblocking_filter_idc = 0, |
|
.pWeightTable = NULL, // &hp->slice_wt, |
|
}; |
|
|
|
hp->vkslice = (VkVideoEncodeH264NaluSliceInfoKHR) { |
|
.sType = VK_STRUCTURE_TYPE_VIDEO_ENCODE_H264_NALU_SLICE_INFO_KHR, |
|
.pNext = NULL, |
|
.constantQp = pic->type == FF_HW_PICTURE_TYPE_B ? enc->fixed_qp_b : |
|
pic->type == FF_HW_PICTURE_TYPE_P ? enc->fixed_qp_p : |
|
enc->unit_opts.fixed_qp_idr, |
|
.pStdSliceHeader = &hp->slice_hdr, |
|
}; |
|
|
|
if (enc->common.opts.rc_mode != VK_VIDEO_ENCODE_RATE_CONTROL_MODE_DISABLED_BIT_KHR) |
|
hp->vkslice.constantQp = 0; |
|
|
|
hp->slice_hdr.slice_qp_delta = hp->vkslice.constantQp - |
|
(enc->units.raw_pps.pic_init_qp_minus26 + 26); |
|
|
|
hp->vkh264pic_info.pNaluSliceEntries = &hp->vkslice; |
|
hp->vkh264pic_info.naluSliceEntryCount = 1; |
|
} |
|
|
|
static void vk_enc_h264_default_ref_pic_list(AVCodecContext *avctx, |
|
FFHWBaseEncodePicture *pic, |
|
FFHWBaseEncodePicture **rpl0, |
|
FFHWBaseEncodePicture **rpl1, |
|
int *rpl_size) |
|
{ |
|
FFHWBaseEncodePicture *prev; |
|
VulkanEncodeH264Picture *hp, *hn, *hc; |
|
int i, j, n = 0; |
|
|
|
prev = pic->prev; |
|
av_assert0(prev); |
|
hp = pic->codec_priv; |
|
|
|
for (i = 0; i < pic->prev->nb_dpb_pics; i++) { |
|
hn = prev->dpb[i]->codec_priv; |
|
av_assert0(hn->frame_num < hp->frame_num); |
|
|
|
if (pic->type == FF_HW_PICTURE_TYPE_P) { |
|
for (j = n; j > 0; j--) { |
|
hc = rpl0[j - 1]->codec_priv; |
|
av_assert0(hc->frame_num != hn->frame_num); |
|
if (hc->frame_num > hn->frame_num) |
|
break; |
|
rpl0[j] = rpl0[j - 1]; |
|
} |
|
rpl0[j] = prev->dpb[i]; |
|
|
|
} else if (pic->type == FF_HW_PICTURE_TYPE_B) { |
|
for (j = n; j > 0; j--) { |
|
hc = rpl0[j - 1]->codec_priv; |
|
av_assert0(hc->pic_order_cnt != hp->pic_order_cnt); |
|
if (hc->pic_order_cnt < hp->pic_order_cnt) { |
|
if (hn->pic_order_cnt > hp->pic_order_cnt || |
|
hn->pic_order_cnt < hc->pic_order_cnt) |
|
break; |
|
} else { |
|
if (hn->pic_order_cnt > hc->pic_order_cnt) |
|
break; |
|
} |
|
rpl0[j] = rpl0[j - 1]; |
|
} |
|
rpl0[j] = prev->dpb[i]; |
|
|
|
for (j = n; j > 0; j--) { |
|
hc = rpl1[j - 1]->codec_priv; |
|
av_assert0(hc->pic_order_cnt != hp->pic_order_cnt); |
|
if (hc->pic_order_cnt > hp->pic_order_cnt) { |
|
if (hn->pic_order_cnt < hp->pic_order_cnt || |
|
hn->pic_order_cnt > hc->pic_order_cnt) |
|
break; |
|
} else { |
|
if (hn->pic_order_cnt < hc->pic_order_cnt) |
|
break; |
|
} |
|
rpl1[j] = rpl1[j - 1]; |
|
} |
|
rpl1[j] = prev->dpb[i]; |
|
} |
|
|
|
++n; |
|
} |
|
|
|
if (pic->type == FF_HW_PICTURE_TYPE_B) { |
|
for (i = 0; i < n; i++) { |
|
if (rpl0[i] != rpl1[i]) |
|
break; |
|
} |
|
if (i == n) |
|
FFSWAP(FFHWBaseEncodePicture *, rpl1[0], rpl1[1]); |
|
} |
|
|
|
if (pic->type == FF_HW_PICTURE_TYPE_P || |
|
pic->type == FF_HW_PICTURE_TYPE_B) { |
|
av_log(avctx, AV_LOG_DEBUG, "Default RefPicList0 for fn=%d/poc=%d:", |
|
hp->frame_num, hp->pic_order_cnt); |
|
for (i = 0; i < n; i++) { |
|
hn = rpl0[i]->codec_priv; |
|
av_log(avctx, AV_LOG_DEBUG, " fn=%d/poc=%d", |
|
hn->frame_num, hn->pic_order_cnt); |
|
} |
|
av_log(avctx, AV_LOG_DEBUG, "\n"); |
|
} |
|
if (pic->type == FF_HW_PICTURE_TYPE_B) { |
|
av_log(avctx, AV_LOG_DEBUG, "Default RefPicList1 for fn=%d/poc=%d:", |
|
hp->frame_num, hp->pic_order_cnt); |
|
for (i = 0; i < n; i++) { |
|
hn = rpl1[i]->codec_priv; |
|
av_log(avctx, AV_LOG_DEBUG, " fn=%d/poc=%d", |
|
hn->frame_num, hn->pic_order_cnt); |
|
} |
|
av_log(avctx, AV_LOG_DEBUG, "\n"); |
|
} |
|
|
|
*rpl_size = n; |
|
} |
|
|
|
static void setup_refs(AVCodecContext *avctx, |
|
FFHWBaseEncodePicture *pic, |
|
VkVideoEncodeInfoKHR *encode_info) |
|
{ |
|
int n, i, j; |
|
VulkanEncodeH264Context *enc = avctx->priv_data; |
|
VulkanEncodeH264Picture *hp = pic->codec_priv; |
|
FFHWBaseEncodePicture *prev = pic->prev; |
|
FFHWBaseEncodePicture *def_l0[MAX_DPB_SIZE], *def_l1[MAX_DPB_SIZE]; |
|
VulkanEncodeH264Picture *href; |
|
|
|
hp->ref_list_info = (StdVideoEncodeH264ReferenceListsInfo) { |
|
.flags = (StdVideoEncodeH264ReferenceListsInfoFlags) { |
|
.ref_pic_list_modification_flag_l0 = 0, |
|
.ref_pic_list_modification_flag_l1 = 0, |
|
/* Reserved */ |
|
}, |
|
/* May be overridden during setup_slices() */ |
|
.num_ref_idx_l0_active_minus1 = pic->nb_refs[0] - 1, |
|
.num_ref_idx_l1_active_minus1 = pic->nb_refs[1] - 1, |
|
/* .RefPicList0 is set in vk_enc_h264_default_ref_pic_list() */ |
|
/* .RefPicList1 is set in vk_enc_h264_default_ref_pic_list() */ |
|
/* Reserved */ |
|
.pRefList0ModOperations = NULL, /* All set below */ |
|
.refList0ModOpCount = 0, |
|
.pRefList1ModOperations = NULL, |
|
.refList1ModOpCount = 0, |
|
.pRefPicMarkingOperations = NULL, |
|
.refPicMarkingOpCount = 0, |
|
}; |
|
|
|
for (i = 0; i < STD_VIDEO_H264_MAX_NUM_LIST_REF; i++) |
|
hp->ref_list_info.RefPicList0[i] = hp->ref_list_info.RefPicList1[i] = -1; |
|
|
|
/* Note: really not sure */ |
|
for (int i = 0; i < pic->nb_refs[0]; i++) { |
|
VkVideoReferenceSlotInfoKHR *slot_info; |
|
slot_info = (VkVideoReferenceSlotInfoKHR *)&encode_info->pReferenceSlots[i]; |
|
hp->ref_list_info.RefPicList0[i] = slot_info->slotIndex; |
|
} |
|
|
|
/* Note: really not sure */ |
|
for (int i = 0; i < pic->nb_refs[1]; i++) { |
|
VkVideoReferenceSlotInfoKHR *slot_info; |
|
slot_info = (VkVideoReferenceSlotInfoKHR *)&encode_info->pReferenceSlots[pic->nb_refs[0] + i]; |
|
hp->ref_list_info.RefPicList1[i] = slot_info->slotIndex; |
|
} |
|
|
|
hp->h264pic_info.pRefLists = &hp->ref_list_info; |
|
|
|
if (pic->is_reference && pic->type != FF_HW_PICTURE_TYPE_IDR) { |
|
FFHWBaseEncodePicture *discard_list[MAX_DPB_SIZE]; |
|
int discard = 0, keep = 0; |
|
|
|
// Discard everything which is in the DPB of the previous frame but |
|
// not in the DPB of this one. |
|
for (i = 0; i < prev->nb_dpb_pics; i++) { |
|
for (j = 0; j < pic->nb_dpb_pics; j++) { |
|
if (prev->dpb[i] == pic->dpb[j]) |
|
break; |
|
} |
|
if (j == pic->nb_dpb_pics) { |
|
discard_list[discard] = prev->dpb[i]; |
|
++discard; |
|
} else { |
|
++keep; |
|
} |
|
} |
|
av_assert0(keep <= enc->units.dpb_frames); |
|
|
|
if (discard == 0) { |
|
hp->h264pic_info.flags.adaptive_ref_pic_marking_mode_flag = 0; |
|
} else { |
|
hp->h264pic_info.flags.adaptive_ref_pic_marking_mode_flag = 1; |
|
for (i = 0; i < discard; i++) { |
|
VulkanEncodeH264Picture *old = discard_list[i]->codec_priv; |
|
av_assert0(old->frame_num < hp->frame_num); |
|
hp->mmco[i] = (StdVideoEncodeH264RefPicMarkingEntry) { |
|
.memory_management_control_operation = 1, |
|
.difference_of_pic_nums_minus1 = hp->frame_num - old->frame_num - 1, |
|
}; |
|
} |
|
hp->mmco[i] = (StdVideoEncodeH264RefPicMarkingEntry) { |
|
.memory_management_control_operation = 0, |
|
}; |
|
hp->ref_list_info.pRefPicMarkingOperations = hp->mmco; |
|
hp->ref_list_info.refPicMarkingOpCount = i + 1; |
|
} |
|
} |
|
|
|
if (pic->type == FF_HW_PICTURE_TYPE_I || pic->type == FF_HW_PICTURE_TYPE_IDR) |
|
return; |
|
|
|
// If the intended references are not the first entries of RefPicListN |
|
// by default, use ref-pic-list-modification to move them there. |
|
vk_enc_h264_default_ref_pic_list(avctx, pic, |
|
def_l0, def_l1, &n); |
|
|
|
if (pic->type == FF_HW_PICTURE_TYPE_P) { |
|
int need_rplm = 0; |
|
for (i = 0; i < pic->nb_refs[0]; i++) { |
|
av_assert0(pic->refs[0][i]); |
|
if (pic->refs[0][i] != (FFHWBaseEncodePicture *)def_l0[i]) |
|
need_rplm = 1; |
|
} |
|
|
|
hp->ref_list_info.flags.ref_pic_list_modification_flag_l0 = need_rplm; |
|
if (need_rplm) { |
|
int pic_num = hp->frame_num; |
|
for (i = 0; i < pic->nb_refs[0]; i++) { |
|
href = pic->refs[0][i]->codec_priv; |
|
av_assert0(href->frame_num != pic_num); |
|
if (href->frame_num < pic_num) { |
|
hp->mods[0][i] = (StdVideoEncodeH264RefListModEntry) { |
|
.modification_of_pic_nums_idc = 0, |
|
.abs_diff_pic_num_minus1 = pic_num - href->frame_num - 1, |
|
}; |
|
} else { |
|
hp->mods[0][i] = (StdVideoEncodeH264RefListModEntry) { |
|
.modification_of_pic_nums_idc = 1, |
|
.abs_diff_pic_num_minus1 = href->frame_num - pic_num - 1, |
|
}; |
|
} |
|
pic_num = href->frame_num; |
|
} |
|
hp->ref_list_info.pRefList0ModOperations = hp->mods[0]; |
|
hp->ref_list_info.refList0ModOpCount = i - 1; |
|
} |
|
} else { |
|
int need_rplm_l0 = 0, need_rplm_l1 = 0; |
|
int n0 = 0, n1 = 0; |
|
for (i = 0; i < pic->nb_refs[0]; i++) { |
|
av_assert0(pic->refs[0][i]); |
|
href = pic->refs[0][i]->codec_priv; |
|
av_assert0(href->pic_order_cnt < hp->pic_order_cnt); |
|
if (pic->refs[0][i] != (FFHWBaseEncodePicture *)def_l0[n0]) |
|
need_rplm_l0 = 1; |
|
++n0; |
|
} |
|
|
|
for (int i = 0; i < pic->nb_refs[1]; i++) { |
|
av_assert0(pic->refs[1][i]); |
|
href = pic->refs[1][i]->codec_priv; |
|
av_assert0(href->pic_order_cnt > hp->pic_order_cnt); |
|
if (pic->refs[1][i] != (FFHWBaseEncodePicture *)def_l1[n1]) |
|
need_rplm_l1 = 1; |
|
++n1; |
|
} |
|
|
|
hp->ref_list_info.flags.ref_pic_list_modification_flag_l0 = need_rplm_l0; |
|
if (need_rplm_l0) { |
|
int pic_num = hp->frame_num; |
|
for (i = j = 0; i < pic->nb_refs[0]; i++) { |
|
href = pic->refs[0][i]->codec_priv; |
|
av_assert0(href->frame_num != pic_num); |
|
if (href->frame_num < pic_num) { |
|
hp->mods[0][j] = (StdVideoEncodeH264RefListModEntry) { |
|
.modification_of_pic_nums_idc = 0, |
|
.abs_diff_pic_num_minus1 = pic_num - href->frame_num - 1, |
|
}; |
|
} else { |
|
hp->mods[0][j] = (StdVideoEncodeH264RefListModEntry) { |
|
.modification_of_pic_nums_idc = 1, |
|
.abs_diff_pic_num_minus1 = href->frame_num - pic_num - 1, |
|
}; |
|
} |
|
pic_num = href->frame_num; |
|
++j; |
|
} |
|
hp->ref_list_info.pRefList0ModOperations = hp->mods[0]; |
|
hp->ref_list_info.refList0ModOpCount = j - 1; |
|
} |
|
|
|
hp->ref_list_info.flags.ref_pic_list_modification_flag_l1 = need_rplm_l1; |
|
if (need_rplm_l1) { |
|
int pic_num = hp->frame_num; |
|
for (i = j = 0; i < pic->nb_refs[1]; i++) { |
|
href = pic->refs[1][i]->codec_priv; |
|
av_assert0(href->frame_num != pic_num); |
|
if (href->frame_num < pic_num) { |
|
hp->mods[1][j] = (StdVideoEncodeH264RefListModEntry) { |
|
.modification_of_pic_nums_idc = 0, |
|
.abs_diff_pic_num_minus1 = pic_num - href->frame_num - 1, |
|
}; |
|
} else { |
|
hp->mods[1][j] = (StdVideoEncodeH264RefListModEntry) { |
|
.modification_of_pic_nums_idc = 1, |
|
.abs_diff_pic_num_minus1 = href->frame_num - pic_num - 1, |
|
}; |
|
} |
|
pic_num = href->frame_num; |
|
++j; |
|
} |
|
hp->ref_list_info.pRefList1ModOperations = hp->mods[1]; |
|
hp->ref_list_info.refList1ModOpCount = j - 1; |
|
} |
|
} |
|
} |
|
|
|
static int init_pic_params(AVCodecContext *avctx, FFHWBaseEncodePicture *pic, |
|
VkVideoEncodeInfoKHR *encode_info) |
|
{ |
|
int err; |
|
FFVulkanEncodePicture *vp = pic->priv; |
|
VulkanEncodeH264Picture *hp = pic->codec_priv; |
|
VkVideoReferenceSlotInfoKHR *ref_slot; |
|
|
|
err = vk_enc_h264_update_pic_info(avctx, pic); |
|
if (err < 0) |
|
return err; |
|
|
|
hp->vkh264pic_info = (VkVideoEncodeH264PictureInfoKHR) { |
|
.sType = VK_STRUCTURE_TYPE_VIDEO_ENCODE_H264_PICTURE_INFO_KHR, |
|
.pNext = NULL, |
|
.pNaluSliceEntries = NULL, // Filled in during setup_slices() |
|
.naluSliceEntryCount = 0, // Filled in during setup_slices() |
|
.pStdPictureInfo = &hp->h264pic_info, |
|
}; |
|
|
|
hp->h264pic_info = (StdVideoEncodeH264PictureInfo) { |
|
.flags = (StdVideoEncodeH264PictureInfoFlags) { |
|
.IdrPicFlag = pic->type == FF_HW_PICTURE_TYPE_IDR, |
|
.is_reference = pic->is_reference, |
|
.no_output_of_prior_pics_flag = 0, |
|
.long_term_reference_flag = 0, |
|
.adaptive_ref_pic_marking_mode_flag = 0, // Filled in during setup_refs() |
|
/* Reserved */ |
|
}, |
|
.seq_parameter_set_id = 0, |
|
.pic_parameter_set_id = 0, |
|
.idr_pic_id = hp->idr_pic_id, |
|
.primary_pic_type = pic->type == FF_HW_PICTURE_TYPE_P ? STD_VIDEO_H264_PICTURE_TYPE_P : |
|
pic->type == FF_HW_PICTURE_TYPE_B ? STD_VIDEO_H264_PICTURE_TYPE_B : |
|
pic->type == FF_HW_PICTURE_TYPE_I ? STD_VIDEO_H264_PICTURE_TYPE_I : |
|
STD_VIDEO_H264_PICTURE_TYPE_IDR, |
|
.frame_num = hp->frame_num, |
|
.PicOrderCnt = hp->pic_order_cnt, |
|
.temporal_id = 0, /* ? */ |
|
/* Reserved */ |
|
.pRefLists = NULL, // Filled in during setup_refs |
|
}; |
|
encode_info->pNext = &hp->vkh264pic_info; |
|
|
|
hp->h264dpb_info = (StdVideoEncodeH264ReferenceInfo) { |
|
.flags = (StdVideoEncodeH264ReferenceInfoFlags) { |
|
.used_for_long_term_reference = 0, |
|
/* Reserved */ |
|
}, |
|
.primary_pic_type = hp->h264pic_info.primary_pic_type, |
|
.FrameNum = hp->h264pic_info.frame_num, |
|
.PicOrderCnt = hp->h264pic_info.PicOrderCnt, |
|
.long_term_pic_num = 0, |
|
.long_term_frame_idx = 0, |
|
.temporal_id = hp->h264pic_info.temporal_id, |
|
}; |
|
hp->vkh264dpb_info = (VkVideoEncodeH264DpbSlotInfoKHR) { |
|
.sType = VK_STRUCTURE_TYPE_VIDEO_ENCODE_H264_DPB_SLOT_INFO_KHR, |
|
.pStdReferenceInfo = &hp->h264dpb_info, |
|
}; |
|
|
|
vp->dpb_slot.pNext = &hp->vkh264dpb_info; |
|
|
|
ref_slot = (VkVideoReferenceSlotInfoKHR *)encode_info->pSetupReferenceSlot; |
|
ref_slot->pNext = &hp->vkh264dpb_info; |
|
|
|
setup_refs(avctx, pic, encode_info); |
|
|
|
setup_slices(avctx, pic); |
|
|
|
return 0; |
|
} |
|
|
|
static int init_profile(AVCodecContext *avctx, |
|
VkVideoProfileInfoKHR *profile, void *pnext) |
|
{ |
|
VkResult ret; |
|
VulkanEncodeH264Context *enc = avctx->priv_data; |
|
FFVulkanEncodeContext *ctx = &enc->common; |
|
FFVulkanContext *s = &ctx->s; |
|
FFVulkanFunctions *vk = &ctx->s.vkfn; |
|
VkVideoEncodeH264CapabilitiesKHR h264_caps = { |
|
.sType = VK_STRUCTURE_TYPE_VIDEO_ENCODE_H264_CAPABILITIES_KHR, |
|
}; |
|
VkVideoEncodeCapabilitiesKHR enc_caps = { |
|
.sType = VK_STRUCTURE_TYPE_VIDEO_ENCODE_CAPABILITIES_KHR, |
|
.pNext = &h264_caps, |
|
}; |
|
VkVideoCapabilitiesKHR caps = { |
|
.sType = VK_STRUCTURE_TYPE_VIDEO_CAPABILITIES_KHR, |
|
.pNext = &enc_caps, |
|
}; |
|
|
|
/* In order of preference */ |
|
int last_supported = AV_PROFILE_UNKNOWN; |
|
static const int known_profiles[] = { |
|
AV_PROFILE_H264_CONSTRAINED_BASELINE, |
|
AV_PROFILE_H264_MAIN, |
|
AV_PROFILE_H264_HIGH, |
|
AV_PROFILE_H264_HIGH_10, |
|
}; |
|
int nb_profiles = FF_ARRAY_ELEMS(known_profiles); |
|
|
|
const AVPixFmtDescriptor *desc = av_pix_fmt_desc_get(s->frames->sw_format); |
|
if (!desc) |
|
return AVERROR(EINVAL); |
|
|
|
if (desc->comp[0].depth == 8) |
|
nb_profiles = 3; |
|
|
|
enc->profile = (VkVideoEncodeH264ProfileInfoKHR) { |
|
.sType = VK_STRUCTURE_TYPE_VIDEO_ENCODE_H264_PROFILE_INFO_KHR, |
|
.pNext = pnext, |
|
.stdProfileIdc = ff_vk_h264_profile_to_vk(avctx->profile), |
|
}; |
|
profile->pNext = &enc->profile; |
|
|
|
/* Set level */ |
|
if (avctx->level == AV_LEVEL_UNKNOWN) |
|
avctx->level = enc->common.opts.level; |
|
|
|
/* User has explicitly specified a profile. */ |
|
if (avctx->profile != AV_PROFILE_UNKNOWN) |
|
return 0; |
|
|
|
av_log(avctx, AV_LOG_DEBUG, "Supported profiles:\n"); |
|
for (int i = 0; i < nb_profiles; i++) { |
|
enc->profile.stdProfileIdc = ff_vk_h264_profile_to_vk(known_profiles[i]); |
|
ret = vk->GetPhysicalDeviceVideoCapabilitiesKHR(s->hwctx->phys_dev, |
|
profile, |
|
&caps); |
|
if (ret == VK_SUCCESS) { |
|
av_log(avctx, AV_LOG_DEBUG, " %s\n", |
|
avcodec_profile_name(avctx->codec_id, known_profiles[i])); |
|
last_supported = known_profiles[i]; |
|
} |
|
} |
|
|
|
if (last_supported == AV_PROFILE_UNKNOWN) { |
|
av_log(avctx, AV_LOG_ERROR, "No supported profiles for given format\n"); |
|
return AVERROR(ENOTSUP); |
|
} |
|
|
|
enc->profile.stdProfileIdc = ff_vk_h264_profile_to_vk(last_supported); |
|
av_log(avctx, AV_LOG_VERBOSE, "Using profile %s\n", |
|
avcodec_profile_name(avctx->codec_id, last_supported)); |
|
avctx->profile = last_supported; |
|
|
|
return 0; |
|
} |
|
|
|
static int init_enc_options(AVCodecContext *avctx) |
|
{ |
|
VulkanEncodeH264Context *enc = avctx->priv_data; |
|
FFHWBaseEncodeH264Opts *unit_opts = &enc->unit_opts; |
|
|
|
if (avctx->rc_buffer_size) |
|
unit_opts->hrd_buffer_size = avctx->rc_buffer_size; |
|
else if (avctx->rc_max_rate > 0) |
|
unit_opts->hrd_buffer_size = avctx->rc_max_rate; |
|
else |
|
unit_opts->hrd_buffer_size = avctx->bit_rate; |
|
|
|
if (avctx->rc_initial_buffer_occupancy) { |
|
if (avctx->rc_initial_buffer_occupancy > unit_opts->hrd_buffer_size) { |
|
av_log(avctx, AV_LOG_ERROR, "Invalid RC buffer settings: " |
|
"must have initial buffer size (%d) <= " |
|
"buffer size (%"PRId64").\n", |
|
avctx->rc_initial_buffer_occupancy, unit_opts->hrd_buffer_size); |
|
return AVERROR(EINVAL); |
|
} |
|
unit_opts->initial_buffer_fullness = avctx->rc_initial_buffer_occupancy; |
|
} else { |
|
unit_opts->initial_buffer_fullness = unit_opts->hrd_buffer_size * 3 / 4; |
|
} |
|
|
|
if (enc->common.opts.rc_mode == VK_VIDEO_ENCODE_RATE_CONTROL_MODE_DISABLED_BIT_KHR) { |
|
/* HRD info is required for timing */ |
|
enc->unit_elems &= ~UNIT_SEI_TIMING; |
|
|
|
enc->fixed_qp_p = av_clip(enc->common.explicit_qp, |
|
enc->caps.minQp, enc->caps.maxQp); |
|
if (avctx->i_quant_factor > 0.0) |
|
unit_opts->fixed_qp_idr = av_clip((avctx->i_quant_factor * enc->fixed_qp_p + |
|
avctx->i_quant_offset) + 0.5, |
|
enc->caps.minQp, enc->caps.maxQp); |
|
else |
|
unit_opts->fixed_qp_idr = enc->fixed_qp_p; |
|
|
|
if (avctx->b_quant_factor > 0.0) |
|
enc->fixed_qp_b = av_clip((avctx->b_quant_factor * enc->fixed_qp_p + |
|
avctx->b_quant_offset) + 0.5, |
|
enc->caps.minQp, enc->caps.maxQp); |
|
else |
|
enc->fixed_qp_b = enc->fixed_qp_p; |
|
|
|
av_log(avctx, AV_LOG_DEBUG, "Using fixed QP = " |
|
"%d / %d / %d for IDR- / P- / B-frames.\n", |
|
unit_opts->fixed_qp_idr, enc->fixed_qp_p, enc->fixed_qp_b); |
|
} else { |
|
unit_opts->fixed_qp_idr = 26; |
|
enc->fixed_qp_p = 26; |
|
enc->fixed_qp_b = 26; |
|
} |
|
|
|
return 0; |
|
} |
|
|
|
static av_cold int init_sequence_headers(AVCodecContext *avctx) |
|
{ |
|
int err; |
|
VulkanEncodeH264Context *enc = avctx->priv_data; |
|
|
|
FFHWBaseEncodeH264 *units = &enc->units; |
|
FFHWBaseEncodeH264Opts *unit_opts = &enc->unit_opts; |
|
|
|
unit_opts->bit_rate = avctx->bit_rate; |
|
unit_opts->mb_width = FFALIGN(avctx->width, 16) / 16; |
|
unit_opts->mb_height = FFALIGN(avctx->height, 16) / 16; |
|
unit_opts->flags = enc->unit_elems & UNIT_SEI_TIMING ? FF_HW_H264_SEI_TIMING : 0; |
|
|
|
/* cabac already set via an option */ |
|
/* fixed_qp_idr initialized in init_enc_options() */ |
|
/* hrd_buffer_size initialized in init_enc_options() */ |
|
/* initial_buffer_fullness initialized in init_enc_options() */ |
|
|
|
err = ff_hw_base_encode_init_params_h264(&enc->common.base, avctx, |
|
units, unit_opts); |
|
if (err < 0) |
|
return err; |
|
|
|
units->raw_sps.seq_scaling_matrix_present_flag = |
|
!!(enc->caps.stdSyntaxFlags & VK_VIDEO_ENCODE_H264_STD_SCALING_MATRIX_PRESENT_FLAG_SET_BIT_KHR); |
|
units->raw_pps.pic_scaling_matrix_present_flag = |
|
!!(enc->caps.stdSyntaxFlags & VK_VIDEO_ENCODE_H264_STD_SCALING_MATRIX_PRESENT_FLAG_SET_BIT_KHR); |
|
units->raw_pps.transform_8x8_mode_flag = |
|
!!(enc->caps.stdSyntaxFlags & VK_VIDEO_ENCODE_H264_STD_TRANSFORM_8X8_MODE_FLAG_SET_BIT_KHR); |
|
|
|
return 0; |
|
} |
|
|
|
typedef struct VulkanH264Units { |
|
StdVideoH264SequenceParameterSet vksps; |
|
StdVideoH264ScalingLists vksps_scaling; |
|
StdVideoH264HrdParameters vksps_vui_header; |
|
StdVideoH264SequenceParameterSetVui vksps_vui; |
|
|
|
StdVideoH264PictureParameterSet vkpps; |
|
StdVideoH264ScalingLists vkpps_scaling; |
|
} VulkanH264Units; |
|
|
|
static av_cold int base_unit_to_vk(AVCodecContext *avctx, |
|
VulkanH264Units *vk_units) |
|
{ |
|
VulkanEncodeH264Context *enc = avctx->priv_data; |
|
|
|
FFHWBaseEncodeH264 *units = &enc->units; |
|
|
|
H264RawSPS *sps = &units->raw_sps; |
|
H264RawHRD *hrd = &sps->vui.nal_hrd_parameters; |
|
StdVideoH264ScalingLists *vksps_scaling = &vk_units->vksps_scaling; |
|
StdVideoH264HrdParameters *vksps_vui_header = &vk_units->vksps_vui_header; |
|
StdVideoH264SequenceParameterSetVui *vksps_vui = &vk_units->vksps_vui; |
|
StdVideoH264SequenceParameterSet *vksps = &vk_units->vksps; |
|
|
|
H264RawPPS *pps = &units->raw_pps; |
|
StdVideoH264ScalingLists *vkpps_scaling = &vk_units->vkpps_scaling; |
|
StdVideoH264PictureParameterSet *vkpps = &vk_units->vkpps; |
|
|
|
*vksps_scaling = (StdVideoH264ScalingLists) { |
|
.scaling_list_present_mask = 0x0, // mask |
|
.use_default_scaling_matrix_mask = 1, |
|
}; |
|
|
|
*vksps_vui_header = (StdVideoH264HrdParameters) { |
|
.cpb_cnt_minus1 = hrd->cpb_cnt_minus1, |
|
.bit_rate_scale = hrd->bit_rate_scale, |
|
.cpb_size_scale = hrd->cpb_size_scale, |
|
/* Reserved */ |
|
/* bit_rate/cpb_size/cbr_flag set below */ |
|
.initial_cpb_removal_delay_length_minus1 = hrd->initial_cpb_removal_delay_length_minus1, |
|
.cpb_removal_delay_length_minus1 = hrd->cpb_removal_delay_length_minus1, |
|
.dpb_output_delay_length_minus1 = hrd->dpb_output_delay_length_minus1, |
|
.time_offset_length = hrd->time_offset_length, |
|
}; |
|
|
|
for (int i = 0; i < H264_MAX_CPB_CNT; i++) { |
|
vksps_vui_header->bit_rate_value_minus1[i] = hrd->bit_rate_value_minus1[i]; |
|
vksps_vui_header->cpb_size_value_minus1[i] = hrd->cpb_size_value_minus1[i]; |
|
vksps_vui_header->cbr_flag[i] = hrd->cbr_flag[i]; |
|
} |
|
|
|
*vksps_vui = (StdVideoH264SequenceParameterSetVui) { |
|
.flags = (StdVideoH264SpsVuiFlags) { |
|
.aspect_ratio_info_present_flag = sps->vui.aspect_ratio_info_present_flag, |
|
.overscan_info_present_flag = sps->vui.overscan_info_present_flag, |
|
.overscan_appropriate_flag = sps->vui.overscan_appropriate_flag, |
|
.video_signal_type_present_flag = sps->vui.video_signal_type_present_flag, |
|
.video_full_range_flag = sps->vui.video_full_range_flag, |
|
.color_description_present_flag = sps->vui.colour_description_present_flag, |
|
.chroma_loc_info_present_flag = sps->vui.chroma_loc_info_present_flag, |
|
.timing_info_present_flag = sps->vui.timing_info_present_flag, |
|
.fixed_frame_rate_flag = sps->vui.fixed_frame_rate_flag, |
|
.bitstream_restriction_flag = sps->vui.bitstream_restriction_flag, |
|
.nal_hrd_parameters_present_flag = sps->vui.nal_hrd_parameters_present_flag, |
|
.vcl_hrd_parameters_present_flag = sps->vui.vcl_hrd_parameters_present_flag, |
|
}, |
|
.aspect_ratio_idc = sps->vui.aspect_ratio_idc, |
|
.sar_width = sps->vui.sar_width, |
|
.sar_height = sps->vui.sar_height, |
|
.video_format = sps->vui.video_format, |
|
.colour_primaries = sps->vui.colour_primaries, |
|
.transfer_characteristics = sps->vui.transfer_characteristics, |
|
.matrix_coefficients = sps->vui.matrix_coefficients, |
|
.num_units_in_tick = sps->vui.num_units_in_tick, |
|
.time_scale = sps->vui.time_scale, |
|
.max_num_reorder_frames = sps->vui.max_num_reorder_frames, |
|
.max_dec_frame_buffering = sps->vui.max_dec_frame_buffering, |
|
.chroma_sample_loc_type_top_field = sps->vui.chroma_sample_loc_type_top_field, |
|
.chroma_sample_loc_type_bottom_field = sps->vui.chroma_sample_loc_type_bottom_field, |
|
/* Reserved */ |
|
.pHrdParameters = vksps_vui_header, |
|
}; |
|
|
|
*vksps = (StdVideoH264SequenceParameterSet) { |
|
.flags = (StdVideoH264SpsFlags) { |
|
.constraint_set0_flag = sps->constraint_set0_flag, |
|
.constraint_set1_flag = sps->constraint_set1_flag, |
|
.constraint_set2_flag = sps->constraint_set2_flag, |
|
.constraint_set3_flag = sps->constraint_set3_flag, |
|
.constraint_set4_flag = sps->constraint_set4_flag, |
|
.constraint_set5_flag = sps->constraint_set5_flag, |
|
.direct_8x8_inference_flag = sps->direct_8x8_inference_flag, |
|
.mb_adaptive_frame_field_flag = sps->mb_adaptive_frame_field_flag, |
|
.frame_mbs_only_flag = sps->frame_mbs_only_flag, |
|
.delta_pic_order_always_zero_flag = sps->delta_pic_order_always_zero_flag, |
|
.separate_colour_plane_flag = sps->separate_colour_plane_flag, |
|
.gaps_in_frame_num_value_allowed_flag = sps->gaps_in_frame_num_allowed_flag, |
|
.qpprime_y_zero_transform_bypass_flag = sps->qpprime_y_zero_transform_bypass_flag, |
|
.frame_cropping_flag = sps->frame_cropping_flag, |
|
.seq_scaling_matrix_present_flag = sps->seq_scaling_matrix_present_flag, |
|
.vui_parameters_present_flag = sps->vui_parameters_present_flag, |
|
}, |
|
.profile_idc = ff_vk_h264_profile_to_vk(sps->profile_idc), |
|
.level_idc = ff_vk_h264_level_to_vk(sps->level_idc), |
|
.chroma_format_idc = sps->chroma_format_idc, |
|
.seq_parameter_set_id = sps->seq_parameter_set_id, |
|
.bit_depth_luma_minus8 = sps->bit_depth_luma_minus8, |
|
.bit_depth_chroma_minus8 = sps->bit_depth_chroma_minus8, |
|
.log2_max_frame_num_minus4 = sps->log2_max_frame_num_minus4, |
|
.pic_order_cnt_type = sps->pic_order_cnt_type, |
|
.offset_for_non_ref_pic = sps->offset_for_non_ref_pic, |
|
.offset_for_top_to_bottom_field = sps->offset_for_top_to_bottom_field, |
|
.log2_max_pic_order_cnt_lsb_minus4 = sps->log2_max_pic_order_cnt_lsb_minus4, |
|
.num_ref_frames_in_pic_order_cnt_cycle = sps->num_ref_frames_in_pic_order_cnt_cycle, |
|
.max_num_ref_frames = sps->max_num_ref_frames, |
|
/* Reserved */ |
|
.pic_width_in_mbs_minus1 = sps->pic_width_in_mbs_minus1, |
|
.pic_height_in_map_units_minus1 = sps->pic_height_in_map_units_minus1, |
|
.frame_crop_left_offset = sps->frame_crop_left_offset, |
|
.frame_crop_right_offset = sps->frame_crop_right_offset, |
|
.frame_crop_top_offset = sps->frame_crop_top_offset, |
|
.frame_crop_bottom_offset = sps->frame_crop_bottom_offset, |
|
/* Reserved */ |
|
.pOffsetForRefFrame = sps->offset_for_ref_frame, |
|
.pScalingLists = vksps_scaling, |
|
.pSequenceParameterSetVui = vksps_vui, |
|
}; |
|
|
|
*vkpps_scaling = (StdVideoH264ScalingLists) { |
|
.scaling_list_present_mask = 0x0, // mask |
|
.use_default_scaling_matrix_mask = 1, |
|
}; |
|
|
|
*vkpps = (StdVideoH264PictureParameterSet) { |
|
.flags = (StdVideoH264PpsFlags) { |
|
.transform_8x8_mode_flag = pps->transform_8x8_mode_flag, |
|
.redundant_pic_cnt_present_flag = pps->redundant_pic_cnt_present_flag, |
|
.constrained_intra_pred_flag = pps->constrained_intra_pred_flag, |
|
.deblocking_filter_control_present_flag = pps->deblocking_filter_control_present_flag, |
|
.weighted_pred_flag = pps->weighted_pred_flag, |
|
.bottom_field_pic_order_in_frame_present_flag = pps->bottom_field_pic_order_in_frame_present_flag, |
|
.entropy_coding_mode_flag = pps->entropy_coding_mode_flag, |
|
.pic_scaling_matrix_present_flag = pps->pic_scaling_matrix_present_flag, |
|
}, |
|
.seq_parameter_set_id = pps->seq_parameter_set_id, |
|
.pic_parameter_set_id = pps->pic_parameter_set_id, |
|
.num_ref_idx_l0_default_active_minus1 = pps->num_ref_idx_l0_default_active_minus1, |
|
.num_ref_idx_l1_default_active_minus1 = pps->num_ref_idx_l1_default_active_minus1, |
|
.weighted_bipred_idc = pps->weighted_bipred_idc, |
|
.pic_init_qp_minus26 = pps->pic_init_qp_minus26, |
|
.pic_init_qs_minus26 = pps->pic_init_qs_minus26, |
|
.chroma_qp_index_offset = pps->chroma_qp_index_offset, |
|
.second_chroma_qp_index_offset = pps->second_chroma_qp_index_offset, |
|
.pScalingLists = vkpps_scaling, |
|
}; |
|
|
|
return 0; |
|
} |
|
|
|
static int create_session_params(AVCodecContext *avctx) |
|
{ |
|
int err; |
|
VulkanEncodeH264Context *enc = avctx->priv_data; |
|
FFVulkanEncodeContext *ctx = &enc->common; |
|
FFVulkanContext *s = &ctx->s; |
|
FFVulkanFunctions *vk = &ctx->s.vkfn; |
|
|
|
VulkanH264Units vk_units = { 0 }; |
|
|
|
VkVideoEncodeH264SessionParametersAddInfoKHR h264_params_info; |
|
VkVideoEncodeH264SessionParametersCreateInfoKHR h264_params; |
|
|
|
/* Convert it to Vulkan */ |
|
err = base_unit_to_vk(avctx, &vk_units); |
|
if (err < 0) { |
|
av_log(avctx, AV_LOG_ERROR, "Unable to convert SPS/PPS units to Vulkan: %s\n", |
|
av_err2str(err)); |
|
return err; |
|
} |
|
|
|
/* Destroy the session params */ |
|
if (ctx->session_params) |
|
vk->DestroyVideoSessionParametersKHR(s->hwctx->act_dev, |
|
ctx->session_params, |
|
s->hwctx->alloc); |
|
|
|
h264_params_info = (VkVideoEncodeH264SessionParametersAddInfoKHR) { |
|
.sType = VK_STRUCTURE_TYPE_VIDEO_ENCODE_H264_SESSION_PARAMETERS_ADD_INFO_KHR, |
|
.pStdSPSs = &vk_units.vksps, |
|
.stdSPSCount = 1, |
|
.pStdPPSs = &vk_units.vkpps, |
|
.stdPPSCount = 1, |
|
}; |
|
h264_params = (VkVideoEncodeH264SessionParametersCreateInfoKHR) { |
|
.sType = VK_STRUCTURE_TYPE_VIDEO_ENCODE_H264_SESSION_PARAMETERS_CREATE_INFO_KHR, |
|
.maxStdSPSCount = 1, |
|
.maxStdPPSCount = 1, |
|
.pParametersAddInfo = &h264_params_info, |
|
}; |
|
|
|
return ff_vulkan_encode_create_session_params(avctx, ctx, &h264_params); |
|
} |
|
|
|
static int parse_feedback_units(AVCodecContext *avctx, |
|
const uint8_t *data, size_t size, |
|
int sps_override, int pps_override) |
|
{ |
|
int err; |
|
VulkanEncodeH264Context *enc = avctx->priv_data; |
|
|
|
CodedBitstreamContext *cbs; |
|
CodedBitstreamFragment au = { 0 }; |
|
|
|
err = ff_cbs_init(&cbs, AV_CODEC_ID_H264, avctx); |
|
if (err < 0) |
|
return err; |
|
|
|
err = ff_cbs_read(cbs, &au, data, size); |
|
if (err < 0) { |
|
av_log(avctx, AV_LOG_ERROR, "Unable to parse feedback units, bad drivers: %s\n", |
|
av_err2str(err)); |
|
return err; |
|
} |
|
|
|
/* If PPS has an override, just copy it entirely. */ |
|
if (pps_override) { |
|
for (int i = 0; i < au.nb_units; i++) { |
|
if (au.units[i].type == H264_NAL_PPS) { |
|
H264RawPPS *pps = au.units[i].content; |
|
memcpy(&enc->units.raw_pps, pps, sizeof(*pps)); |
|
break; |
|
} |
|
} |
|
} |
|
|
|
ff_cbs_fragment_free(&au); |
|
ff_cbs_close(&cbs); |
|
|
|
return 0; |
|
} |
|
|
|
static int init_base_units(AVCodecContext *avctx) |
|
{ |
|
int err; |
|
VkResult ret; |
|
VulkanEncodeH264Context *enc = avctx->priv_data; |
|
FFVulkanEncodeContext *ctx = &enc->common; |
|
FFVulkanContext *s = &ctx->s; |
|
FFVulkanFunctions *vk = &ctx->s.vkfn; |
|
|
|
VkVideoEncodeH264SessionParametersGetInfoKHR h264_params_info; |
|
VkVideoEncodeSessionParametersGetInfoKHR params_info; |
|
VkVideoEncodeH264SessionParametersFeedbackInfoKHR h264_params_feedback; |
|
VkVideoEncodeSessionParametersFeedbackInfoKHR params_feedback; |
|
|
|
void *data = NULL; |
|
size_t data_size = 0; |
|
|
|
/* Generate SPS/PPS unit info */ |
|
err = init_sequence_headers(avctx); |
|
if (err < 0) { |
|
av_log(avctx, AV_LOG_ERROR, "Unable to initialize SPS/PPS units: %s\n", |
|
av_err2str(err)); |
|
return err; |
|
} |
|
|
|
/* Create session parameters from them */ |
|
err = create_session_params(avctx); |
|
if (err < 0) |
|
return err; |
|
|
|
h264_params_info = (VkVideoEncodeH264SessionParametersGetInfoKHR) { |
|
.sType = VK_STRUCTURE_TYPE_VIDEO_ENCODE_H264_SESSION_PARAMETERS_GET_INFO_KHR, |
|
.writeStdSPS = 1, |
|
.writeStdPPS = 1, |
|
.stdSPSId = enc->units.raw_sps.seq_parameter_set_id, |
|
.stdPPSId = enc->units.raw_pps.pic_parameter_set_id, |
|
}; |
|
params_info = (VkVideoEncodeSessionParametersGetInfoKHR) { |
|
.sType = VK_STRUCTURE_TYPE_VIDEO_ENCODE_SESSION_PARAMETERS_GET_INFO_KHR, |
|
.pNext = &h264_params_info, |
|
.videoSessionParameters = ctx->session_params, |
|
}; |
|
|
|
h264_params_feedback = (VkVideoEncodeH264SessionParametersFeedbackInfoKHR) { |
|
.sType = VK_STRUCTURE_TYPE_VIDEO_ENCODE_H264_SESSION_PARAMETERS_FEEDBACK_INFO_KHR, |
|
}; |
|
params_feedback = (VkVideoEncodeSessionParametersFeedbackInfoKHR) { |
|
.sType = VK_STRUCTURE_TYPE_VIDEO_ENCODE_SESSION_PARAMETERS_FEEDBACK_INFO_KHR, |
|
.pNext = &h264_params_feedback, |
|
}; |
|
|
|
ret = vk->GetEncodedVideoSessionParametersKHR(s->hwctx->act_dev, ¶ms_info, |
|
¶ms_feedback, |
|
&data_size, data); |
|
if (ret == VK_INCOMPLETE || |
|
(ret == VK_SUCCESS) && (data_size > 0)) { |
|
data = av_mallocz(data_size); |
|
if (!data) |
|
return AVERROR(ENOMEM); |
|
} else { |
|
av_log(avctx, AV_LOG_ERROR, "Unable to get feedback for H.264 units = %"SIZE_SPECIFIER"\n", data_size); |
|
return err; |
|
} |
|
|
|
ret = vk->GetEncodedVideoSessionParametersKHR(s->hwctx->act_dev, ¶ms_info, |
|
¶ms_feedback, |
|
&data_size, data); |
|
if (ret != VK_SUCCESS) { |
|
av_log(avctx, AV_LOG_ERROR, "Error writing feedback units\n"); |
|
return err; |
|
} |
|
|
|
av_log(avctx, AV_LOG_VERBOSE, "Feedback units written, overrides: %i (SPS: %i PPS: %i)\n", |
|
params_feedback.hasOverrides, |
|
h264_params_feedback.hasStdSPSOverrides, |
|
h264_params_feedback.hasStdPPSOverrides); |
|
|
|
params_feedback.hasOverrides = 1; |
|
h264_params_feedback.hasStdPPSOverrides = 1; |
|
|
|
/* No need to sync any overrides */ |
|
if (!params_feedback.hasOverrides) |
|
return 0; |
|
|
|
/* Parse back tne units and override */ |
|
err = parse_feedback_units(avctx, data, data_size, |
|
h264_params_feedback.hasStdSPSOverrides, |
|
h264_params_feedback.hasStdPPSOverrides); |
|
if (err < 0) |
|
return err; |
|
|
|
/* Create final session parameters */ |
|
err = create_session_params(avctx); |
|
if (err < 0) |
|
return err; |
|
|
|
return 0; |
|
} |
|
|
|
static int vulkan_encode_h264_add_nal(AVCodecContext *avctx, |
|
CodedBitstreamFragment *au, |
|
void *nal_unit) |
|
{ |
|
H264RawNALUnitHeader *header = nal_unit; |
|
|
|
int err = ff_cbs_insert_unit_content(au, -1, |
|
header->nal_unit_type, nal_unit, NULL); |
|
if (err < 0) |
|
av_log(avctx, AV_LOG_ERROR, "Failed to add NAL unit: " |
|
"type = %d.\n", header->nal_unit_type); |
|
|
|
return err; |
|
} |
|
|
|
static int write_access_unit(AVCodecContext *avctx, |
|
uint8_t *data, size_t *data_len, |
|
CodedBitstreamFragment *au) |
|
{ |
|
VulkanEncodeH264Context *enc = avctx->priv_data; |
|
|
|
int err = ff_cbs_write_fragment_data(enc->cbs, au); |
|
if (err < 0) { |
|
av_log(avctx, AV_LOG_ERROR, "Failed to write packed header.\n"); |
|
return err; |
|
} |
|
|
|
if (*data_len < au->data_size) { |
|
av_log(avctx, AV_LOG_ERROR, "Access unit too large: %zu < %zu.\n", |
|
*data_len, au->data_size); |
|
return AVERROR(ENOSPC); |
|
} |
|
|
|
memcpy(data, au->data, au->data_size); |
|
*data_len = au->data_size; |
|
|
|
return 0; |
|
} |
|
|
|
static int write_sequence_headers(AVCodecContext *avctx, |
|
FFHWBaseEncodePicture *base_pic, |
|
uint8_t *data, size_t *data_len) |
|
{ |
|
int err; |
|
VulkanEncodeH264Context *enc = avctx->priv_data; |
|
VulkanEncodeH264Picture *hp = base_pic ? base_pic->codec_priv : NULL; |
|
CodedBitstreamFragment *au = &enc->current_access_unit; |
|
|
|
if (hp && hp->units_needed & UNIT_AUD) { |
|
err = vulkan_encode_h264_add_nal(avctx, au, &enc->raw_aud); |
|
if (err < 0) |
|
goto fail; |
|
} |
|
|
|
err = vulkan_encode_h264_add_nal(avctx, au, &enc->units.raw_sps); |
|
if (err < 0) |
|
goto fail; |
|
|
|
err = vulkan_encode_h264_add_nal(avctx, au, &enc->units.raw_pps); |
|
if (err < 0) |
|
goto fail; |
|
|
|
err = write_access_unit(avctx, data, data_len, au); |
|
fail: |
|
ff_cbs_fragment_reset(au); |
|
return err; |
|
} |
|
|
|
static int write_extra_headers(AVCodecContext *avctx, |
|
FFHWBaseEncodePicture *base_pic, |
|
uint8_t *data, size_t *data_len) |
|
{ |
|
int err; |
|
VulkanEncodeH264Context *enc = avctx->priv_data; |
|
VulkanEncodeH264Picture *hp = base_pic->codec_priv; |
|
CodedBitstreamFragment *au = &enc->current_access_unit; |
|
|
|
if (hp->units_needed & UNIT_AUD) { |
|
err = vulkan_encode_h264_add_nal(avctx, au, &enc->raw_aud); |
|
if (err < 0) |
|
goto fail; |
|
} |
|
|
|
if (hp->units_needed & UNIT_SEI_IDENTIFIER) { |
|
err = ff_cbs_sei_add_message(enc->cbs, au, 1, |
|
SEI_TYPE_USER_DATA_UNREGISTERED, |
|
&enc->sei_identifier, NULL); |
|
if (err < 0) |
|
goto fail; |
|
} |
|
|
|
if (hp->units_needed & UNIT_SEI_TIMING) { |
|
if (base_pic->type == FF_HW_PICTURE_TYPE_IDR) { |
|
err = ff_cbs_sei_add_message(enc->cbs, au, 1, |
|
SEI_TYPE_BUFFERING_PERIOD, |
|
&enc->units.sei_buffering_period, NULL); |
|
if (err < 0) |
|
goto fail; |
|
} |
|
err = ff_cbs_sei_add_message(enc->cbs, au, 1, |
|
SEI_TYPE_PIC_TIMING, |
|
&enc->sei_pic_timing, NULL); |
|
if (err < 0) |
|
goto fail; |
|
} |
|
|
|
if (hp->units_needed & UNIT_SEI_RECOVERY) { |
|
err = ff_cbs_sei_add_message(enc->cbs, au, 1, |
|
SEI_TYPE_RECOVERY_POINT, |
|
&enc->sei_recovery_point, NULL); |
|
if (err < 0) |
|
goto fail; |
|
} |
|
|
|
if (hp->units_needed & UNIT_SEI_A53_CC) { |
|
err = ff_cbs_sei_add_message(enc->cbs, au, 1, |
|
SEI_TYPE_USER_DATA_REGISTERED_ITU_T_T35, |
|
&enc->sei_a53cc, NULL); |
|
if (err < 0) |
|
goto fail; |
|
} |
|
|
|
if (hp->units_needed) { |
|
err = write_access_unit(avctx, data, data_len, au); |
|
if (err < 0) |
|
goto fail; |
|
} else { |
|
*data_len = 0; |
|
} |
|
|
|
fail: |
|
ff_cbs_fragment_reset(au); |
|
return err; |
|
} |
|
|
|
static int write_filler(AVCodecContext *avctx, uint32_t filler, |
|
uint8_t *data, size_t *data_len) |
|
{ |
|
int err; |
|
VulkanEncodeH264Context *enc = avctx->priv_data; |
|
CodedBitstreamFragment *au = &enc->current_access_unit; |
|
|
|
H264RawFiller raw_filler = { |
|
.nal_unit_header = { |
|
.nal_unit_type = H264_NAL_FILLER_DATA, |
|
}, |
|
.filler_size = filler, |
|
}; |
|
|
|
err = vulkan_encode_h264_add_nal(avctx, au, &raw_filler); |
|
if (err < 0) |
|
goto fail; |
|
|
|
err = write_access_unit(avctx, data, data_len, au); |
|
fail: |
|
ff_cbs_fragment_reset(au); |
|
return err; |
|
} |
|
|
|
static const FFVulkanCodec enc_cb = { |
|
.flags = FF_HW_FLAG_B_PICTURES | |
|
FF_HW_FLAG_B_PICTURE_REFERENCES | |
|
FF_HW_FLAG_NON_IDR_KEY_PICTURES, |
|
.picture_priv_data_size = sizeof(VulkanEncodeH264Picture), |
|
.filler_header_size = 6, |
|
.init_profile = init_profile, |
|
.init_pic_rc = init_pic_rc, |
|
.init_pic_params = init_pic_params, |
|
.write_sequence_headers = write_sequence_headers, |
|
.write_extra_headers = write_extra_headers, |
|
.write_filler = write_filler, |
|
}; |
|
|
|
static av_cold int vulkan_encode_h264_init(AVCodecContext *avctx) |
|
{ |
|
int err, ref_l0, ref_l1; |
|
VulkanEncodeH264Context *enc = avctx->priv_data; |
|
FFVulkanEncodeContext *ctx = &enc->common; |
|
FFVulkanContext *s = &ctx->s; |
|
FFHWBaseEncodeContext *base_ctx = &ctx->base; |
|
int flags; |
|
|
|
if (avctx->profile == AV_PROFILE_UNKNOWN) |
|
avctx->profile = enc->common.opts.profile; |
|
|
|
enc->caps = (VkVideoEncodeH264CapabilitiesKHR) { |
|
.sType = VK_STRUCTURE_TYPE_VIDEO_ENCODE_H264_CAPABILITIES_KHR, |
|
}; |
|
|
|
enc->quality_props = (VkVideoEncodeH264QualityLevelPropertiesKHR) { |
|
.sType = VK_STRUCTURE_TYPE_VIDEO_ENCODE_H264_QUALITY_LEVEL_PROPERTIES_KHR, |
|
}; |
|
|
|
err = ff_vulkan_encode_init(avctx, &enc->common, |
|
&ff_vk_enc_h264_desc, &enc_cb, |
|
&enc->caps, &enc->quality_props); |
|
if (err < 0) |
|
return err; |
|
|
|
av_log(avctx, AV_LOG_VERBOSE, "H264 encoder capabilities:\n"); |
|
av_log(avctx, AV_LOG_VERBOSE, " Standard capability flags:\n"); |
|
av_log(avctx, AV_LOG_VERBOSE, " separate_color_plane: %i\n", |
|
!!(enc->caps.stdSyntaxFlags & VK_VIDEO_ENCODE_H264_STD_SEPARATE_COLOR_PLANE_FLAG_SET_BIT_KHR)); |
|
av_log(avctx, AV_LOG_VERBOSE, " qprime_y_zero_transform_bypass: %i\n", |
|
!!(enc->caps.stdSyntaxFlags & VK_VIDEO_ENCODE_H264_STD_QPPRIME_Y_ZERO_TRANSFORM_BYPASS_FLAG_SET_BIT_KHR)); |
|
av_log(avctx, AV_LOG_VERBOSE, " scaling_lists: %i\n", |
|
!!(enc->caps.stdSyntaxFlags & VK_VIDEO_ENCODE_H264_STD_SCALING_MATRIX_PRESENT_FLAG_SET_BIT_KHR)); |
|
av_log(avctx, AV_LOG_VERBOSE, " chroma_qp_index_offset: %i\n", |
|
!!(enc->caps.stdSyntaxFlags & VK_VIDEO_ENCODE_H264_STD_CHROMA_QP_INDEX_OFFSET_BIT_KHR)); |
|
av_log(avctx, AV_LOG_VERBOSE, " second_chroma_qp_index_offset: %i\n", |
|
!!(enc->caps.stdSyntaxFlags & VK_VIDEO_ENCODE_H264_STD_SECOND_CHROMA_QP_INDEX_OFFSET_BIT_KHR)); |
|
av_log(avctx, AV_LOG_VERBOSE, " pic_init_qp: %i\n", |
|
!!(enc->caps.stdSyntaxFlags & VK_VIDEO_ENCODE_H264_STD_PIC_INIT_QP_MINUS26_BIT_KHR)); |
|
av_log(avctx, AV_LOG_VERBOSE, " weighted:%s%s%s\n", |
|
enc->caps.stdSyntaxFlags & VK_VIDEO_ENCODE_H264_STD_WEIGHTED_PRED_FLAG_SET_BIT_KHR ? |
|
" pred" : "", |
|
enc->caps.stdSyntaxFlags & VK_VIDEO_ENCODE_H264_STD_WEIGHTED_BIPRED_IDC_EXPLICIT_BIT_KHR ? |
|
" bipred_explicit" : "", |
|
enc->caps.stdSyntaxFlags & VK_VIDEO_ENCODE_H264_STD_WEIGHTED_BIPRED_IDC_IMPLICIT_BIT_KHR ? |
|
" bipred_implicit" : ""); |
|
av_log(avctx, AV_LOG_VERBOSE, " 8x8_transforms: %i\n", |
|
!!(enc->caps.stdSyntaxFlags & VK_VIDEO_ENCODE_H264_STD_TRANSFORM_8X8_MODE_FLAG_SET_BIT_KHR)); |
|
av_log(avctx, AV_LOG_VERBOSE, " disable_direct_spatial_mv_pred: %i\n", |
|
!!(enc->caps.stdSyntaxFlags & VK_VIDEO_ENCODE_H264_STD_DIRECT_SPATIAL_MV_PRED_FLAG_UNSET_BIT_KHR)); |
|
av_log(avctx, AV_LOG_VERBOSE, " coder:%s%s\n", |
|
enc->caps.stdSyntaxFlags & VK_VIDEO_ENCODE_H264_STD_ENTROPY_CODING_MODE_FLAG_UNSET_BIT_KHR ? |
|
" cabac" : "", |
|
enc->caps.stdSyntaxFlags & VK_VIDEO_ENCODE_H264_STD_ENTROPY_CODING_MODE_FLAG_SET_BIT_KHR ? |
|
" cavlc" : ""); |
|
av_log(avctx, AV_LOG_VERBOSE, " direct_8x8_inference: %i\n", |
|
!!(enc->caps.stdSyntaxFlags & VK_VIDEO_ENCODE_H264_STD_DIRECT_8X8_INFERENCE_FLAG_UNSET_BIT_KHR)); |
|
av_log(avctx, AV_LOG_VERBOSE, " constrained_intra_pred: %i\n", |
|
!!(enc->caps.stdSyntaxFlags & VK_VIDEO_ENCODE_H264_STD_CONSTRAINED_INTRA_PRED_FLAG_SET_BIT_KHR)); |
|
av_log(avctx, AV_LOG_VERBOSE, " deblock:%s%s%s\n", |
|
enc->caps.stdSyntaxFlags & VK_VIDEO_ENCODE_H264_STD_DEBLOCKING_FILTER_DISABLED_BIT_KHR ? |
|
" filter_disabling" : "", |
|
enc->caps.stdSyntaxFlags & VK_VIDEO_ENCODE_H264_STD_DEBLOCKING_FILTER_ENABLED_BIT_KHR ? |
|
" filter_enabling" : "", |
|
enc->caps.stdSyntaxFlags & VK_VIDEO_ENCODE_H264_STD_DEBLOCKING_FILTER_PARTIAL_BIT_KHR ? |
|
" filter_partial" : ""); |
|
|
|
av_log(avctx, AV_LOG_VERBOSE, " Capability flags:\n"); |
|
av_log(avctx, AV_LOG_VERBOSE, " hdr_compliance: %i\n", |
|
!!(enc->caps.flags & VK_VIDEO_ENCODE_H264_CAPABILITY_HRD_COMPLIANCE_BIT_KHR)); |
|
av_log(avctx, AV_LOG_VERBOSE, " pred_weight_table_generated: %i\n", |
|
!!(enc->caps.flags & VK_VIDEO_ENCODE_H264_CAPABILITY_PREDICTION_WEIGHT_TABLE_GENERATED_BIT_KHR)); |
|
av_log(avctx, AV_LOG_VERBOSE, " row_unaligned_slice: %i\n", |
|
!!(enc->caps.flags & VK_VIDEO_ENCODE_H264_CAPABILITY_ROW_UNALIGNED_SLICE_BIT_KHR)); |
|
av_log(avctx, AV_LOG_VERBOSE, " different_slice_type: %i\n", |
|
!!(enc->caps.flags & VK_VIDEO_ENCODE_H264_CAPABILITY_DIFFERENT_SLICE_TYPE_BIT_KHR)); |
|
av_log(avctx, AV_LOG_VERBOSE, " b_frame_in_l0_list: %i\n", |
|
!!(enc->caps.flags & VK_VIDEO_ENCODE_H264_CAPABILITY_B_FRAME_IN_L0_LIST_BIT_KHR)); |
|
av_log(avctx, AV_LOG_VERBOSE, " b_frame_in_l1_list: %i\n", |
|
!!(enc->caps.flags & VK_VIDEO_ENCODE_H264_CAPABILITY_B_FRAME_IN_L1_LIST_BIT_KHR)); |
|
av_log(avctx, AV_LOG_VERBOSE, " per_pict_type_min_max_qp: %i\n", |
|
!!(enc->caps.flags & VK_VIDEO_ENCODE_H264_CAPABILITY_PER_PICTURE_TYPE_MIN_MAX_QP_BIT_KHR)); |
|
av_log(avctx, AV_LOG_VERBOSE, " per_slice_constant_qp: %i\n", |
|
!!(enc->caps.flags & VK_VIDEO_ENCODE_H264_CAPABILITY_PER_SLICE_CONSTANT_QP_BIT_KHR)); |
|
av_log(avctx, AV_LOG_VERBOSE, " generate_prefix_nalu: %i\n", |
|
!!(enc->caps.flags & VK_VIDEO_ENCODE_H264_CAPABILITY_GENERATE_PREFIX_NALU_BIT_KHR)); |
|
|
|
av_log(avctx, AV_LOG_VERBOSE, " Capabilities:\n"); |
|
av_log(avctx, AV_LOG_VERBOSE, " maxLevelIdc: %i\n", |
|
enc->caps.maxLevelIdc); |
|
av_log(avctx, AV_LOG_VERBOSE, " maxSliceCount: %i\n", |
|
enc->caps.maxSliceCount); |
|
av_log(avctx, AV_LOG_VERBOSE, " max(P/B)PictureL0ReferenceCount: %i P's; %i B's\n", |
|
enc->caps.maxPPictureL0ReferenceCount, |
|
enc->caps.maxBPictureL0ReferenceCount); |
|
av_log(avctx, AV_LOG_VERBOSE, " maxL1ReferenceCount: %i\n", |
|
enc->caps.maxL1ReferenceCount); |
|
av_log(avctx, AV_LOG_VERBOSE, " maxTemporalLayerCount: %i\n", |
|
enc->caps.maxTemporalLayerCount); |
|
av_log(avctx, AV_LOG_VERBOSE, " expectDyadicTemporalLayerPattern: %i\n", |
|
enc->caps.expectDyadicTemporalLayerPattern); |
|
av_log(avctx, AV_LOG_VERBOSE, " min/max Qp: [%i, %i]\n", |
|
enc->caps.minQp, enc->caps.maxQp); |
|
av_log(avctx, AV_LOG_VERBOSE, " prefersGopRemainingFrames: %i\n", |
|
enc->caps.prefersGopRemainingFrames); |
|
av_log(avctx, AV_LOG_VERBOSE, " requiresGopRemainingFrames: %i\n", |
|
enc->caps.requiresGopRemainingFrames); |
|
|
|
err = init_enc_options(avctx); |
|
if (err < 0) |
|
return err; |
|
|
|
flags = ctx->codec->flags; |
|
if (!enc->caps.maxPPictureL0ReferenceCount && |
|
!enc->caps.maxBPictureL0ReferenceCount && |
|
!enc->caps.maxL1ReferenceCount) { |
|
/* Intra-only */ |
|
flags |= FF_HW_FLAG_INTRA_ONLY; |
|
ref_l0 = ref_l1 = 0; |
|
} else if (!enc->caps.maxPPictureL0ReferenceCount) { |
|
/* No P-frames? How. */ |
|
base_ctx->p_to_gpb = 1; |
|
ref_l0 = enc->caps.maxBPictureL0ReferenceCount; |
|
ref_l1 = enc->caps.maxL1ReferenceCount; |
|
} else if (!enc->caps.maxBPictureL0ReferenceCount && |
|
!enc->caps.maxL1ReferenceCount) { |
|
/* No B-frames */ |
|
flags &= ~(FF_HW_FLAG_B_PICTURES | FF_HW_FLAG_B_PICTURE_REFERENCES); |
|
ref_l0 = enc->caps.maxPPictureL0ReferenceCount; |
|
ref_l1 = 0; |
|
} else { |
|
/* P and B frames */ |
|
ref_l0 = FFMIN(enc->caps.maxPPictureL0ReferenceCount, |
|
enc->caps.maxBPictureL0ReferenceCount); |
|
ref_l1 = enc->caps.maxL1ReferenceCount; |
|
} |
|
|
|
err = ff_hw_base_init_gop_structure(base_ctx, avctx, ref_l0, ref_l1, |
|
flags, 0); |
|
if (err < 0) |
|
return err; |
|
|
|
base_ctx->output_delay = base_ctx->b_per_p; |
|
base_ctx->decode_delay = base_ctx->max_b_depth; |
|
|
|
/* Prepare SEI */ |
|
if (enc->unit_elems & UNIT_SEI_IDENTIFIER) { |
|
int len; |
|
|
|
memcpy(enc->sei_identifier.uuid_iso_iec_11578, |
|
vulkan_encode_h264_sei_identifier_uuid, |
|
sizeof(enc->sei_identifier.uuid_iso_iec_11578)); |
|
|
|
len = snprintf(NULL, 0, |
|
"%s / Vulkan video %i.%i.%i / %s %i.%i.%i / %s", |
|
LIBAVCODEC_IDENT, |
|
CODEC_VER(ff_vk_enc_h264_desc.ext_props.specVersion), |
|
s->driver_props.driverName, |
|
CODEC_VER(s->props.properties.driverVersion), |
|
s->props.properties.deviceName); |
|
|
|
if (len >= 0) { |
|
enc->sei_identifier_string = av_malloc(len + 1); |
|
if (!enc->sei_identifier_string) |
|
return AVERROR(ENOMEM); |
|
|
|
len = snprintf(enc->sei_identifier_string, len + 1, |
|
"%s / Vulkan video %i.%i.%i / %s %i.%i.%i / %s", |
|
LIBAVCODEC_IDENT, |
|
CODEC_VER(ff_vk_enc_h264_desc.ext_props.specVersion), |
|
s->driver_props.driverName, |
|
CODEC_VER(s->props.properties.driverVersion), |
|
s->props.properties.deviceName); |
|
|
|
enc->sei_identifier.data = enc->sei_identifier_string; |
|
enc->sei_identifier.data_length = len + 1; |
|
} |
|
} |
|
|
|
/* Init CBS */ |
|
err = ff_cbs_init(&enc->cbs, AV_CODEC_ID_H264, avctx); |
|
if (err < 0) |
|
return err; |
|
|
|
/* Create units and session parameters */ |
|
err = init_base_units(avctx); |
|
if (err < 0) |
|
return err; |
|
|
|
/* Write out extradata */ |
|
err = ff_vulkan_write_global_header(avctx, &enc->common); |
|
if (err < 0) |
|
return err; |
|
|
|
return 0; |
|
} |
|
|
|
static av_cold int vulkan_encode_h264_close(AVCodecContext *avctx) |
|
{ |
|
VulkanEncodeH264Context *enc = avctx->priv_data; |
|
ff_vulkan_encode_uninit(&enc->common); |
|
return 0; |
|
} |
|
|
|
#define OFFSET(x) offsetof(VulkanEncodeH264Context, x) |
|
#define FLAGS (AV_OPT_FLAG_VIDEO_PARAM | AV_OPT_FLAG_ENCODING_PARAM) |
|
static const AVOption vulkan_encode_h264_options[] = { |
|
HW_BASE_ENCODE_COMMON_OPTIONS, |
|
VULKAN_ENCODE_COMMON_OPTIONS, |
|
|
|
{ "profile", "Set profile (profile_idc and constraint_set*_flag)", |
|
OFFSET(common.opts.profile), AV_OPT_TYPE_INT, |
|
{ .i64 = AV_PROFILE_UNKNOWN }, AV_PROFILE_UNKNOWN, 0xffff, FLAGS, .unit = "profile" }, |
|
|
|
#define PROFILE(name, value) name, NULL, 0, AV_OPT_TYPE_CONST, \ |
|
{ .i64 = value }, 0, 0, FLAGS, .unit = "profile" |
|
{ PROFILE("constrained_baseline", AV_PROFILE_H264_CONSTRAINED_BASELINE) }, |
|
{ PROFILE("main", AV_PROFILE_H264_MAIN) }, |
|
{ PROFILE("high", AV_PROFILE_H264_HIGH) }, |
|
{ PROFILE("high444p", AV_PROFILE_H264_HIGH_10) }, |
|
#undef PROFILE |
|
|
|
{ "level", "Set level (level_idc)", |
|
OFFSET(common.opts.level), AV_OPT_TYPE_INT, |
|
{ .i64 = AV_LEVEL_UNKNOWN }, AV_LEVEL_UNKNOWN, 0xff, FLAGS, .unit = "level" }, |
|
|
|
#define LEVEL(name, value) name, NULL, 0, AV_OPT_TYPE_CONST, \ |
|
{ .i64 = value }, 0, 0, FLAGS, .unit = "level" |
|
{ LEVEL("1", 10) }, |
|
{ LEVEL("1.1", 11) }, |
|
{ LEVEL("1.2", 12) }, |
|
{ LEVEL("1.3", 13) }, |
|
{ LEVEL("2", 20) }, |
|
{ LEVEL("2.1", 21) }, |
|
{ LEVEL("2.2", 22) }, |
|
{ LEVEL("3", 30) }, |
|
{ LEVEL("3.1", 31) }, |
|
{ LEVEL("3.2", 32) }, |
|
{ LEVEL("4", 40) }, |
|
{ LEVEL("4.1", 41) }, |
|
{ LEVEL("4.2", 42) }, |
|
{ LEVEL("5", 50) }, |
|
{ LEVEL("5.1", 51) }, |
|
{ LEVEL("5.2", 52) }, |
|
{ LEVEL("6", 60) }, |
|
{ LEVEL("6.1", 61) }, |
|
{ LEVEL("6.2", 62) }, |
|
#undef LEVEL |
|
|
|
{ "coder", "Entropy coder type", OFFSET(unit_opts.cabac), AV_OPT_TYPE_INT, { .i64 = 1 }, 0, 1, FLAGS, "coder" }, |
|
{ "cabac", NULL, 0, AV_OPT_TYPE_CONST, { .i64 = 1 }, INT_MIN, INT_MAX, FLAGS, "coder" }, |
|
{ "vlc", NULL, 0, AV_OPT_TYPE_CONST, { .i64 = 0 }, INT_MIN, INT_MAX, FLAGS, "coder" }, |
|
|
|
{ "units", "Set units to include", OFFSET(unit_elems), AV_OPT_TYPE_FLAGS, { .i64 = UNIT_AUD | UNIT_SEI_IDENTIFIER | UNIT_SEI_RECOVERY | UNIT_SEI_TIMING | UNIT_SEI_A53_CC }, 0, INT_MAX, FLAGS, "units" }, |
|
{ "aud", "Include AUD units", 0, AV_OPT_TYPE_CONST, { .i64 = UNIT_AUD }, INT_MIN, INT_MAX, FLAGS, "units" }, |
|
{ "identifier", "Include encoder version identifier", 0, AV_OPT_TYPE_CONST, { .i64 = UNIT_SEI_IDENTIFIER }, INT_MIN, INT_MAX, FLAGS, "units" }, |
|
{ "timing", "Include timing parameters (buffering_period and pic_timing)", 0, AV_OPT_TYPE_CONST, { .i64 = UNIT_SEI_TIMING }, INT_MIN, INT_MAX, FLAGS, "units" }, |
|
{ "recovery", "Include recovery points where appropriate", 0, AV_OPT_TYPE_CONST, { .i64 = UNIT_SEI_RECOVERY }, INT_MIN, INT_MAX, FLAGS, "units" }, |
|
{ "a53_cc", "Include A/53 caption data", 0, AV_OPT_TYPE_CONST, { .i64 = UNIT_SEI_A53_CC }, INT_MIN, INT_MAX, FLAGS, "units" }, |
|
|
|
{ NULL }, |
|
}; |
|
|
|
static const FFCodecDefault vulkan_encode_h264_defaults[] = { |
|
{ "b", "0" }, |
|
{ "bf", "2" }, |
|
{ "g", "300" }, |
|
{ "i_qfactor", "1" }, |
|
{ "i_qoffset", "0" }, |
|
{ "b_qfactor", "1" }, |
|
{ "b_qoffset", "0" }, |
|
{ "qmin", "-1" }, |
|
{ "qmax", "-1" }, |
|
{ NULL }, |
|
}; |
|
|
|
static const AVClass vulkan_encode_h264_class = { |
|
.class_name = "h264_vulkan", |
|
.item_name = av_default_item_name, |
|
.option = vulkan_encode_h264_options, |
|
.version = LIBAVUTIL_VERSION_INT, |
|
}; |
|
|
|
const FFCodec ff_h264_vulkan_encoder = { |
|
.p.name = "h264_vulkan", |
|
CODEC_LONG_NAME("H.264/AVC (Vulkan)"), |
|
.p.type = AVMEDIA_TYPE_VIDEO, |
|
.p.id = AV_CODEC_ID_H264, |
|
.priv_data_size = sizeof(VulkanEncodeH264Context), |
|
.init = &vulkan_encode_h264_init, |
|
FF_CODEC_RECEIVE_PACKET_CB(&ff_vulkan_encode_receive_packet), |
|
.close = &vulkan_encode_h264_close, |
|
.p.priv_class = &vulkan_encode_h264_class, |
|
.p.capabilities = AV_CODEC_CAP_DELAY | |
|
AV_CODEC_CAP_HARDWARE | |
|
AV_CODEC_CAP_DR1 | |
|
AV_CODEC_CAP_ENCODER_FLUSH | |
|
AV_CODEC_CAP_ENCODER_REORDERED_OPAQUE, |
|
.caps_internal = FF_CODEC_CAP_INIT_CLEANUP, |
|
.defaults = vulkan_encode_h264_defaults, |
|
.p.pix_fmts = (const enum AVPixelFormat[]) { |
|
AV_PIX_FMT_VULKAN, |
|
AV_PIX_FMT_NONE, |
|
}, |
|
.hw_configs = ff_vulkan_encode_hw_configs, |
|
.p.wrapper_name = "vulkan", |
|
};
|
|
|