2022-03-21 19:21:58 +03:00
|
|
|
package mpeg
|
|
|
|
|
|
|
|
// https://www.itu.int/rec/T-REC-H.265
|
|
|
|
|
|
|
|
import (
|
|
|
|
"github.com/wader/fq/format"
|
|
|
|
"github.com/wader/fq/pkg/decode"
|
2022-07-16 19:39:57 +03:00
|
|
|
"github.com/wader/fq/pkg/interp"
|
2022-03-21 19:21:58 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
func init() {
|
2023-03-29 01:36:55 +03:00
|
|
|
interp.RegisterFormat(
|
2023-05-01 14:19:04 +03:00
|
|
|
format.HEVC_VPS,
|
2023-03-29 01:36:55 +03:00
|
|
|
&decode.Format{
|
|
|
|
Description: "H.265/HEVC Video Parameter Set",
|
|
|
|
DecodeFn: hevcVPSDecode,
|
|
|
|
})
|
2022-03-21 19:21:58 +03:00
|
|
|
}
|
|
|
|
|
2022-07-22 23:48:25 +03:00
|
|
|
const maxVpsLayers = 1000
|
|
|
|
|
2022-03-21 19:21:58 +03:00
|
|
|
// H.265 page 33
|
2023-02-18 23:10:48 +03:00
|
|
|
func hevcVPSDecode(d *decode.D) any {
|
2022-03-21 19:21:58 +03:00
|
|
|
d.FieldU4("vps_video_parameter_set_id")
|
|
|
|
d.FieldBool("vps_base_layer_internal_flag")
|
|
|
|
d.FieldBool("vps_base_layer_available_flag")
|
|
|
|
d.FieldU6("vps_max_layers_minus1")
|
|
|
|
vpsMaxSubLayersMinus1 := d.FieldU3("vps_max_sub_layers_minus1")
|
|
|
|
d.FieldBool("vps_temporal_id_nesting_flag")
|
|
|
|
d.FieldU16("vps_reserved_0xffff_16bits")
|
|
|
|
profileTierLevelDecode(d, true, vpsMaxSubLayersMinus1)
|
|
|
|
vpsSubLayerOrderingInfoPresentFlag := d.FieldBool("vps_sub_layer_ordering_info_present_flag")
|
|
|
|
d.FieldArray("vps_sub_layer_ordering_infos", func(d *decode.D) {
|
|
|
|
i := vpsMaxSubLayersMinus1
|
|
|
|
if vpsSubLayerOrderingInfoPresentFlag {
|
|
|
|
i = 0
|
|
|
|
}
|
|
|
|
for ; i <= vpsMaxSubLayersMinus1; i++ {
|
|
|
|
d.FieldStruct("sps_sub_layer_ordering_info", func(d *decode.D) {
|
2022-09-30 14:58:23 +03:00
|
|
|
d.FieldUintFn("sps_max_dec_pic_buffering_minus1", uEV)
|
|
|
|
d.FieldUintFn("sps_max_num_reorder_pics", uEV)
|
|
|
|
d.FieldUintFn("sps_max_latency_increase_plus1", uEV)
|
2022-03-21 19:21:58 +03:00
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|
|
|
|
vpsMaxLayerID := d.FieldU6("vps_max_layer_id")
|
2022-09-30 14:58:23 +03:00
|
|
|
vpsNumLayerSetsMinus1 := d.FieldUintFn("vps_num_layer_sets_minus1", uEV)
|
2022-07-22 23:48:25 +03:00
|
|
|
if vpsNumLayerSetsMinus1 > maxVpsLayers {
|
|
|
|
d.Errorf("too many vps layers %d > %d", vpsNumLayerSetsMinus1, maxVpsLayers)
|
|
|
|
}
|
2022-03-21 19:21:58 +03:00
|
|
|
d.FieldArray("layer_id_included_sets_flags", func(d *decode.D) {
|
|
|
|
for i := uint64(0); i <= vpsNumLayerSetsMinus1; i++ {
|
|
|
|
d.FieldArray("layer_id_included_sets_flags", func(d *decode.D) {
|
|
|
|
for j := uint64(0); j <= vpsMaxLayerID; j++ {
|
|
|
|
d.FieldBool("layer_id_included_flag_sets_flag")
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|
|
|
|
vpsTimingInfoPresentFlag := d.FieldBool("vps_timing_info_present_flag")
|
|
|
|
if vpsTimingInfoPresentFlag {
|
|
|
|
d.FieldU32("vps_num_units_in_tick")
|
|
|
|
d.FieldU32("vps_time_scale")
|
|
|
|
vpsPocProportionalToTimingFlag := d.FieldBool("vps_poc_proportional_to_timing_flag")
|
|
|
|
if vpsPocProportionalToTimingFlag {
|
2022-09-30 14:58:23 +03:00
|
|
|
d.FieldUintFn("vps_num_ticks_poc_diff_one_minus1", uEV)
|
2022-03-21 19:21:58 +03:00
|
|
|
}
|
|
|
|
vpsHrdParametersPresentFlag := d.FieldBool("vps_hrd_parameters_present_flag")
|
|
|
|
if vpsHrdParametersPresentFlag {
|
|
|
|
hevcHrdParameters(d, true, vpsMaxSubLayersMinus1)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// TODO:
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|