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