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

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, 8 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
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/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 &param_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
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698