1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker * Copyright 2015 The Android Open Source Project
3*c8dee2aaSAndroid Build Coastguard Worker *
4*c8dee2aaSAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker */
7*c8dee2aaSAndroid Build Coastguard Worker
8*c8dee2aaSAndroid Build Coastguard Worker #ifndef SkCodecPriv_DEFINED
9*c8dee2aaSAndroid Build Coastguard Worker #define SkCodecPriv_DEFINED
10*c8dee2aaSAndroid Build Coastguard Worker
11*c8dee2aaSAndroid Build Coastguard Worker #include "include/codec/SkEncodedOrigin.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkImageInfo.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkTypes.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/SkColorData.h"
15*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/SkEncodedInfo.h"
16*c8dee2aaSAndroid Build Coastguard Worker #include "src/codec/SkColorPalette.h"
17*c8dee2aaSAndroid Build Coastguard Worker
18*c8dee2aaSAndroid Build Coastguard Worker #include <string_view>
19*c8dee2aaSAndroid Build Coastguard Worker
20*c8dee2aaSAndroid Build Coastguard Worker #ifdef SK_PRINT_CODEC_MESSAGES
21*c8dee2aaSAndroid Build Coastguard Worker #define SkCodecPrintf SkDebugf
22*c8dee2aaSAndroid Build Coastguard Worker #else
23*c8dee2aaSAndroid Build Coastguard Worker #define SkCodecPrintf(...)
24*c8dee2aaSAndroid Build Coastguard Worker #endif
25*c8dee2aaSAndroid Build Coastguard Worker
26*c8dee2aaSAndroid Build Coastguard Worker // Defined in SkCodec.cpp
27*c8dee2aaSAndroid Build Coastguard Worker bool sk_select_xform_format(SkColorType colorType, bool forColorTable,
28*c8dee2aaSAndroid Build Coastguard Worker skcms_PixelFormat* outFormat);
29*c8dee2aaSAndroid Build Coastguard Worker
30*c8dee2aaSAndroid Build Coastguard Worker // FIXME: Consider sharing with dm, nanbench, and tools.
get_scale_from_sample_size(int sampleSize)31*c8dee2aaSAndroid Build Coastguard Worker static inline float get_scale_from_sample_size(int sampleSize) {
32*c8dee2aaSAndroid Build Coastguard Worker return 1.0f / ((float) sampleSize);
33*c8dee2aaSAndroid Build Coastguard Worker }
34*c8dee2aaSAndroid Build Coastguard Worker
is_valid_subset(const SkIRect & subset,const SkISize & imageDims)35*c8dee2aaSAndroid Build Coastguard Worker static inline bool is_valid_subset(const SkIRect& subset, const SkISize& imageDims) {
36*c8dee2aaSAndroid Build Coastguard Worker return SkIRect::MakeSize(imageDims).contains(subset);
37*c8dee2aaSAndroid Build Coastguard Worker }
38*c8dee2aaSAndroid Build Coastguard Worker
39*c8dee2aaSAndroid Build Coastguard Worker /*
40*c8dee2aaSAndroid Build Coastguard Worker * returns a scaled dimension based on the original dimension and the sampleSize
41*c8dee2aaSAndroid Build Coastguard Worker * NOTE: we round down here for scaled dimension to match the behavior of SkImageDecoder
42*c8dee2aaSAndroid Build Coastguard Worker * FIXME: I think we should call this get_sampled_dimension().
43*c8dee2aaSAndroid Build Coastguard Worker */
get_scaled_dimension(int srcDimension,int sampleSize)44*c8dee2aaSAndroid Build Coastguard Worker static inline int get_scaled_dimension(int srcDimension, int sampleSize) {
45*c8dee2aaSAndroid Build Coastguard Worker if (sampleSize > srcDimension) {
46*c8dee2aaSAndroid Build Coastguard Worker return 1;
47*c8dee2aaSAndroid Build Coastguard Worker }
48*c8dee2aaSAndroid Build Coastguard Worker return srcDimension / sampleSize;
49*c8dee2aaSAndroid Build Coastguard Worker }
50*c8dee2aaSAndroid Build Coastguard Worker
51*c8dee2aaSAndroid Build Coastguard Worker /*
52*c8dee2aaSAndroid Build Coastguard Worker * Returns the first coordinate that we will keep during a scaled decode.
53*c8dee2aaSAndroid Build Coastguard Worker * The output can be interpreted as an x-coordinate or a y-coordinate.
54*c8dee2aaSAndroid Build Coastguard Worker *
55*c8dee2aaSAndroid Build Coastguard Worker * This does not need to be called and is not called when sampleFactor == 1.
56*c8dee2aaSAndroid Build Coastguard Worker */
get_start_coord(int sampleFactor)57*c8dee2aaSAndroid Build Coastguard Worker static inline int get_start_coord(int sampleFactor) { return sampleFactor / 2; }
58*c8dee2aaSAndroid Build Coastguard Worker
59*c8dee2aaSAndroid Build Coastguard Worker /*
60*c8dee2aaSAndroid Build Coastguard Worker * Given a coordinate in the original image, this returns the corresponding
61*c8dee2aaSAndroid Build Coastguard Worker * coordinate in the scaled image. This function is meaningless if
62*c8dee2aaSAndroid Build Coastguard Worker * IsCoordNecessary returns false.
63*c8dee2aaSAndroid Build Coastguard Worker * The output can be interpreted as an x-coordinate or a y-coordinate.
64*c8dee2aaSAndroid Build Coastguard Worker *
65*c8dee2aaSAndroid Build Coastguard Worker * This does not need to be called and is not called when sampleFactor == 1.
66*c8dee2aaSAndroid Build Coastguard Worker */
get_dst_coord(int srcCoord,int sampleFactor)67*c8dee2aaSAndroid Build Coastguard Worker static inline int get_dst_coord(int srcCoord, int sampleFactor) { return srcCoord / sampleFactor; }
68*c8dee2aaSAndroid Build Coastguard Worker
69*c8dee2aaSAndroid Build Coastguard Worker /*
70*c8dee2aaSAndroid Build Coastguard Worker * When scaling, we will discard certain y-coordinates (rows) and
71*c8dee2aaSAndroid Build Coastguard Worker * x-coordinates (columns). This function returns true if we should keep the
72*c8dee2aaSAndroid Build Coastguard Worker * coordinate and false otherwise.
73*c8dee2aaSAndroid Build Coastguard Worker * The inputs may be x-coordinates or y-coordinates.
74*c8dee2aaSAndroid Build Coastguard Worker *
75*c8dee2aaSAndroid Build Coastguard Worker * This does not need to be called and is not called when sampleFactor == 1.
76*c8dee2aaSAndroid Build Coastguard Worker */
is_coord_necessary(int srcCoord,int sampleFactor,int scaledDim)77*c8dee2aaSAndroid Build Coastguard Worker static inline bool is_coord_necessary(int srcCoord, int sampleFactor, int scaledDim) {
78*c8dee2aaSAndroid Build Coastguard Worker // Get the first coordinate that we want to keep
79*c8dee2aaSAndroid Build Coastguard Worker int startCoord = get_start_coord(sampleFactor);
80*c8dee2aaSAndroid Build Coastguard Worker
81*c8dee2aaSAndroid Build Coastguard Worker // Return false on edge cases
82*c8dee2aaSAndroid Build Coastguard Worker if (srcCoord < startCoord || get_dst_coord(srcCoord, sampleFactor) >= scaledDim) {
83*c8dee2aaSAndroid Build Coastguard Worker return false;
84*c8dee2aaSAndroid Build Coastguard Worker }
85*c8dee2aaSAndroid Build Coastguard Worker
86*c8dee2aaSAndroid Build Coastguard Worker // Every sampleFactor rows are necessary
87*c8dee2aaSAndroid Build Coastguard Worker return ((srcCoord - startCoord) % sampleFactor) == 0;
88*c8dee2aaSAndroid Build Coastguard Worker }
89*c8dee2aaSAndroid Build Coastguard Worker
valid_alpha(SkAlphaType dstAlpha,bool srcIsOpaque)90*c8dee2aaSAndroid Build Coastguard Worker static inline bool valid_alpha(SkAlphaType dstAlpha, bool srcIsOpaque) {
91*c8dee2aaSAndroid Build Coastguard Worker if (kUnknown_SkAlphaType == dstAlpha) {
92*c8dee2aaSAndroid Build Coastguard Worker return false;
93*c8dee2aaSAndroid Build Coastguard Worker }
94*c8dee2aaSAndroid Build Coastguard Worker
95*c8dee2aaSAndroid Build Coastguard Worker if (srcIsOpaque) {
96*c8dee2aaSAndroid Build Coastguard Worker if (kOpaque_SkAlphaType != dstAlpha) {
97*c8dee2aaSAndroid Build Coastguard Worker SkCodecPrintf("Warning: an opaque image should be decoded as opaque "
98*c8dee2aaSAndroid Build Coastguard Worker "- it is being decoded as non-opaque, which will draw slower\n");
99*c8dee2aaSAndroid Build Coastguard Worker }
100*c8dee2aaSAndroid Build Coastguard Worker return true;
101*c8dee2aaSAndroid Build Coastguard Worker }
102*c8dee2aaSAndroid Build Coastguard Worker
103*c8dee2aaSAndroid Build Coastguard Worker return dstAlpha != kOpaque_SkAlphaType;
104*c8dee2aaSAndroid Build Coastguard Worker }
105*c8dee2aaSAndroid Build Coastguard Worker
106*c8dee2aaSAndroid Build Coastguard Worker /*
107*c8dee2aaSAndroid Build Coastguard Worker * If there is a color table, get a pointer to the colors, otherwise return nullptr
108*c8dee2aaSAndroid Build Coastguard Worker */
get_color_ptr(SkColorPalette * colorTable)109*c8dee2aaSAndroid Build Coastguard Worker static inline const SkPMColor* get_color_ptr(SkColorPalette* colorTable) {
110*c8dee2aaSAndroid Build Coastguard Worker return nullptr != colorTable ? colorTable->readColors() : nullptr;
111*c8dee2aaSAndroid Build Coastguard Worker }
112*c8dee2aaSAndroid Build Coastguard Worker
113*c8dee2aaSAndroid Build Coastguard Worker /*
114*c8dee2aaSAndroid Build Coastguard Worker * Compute row bytes for an image using pixels per byte
115*c8dee2aaSAndroid Build Coastguard Worker */
compute_row_bytes_ppb(int width,uint32_t pixelsPerByte)116*c8dee2aaSAndroid Build Coastguard Worker static inline size_t compute_row_bytes_ppb(int width, uint32_t pixelsPerByte) {
117*c8dee2aaSAndroid Build Coastguard Worker return (width + pixelsPerByte - 1) / pixelsPerByte;
118*c8dee2aaSAndroid Build Coastguard Worker }
119*c8dee2aaSAndroid Build Coastguard Worker
120*c8dee2aaSAndroid Build Coastguard Worker /*
121*c8dee2aaSAndroid Build Coastguard Worker * Compute row bytes for an image using bytes per pixel
122*c8dee2aaSAndroid Build Coastguard Worker */
compute_row_bytes_bpp(int width,uint32_t bytesPerPixel)123*c8dee2aaSAndroid Build Coastguard Worker static inline size_t compute_row_bytes_bpp(int width, uint32_t bytesPerPixel) {
124*c8dee2aaSAndroid Build Coastguard Worker return width * bytesPerPixel;
125*c8dee2aaSAndroid Build Coastguard Worker }
126*c8dee2aaSAndroid Build Coastguard Worker
127*c8dee2aaSAndroid Build Coastguard Worker /*
128*c8dee2aaSAndroid Build Coastguard Worker * Compute row bytes for an image
129*c8dee2aaSAndroid Build Coastguard Worker */
compute_row_bytes(int width,uint32_t bitsPerPixel)130*c8dee2aaSAndroid Build Coastguard Worker static inline size_t compute_row_bytes(int width, uint32_t bitsPerPixel) {
131*c8dee2aaSAndroid Build Coastguard Worker if (bitsPerPixel < 16) {
132*c8dee2aaSAndroid Build Coastguard Worker SkASSERT(0 == 8 % bitsPerPixel);
133*c8dee2aaSAndroid Build Coastguard Worker const uint32_t pixelsPerByte = 8 / bitsPerPixel;
134*c8dee2aaSAndroid Build Coastguard Worker return compute_row_bytes_ppb(width, pixelsPerByte);
135*c8dee2aaSAndroid Build Coastguard Worker } else {
136*c8dee2aaSAndroid Build Coastguard Worker SkASSERT(0 == bitsPerPixel % 8);
137*c8dee2aaSAndroid Build Coastguard Worker const uint32_t bytesPerPixel = bitsPerPixel / 8;
138*c8dee2aaSAndroid Build Coastguard Worker return compute_row_bytes_bpp(width, bytesPerPixel);
139*c8dee2aaSAndroid Build Coastguard Worker }
140*c8dee2aaSAndroid Build Coastguard Worker }
141*c8dee2aaSAndroid Build Coastguard Worker
142*c8dee2aaSAndroid Build Coastguard Worker /*
143*c8dee2aaSAndroid Build Coastguard Worker * Get a byte from a buffer
144*c8dee2aaSAndroid Build Coastguard Worker * This method is unsafe, the caller is responsible for performing a check
145*c8dee2aaSAndroid Build Coastguard Worker */
get_byte(const uint8_t * buffer,uint32_t i)146*c8dee2aaSAndroid Build Coastguard Worker static inline uint8_t get_byte(const uint8_t* buffer, uint32_t i) {
147*c8dee2aaSAndroid Build Coastguard Worker return buffer[i];
148*c8dee2aaSAndroid Build Coastguard Worker }
149*c8dee2aaSAndroid Build Coastguard Worker
150*c8dee2aaSAndroid Build Coastguard Worker /*
151*c8dee2aaSAndroid Build Coastguard Worker * Get a short from a buffer
152*c8dee2aaSAndroid Build Coastguard Worker * This method is unsafe, the caller is responsible for performing a check
153*c8dee2aaSAndroid Build Coastguard Worker */
get_short(const uint8_t * buffer,uint32_t i)154*c8dee2aaSAndroid Build Coastguard Worker static inline uint16_t get_short(const uint8_t* buffer, uint32_t i) {
155*c8dee2aaSAndroid Build Coastguard Worker uint16_t result;
156*c8dee2aaSAndroid Build Coastguard Worker memcpy(&result, &(buffer[i]), 2);
157*c8dee2aaSAndroid Build Coastguard Worker #ifdef SK_CPU_BENDIAN
158*c8dee2aaSAndroid Build Coastguard Worker return SkEndianSwap16(result);
159*c8dee2aaSAndroid Build Coastguard Worker #else
160*c8dee2aaSAndroid Build Coastguard Worker return result;
161*c8dee2aaSAndroid Build Coastguard Worker #endif
162*c8dee2aaSAndroid Build Coastguard Worker }
163*c8dee2aaSAndroid Build Coastguard Worker
164*c8dee2aaSAndroid Build Coastguard Worker /*
165*c8dee2aaSAndroid Build Coastguard Worker * Get an int from a buffer
166*c8dee2aaSAndroid Build Coastguard Worker * This method is unsafe, the caller is responsible for performing a check
167*c8dee2aaSAndroid Build Coastguard Worker */
get_int(const uint8_t * buffer,uint32_t i)168*c8dee2aaSAndroid Build Coastguard Worker static inline uint32_t get_int(const uint8_t* buffer, uint32_t i) {
169*c8dee2aaSAndroid Build Coastguard Worker uint32_t result;
170*c8dee2aaSAndroid Build Coastguard Worker memcpy(&result, &(buffer[i]), 4);
171*c8dee2aaSAndroid Build Coastguard Worker #ifdef SK_CPU_BENDIAN
172*c8dee2aaSAndroid Build Coastguard Worker return SkEndianSwap32(result);
173*c8dee2aaSAndroid Build Coastguard Worker #else
174*c8dee2aaSAndroid Build Coastguard Worker return result;
175*c8dee2aaSAndroid Build Coastguard Worker #endif
176*c8dee2aaSAndroid Build Coastguard Worker }
177*c8dee2aaSAndroid Build Coastguard Worker
178*c8dee2aaSAndroid Build Coastguard Worker /*
179*c8dee2aaSAndroid Build Coastguard Worker * @param data Buffer to read bytes from
180*c8dee2aaSAndroid Build Coastguard Worker * @param isLittleEndian Output parameter
181*c8dee2aaSAndroid Build Coastguard Worker * Indicates if the data is little endian
182*c8dee2aaSAndroid Build Coastguard Worker * Is unaffected on false returns
183*c8dee2aaSAndroid Build Coastguard Worker */
is_valid_endian_marker(const uint8_t * data,bool * isLittleEndian)184*c8dee2aaSAndroid Build Coastguard Worker static inline bool is_valid_endian_marker(const uint8_t* data, bool* isLittleEndian) {
185*c8dee2aaSAndroid Build Coastguard Worker // II indicates Intel (little endian) and MM indicates motorola (big endian).
186*c8dee2aaSAndroid Build Coastguard Worker if (('I' != data[0] || 'I' != data[1]) && ('M' != data[0] || 'M' != data[1])) {
187*c8dee2aaSAndroid Build Coastguard Worker return false;
188*c8dee2aaSAndroid Build Coastguard Worker }
189*c8dee2aaSAndroid Build Coastguard Worker
190*c8dee2aaSAndroid Build Coastguard Worker *isLittleEndian = ('I' == data[0]);
191*c8dee2aaSAndroid Build Coastguard Worker return true;
192*c8dee2aaSAndroid Build Coastguard Worker }
193*c8dee2aaSAndroid Build Coastguard Worker
get_endian_short(const uint8_t * data,bool littleEndian)194*c8dee2aaSAndroid Build Coastguard Worker static inline uint16_t get_endian_short(const uint8_t* data, bool littleEndian) {
195*c8dee2aaSAndroid Build Coastguard Worker if (littleEndian) {
196*c8dee2aaSAndroid Build Coastguard Worker return (data[1] << 8) | (data[0]);
197*c8dee2aaSAndroid Build Coastguard Worker }
198*c8dee2aaSAndroid Build Coastguard Worker
199*c8dee2aaSAndroid Build Coastguard Worker return (data[0] << 8) | (data[1]);
200*c8dee2aaSAndroid Build Coastguard Worker }
201*c8dee2aaSAndroid Build Coastguard Worker
get_endian_int(const uint8_t * data,bool littleEndian)202*c8dee2aaSAndroid Build Coastguard Worker static inline uint32_t get_endian_int(const uint8_t* data, bool littleEndian) {
203*c8dee2aaSAndroid Build Coastguard Worker if (littleEndian) {
204*c8dee2aaSAndroid Build Coastguard Worker return (data[3] << 24) | (data[2] << 16) | (data[1] << 8) | (data[0]);
205*c8dee2aaSAndroid Build Coastguard Worker }
206*c8dee2aaSAndroid Build Coastguard Worker
207*c8dee2aaSAndroid Build Coastguard Worker return (data[0] << 24) | (data[1] << 16) | (data[2] << 8) | (data[3]);
208*c8dee2aaSAndroid Build Coastguard Worker }
209*c8dee2aaSAndroid Build Coastguard Worker
premultiply_argb_as_rgba(U8CPU a,U8CPU r,U8CPU g,U8CPU b)210*c8dee2aaSAndroid Build Coastguard Worker static inline SkPMColor premultiply_argb_as_rgba(U8CPU a, U8CPU r, U8CPU g, U8CPU b) {
211*c8dee2aaSAndroid Build Coastguard Worker if (a != 255) {
212*c8dee2aaSAndroid Build Coastguard Worker r = SkMulDiv255Round(r, a);
213*c8dee2aaSAndroid Build Coastguard Worker g = SkMulDiv255Round(g, a);
214*c8dee2aaSAndroid Build Coastguard Worker b = SkMulDiv255Round(b, a);
215*c8dee2aaSAndroid Build Coastguard Worker }
216*c8dee2aaSAndroid Build Coastguard Worker
217*c8dee2aaSAndroid Build Coastguard Worker return SkPackARGB_as_RGBA(a, r, g, b);
218*c8dee2aaSAndroid Build Coastguard Worker }
219*c8dee2aaSAndroid Build Coastguard Worker
premultiply_argb_as_bgra(U8CPU a,U8CPU r,U8CPU g,U8CPU b)220*c8dee2aaSAndroid Build Coastguard Worker static inline SkPMColor premultiply_argb_as_bgra(U8CPU a, U8CPU r, U8CPU g, U8CPU b) {
221*c8dee2aaSAndroid Build Coastguard Worker if (a != 255) {
222*c8dee2aaSAndroid Build Coastguard Worker r = SkMulDiv255Round(r, a);
223*c8dee2aaSAndroid Build Coastguard Worker g = SkMulDiv255Round(g, a);
224*c8dee2aaSAndroid Build Coastguard Worker b = SkMulDiv255Round(b, a);
225*c8dee2aaSAndroid Build Coastguard Worker }
226*c8dee2aaSAndroid Build Coastguard Worker
227*c8dee2aaSAndroid Build Coastguard Worker return SkPackARGB_as_BGRA(a, r, g, b);
228*c8dee2aaSAndroid Build Coastguard Worker }
229*c8dee2aaSAndroid Build Coastguard Worker
is_rgba(SkColorType colorType)230*c8dee2aaSAndroid Build Coastguard Worker static inline bool is_rgba(SkColorType colorType) {
231*c8dee2aaSAndroid Build Coastguard Worker #ifdef SK_PMCOLOR_IS_RGBA
232*c8dee2aaSAndroid Build Coastguard Worker return (kBGRA_8888_SkColorType != colorType);
233*c8dee2aaSAndroid Build Coastguard Worker #else
234*c8dee2aaSAndroid Build Coastguard Worker return (kRGBA_8888_SkColorType == colorType);
235*c8dee2aaSAndroid Build Coastguard Worker #endif
236*c8dee2aaSAndroid Build Coastguard Worker }
237*c8dee2aaSAndroid Build Coastguard Worker
238*c8dee2aaSAndroid Build Coastguard Worker // Method for coverting to a 32 bit pixel.
239*c8dee2aaSAndroid Build Coastguard Worker typedef uint32_t (*PackColorProc)(U8CPU a, U8CPU r, U8CPU g, U8CPU b);
240*c8dee2aaSAndroid Build Coastguard Worker
choose_pack_color_proc(bool isPremul,SkColorType colorType)241*c8dee2aaSAndroid Build Coastguard Worker static inline PackColorProc choose_pack_color_proc(bool isPremul, SkColorType colorType) {
242*c8dee2aaSAndroid Build Coastguard Worker bool isRGBA = is_rgba(colorType);
243*c8dee2aaSAndroid Build Coastguard Worker if (isPremul) {
244*c8dee2aaSAndroid Build Coastguard Worker if (isRGBA) {
245*c8dee2aaSAndroid Build Coastguard Worker return &premultiply_argb_as_rgba;
246*c8dee2aaSAndroid Build Coastguard Worker } else {
247*c8dee2aaSAndroid Build Coastguard Worker return &premultiply_argb_as_bgra;
248*c8dee2aaSAndroid Build Coastguard Worker }
249*c8dee2aaSAndroid Build Coastguard Worker } else {
250*c8dee2aaSAndroid Build Coastguard Worker if (isRGBA) {
251*c8dee2aaSAndroid Build Coastguard Worker return &SkPackARGB_as_RGBA;
252*c8dee2aaSAndroid Build Coastguard Worker } else {
253*c8dee2aaSAndroid Build Coastguard Worker return &SkPackARGB_as_BGRA;
254*c8dee2aaSAndroid Build Coastguard Worker }
255*c8dee2aaSAndroid Build Coastguard Worker }
256*c8dee2aaSAndroid Build Coastguard Worker }
257*c8dee2aaSAndroid Build Coastguard Worker
258*c8dee2aaSAndroid Build Coastguard Worker namespace SkCodecs {
259*c8dee2aaSAndroid Build Coastguard Worker bool HasDecoder(std::string_view id);
260*c8dee2aaSAndroid Build Coastguard Worker }
261*c8dee2aaSAndroid Build Coastguard Worker
262*c8dee2aaSAndroid Build Coastguard Worker #endif // SkCodecPriv_DEFINED
263