1 | /* |
2 | * Copyright 2015 Google Inc. |
3 | * |
4 | * Use of this source code is governed by a BSD-style license that can be |
5 | * found in the LICENSE file. |
6 | */ |
7 | |
8 | #include "src/pdf/SkJpegInfo.h" |
9 | |
10 | #include "include/private/SkTo.h" |
11 | |
12 | #ifndef SK_CODEC_DECODES_JPEG |
13 | |
14 | namespace { |
15 | class JpegSegment { |
16 | public: |
17 | JpegSegment(const void* data, size_t size) |
18 | : fData(static_cast<const char*>(data)) |
19 | , fSize(size) |
20 | , fOffset(0) |
21 | , fLength(0) {} |
22 | bool read() { |
23 | if (!this->readBigendianUint16(&fMarker)) { |
24 | return false; |
25 | } |
26 | if (JpegSegment::StandAloneMarker(fMarker)) { |
27 | fLength = 0; |
28 | fBuffer = nullptr; |
29 | return true; |
30 | } |
31 | if (!this->readBigendianUint16(&fLength) || fLength < 2) { |
32 | return false; |
33 | } |
34 | fLength -= 2; // Length includes itself for some reason. |
35 | if (fOffset + fLength > fSize) { |
36 | return false; // Segment too long. |
37 | } |
38 | fBuffer = &fData[fOffset]; |
39 | fOffset += fLength; |
40 | return true; |
41 | } |
42 | |
43 | bool isSOF() { |
44 | return (fMarker & 0xFFF0) == 0xFFC0 && fMarker != 0xFFC4 && |
45 | fMarker != 0xFFC8 && fMarker != 0xFFCC; |
46 | } |
47 | uint16_t marker() { return fMarker; } |
48 | uint16_t length() { return fLength; } |
49 | const char* data() { return fBuffer; } |
50 | |
51 | static uint16_t GetBigendianUint16(const char* ptr) { |
52 | // "the most significant byte shall come first" |
53 | return (static_cast<uint8_t>(ptr[0]) << 8) | |
54 | static_cast<uint8_t>(ptr[1]); |
55 | } |
56 | |
57 | private: |
58 | const char* const fData; |
59 | const size_t fSize; |
60 | size_t fOffset; |
61 | const char* fBuffer; |
62 | uint16_t fMarker; |
63 | uint16_t fLength; |
64 | |
65 | bool readBigendianUint16(uint16_t* value) { |
66 | if (fOffset + 2 > fSize) { |
67 | return false; |
68 | } |
69 | *value = JpegSegment::GetBigendianUint16(&fData[fOffset]); |
70 | fOffset += 2; |
71 | return true; |
72 | } |
73 | static bool StandAloneMarker(uint16_t marker) { |
74 | // RST[m] markers or SOI, EOI, TEM |
75 | return (marker & 0xFFF8) == 0xFFD0 || marker == 0xFFD8 || |
76 | marker == 0xFFD9 || marker == 0xFF01; |
77 | } |
78 | }; |
79 | } // namespace |
80 | |
81 | bool SkGetJpegInfo(const void* data, size_t len, |
82 | SkISize* size, |
83 | SkEncodedInfo::Color* colorType, |
84 | SkEncodedOrigin* orientation) { |
85 | static const uint16_t kSOI = 0xFFD8; |
86 | static const uint16_t kAPP0 = 0xFFE0; |
87 | JpegSegment segment(data, len); |
88 | if (!segment.read() || segment.marker() != kSOI) { |
89 | return false; // not a JPEG |
90 | } |
91 | if (!segment.read() || segment.marker() != kAPP0) { |
92 | return false; // not an APP0 segment |
93 | } |
94 | static const char kJfif[] = {'J', 'F', 'I', 'F', '\0'}; |
95 | SkASSERT(segment.data()); |
96 | if (SkToSizeT(segment.length()) < sizeof(kJfif) || |
97 | 0 != memcmp(segment.data(), kJfif, sizeof(kJfif))) { |
98 | return false; // Not JFIF JPEG |
99 | } |
100 | do { |
101 | if (!segment.read()) { |
102 | return false; // malformed JPEG |
103 | } |
104 | } while (!segment.isSOF()); |
105 | if (segment.length() < 6) { |
106 | return false; // SOF segment is short |
107 | } |
108 | if (8 != segment.data()[0]) { |
109 | return false; // Only support 8-bit precision |
110 | } |
111 | int numberOfComponents = segment.data()[5]; |
112 | if (numberOfComponents != 1 && numberOfComponents != 3) { |
113 | return false; // Invalid JFIF |
114 | } |
115 | if (size) { |
116 | *size = {JpegSegment::GetBigendianUint16(&segment.data()[3]), |
117 | JpegSegment::GetBigendianUint16(&segment.data()[1])}; |
118 | } |
119 | if (colorType) { |
120 | *colorType = numberOfComponents == 3 ? SkEncodedInfo::kYUV_Color |
121 | : SkEncodedInfo::kGray_Color; |
122 | } |
123 | if (orientation) { |
124 | *orientation = kTopLeft_SkEncodedOrigin; |
125 | } |
126 | return true; |
127 | } |
128 | #endif // SK_CODEC_DECODES_JPEG |
129 | |