7#include <packager/media/codecs/hevc_decoder_configuration_record.h>
9#include <absl/log/check.h>
10#include <absl/strings/escaping.h>
11#include <absl/strings/str_format.h>
12#include <absl/strings/str_join.h>
14#include <packager/media/base/buffer_reader.h>
15#include <packager/media/base/rcheck.h>
16#include <packager/media/codecs/h265_parser.h>
17#include <packager/utils/bytes_to_string_view.h>
25std::string GeneralProfileSpaceAsString(uint8_t general_profile_space) {
26 switch (general_profile_space) {
36 LOG(WARNING) <<
"Unexpected general_profile_space "
37 << general_profile_space;
42std::string TrimLeadingZeros(
const std::string& str) {
43 DCHECK_GT(str.size(), 0u);
44 for (
size_t i = 0; i < str.size(); ++i) {
45 if (str[i] ==
'0')
continue;
56std::string ReverseBitsAndHexEncode(uint32_t x) {
57 x = ((x & 0x55555555) << 1) | ((x & 0xAAAAAAAA) >> 1);
58 x = ((x & 0x33333333) << 2) | ((x & 0xCCCCCCCC) >> 2);
59 x = ((x & 0x0F0F0F0F) << 4) | ((x & 0xF0F0F0F0) >> 4);
60 const uint8_t bytes[] = {
static_cast<uint8_t
>(x & 0xFF),
61 static_cast<uint8_t
>((x >> 8) & 0xFF),
62 static_cast<uint8_t
>((x >> 16) & 0xFF),
63 static_cast<uint8_t
>((x >> 24) & 0xFF)};
64 return TrimLeadingZeros(absl::BytesToHexString(
70HEVCDecoderConfigurationRecord::HEVCDecoderConfigurationRecord() =
default;
72HEVCDecoderConfigurationRecord::~HEVCDecoderConfigurationRecord() =
default;
74bool HEVCDecoderConfigurationRecord::ParseInternal() {
77 uint8_t profile_indication = 0;
78 uint8_t length_size_minus_one = 0;
79 uint8_t num_of_arrays = 0;
81 RCHECK(reader.Read1(&version_) && version_ == 1 &&
82 reader.Read1(&profile_indication) &&
83 reader.Read4(&general_profile_compatibility_flags_) &&
84 reader.ReadToVector(&general_constraint_indicator_flags_, 6) &&
85 reader.Read1(&general_level_idc_) &&
86 reader.SkipBytes(8) &&
87 reader.Read1(&length_size_minus_one) &&
88 reader.Read1(&num_of_arrays));
90 general_profile_space_ = profile_indication >> 6;
91 RCHECK(general_profile_space_ <= 3u);
92 general_tier_flag_ = ((profile_indication >> 5) & 1) == 1;
93 general_profile_idc_ = profile_indication & 0x1f;
95 RCHECK(reader.Read1(&version_) && version_ == 1 &&
96 reader.SkipBytes(3) &&
97 reader.Read1(&length_size_minus_one) &&
98 reader.Read1(&num_of_arrays));
100 if ((length_size_minus_one & 0x3) == 2) {
101 LOG(ERROR) <<
"Invalid NALU length size.";
106 if (parser_ ==
nullptr) {
107 if (internal_parser_used_ ==
true)
108 parser_ = &internal_parser_;
110 LOG(ERROR) <<
"Internal parser is not used, but parser_ is not set!";
115 for (
int i = 0; i < num_of_arrays; i++) {
116 uint8_t nal_unit_type;
118 RCHECK(reader.Read1(&nal_unit_type));
119 nal_unit_type &= 0x3f;
120 RCHECK(reader.Read2(&num_nalus));
121 for (
int j = 0; j < num_nalus; j++) {
122 uint16_t nalu_length;
123 RCHECK(reader.Read2(&nalu_length));
124 uint64_t nalu_offset = reader.pos();
125 RCHECK(reader.SkipBytes(nalu_length));
128 RCHECK(
nalu.Initialize(Nalu::kH265,
data() + nalu_offset, nalu_length));
129 RCHECK(
nalu.
type() == nal_unit_type);
132 if (
nalu.
type() == Nalu::H265_VPS) {
134 RCHECK(parser_->
ParseVps(
nalu, &vps_id) == H265Parser::kOk);
135 }
else if (
nalu.
type() == Nalu::H265_SPS) {
137 RCHECK(parser_->
ParseSps(
nalu, &sps_id) == H265Parser::kOk);
138 const H265Sps* sps = parser_->
GetSps(sps_id);
141 sps->vui_parameters.transfer_characteristics);
146 const int* general_profile_tier_level_data =
147 sps->general_profile_tier_level_data;
148 general_profile_space_ =
149 (general_profile_tier_level_data[0] & 0xFF) >> 6;
150 RCHECK(general_profile_space_ <= 3u);
152 ((general_profile_tier_level_data[0] & 0x3F) >> 5) == 1;
153 general_profile_idc_ = general_profile_tier_level_data[0] & 0x1F;
154 general_profile_compatibility_flags_ =
155 ((general_profile_tier_level_data[1] & 0xFF) << 24) |
156 ((general_profile_tier_level_data[2] & 0xFF) << 16) |
157 ((general_profile_tier_level_data[3] & 0xFF) << 8) |
158 (general_profile_tier_level_data[4] & 0xFF);
159 general_constraint_indicator_flags_.resize(6);
160 for (
int k = 0; k < 6; ++k) {
161 general_constraint_indicator_flags_[i] =
162 general_profile_tier_level_data[5 + i] & 0xFF;
164 general_level_idc_ = general_profile_tier_level_data[11] & 0xFF;
175 FourCC codec_fourcc)
const {
177 std::vector<std::string> fields;
178 fields.push_back(FourCCToString(codec_fourcc));
179 fields.push_back(GeneralProfileSpaceAsString(general_profile_space_) +
180 absl::StrFormat(
"%d", general_profile_idc_));
182 ReverseBitsAndHexEncode(general_profile_compatibility_flags_));
183 fields.push_back((general_tier_flag_ ?
"H" :
"L") +
184 absl::StrFormat(
"%d", general_level_idc_));
187 std::vector<uint8_t> constraints = general_constraint_indicator_flags_;
188 size_t size = constraints.size();
189 for (; size > 0; --size) {
190 if (constraints[size - 1] != 0)
break;
192 constraints.resize(size);
193 for (uint8_t constraint : constraints)
194 fields.push_back(TrimLeadingZeros(
197 return absl::StrJoin(fields,
".");
200bool HEVCDecoderConfigurationRecord::ParseLHEVCConfig(
201 const std::vector<uint8_t>& data) {
All the methods that are virtual are virtual for mocking.
std::string_view byte_array_to_string_view(const uint8_t *bytes, size_t bytes_size)
Convert byte array to string_view.