Add ICU message format support
[chromium-blink-merge.git] / media / filters / jpeg_parser.cc
blobbb07bd710aa54f8d6444e4c9af31fd7dad320f59
1 // Copyright 2015 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "media/filters/jpeg_parser.h"
7 #include "base/big_endian.h"
8 #include "base/logging.h"
10 using base::BigEndianReader;
12 #define READ_U8_OR_RETURN_FALSE(out) \
13 do { \
14 uint8_t _out; \
15 if (!reader.ReadU8(&_out)) { \
16 DVLOG(1) \
17 << "Error in stream: unexpected EOS while trying to read " #out; \
18 return false; \
19 } \
20 *(out) = _out; \
21 } while (0)
23 #define READ_U16_OR_RETURN_FALSE(out) \
24 do { \
25 uint16_t _out; \
26 if (!reader.ReadU16(&_out)) { \
27 DVLOG(1) \
28 << "Error in stream: unexpected EOS while trying to read " #out; \
29 return false; \
30 } \
31 *(out) = _out; \
32 } while (0)
34 namespace media {
36 static bool InRange(int value, int a, int b) {
37 return a <= value && value <= b;
40 // Round up |value| to multiple of |mul|. |value| must be non-negative.
41 // |mul| must be positive.
42 static int RoundUp(int value, int mul) {
43 DCHECK_GE(value, 0);
44 DCHECK_GE(mul, 1);
45 return (value + mul - 1) / mul * mul;
48 // |frame_header| is already initialized to 0 in ParseJpegPicture.
49 static bool ParseSOF(const char* buffer,
50 size_t length,
51 JpegFrameHeader* frame_header) {
52 // Spec B.2.2 Frame header syntax
53 DCHECK(buffer);
54 DCHECK(frame_header);
55 BigEndianReader reader(buffer, length);
57 uint8_t precision;
58 READ_U8_OR_RETURN_FALSE(&precision);
59 READ_U16_OR_RETURN_FALSE(&frame_header->visible_height);
60 READ_U16_OR_RETURN_FALSE(&frame_header->visible_width);
61 READ_U8_OR_RETURN_FALSE(&frame_header->num_components);
63 if (precision != 8) {
64 DLOG(ERROR) << "Only support 8-bit precision, not "
65 << static_cast<int>(precision) << " bit for baseline";
66 return false;
68 if (!InRange(frame_header->num_components, 1,
69 arraysize(frame_header->components))) {
70 DLOG(ERROR) << "num_components="
71 << static_cast<int>(frame_header->num_components)
72 << " is not supported";
73 return false;
76 int max_h_factor = 0;
77 int max_v_factor = 0;
78 for (size_t i = 0; i < frame_header->num_components; i++) {
79 JpegComponent& component = frame_header->components[i];
80 READ_U8_OR_RETURN_FALSE(&component.id);
81 if (component.id > frame_header->num_components) {
82 DLOG(ERROR) << "component id (" << static_cast<int>(component.id)
83 << ") should be <= num_components ("
84 << static_cast<int>(frame_header->num_components) << ")";
85 return false;
87 uint8_t hv;
88 READ_U8_OR_RETURN_FALSE(&hv);
89 component.horizontal_sampling_factor = hv / 16;
90 component.vertical_sampling_factor = hv % 16;
91 if (component.horizontal_sampling_factor > max_h_factor)
92 max_h_factor = component.horizontal_sampling_factor;
93 if (component.vertical_sampling_factor > max_v_factor)
94 max_v_factor = component.vertical_sampling_factor;
95 if (!InRange(component.horizontal_sampling_factor, 1, 4)) {
96 DVLOG(1) << "Invalid horizontal sampling factor "
97 << static_cast<int>(component.horizontal_sampling_factor);
98 return false;
100 if (!InRange(component.vertical_sampling_factor, 1, 4)) {
101 DVLOG(1) << "Invalid vertical sampling factor "
102 << static_cast<int>(component.horizontal_sampling_factor);
103 return false;
105 READ_U8_OR_RETURN_FALSE(&component.quantization_table_selector);
108 // The size of data unit is 8*8 and the coded size should be extended
109 // to complete minimum coded unit, MCU. See Spec A.2.
110 frame_header->coded_width =
111 RoundUp(frame_header->visible_width, max_h_factor * 8);
112 frame_header->coded_height =
113 RoundUp(frame_header->visible_height, max_v_factor * 8);
115 return true;
118 // |q_table| is already initialized to 0 in ParseJpegPicture.
119 static bool ParseDQT(const char* buffer,
120 size_t length,
121 JpegQuantizationTable* q_table) {
122 // Spec B.2.4.1 Quantization table-specification syntax
123 DCHECK(buffer);
124 DCHECK(q_table);
125 BigEndianReader reader(buffer, length);
126 while (reader.remaining() > 0) {
127 uint8_t precision_and_table_id;
128 READ_U8_OR_RETURN_FALSE(&precision_and_table_id);
129 uint8_t precision = precision_and_table_id / 16;
130 uint8_t table_id = precision_and_table_id % 16;
131 if (!InRange(precision, 0, 1)) {
132 DVLOG(1) << "Invalid precision " << static_cast<int>(precision);
133 return false;
135 if (precision == 1) { // 1 means 16-bit precision
136 DLOG(ERROR) << "An 8-bit DCT-based process shall not use a 16-bit "
137 << "precision quantization table";
138 return false;
140 if (table_id >= kJpegMaxQuantizationTableNum) {
141 DLOG(ERROR) << "Quantization table id (" << static_cast<int>(table_id)
142 << ") exceeded " << kJpegMaxQuantizationTableNum;
143 return false;
146 if (!reader.ReadBytes(&q_table[table_id].value,
147 sizeof(q_table[table_id].value)))
148 return false;
149 q_table[table_id].valid = true;
151 return true;
154 // |dc_table| and |ac_table| are already initialized to 0 in ParseJpegPicture.
155 static bool ParseDHT(const char* buffer,
156 size_t length,
157 JpegHuffmanTable* dc_table,
158 JpegHuffmanTable* ac_table) {
159 // Spec B.2.4.2 Huffman table-specification syntax
160 DCHECK(buffer);
161 DCHECK(dc_table);
162 DCHECK(ac_table);
163 BigEndianReader reader(buffer, length);
164 while (reader.remaining() > 0) {
165 uint8_t table_class_and_id;
166 READ_U8_OR_RETURN_FALSE(&table_class_and_id);
167 int table_class = table_class_and_id / 16;
168 int table_id = table_class_and_id % 16;
169 if (!InRange(table_class, 0, 1)) {
170 DVLOG(1) << "Invalid table class " << table_class;
171 return false;
173 if (table_id >= 2) {
174 DLOG(ERROR) << "Table id(" << table_id
175 << ") >= 2 is invalid for baseline profile";
176 return false;
179 JpegHuffmanTable* table;
180 if (table_class == 1)
181 table = &ac_table[table_id];
182 else
183 table = &dc_table[table_id];
185 size_t count = 0;
186 if (!reader.ReadBytes(&table->code_length, sizeof(table->code_length)))
187 return false;
188 for (size_t i = 0; i < arraysize(table->code_length); i++)
189 count += table->code_length[i];
191 if (!InRange(count, 0, sizeof(table->code_value))) {
192 DVLOG(1) << "Invalid code count " << count;
193 return false;
195 if (!reader.ReadBytes(&table->code_value, count))
196 return false;
197 table->valid = true;
199 return true;
202 static bool ParseDRI(const char* buffer,
203 size_t length,
204 uint16_t* restart_interval) {
205 // Spec B.2.4.4 Restart interval definition syntax
206 DCHECK(buffer);
207 DCHECK(restart_interval);
208 BigEndianReader reader(buffer, length);
209 return reader.ReadU16(restart_interval) && reader.remaining() == 0;
212 // |scan| is already initialized to 0 in ParseJpegPicture.
213 static bool ParseSOS(const char* buffer,
214 size_t length,
215 const JpegFrameHeader& frame_header,
216 JpegScanHeader* scan) {
217 // Spec B.2.3 Scan header syntax
218 DCHECK(buffer);
219 DCHECK(scan);
220 BigEndianReader reader(buffer, length);
221 READ_U8_OR_RETURN_FALSE(&scan->num_components);
222 if (scan->num_components != frame_header.num_components) {
223 DLOG(ERROR) << "The number of scan components ("
224 << static_cast<int>(scan->num_components)
225 << ") mismatches the number of image components ("
226 << static_cast<int>(frame_header.num_components) << ")";
227 return false;
230 for (int i = 0; i < scan->num_components; i++) {
231 JpegScanHeader::Component* component = &scan->components[i];
232 READ_U8_OR_RETURN_FALSE(&component->component_selector);
233 uint8_t dc_and_ac_selector;
234 READ_U8_OR_RETURN_FALSE(&dc_and_ac_selector);
235 component->dc_selector = dc_and_ac_selector / 16;
236 component->ac_selector = dc_and_ac_selector % 16;
237 if (component->component_selector != frame_header.components[i].id) {
238 DLOG(ERROR) << "component selector mismatches image component id";
239 return false;
241 if (component->dc_selector >= kJpegMaxHuffmanTableNumBaseline) {
242 DLOG(ERROR) << "DC selector (" << static_cast<int>(component->dc_selector)
243 << ") should be 0 or 1 for baseline mode";
244 return false;
246 if (component->ac_selector >= kJpegMaxHuffmanTableNumBaseline) {
247 DLOG(ERROR) << "AC selector (" << static_cast<int>(component->ac_selector)
248 << ") should be 0 or 1 for baseline mode";
249 return false;
253 // Unused fields, only for value checking.
254 uint8_t spectral_selection_start;
255 uint8_t spectral_selection_end;
256 uint8_t point_transform;
257 READ_U8_OR_RETURN_FALSE(&spectral_selection_start);
258 READ_U8_OR_RETURN_FALSE(&spectral_selection_end);
259 READ_U8_OR_RETURN_FALSE(&point_transform);
260 if (spectral_selection_start != 0 || spectral_selection_end != 63) {
261 DLOG(ERROR) << "Spectral selection should be 0,63 for baseline mode";
262 return false;
264 if (point_transform != 0) {
265 DLOG(ERROR) << "Point transform should be 0 for baseline mode";
266 return false;
269 return true;
272 // |result| is already initialized to 0 in ParseJpegPicture.
273 static bool ParseSOI(const char* buffer,
274 size_t length,
275 JpegParseResult* result) {
276 // Spec B.2.1 High-level syntax
277 DCHECK(buffer);
278 DCHECK(result);
279 BigEndianReader reader(buffer, length);
280 uint8_t marker1;
281 uint8_t marker2;
282 bool has_marker_dqt = false;
283 bool has_marker_sos = false;
285 // Once reached SOS, all neccesary data are parsed.
286 while (!has_marker_sos) {
287 READ_U8_OR_RETURN_FALSE(&marker1);
288 if (marker1 != JPEG_MARKER_PREFIX)
289 return false;
291 do {
292 READ_U8_OR_RETURN_FALSE(&marker2);
293 } while (marker2 == JPEG_MARKER_PREFIX); // skip fill bytes
295 uint16_t size;
296 READ_U16_OR_RETURN_FALSE(&size);
297 // The size includes the size field itself.
298 if (size < sizeof(size)) {
299 DLOG(ERROR) << "Ill-formed JPEG. Segment size (" << size
300 << ") is smaller than size field (" << sizeof(size) << ")";
301 return false;
303 size -= sizeof(size);
305 if (reader.remaining() < size) {
306 DLOG(ERROR) << "Ill-formed JPEG. Remaining size (" << reader.remaining()
307 << ") is smaller than header specified (" << size << ")";
308 return false;
311 switch (marker2) {
312 case JPEG_SOF0:
313 if (!ParseSOF(reader.ptr(), size, &result->frame_header)) {
314 DLOG(ERROR) << "ParseSOF failed";
315 return false;
317 break;
318 case JPEG_SOF1:
319 case JPEG_SOF2:
320 case JPEG_SOF3:
321 case JPEG_SOF5:
322 case JPEG_SOF6:
323 case JPEG_SOF7:
324 case JPEG_SOF9:
325 case JPEG_SOF10:
326 case JPEG_SOF11:
327 case JPEG_SOF13:
328 case JPEG_SOF14:
329 case JPEG_SOF15:
330 DLOG(ERROR) << "Only SOF0 (baseline) is supported, but got SOF"
331 << (marker2 - JPEG_SOF0);
332 return false;
333 case JPEG_DQT:
334 if (!ParseDQT(reader.ptr(), size, result->q_table)) {
335 DLOG(ERROR) << "ParseDQT failed";
336 return false;
338 has_marker_dqt = true;
339 break;
340 case JPEG_DHT:
341 if (!ParseDHT(reader.ptr(), size, result->dc_table, result->ac_table)) {
342 DLOG(ERROR) << "ParseDHT failed";
343 return false;
345 break;
346 case JPEG_DRI:
347 if (!ParseDRI(reader.ptr(), size, &result->restart_interval)) {
348 DLOG(ERROR) << "ParseDRI failed";
349 return false;
351 break;
352 case JPEG_SOS:
353 if (!ParseSOS(reader.ptr(), size, result->frame_header,
354 &result->scan)) {
355 DLOG(ERROR) << "ParseSOS failed";
356 return false;
358 has_marker_sos = true;
359 break;
360 default:
361 DVLOG(4) << "unknown marker " << static_cast<int>(marker2);
362 break;
364 reader.Skip(size);
367 if (!has_marker_dqt) {
368 DLOG(ERROR) << "No DQT marker found";
369 return false;
372 // Scan data follows scan header immediately.
373 result->data = reader.ptr();
374 result->data_size = reader.remaining();
376 return true;
379 bool ParseJpegPicture(const uint8_t* buffer,
380 size_t length,
381 JpegParseResult* result) {
382 DCHECK(buffer);
383 DCHECK(result);
384 BigEndianReader reader(reinterpret_cast<const char*>(buffer), length);
385 memset(result, 0, sizeof(JpegParseResult));
387 uint8_t marker1, marker2;
388 READ_U8_OR_RETURN_FALSE(&marker1);
389 READ_U8_OR_RETURN_FALSE(&marker2);
390 if (marker1 != JPEG_MARKER_PREFIX || marker2 != JPEG_SOI) {
391 DLOG(ERROR) << "Not a JPEG";
392 return false;
395 return ParseSOI(reader.ptr(), reader.remaining(), result);
398 } // namespace media