OLD | NEW |
1 // Copyright 2014 The Chromium Authors. All rights reserved. | 1 // Copyright 2014 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "media/formats/mp4/avc.h" | 5 #include "media/formats/mp4/avc.h" |
6 | 6 |
7 #include <algorithm> | 7 #include <algorithm> |
8 #include <vector> | 8 #include <vector> |
9 | 9 |
| 10 #include "base/logging.h" |
| 11 #include "media/filters/h264_parser.h" |
10 #include "media/formats/mp4/box_definitions.h" | 12 #include "media/formats/mp4/box_definitions.h" |
11 #include "media/formats/mp4/box_reader.h" | 13 #include "media/formats/mp4/box_reader.h" |
12 | 14 |
13 namespace media { | 15 namespace media { |
14 namespace mp4 { | 16 namespace mp4 { |
15 | 17 |
16 static const uint8 kAnnexBStartCode[] = {0, 0, 0, 1}; | 18 static const uint8 kAnnexBStartCode[] = {0, 0, 0, 1}; |
17 static const int kAnnexBStartCodeSize = 4; | 19 static const int kAnnexBStartCodeSize = 4; |
18 | 20 |
19 static bool ConvertAVCToAnnexBInPlaceForLengthSize4(std::vector<uint8>* buf) { | 21 static bool ConvertAVCToAnnexBInPlaceForLengthSize4(std::vector<uint8>* buf) { |
20 const int kLengthSize = 4; | 22 const int kLengthSize = 4; |
21 size_t pos = 0; | 23 size_t pos = 0; |
22 while (pos + kLengthSize < buf->size()) { | 24 while (pos + kLengthSize < buf->size()) { |
23 int nal_size = (*buf)[pos]; | 25 uint32 nal_size = (*buf)[pos]; |
24 nal_size = (nal_size << 8) + (*buf)[pos+1]; | 26 nal_size = (nal_size << 8) + (*buf)[pos+1]; |
25 nal_size = (nal_size << 8) + (*buf)[pos+2]; | 27 nal_size = (nal_size << 8) + (*buf)[pos+2]; |
26 nal_size = (nal_size << 8) + (*buf)[pos+3]; | 28 nal_size = (nal_size << 8) + (*buf)[pos+3]; |
| 29 |
| 30 if (nal_size == 0) { |
| 31 DVLOG(1) << __FUNCTION__ << " nal_size is 0"; |
| 32 return false; |
| 33 } |
| 34 |
27 std::copy(kAnnexBStartCode, kAnnexBStartCode + kAnnexBStartCodeSize, | 35 std::copy(kAnnexBStartCode, kAnnexBStartCode + kAnnexBStartCodeSize, |
28 buf->begin() + pos); | 36 buf->begin() + pos); |
29 pos += kLengthSize + nal_size; | 37 pos += kLengthSize + nal_size; |
30 } | 38 } |
31 return pos == buf->size(); | 39 return pos == buf->size(); |
32 } | 40 } |
33 | 41 |
34 // static | 42 // static |
35 bool AVC::ConvertFrameToAnnexB(int length_size, std::vector<uint8>* buffer) { | 43 bool AVC::ConvertFrameToAnnexB(int length_size, std::vector<uint8>* buffer) { |
36 RCHECK(length_size == 1 || length_size == 2 || length_size == 4); | 44 RCHECK(length_size == 1 || length_size == 2 || length_size == 4); |
37 | 45 |
38 if (length_size == 4) | 46 if (length_size == 4) |
39 return ConvertAVCToAnnexBInPlaceForLengthSize4(buffer); | 47 return ConvertAVCToAnnexBInPlaceForLengthSize4(buffer); |
40 | 48 |
41 std::vector<uint8> temp; | 49 std::vector<uint8> temp; |
42 temp.swap(*buffer); | 50 temp.swap(*buffer); |
43 buffer->reserve(temp.size() + 32); | 51 buffer->reserve(temp.size() + 32); |
44 | 52 |
45 size_t pos = 0; | 53 size_t pos = 0; |
46 while (pos + length_size < temp.size()) { | 54 while (pos + length_size < temp.size()) { |
47 int nal_size = temp[pos]; | 55 int nal_size = temp[pos]; |
48 if (length_size == 2) nal_size = (nal_size << 8) + temp[pos+1]; | 56 if (length_size == 2) nal_size = (nal_size << 8) + temp[pos+1]; |
49 pos += length_size; | 57 pos += length_size; |
50 | 58 |
| 59 if (nal_size == 0) { |
| 60 DVLOG(1) << __FUNCTION__ << " nal_size is 0"; |
| 61 return false; |
| 62 } |
| 63 |
51 RCHECK(pos + nal_size <= temp.size()); | 64 RCHECK(pos + nal_size <= temp.size()); |
52 buffer->insert(buffer->end(), kAnnexBStartCode, | 65 buffer->insert(buffer->end(), kAnnexBStartCode, |
53 kAnnexBStartCode + kAnnexBStartCodeSize); | 66 kAnnexBStartCode + kAnnexBStartCodeSize); |
54 buffer->insert(buffer->end(), temp.begin() + pos, | 67 buffer->insert(buffer->end(), temp.begin() + pos, |
55 temp.begin() + pos + nal_size); | 68 temp.begin() + pos + nal_size); |
56 pos += nal_size; | 69 pos += nal_size; |
57 } | 70 } |
58 return pos == temp.size(); | 71 return pos == temp.size(); |
59 } | 72 } |
60 | 73 |
61 // static | 74 // static |
| 75 bool AVC::InsertParamSetsAnnexB(const AVCDecoderConfigurationRecord& avc_config, |
| 76 std::vector<uint8>* buffer) { |
| 77 DCHECK(AVC::IsValidAnnexB(*buffer)); |
| 78 |
| 79 scoped_ptr<H264Parser> parser(new H264Parser()); |
| 80 const uint8* start = &(*buffer)[0]; |
| 81 parser->SetStream(start, buffer->size()); |
| 82 |
| 83 H264NALU nalu; |
| 84 if (parser->AdvanceToNextNALU(&nalu) != H264Parser::kOk) |
| 85 return false; |
| 86 |
| 87 std::vector<uint8>::iterator config_insert_point = buffer->begin(); |
| 88 |
| 89 if (nalu.nal_unit_type == H264NALU::kAUD) { |
| 90 // Move insert point to just after the AUD. |
| 91 config_insert_point += (nalu.data + nalu.size) - start; |
| 92 |
| 93 // Move to the next NALU so the SPS/PPS checks below will catch |
| 94 // existing SPS/PPS that may occur right after the AUD. |
| 95 if (parser->AdvanceToNextNALU(&nalu) != H264Parser::kOk) |
| 96 return false; |
| 97 } |
| 98 |
| 99 if (nalu.nal_unit_type == H264NALU::kSPS || |
| 100 nalu.nal_unit_type == H264NALU::kPPS) { |
| 101 // Param sets are already in |buffer| so just use those instead |
| 102 // since they are likely overrides for the parameters in the |
| 103 // decoder configuration. |
| 104 return true; |
| 105 } |
| 106 |
| 107 // Clear |parser| and |start| since they aren't needed anymore and |
| 108 // will hold stale pointers once the insert happens. |
| 109 parser.reset(); |
| 110 start = NULL; |
| 111 |
| 112 std::vector<uint8> param_sets; |
| 113 RCHECK(AVC::ConvertConfigToAnnexB(avc_config, ¶m_sets)); |
| 114 |
| 115 buffer->insert(config_insert_point, |
| 116 param_sets.begin(), param_sets.end()); |
| 117 |
| 118 DCHECK(AVC::IsValidAnnexB(*buffer)); |
| 119 return true; |
| 120 } |
| 121 |
| 122 // static |
62 bool AVC::ConvertConfigToAnnexB( | 123 bool AVC::ConvertConfigToAnnexB( |
63 const AVCDecoderConfigurationRecord& avc_config, | 124 const AVCDecoderConfigurationRecord& avc_config, |
64 std::vector<uint8>* buffer) { | 125 std::vector<uint8>* buffer) { |
65 DCHECK(buffer->empty()); | 126 DCHECK(buffer->empty()); |
66 buffer->clear(); | 127 buffer->clear(); |
67 int total_size = 0; | 128 int total_size = 0; |
68 for (size_t i = 0; i < avc_config.sps_list.size(); i++) | 129 for (size_t i = 0; i < avc_config.sps_list.size(); i++) |
69 total_size += avc_config.sps_list[i].size() + kAnnexBStartCodeSize; | 130 total_size += avc_config.sps_list[i].size() + kAnnexBStartCodeSize; |
70 for (size_t i = 0; i < avc_config.pps_list.size(); i++) | 131 for (size_t i = 0; i < avc_config.pps_list.size(); i++) |
71 total_size += avc_config.pps_list[i].size() + kAnnexBStartCodeSize; | 132 total_size += avc_config.pps_list[i].size() + kAnnexBStartCodeSize; |
72 buffer->reserve(total_size); | 133 buffer->reserve(total_size); |
73 | 134 |
74 for (size_t i = 0; i < avc_config.sps_list.size(); i++) { | 135 for (size_t i = 0; i < avc_config.sps_list.size(); i++) { |
75 buffer->insert(buffer->end(), kAnnexBStartCode, | 136 buffer->insert(buffer->end(), kAnnexBStartCode, |
76 kAnnexBStartCode + kAnnexBStartCodeSize); | 137 kAnnexBStartCode + kAnnexBStartCodeSize); |
77 buffer->insert(buffer->end(), avc_config.sps_list[i].begin(), | 138 buffer->insert(buffer->end(), avc_config.sps_list[i].begin(), |
78 avc_config.sps_list[i].end()); | 139 avc_config.sps_list[i].end()); |
79 } | 140 } |
80 | 141 |
81 for (size_t i = 0; i < avc_config.pps_list.size(); i++) { | 142 for (size_t i = 0; i < avc_config.pps_list.size(); i++) { |
82 buffer->insert(buffer->end(), kAnnexBStartCode, | 143 buffer->insert(buffer->end(), kAnnexBStartCode, |
83 kAnnexBStartCode + kAnnexBStartCodeSize); | 144 kAnnexBStartCode + kAnnexBStartCodeSize); |
84 buffer->insert(buffer->end(), avc_config.pps_list[i].begin(), | 145 buffer->insert(buffer->end(), avc_config.pps_list[i].begin(), |
85 avc_config.pps_list[i].end()); | 146 avc_config.pps_list[i].end()); |
86 } | 147 } |
87 return true; | 148 return true; |
88 } | 149 } |
89 | 150 |
| 151 // Verifies AnnexB NALU order according to ISO/IEC 14496-10 Section 7.4.1.2.3 |
| 152 bool AVC::IsValidAnnexB(const std::vector<uint8>& buffer) { |
| 153 DVLOG(1) << __FUNCTION__; |
| 154 |
| 155 if (buffer.empty()) |
| 156 return true; |
| 157 |
| 158 H264Parser parser; |
| 159 parser.SetStream(&buffer[0], buffer.size()); |
| 160 |
| 161 typedef enum { |
| 162 kExpectAUD, |
| 163 kExpectSPS, |
| 164 kExpectSPSExt, |
| 165 kExpectPPS, |
| 166 kExpectSEI, |
| 167 kExpectReserved, // nal_unit_types 14-18 |
| 168 kExpectVCL, // nal_unit_types 1-5 |
| 169 kExpectEOStream, |
| 170 kExpectNoMoreData, |
| 171 } NALUOrderStates; |
| 172 |
| 173 |
| 174 H264NALU nalu; |
| 175 NALUOrderStates order_state = kExpectAUD; |
| 176 for (bool done = false; !done;) { |
| 177 switch (parser.AdvanceToNextNALU(&nalu)) { |
| 178 case H264Parser::kOk: |
| 179 DVLOG(1) << " nal_unit_type " << nalu.nal_unit_type; |
| 180 |
| 181 switch (nalu.nal_unit_type) { |
| 182 case H264NALU::kAUD: |
| 183 if (order_state > kExpectAUD) { |
| 184 DVLOG(1) << "Unexpected AUD in order_state " << order_state; |
| 185 return false; |
| 186 } |
| 187 |
| 188 order_state = kExpectSPS; |
| 189 break; |
| 190 |
| 191 case H264NALU::kSPS: |
| 192 if (order_state > kExpectSPSExt) { |
| 193 DVLOG(1) << "Unexpected SPS in order_state " << order_state; |
| 194 return false; |
| 195 } |
| 196 order_state = kExpectSPSExt; |
| 197 break; |
| 198 |
| 199 case H264NALU::kSPSExt: |
| 200 if (order_state != kExpectSPSExt) { |
| 201 DVLOG(1) << "Unexpected SPS extension in order_state " |
| 202 << order_state; |
| 203 return false; |
| 204 order_state = kExpectSPS; |
| 205 } |
| 206 break; |
| 207 |
| 208 case H264NALU::kPPS: |
| 209 if (order_state > kExpectPPS) { |
| 210 DVLOG(1) << "Unexpected PPS in order_state " << order_state; |
| 211 return false; |
| 212 } |
| 213 order_state = kExpectPPS; |
| 214 break; |
| 215 |
| 216 case H264NALU::kSEIMessage: |
| 217 if (order_state > kExpectSEI) { |
| 218 DVLOG(1) << "Unexpected SEI in order_state " << order_state; |
| 219 return false; |
| 220 } |
| 221 |
| 222 if (order_state < kExpectSEI) |
| 223 order_state = kExpectSEI; |
| 224 break; |
| 225 |
| 226 case H264NALU::kReserved14: |
| 227 case H264NALU::kReserved15: |
| 228 case H264NALU::kReserved16: |
| 229 case H264NALU::kReserved17: |
| 230 case H264NALU::kReserved18: |
| 231 if (order_state > kExpectReserved) { |
| 232 DVLOG(1) << "Unexpected reserved NALU in order_state " |
| 233 << order_state; |
| 234 return false; |
| 235 } |
| 236 |
| 237 if (order_state < kExpectReserved) |
| 238 order_state = kExpectReserved; |
| 239 break; |
| 240 |
| 241 case H264NALU::kNonIDRSlice: |
| 242 case H264NALU::kSliceDataA: |
| 243 case H264NALU::kSliceDataB: |
| 244 case H264NALU::kSliceDataC: |
| 245 case H264NALU::kIDRSlice: |
| 246 if (order_state > kExpectVCL) { |
| 247 DVLOG(1) << "Unexpected VCL in order_state " << order_state; |
| 248 return false; |
| 249 } |
| 250 |
| 251 if (order_state < kExpectVCL) |
| 252 order_state = kExpectVCL; |
| 253 break; |
| 254 |
| 255 case H264NALU::kCodedSliceAux: |
| 256 if (order_state != kExpectVCL) { |
| 257 DVLOG(1) << "Unexpected extension in order_state " << order_state; |
| 258 return false; |
| 259 } |
| 260 break; |
| 261 |
| 262 case H264NALU::kEOSeq: |
| 263 if (order_state != kExpectVCL) { |
| 264 DVLOG(1) << "Unexpected EOSeq in order_state " << order_state; |
| 265 return false; |
| 266 } |
| 267 |
| 268 order_state = kExpectEOStream; |
| 269 break; |
| 270 |
| 271 case H264NALU::kEOStream: |
| 272 if (order_state < kExpectVCL) { |
| 273 DVLOG(1) << "Unexpected EOStream in order_state " << order_state; |
| 274 return false; |
| 275 } |
| 276 |
| 277 order_state = kExpectNoMoreData; |
| 278 break; |
| 279 |
| 280 case H264NALU::kFiller: |
| 281 case H264NALU::kUnspecified: |
| 282 if (order_state < kExpectVCL) |
| 283 return false; |
| 284 break; |
| 285 |
| 286 default: |
| 287 DCHECK_GE(nalu.nal_unit_type, 20); |
| 288 |
| 289 if (nalu.nal_unit_type >= 20 && nalu.nal_unit_type <= 31 && |
| 290 order_state != kExpectVCL) { |
| 291 DVLOG(1) << "Unexpected reserved/unspecified unit in order_state " |
| 292 << order_state; |
| 293 return false; |
| 294 } |
| 295 |
| 296 continue; |
| 297 }; |
| 298 break; |
| 299 |
| 300 case H264Parser::kInvalidStream: |
| 301 case H264Parser::kUnsupportedStream: |
| 302 return false; |
| 303 |
| 304 case H264Parser::kEOStream: |
| 305 done = true; |
| 306 } |
| 307 } |
| 308 |
| 309 return order_state >= kExpectVCL; |
| 310 } |
| 311 |
90 } // namespace mp4 | 312 } // namespace mp4 |
91 } // namespace media | 313 } // namespace media |
OLD | NEW |