Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(593)

Side by Side Diff: media/formats/mp4/avc.cc

Issue 246853005: Fix SPS/PPS insertion logic in MP4StreamParser. (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: Address CR comments. Created 6 years, 7 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
« no previous file with comments | « media/formats/mp4/avc.h ('k') | media/formats/mp4/avc_unittest.cc » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
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/base/decrypt_config.h"
12 #include "media/filters/h264_parser.h"
10 #include "media/formats/mp4/box_definitions.h" 13 #include "media/formats/mp4/box_definitions.h"
11 #include "media/formats/mp4/box_reader.h" 14 #include "media/formats/mp4/box_reader.h"
12 15
13 namespace media { 16 namespace media {
14 namespace mp4 { 17 namespace mp4 {
15 18
16 static const uint8 kAnnexBStartCode[] = {0, 0, 0, 1}; 19 static const uint8 kAnnexBStartCode[] = {0, 0, 0, 1};
17 static const int kAnnexBStartCodeSize = 4; 20 static const int kAnnexBStartCodeSize = 4;
18 21
19 static bool ConvertAVCToAnnexBInPlaceForLengthSize4(std::vector<uint8>* buf) { 22 static bool ConvertAVCToAnnexBInPlaceForLengthSize4(std::vector<uint8>* buf) {
20 const int kLengthSize = 4; 23 const int kLengthSize = 4;
21 size_t pos = 0; 24 size_t pos = 0;
22 while (pos + kLengthSize < buf->size()) { 25 while (pos + kLengthSize < buf->size()) {
23 int nal_size = (*buf)[pos]; 26 uint32 nal_size = (*buf)[pos];
24 nal_size = (nal_size << 8) + (*buf)[pos+1]; 27 nal_size = (nal_size << 8) + (*buf)[pos+1];
25 nal_size = (nal_size << 8) + (*buf)[pos+2]; 28 nal_size = (nal_size << 8) + (*buf)[pos+2];
26 nal_size = (nal_size << 8) + (*buf)[pos+3]; 29 nal_size = (nal_size << 8) + (*buf)[pos+3];
30
31 if (nal_size == 0) {
32 DVLOG(1) << "nal_size is 0";
33 return false;
34 }
35
27 std::copy(kAnnexBStartCode, kAnnexBStartCode + kAnnexBStartCodeSize, 36 std::copy(kAnnexBStartCode, kAnnexBStartCode + kAnnexBStartCodeSize,
28 buf->begin() + pos); 37 buf->begin() + pos);
29 pos += kLengthSize + nal_size; 38 pos += kLengthSize + nal_size;
30 } 39 }
31 return pos == buf->size(); 40 return pos == buf->size();
32 } 41 }
33 42
34 // static 43 // static
35 bool AVC::ConvertFrameToAnnexB(int length_size, std::vector<uint8>* buffer) { 44 bool AVC::ConvertFrameToAnnexB(int length_size, std::vector<uint8>* buffer) {
36 RCHECK(length_size == 1 || length_size == 2 || length_size == 4); 45 RCHECK(length_size == 1 || length_size == 2 || length_size == 4);
37 46
38 if (length_size == 4) 47 if (length_size == 4)
39 return ConvertAVCToAnnexBInPlaceForLengthSize4(buffer); 48 return ConvertAVCToAnnexBInPlaceForLengthSize4(buffer);
40 49
41 std::vector<uint8> temp; 50 std::vector<uint8> temp;
42 temp.swap(*buffer); 51 temp.swap(*buffer);
43 buffer->reserve(temp.size() + 32); 52 buffer->reserve(temp.size() + 32);
44 53
45 size_t pos = 0; 54 size_t pos = 0;
46 while (pos + length_size < temp.size()) { 55 while (pos + length_size < temp.size()) {
47 int nal_size = temp[pos]; 56 int nal_size = temp[pos];
48 if (length_size == 2) nal_size = (nal_size << 8) + temp[pos+1]; 57 if (length_size == 2) nal_size = (nal_size << 8) + temp[pos+1];
49 pos += length_size; 58 pos += length_size;
50 59
60 if (nal_size == 0) {
61 DVLOG(1) << "nal_size is 0";
62 return false;
63 }
64
51 RCHECK(pos + nal_size <= temp.size()); 65 RCHECK(pos + nal_size <= temp.size());
52 buffer->insert(buffer->end(), kAnnexBStartCode, 66 buffer->insert(buffer->end(), kAnnexBStartCode,
53 kAnnexBStartCode + kAnnexBStartCodeSize); 67 kAnnexBStartCode + kAnnexBStartCodeSize);
54 buffer->insert(buffer->end(), temp.begin() + pos, 68 buffer->insert(buffer->end(), temp.begin() + pos,
55 temp.begin() + pos + nal_size); 69 temp.begin() + pos + nal_size);
56 pos += nal_size; 70 pos += nal_size;
57 } 71 }
58 return pos == temp.size(); 72 return pos == temp.size();
59 } 73 }
60 74
61 // static 75 // static
76 bool AVC::InsertParamSetsAnnexB(const AVCDecoderConfigurationRecord& avc_config,
77 std::vector<uint8>* buffer,
78 std::vector<SubsampleEntry>* subsamples) {
79 DCHECK(AVC::IsValidAnnexB(*buffer));
80
81 scoped_ptr<H264Parser> parser(new H264Parser());
82 const uint8* start = &(*buffer)[0];
83 parser->SetStream(start, buffer->size());
84
85 H264NALU nalu;
86 if (parser->AdvanceToNextNALU(&nalu) != H264Parser::kOk)
87 return false;
88
89 std::vector<uint8>::iterator config_insert_point = buffer->begin();
90 std::vector<SubsampleEntry>::iterator subsamples_insert_point =
91 subsamples->begin();
92
93 if (nalu.nal_unit_type == H264NALU::kAUD) {
94 // Move insert point to just after the AUD.
95 config_insert_point += (nalu.data + nalu.size) - start;
96
97 if (!subsamples->empty()) {
98 DCHECK_EQ((*subsamples)[0].clear_bytes + (*subsamples)[0].cypher_bytes,
99 config_insert_point - buffer->begin());
damienv1 2014/04/25 00:11:39 I first suggested a DCHECK. But if the stream is m
acolwell GONE FROM CHROMIUM 2014/04/25 00:29:15 Done.
100
101 subsamples_insert_point++;
102 }
103
104 }
105
106 // Clear |parser| and |start| since they aren't needed anymore and
107 // will hold stale pointers once the insert happens.
108 parser.reset();
109 start = NULL;
110
111 std::vector<uint8> param_sets;
112 std::vector<SubsampleEntry> config_subsamples;
113 RCHECK(AVC::ConvertConfigToAnnexB(avc_config,
114 &param_sets,
115 &config_subsamples));
116
117 if (!subsamples->empty()) {
118 subsamples->insert(subsamples_insert_point,
119 config_subsamples.begin(),
120 config_subsamples.end());
121 }
122
123 buffer->insert(config_insert_point,
124 param_sets.begin(), param_sets.end());
125
126 DCHECK(AVC::IsValidAnnexB(*buffer));
127 return true;
128 }
129
130 // static
62 bool AVC::ConvertConfigToAnnexB( 131 bool AVC::ConvertConfigToAnnexB(
63 const AVCDecoderConfigurationRecord& avc_config, 132 const AVCDecoderConfigurationRecord& avc_config,
64 std::vector<uint8>* buffer) { 133 std::vector<uint8>* buffer,
134 std::vector<SubsampleEntry>* subsamples) {
65 DCHECK(buffer->empty()); 135 DCHECK(buffer->empty());
66 buffer->clear(); 136 buffer->clear();
67 int total_size = 0; 137 int total_size = 0;
68 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++)
69 total_size += avc_config.sps_list[i].size() + kAnnexBStartCodeSize; 139 total_size += avc_config.sps_list[i].size() + kAnnexBStartCodeSize;
70 for (size_t i = 0; i < avc_config.pps_list.size(); i++) 140 for (size_t i = 0; i < avc_config.pps_list.size(); i++)
71 total_size += avc_config.pps_list[i].size() + kAnnexBStartCodeSize; 141 total_size += avc_config.pps_list[i].size() + kAnnexBStartCodeSize;
72 buffer->reserve(total_size); 142 buffer->reserve(total_size);
73 143
74 for (size_t i = 0; i < avc_config.sps_list.size(); i++) { 144 for (size_t i = 0; i < avc_config.sps_list.size(); i++) {
75 buffer->insert(buffer->end(), kAnnexBStartCode, 145 buffer->insert(buffer->end(), kAnnexBStartCode,
76 kAnnexBStartCode + kAnnexBStartCodeSize); 146 kAnnexBStartCode + kAnnexBStartCodeSize);
77 buffer->insert(buffer->end(), avc_config.sps_list[i].begin(), 147 buffer->insert(buffer->end(), avc_config.sps_list[i].begin(),
78 avc_config.sps_list[i].end()); 148 avc_config.sps_list[i].end());
149
150 SubsampleEntry entry;
151 entry.clear_bytes = kAnnexBStartCodeSize + avc_config.sps_list[i].size();
152 entry.cypher_bytes = 0;
153 subsamples->push_back(entry);
79 } 154 }
80 155
81 for (size_t i = 0; i < avc_config.pps_list.size(); i++) { 156 for (size_t i = 0; i < avc_config.pps_list.size(); i++) {
82 buffer->insert(buffer->end(), kAnnexBStartCode, 157 buffer->insert(buffer->end(), kAnnexBStartCode,
83 kAnnexBStartCode + kAnnexBStartCodeSize); 158 kAnnexBStartCode + kAnnexBStartCodeSize);
84 buffer->insert(buffer->end(), avc_config.pps_list[i].begin(), 159 buffer->insert(buffer->end(), avc_config.pps_list[i].begin(),
85 avc_config.pps_list[i].end()); 160 avc_config.pps_list[i].end());
161
162 SubsampleEntry entry;
163 entry.clear_bytes = kAnnexBStartCodeSize + avc_config.pps_list[i].size();
164 entry.cypher_bytes = 0;
165 subsamples->push_back(entry);
86 } 166 }
87 return true; 167 return true;
88 } 168 }
89 169
170 // Verifies AnnexB NALU order according to ISO/IEC 14496-10 Section 7.4.1.2.3
171 bool AVC::IsValidAnnexB(const std::vector<uint8>& buffer) {
172 DVLOG(1) << __FUNCTION__;
173
174 if (buffer.empty())
175 return true;
176
177 H264Parser parser;
178 parser.SetStream(&buffer[0], buffer.size());
179
180 typedef enum {
181 kAUDAllowed,
182 kBeforeFirstVCL, // VCL == nal_unit_types 1-5
183 kAfterFirstVCL,
184 kEOStreamAllowed,
185 kNoMoreDataAllowed,
186 } NALUOrderState;
187
188 H264NALU nalu;
189 NALUOrderState order_state = kAUDAllowed;
190 int last_nalu_type = H264NALU::kUnspecified;
191 bool done = false;
192 while (!done) {
193 switch (parser.AdvanceToNextNALU(&nalu)) {
194 case H264Parser::kOk:
195 DVLOG(1) << "nal_unit_type " << nalu.nal_unit_type;
196
197 switch (nalu.nal_unit_type) {
198 case H264NALU::kAUD:
199 if (order_state > kAUDAllowed) {
200 DVLOG(1) << "Unexpected AUD in order_state " << order_state;
201 return false;
202 }
203
204 order_state = kBeforeFirstVCL;
205 break;
206
207 case H264NALU::kSEIMessage:
208 case H264NALU::kReserved14:
209 case H264NALU::kReserved15:
210 case H264NALU::kReserved16:
211 case H264NALU::kReserved17:
212 case H264NALU::kReserved18:
213 case H264NALU::kPPS:
214 case H264NALU::kSPS:
215 if (order_state > kBeforeFirstVCL) {
216 DVLOG(1) << "Unexpected NALU type " << nalu.nal_unit_type
217 << " in order_state " << order_state;
218 return false;
219 }
220
221 order_state = kBeforeFirstVCL;
222 break;
223
224 case H264NALU::kSPSExt:
225 if (last_nalu_type != H264NALU::kSPS) {
226 DVLOG(1) << "SPS extension does not follow an SPS.";
227 return false;
228 }
229 break;
230
231 case H264NALU::kNonIDRSlice:
232 case H264NALU::kSliceDataA:
233 case H264NALU::kSliceDataB:
234 case H264NALU::kSliceDataC:
235 case H264NALU::kIDRSlice:
236 if (order_state > kAfterFirstVCL) {
237 DVLOG(1) << "Unexpected VCL in order_state " << order_state;
238 return false;
239 }
240
241 order_state = kAfterFirstVCL;
242 break;
243
244 case H264NALU::kCodedSliceAux:
245 if (order_state != kAfterFirstVCL) {
246 DVLOG(1) << "Unexpected extension in order_state " << order_state;
247 return false;
248 }
249 break;
250
251 case H264NALU::kEOSeq:
252 if (order_state != kAfterFirstVCL) {
253 DVLOG(1) << "Unexpected EOSeq in order_state " << order_state;
254 return false;
255 }
256
257 order_state = kEOStreamAllowed;
258 break;
259
260 case H264NALU::kEOStream:
261 if (order_state < kAfterFirstVCL) {
262 DVLOG(1) << "Unexpected EOStream in order_state " << order_state;
263 return false;
264 }
265
damienv1 2014/04/25 00:11:39 nit: I would remove the blank line here and in eve
acolwell GONE FROM CHROMIUM 2014/04/25 00:29:15 Done.
266 order_state = kNoMoreDataAllowed;
267 break;
268
269 case H264NALU::kFiller:
270 case H264NALU::kUnspecified:
271 if (!(order_state >= kAfterFirstVCL &&
272 order_state < kEOStreamAllowed)) {
273 DVLOG(1) << "Unexpected NALU type " << nalu.nal_unit_type
274 << " in order_state " << order_state;
275 return false;
276 }
277 break;
278
279 default:
280 DCHECK_GE(nalu.nal_unit_type, 20);
281
282 if (nalu.nal_unit_type >= 20 && nalu.nal_unit_type <= 31 &&
283 order_state != kAfterFirstVCL) {
284 DVLOG(1) << "Unexpected NALU type " << nalu.nal_unit_type
285 << " in order_state " << order_state;
286 return false;
287 }
288 }
289 last_nalu_type = nalu.nal_unit_type;
290 break;
291
292 case H264Parser::kInvalidStream:
293 return false;
294
295 case H264Parser::kUnsupportedStream:
296 NOTREACHED() << "AdvanceToNextNALU() returned kUnsupportedStream!";
297 return false;
298
299 case H264Parser::kEOStream:
300 done = true;
301 }
302 }
303
304 return order_state >= kAfterFirstVCL;
305 }
306
90 } // namespace mp4 307 } // namespace mp4
91 } // namespace media 308 } // namespace media
OLDNEW
« no previous file with comments | « media/formats/mp4/avc.h ('k') | media/formats/mp4/avc_unittest.cc » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698