1
1
mirror of https://github.com/wader/fq.git synced 2024-12-25 22:34:14 +03:00
fq/format/mpeg/hevc_sps.go
2022-07-19 18:33:50 +02:00

333 lines
12 KiB
Go

package mpeg
// https://www.itu.int/rec/T-REC-H.265
import (
"github.com/wader/fq/format"
"github.com/wader/fq/pkg/decode"
"github.com/wader/fq/pkg/interp"
)
func init() {
interp.RegisterFormat(decode.Format{
Name: format.HEVC_SPS,
Description: "H.265/HEVC Sequence Parameter Set",
DecodeFn: hevcSPSDecode,
})
}
func profileLayerDecode(d *decode.D, prefix string, profilePresent bool, levelPresent bool, isSublayer bool) {
if profilePresent {
d.FieldU2(prefix + "profile_space")
d.FieldU1(prefix + "tier_flag")
generalProfileIdc := d.FieldU5(prefix + "profile_idc")
var generalProfileCompatibilityFlags [32]bool
d.FieldArray(prefix+"profile_compatibility_flags", func(d *decode.D) {
for j := 0; j < 32; j++ {
generalProfileCompatibilityFlags[j] = d.FieldBool(prefix + "profile_compatibility_flag")
}
})
d.FieldBool(prefix + "progressive_source_flag")
d.FieldBool(prefix + "interlaced_source_flag")
d.FieldBool(prefix + "non_packed_constraint_flag")
d.FieldBool(prefix + "frame_only_constraint_flag")
if generalProfileIdc == 4 || generalProfileCompatibilityFlags[4] ||
generalProfileIdc == 5 || generalProfileCompatibilityFlags[5] ||
generalProfileIdc == 6 || generalProfileCompatibilityFlags[6] ||
generalProfileIdc == 7 || generalProfileCompatibilityFlags[7] ||
generalProfileIdc == 8 || generalProfileCompatibilityFlags[8] ||
generalProfileIdc == 9 || generalProfileCompatibilityFlags[9] ||
generalProfileIdc == 10 || generalProfileCompatibilityFlags[10] {
d.FieldBool(prefix + "max_12bit_constraint_flag")
d.FieldBool(prefix + "max_10bit_constraint_flag")
d.FieldBool(prefix + "max_8bit_constraint_flag")
d.FieldBool(prefix + "max_422chroma_constraint_flag")
d.FieldBool(prefix + "max_420chroma_constraint_flag")
d.FieldBool(prefix + "max_monochrome_constraint_flag")
d.FieldBool(prefix + "intra_constraint_flag")
d.FieldBool(prefix + "one_picture_only_constraint_flag")
d.FieldBool(prefix + "lower_bit_rate_constraint_flag")
if generalProfileIdc == 5 || generalProfileCompatibilityFlags[5] ||
(!isSublayer &&
(generalProfileIdc == 9 || generalProfileCompatibilityFlags[9] ||
generalProfileIdc == 10 || generalProfileCompatibilityFlags[10])) {
d.FieldBool(prefix + "max_14bit_constraint_flag")
d.FieldU33(prefix + "reserved_zero_33bits")
} else {
d.FieldU34(prefix + "reserved_zero_34bits")
}
} else {
d.FieldU43(prefix + "reserved_zero_43bits")
}
if (generalProfileIdc >= 1 && generalProfileIdc <= 5) ||
generalProfileIdc == 9 ||
generalProfileCompatibilityFlags[1] || generalProfileCompatibilityFlags[2] ||
generalProfileCompatibilityFlags[3] || generalProfileCompatibilityFlags[4] ||
generalProfileCompatibilityFlags[5] || generalProfileCompatibilityFlags[9] {
d.FieldBool(prefix + "inbld_flag")
} else {
d.FieldBool(prefix + "reserved_zero_bit")
}
}
if levelPresent {
d.FieldU8(prefix + "level_idc")
}
}
// H.265 page 41
func profileTierLevelDecode(d *decode.D, profilePresentFlag bool, maxNumSubLayersMinus1 uint64) {
profileLayerDecode(d, "general_", profilePresentFlag, true, false)
subLayerProfilePresentFlags := make([]bool, maxNumSubLayersMinus1)
subLayerLevelPresentFlags := make([]bool, maxNumSubLayersMinus1)
d.FieldArray("sub_layer_presents", func(d *decode.D) {
for i := uint64(0); i < maxNumSubLayersMinus1; i++ {
d.FieldStruct("sub_layer_present", func(d *decode.D) {
subLayerProfilePresentFlags[i] = d.FieldBool("sub_layer_profile_present_flag")
subLayerLevelPresentFlags[i] = d.FieldBool("sub_layer_level_present_flag")
})
}
})
if maxNumSubLayersMinus1 > 0 {
for i := maxNumSubLayersMinus1; i < 8; i++ {
d.FieldArray("reserved_zero_2bits", func(d *decode.D) {
d.FieldU33("reserved_zero_2bits")
})
}
}
d.FieldArray("sub_layers", func(d *decode.D) {
for i := uint64(0); i < maxNumSubLayersMinus1; i++ {
d.FieldStruct("sub_layer", func(d *decode.D) {
profileLayerDecode(d, "", subLayerProfilePresentFlags[i], subLayerProfilePresentFlags[i], true)
})
}
})
}
func hevcSubLayerHrdParameters(d *decode.D, subPicHrdParamsPresentFlag bool, cpbCntMinus1 int) {
for i := 0; i <= cpbCntMinus1; i++ {
d.FieldStruct("parameters", func(d *decode.D) {
d.FieldUFn("bit_rate_value_minus1", uEV)
d.FieldUFn("cpb_size_value_minus1", uEV)
if subPicHrdParamsPresentFlag {
d.FieldUFn("cpb_size_du_value_minus1", uEV)
d.FieldUFn("bit_rate_du_value_minus1", uEV)
}
d.FieldBool("cbr_flag")
})
}
}
func hevcHrdParameters(d *decode.D, commonInfPresentFlag bool, maxNumSubLayersMinus1 uint64) {
var nalHrdParametersPresentFlag bool
var vclHrdParametersPresentFlag bool
var subPicHrdParamsPresentFlag bool
if commonInfPresentFlag {
nalHrdParametersPresentFlag = d.FieldBool("nal_hrd_parameters_present_flag")
vclHrdParametersPresentFlag = d.FieldBool("vcl_hrd_parameters_present_flag")
if nalHrdParametersPresentFlag && vclHrdParametersPresentFlag {
subPicHrdParamsPresentFlag = d.FieldBool("sub_pic_hrd_params_present_flag")
if subPicHrdParamsPresentFlag {
d.FieldU8("tick_divisor_minus2")
d.FieldU5("sar_wdu_cpb_removal_delay_increment_length_minus1idth")
d.FieldBool("sub_pic_cpb_params_in_pic_timing_sei_flag")
d.FieldU5("dpb_output_delay_du_length_minus1")
}
d.FieldU8("tick_divisor_minus2")
d.FieldU8("tick_divisor_minus2")
if subPicHrdParamsPresentFlag {
d.FieldU4("cpb_size_du_scale")
}
d.FieldU5("initial_cpb_removal_delay_length_minus1")
d.FieldU5("au_cpb_removal_delay_length_minus1")
d.FieldU5("dpb_output_delay_length_minus1")
}
}
d.FieldArray("sub_layers", func(d *decode.D) {
for i := uint64(0); i < maxNumSubLayersMinus1; i++ {
d.FieldStruct("sub_layer", func(d *decode.D) {
fixedPicRateGeneralFlag := d.FieldBool("fixed_pic_rate_general_flag")
var fixedPicRateWithinCvsFlag bool
if !fixedPicRateGeneralFlag {
fixedPicRateWithinCvsFlag = d.FieldBool("fixed_pic_rate_within_cvs_flag")
}
var lowDelayHrdFlag bool
if fixedPicRateWithinCvsFlag {
d.FieldUFn("elemental_duration_in_tc_minus1", uEV)
} else {
lowDelayHrdFlag = d.FieldBool("low_delay_hrd_flag")
}
var cpbCntMinus1 int
if !lowDelayHrdFlag {
cpbCntMinus1 = int(d.FieldUFn("cpb_cnt_minus1", uEV))
}
if nalHrdParametersPresentFlag {
hevcSubLayerHrdParameters(d, subPicHrdParamsPresentFlag, cpbCntMinus1)
}
if vclHrdParametersPresentFlag {
hevcSubLayerHrdParameters(d, subPicHrdParamsPresentFlag, cpbCntMinus1)
}
})
}
})
}
func hevcVuiParameters(d *decode.D, spsMaxSubLayersMinus1 uint64) {
aspectRatioInfoPresentFlag := d.FieldBool("aspect_ratio_info_present_flag")
if aspectRatioInfoPresentFlag {
aspectRatioIdc := d.FieldU8("aspect_ratio_idc", avcAspectRatioIdcMap)
const extendedSAR = 255
if aspectRatioIdc == extendedSAR {
d.FieldU16("sar_width")
d.FieldU16("sar_height")
}
}
overscanInfoPresentFlag := d.FieldBool("overscan_info_present_flag")
if overscanInfoPresentFlag {
d.FieldBool("overscan_appropriate_flag")
}
videoSignalTypePresentFlag := d.FieldBool("video_signal_type_present_flag")
if videoSignalTypePresentFlag {
d.FieldU3("video_format", avcVideoFormatMap)
d.FieldBool("video_full_range_flag")
colourDescriptionPresentFlag := d.FieldBool("colour_description_present_flag")
if colourDescriptionPresentFlag {
d.FieldU8("colour_primaries", format.ISO_23091_2_ColourPrimariesMap)
d.FieldU8("transfer_characteristics", format.ISO_23091_2_TransferCharacteristicMap)
d.FieldU8("matrix_coefficients", format.ISO_23091_2_MatrixCoefficients)
}
}
chromaLocInfoPresentFlag := d.FieldBool("chroma_loc_info_present_flag")
if chromaLocInfoPresentFlag {
d.FieldUFn("chroma_sample_loc_type_top_field", uEV)
d.FieldUFn("chroma_sample_loc_type_bottom_field", uEV)
}
d.FieldBool("neutral_chroma_indication_flag")
d.FieldBool("field_seq_flag")
d.FieldBool("frame_field_info_present_flag")
defaultDisplayWindowFlag := d.FieldBool("default_display_window_flag")
if defaultDisplayWindowFlag {
d.FieldUFn("def_disp_win_left_offset", uEV)
d.FieldUFn("def_disp_win_right_offset", uEV)
d.FieldUFn("def_disp_win_top_offset", uEV)
d.FieldUFn("def_disp_win_bottom_offset", uEV)
}
vuiTimingInfoPresentFlag := d.FieldBool("vui_timing_info_present_flag")
if vuiTimingInfoPresentFlag {
d.FieldU32("vui_num_units_in_tick")
d.FieldU32("vui_time_scale")
vuiPocProportionalToTimingFlag := d.FieldBool("vui_poc_proportional_to_timing_flag")
if vuiPocProportionalToTimingFlag {
d.FieldUFn("vui_num_ticks_poc_diff_one_minus1", uEV)
}
vuiHrdParametersPresentFlag := d.FieldBool("vui_hrd_parameters_present_flag")
if vuiHrdParametersPresentFlag {
hevcHrdParameters(d, true, spsMaxSubLayersMinus1)
}
}
bitstreamRestrictionFlag := d.FieldBool("bitstream_restriction_flag")
if bitstreamRestrictionFlag {
d.FieldBool("tiles_fixed_structure_flag")
d.FieldBool("motion_vectors_over_pic_boundaries_flag")
d.FieldBool("restricted_ref_pic_lists_flag")
d.FieldUFn("min_spatial_segmentation_idc", uEV)
d.FieldUFn("max_bytes_per_pic_denom", uEV)
d.FieldUFn("max_bits_per_min_cu_denom", uEV)
d.FieldUFn("log2_max_mv_length_horizontal", uEV)
d.FieldUFn("log2_max_mv_length_vertical", uEV)
}
}
// H.265 page 34
func hevcSPSDecode(d *decode.D, _ any) any {
d.FieldU4("sps_video_parameter_set_id")
spsMaxSubLayersMinus1 := d.FieldU3("sps_max_sub_layers_minus1")
d.FieldBool("sps_temporal_id_nesting_flag")
profileTierLevelDecode(d, true, spsMaxSubLayersMinus1)
d.FieldUFn("sps_seq_parameter_set_id", uEV)
chromaFormatIdc := d.FieldUFn("chroma_format_idc", uEV, chromaFormatMap)
if chromaFormatIdc == 3 {
d.FieldBool("separate_colour_plane_flag")
}
d.FieldUFn("pic_width_in_luma_samples", uEV)
d.FieldUFn("pic_height_in_luma_samples", uEV)
conformanceWindowFlag := d.FieldBool("conformance_window_flag")
if conformanceWindowFlag {
d.FieldUFn("conf_win_left_offset", uEV)
d.FieldUFn("conf_win_right_offset", uEV)
d.FieldUFn("conf_win_top_offset", uEV)
d.FieldUFn("conf_win_bottom_offset", uEV)
}
d.FieldUFn("bit_depth_luma_minus8", uEV)
d.FieldUFn("bit_depth_chroma_minus8", uEV)
d.FieldUFn("log2_max_pic_order_cnt_lsb_minus4", uEV)
spsSubLayerOrderingInfoPresentFlag := d.FieldBool("sps_sub_layer_ordering_info_present_flag")
d.FieldArray("sps_sub_layer_ordering_infos", func(d *decode.D) {
i := spsMaxSubLayersMinus1
if spsSubLayerOrderingInfoPresentFlag {
i = 0
}
for ; i <= spsMaxSubLayersMinus1; i++ {
d.FieldStruct("sps_sub_layer_ordering_info", func(d *decode.D) {
d.FieldUFn("sps_max_dec_pic_buffering_minus1", uEV)
d.FieldUFn("sps_max_num_reorder_pics", uEV)
d.FieldUFn("sps_max_latency_increase_plus1", uEV)
})
}
})
d.FieldUFn("log2_min_luma_coding_block_size_minus3", uEV)
d.FieldUFn("log2_diff_max_min_luma_coding_block_size", uEV)
d.FieldUFn("log2_min_luma_transform_block_size_minus2", uEV)
d.FieldUFn("log2_diff_max_min_luma_transform_block_size", uEV)
d.FieldUFn("max_transform_hierarchy_depth_inter", uEV)
d.FieldUFn("max_transform_hierarchy_depth_intra", uEV)
scalingListEnabledFlag := d.FieldBool("scaling_list_enabled_flag")
if scalingListEnabledFlag {
spsScalingListDataPresentFlag := d.FieldBool("sps_scaling_list_data_present_flag")
if spsScalingListDataPresentFlag {
// TODO: scaling_list_data
return nil
}
}
d.FieldBool("amp_enabled_flag")
d.FieldBool("sample_adaptive_offset_enabled_flag")
pcmEnabledFlag := d.FieldBool("pcm_enabled_flag")
if pcmEnabledFlag {
d.FieldU4("pcm_sample_bit_depth_luma_minus1")
d.FieldU4("pcm_sample_bit_depth_chroma_minus1")
d.FieldUFn("log2_min_pcm_luma_coding_block_size_minus3", uEV)
d.FieldUFn("log2_diff_max_min_pcm_luma_coding_block_size", uEV)
d.FieldBool("pcm_loop_filter_disabled_flag")
}
numShortTermRefPicSets := d.FieldUFn("num_short_term_ref_pic_sets", uEV)
if numShortTermRefPicSets > 0 {
// TODO
return nil
}
longTermRefPicsPresentFlag := d.FieldBool("long_term_ref_pics_present_flag")
if longTermRefPicsPresentFlag {
// TODO
return nil
}
d.FieldBool("sps_temporal_mvp_enabled_flag")
d.FieldBool("strong_intra_smoothing_enabled_flag")
vuiParametersPresentFlag := d.FieldBool("vui_parameters_present_flag")
if vuiParametersPresentFlag {
d.FieldStruct("vui_parameters", func(d *decode.D) { hevcVuiParameters(d, spsMaxSubLayersMinus1) })
}
spsExtensionPresentFlag := d.FieldBool("sps_extension_present_flag")
if spsExtensionPresentFlag {
d.FieldU1("sps_range_extension_flag")
d.FieldU1("sps_multilayer_extension_flag")
d.FieldU1("sps_3d_extension_flag")
d.FieldU1("sps_scc_extension_flag")
d.FieldU4("sps_extension_4bits")
}
// TODO
return nil
}