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) << " nal_size is 0"; | |
scherkus (not reviewing)
2014/04/24 17:16:35
nit: remove leading space?
acolwell GONE FROM CHROMIUM
2014/04/24 23:52:15
Done.
| |
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) << " nal_size is 0"; | |
scherkus (not reviewing)
2014/04/24 17:16:35
ditto
acolwell GONE FROM CHROMIUM
2014/04/24 23:52:15
Done.
| |
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 std::vector<SubsampleEntry>* subsamples) { | |
78 DCHECK(AVC::IsValidAnnexB(*buffer)); | |
79 | |
80 scoped_ptr<H264Parser> parser(new H264Parser()); | |
81 const uint8* start = &(*buffer)[0]; | |
82 parser->SetStream(start, buffer->size()); | |
83 | |
84 H264NALU nalu; | |
85 if (parser->AdvanceToNextNALU(&nalu) != H264Parser::kOk) | |
86 return false; | |
87 | |
88 std::vector<uint8>::iterator config_insert_point = buffer->begin(); | |
89 std::vector<SubsampleEntry>::iterator subsamples_insert_point = | |
90 subsamples->begin(); | |
91 | |
92 if (nalu.nal_unit_type == H264NALU::kAUD) { | |
93 // Move insert point to just after the AUD. | |
94 config_insert_point += (nalu.data + nalu.size) - start; | |
95 | |
96 if (!subsamples->empty()) | |
97 subsamples_insert_point++; | |
damienv1
2014/04/24 00:03:58
I don't think the logic is correct here and will n
acolwell GONE FROM CHROMIUM
2014/04/24 23:52:15
Left code as is based on our offline discussion. I
| |
98 } | |
99 | |
100 // Clear |parser| and |start| since they aren't needed anymore and | |
101 // will hold stale pointers once the insert happens. | |
102 parser.reset(); | |
103 start = NULL; | |
104 | |
105 std::vector<uint8> param_sets; | |
106 std::vector<SubsampleEntry> config_subsamples; | |
107 RCHECK(AVC::ConvertConfigToAnnexB(avc_config, | |
108 ¶m_sets, | |
109 &config_subsamples)); | |
110 | |
111 if (!subsamples->empty()) { | |
112 subsamples->insert(subsamples_insert_point, | |
113 config_subsamples.begin(), | |
114 config_subsamples.end()); | |
115 } | |
116 | |
117 buffer->insert(config_insert_point, | |
118 param_sets.begin(), param_sets.end()); | |
119 | |
120 DCHECK(AVC::IsValidAnnexB(*buffer)); | |
121 return true; | |
122 } | |
123 | |
124 // static | |
62 bool AVC::ConvertConfigToAnnexB( | 125 bool AVC::ConvertConfigToAnnexB( |
63 const AVCDecoderConfigurationRecord& avc_config, | 126 const AVCDecoderConfigurationRecord& avc_config, |
64 std::vector<uint8>* buffer) { | 127 std::vector<uint8>* buffer, |
128 std::vector<SubsampleEntry>* subsamples) { | |
65 DCHECK(buffer->empty()); | 129 DCHECK(buffer->empty()); |
66 buffer->clear(); | 130 buffer->clear(); |
67 int total_size = 0; | 131 int total_size = 0; |
68 for (size_t i = 0; i < avc_config.sps_list.size(); i++) | 132 for (size_t i = 0; i < avc_config.sps_list.size(); i++) |
69 total_size += avc_config.sps_list[i].size() + kAnnexBStartCodeSize; | 133 total_size += avc_config.sps_list[i].size() + kAnnexBStartCodeSize; |
70 for (size_t i = 0; i < avc_config.pps_list.size(); i++) | 134 for (size_t i = 0; i < avc_config.pps_list.size(); i++) |
71 total_size += avc_config.pps_list[i].size() + kAnnexBStartCodeSize; | 135 total_size += avc_config.pps_list[i].size() + kAnnexBStartCodeSize; |
72 buffer->reserve(total_size); | 136 buffer->reserve(total_size); |
73 | 137 |
74 for (size_t i = 0; i < avc_config.sps_list.size(); i++) { | 138 for (size_t i = 0; i < avc_config.sps_list.size(); i++) { |
75 buffer->insert(buffer->end(), kAnnexBStartCode, | 139 buffer->insert(buffer->end(), kAnnexBStartCode, |
76 kAnnexBStartCode + kAnnexBStartCodeSize); | 140 kAnnexBStartCode + kAnnexBStartCodeSize); |
77 buffer->insert(buffer->end(), avc_config.sps_list[i].begin(), | 141 buffer->insert(buffer->end(), avc_config.sps_list[i].begin(), |
78 avc_config.sps_list[i].end()); | 142 avc_config.sps_list[i].end()); |
143 | |
144 SubsampleEntry entry; | |
145 entry.clear_bytes = kAnnexBStartCodeSize + avc_config.sps_list[i].size(); | |
146 entry.cypher_bytes = 0; | |
147 subsamples->push_back(entry); | |
79 } | 148 } |
80 | 149 |
81 for (size_t i = 0; i < avc_config.pps_list.size(); i++) { | 150 for (size_t i = 0; i < avc_config.pps_list.size(); i++) { |
82 buffer->insert(buffer->end(), kAnnexBStartCode, | 151 buffer->insert(buffer->end(), kAnnexBStartCode, |
83 kAnnexBStartCode + kAnnexBStartCodeSize); | 152 kAnnexBStartCode + kAnnexBStartCodeSize); |
84 buffer->insert(buffer->end(), avc_config.pps_list[i].begin(), | 153 buffer->insert(buffer->end(), avc_config.pps_list[i].begin(), |
85 avc_config.pps_list[i].end()); | 154 avc_config.pps_list[i].end()); |
155 | |
156 SubsampleEntry entry; | |
157 entry.clear_bytes = kAnnexBStartCodeSize + avc_config.pps_list[i].size(); | |
158 entry.cypher_bytes = 0; | |
159 subsamples->push_back(entry); | |
86 } | 160 } |
87 return true; | 161 return true; |
88 } | 162 } |
89 | 163 |
164 // Verifies AnnexB NALU order according to ISO/IEC 14496-10 Section 7.4.1.2.3 | |
165 bool AVC::IsValidAnnexB(const std::vector<uint8>& buffer) { | |
166 DVLOG(1) << __FUNCTION__; | |
167 | |
168 if (buffer.empty()) | |
169 return true; | |
170 | |
171 H264Parser parser; | |
172 parser.SetStream(&buffer[0], buffer.size()); | |
173 | |
174 typedef enum { | |
175 kAUDAllowed, | |
176 kBeforeFirstVCL, // VCL == nal_unit_types 1-5 | |
scherkus (not reviewing)
2014/04/24 17:16:35
nit: two spaces before // comments
acolwell GONE FROM CHROMIUM
2014/04/24 23:52:15
Done.
| |
177 kAfterFirstVCL, | |
178 kEOStreamAllowed, | |
179 kNoMoreDataAllowed, | |
180 } NALUOrderState; | |
181 | |
182 H264NALU nalu; | |
183 NALUOrderState order_state = kAUDAllowed; | |
184 int last_nalu_type = H264NALU::kUnspecified; | |
185 for (bool done = false; !done;) { | |
damienv1
2014/04/24 00:03:58
nit: would personally write a while loop here, but
scherkus (not reviewing)
2014/04/24 17:16:35
+1
acolwell GONE FROM CHROMIUM
2014/04/24 23:52:15
Done.
| |
186 switch (parser.AdvanceToNextNALU(&nalu)) { | |
187 case H264Parser::kOk: | |
188 DVLOG(1) << " nal_unit_type " << nalu.nal_unit_type; | |
scherkus (not reviewing)
2014/04/24 17:16:35
ditto w/ removing leading space?
acolwell GONE FROM CHROMIUM
2014/04/24 23:52:15
Done.
| |
189 | |
190 switch (nalu.nal_unit_type) { | |
191 case H264NALU::kAUD: | |
192 if (order_state > kAUDAllowed) { | |
193 DVLOG(1) << "Unexpected AUD in order_state " << order_state; | |
194 return false; | |
195 } | |
196 | |
197 order_state = kBeforeFirstVCL; | |
198 break; | |
199 | |
200 case H264NALU::kSEIMessage: | |
201 case H264NALU::kReserved14: | |
202 case H264NALU::kReserved15: | |
203 case H264NALU::kReserved16: | |
204 case H264NALU::kReserved17: | |
205 case H264NALU::kReserved18: | |
206 case H264NALU::kPPS: | |
207 case H264NALU::kSPS: | |
208 if (order_state > kBeforeFirstVCL) { | |
209 DVLOG(1) << "Unexpected NALU type " << nalu.nal_unit_type | |
210 << " in order_state " << order_state; | |
211 return false; | |
212 } | |
213 | |
214 order_state = kBeforeFirstVCL; | |
215 break; | |
216 | |
217 case H264NALU::kSPSExt: | |
218 if (last_nalu_type != H264NALU::kSPS) { | |
219 DVLOG(1) << "SPS extension does not follow an SPS."; | |
220 return false; | |
221 } | |
222 break; | |
223 | |
224 case H264NALU::kNonIDRSlice: | |
225 case H264NALU::kSliceDataA: | |
226 case H264NALU::kSliceDataB: | |
227 case H264NALU::kSliceDataC: | |
228 case H264NALU::kIDRSlice: | |
229 if (order_state > kAfterFirstVCL) { | |
230 DVLOG(1) << "Unexpected VCL in order_state " << order_state; | |
231 return false; | |
232 } | |
233 | |
234 order_state = kAfterFirstVCL; | |
235 break; | |
236 | |
237 case H264NALU::kCodedSliceAux: | |
238 if (order_state != kAfterFirstVCL) { | |
239 DVLOG(1) << "Unexpected extension in order_state " << order_state; | |
240 return false; | |
241 } | |
242 break; | |
243 | |
244 case H264NALU::kEOSeq: | |
245 if (order_state != kAfterFirstVCL) { | |
246 DVLOG(1) << "Unexpected EOSeq in order_state " << order_state; | |
247 return false; | |
248 } | |
249 | |
250 order_state = kEOStreamAllowed; | |
251 break; | |
252 | |
253 case H264NALU::kEOStream: | |
254 if (order_state < kAfterFirstVCL) { | |
255 DVLOG(1) << "Unexpected EOStream in order_state " << order_state; | |
256 return false; | |
257 } | |
258 | |
259 order_state = kNoMoreDataAllowed; | |
260 break; | |
261 | |
262 case H264NALU::kFiller: | |
263 case H264NALU::kUnspecified: | |
264 if (!(order_state >= kAfterFirstVCL && | |
265 order_state < kEOStreamAllowed)) { | |
266 DVLOG(1) << "Unexpected NALU type " << nalu.nal_unit_type | |
267 << " in order_state " << order_state; | |
268 return false; | |
269 } | |
270 break; | |
271 | |
272 default: | |
273 DCHECK_GE(nalu.nal_unit_type, 20); | |
274 | |
275 if (nalu.nal_unit_type >= 20 && nalu.nal_unit_type <= 31 && | |
276 order_state != kAfterFirstVCL) { | |
277 DVLOG(1) << "Unexpected NALU type " << nalu.nal_unit_type | |
278 << " in order_state " << order_state; | |
279 return false; | |
280 } | |
281 } | |
282 last_nalu_type = nalu.nal_unit_type; | |
283 break; | |
284 | |
285 case H264Parser::kInvalidStream: | |
286 return false; | |
287 | |
288 case H264Parser::kUnsupportedStream: | |
289 NOTREACHED() << "AdvanceToNextNALU() returned kUnsupportedStream!"; | |
290 return false; | |
291 | |
292 case H264Parser::kEOStream: | |
293 done = true; | |
294 } | |
295 } | |
296 | |
297 return order_state >= kAfterFirstVCL; | |
298 } | |
299 | |
90 } // namespace mp4 | 300 } // namespace mp4 |
91 } // namespace media | 301 } // namespace media |
OLD | NEW |