blob: 2be04ff5a7a966d7b4e1cb6f94a3b6d0c3a68438 [file] [log] [blame]
reed@google.com3443fd82013-11-13 19:09:13 +00001/*
2 * Copyright 2013 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#ifndef SkImageInfo_DEFINED
9#define SkImageInfo_DEFINED
10
msarett23c51102016-05-27 07:39:02 -070011#include "SkColorSpace.h"
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +000012#include "SkMath.h"
halcanaryf622a6c2014-10-24 12:54:53 -070013#include "SkRect.h"
commit-bot@chromium.org32678d92014-01-15 02:38:22 +000014#include "SkSize.h"
reed@google.com3443fd82013-11-13 19:09:13 +000015
commit-bot@chromium.org8b0e8ac2014-01-30 18:58:24 +000016class SkReadBuffer;
halcanaryf622a6c2014-10-24 12:54:53 -070017class SkWriteBuffer;
reed@google.com9230ea22013-12-09 22:01:03 +000018
reed@google.com3443fd82013-11-13 19:09:13 +000019/**
bsalomonafa95e22015-10-12 10:39:46 -070020 * Describes how to interpret the alpha component of a pixel.
reed@google.com3443fd82013-11-13 19:09:13 +000021 */
22enum SkAlphaType {
reed44977482015-02-27 10:23:00 -080023 kUnknown_SkAlphaType,
skia.committer@gmail.com73a5d532013-11-14 07:02:31 +000024
reed@google.com3443fd82013-11-13 19:09:13 +000025 /**
26 * All pixels are stored as opaque. This differs slightly from kIgnore in
27 * that kOpaque has correct "opaque" values stored in the pixels, while
28 * kIgnore may not, but in both cases the caller should treat the pixels
29 * as opaque.
30 */
31 kOpaque_SkAlphaType,
skia.committer@gmail.com73a5d532013-11-14 07:02:31 +000032
reed@google.com3443fd82013-11-13 19:09:13 +000033 /**
34 * All pixels have their alpha premultiplied in their color components.
35 * This is the natural format for the rendering target pixels.
36 */
37 kPremul_SkAlphaType,
skia.committer@gmail.com73a5d532013-11-14 07:02:31 +000038
reed@google.com3443fd82013-11-13 19:09:13 +000039 /**
40 * All pixels have their color components stored without any regard to the
41 * alpha. e.g. this is the default configuration for PNG images.
42 *
43 * This alpha-type is ONLY supported for input images. Rendering cannot
44 * generate this on output.
45 */
46 kUnpremul_SkAlphaType,
skia.committer@gmail.com73a5d532013-11-14 07:02:31 +000047
reed@google.com3443fd82013-11-13 19:09:13 +000048 kLastEnum_SkAlphaType = kUnpremul_SkAlphaType
49};
50
51static inline bool SkAlphaTypeIsOpaque(SkAlphaType at) {
reed44977482015-02-27 10:23:00 -080052 return kOpaque_SkAlphaType == at;
reed@google.com3443fd82013-11-13 19:09:13 +000053}
54
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +000055static inline bool SkAlphaTypeIsValid(unsigned value) {
56 return value <= kLastEnum_SkAlphaType;
57}
58
reed@google.com3443fd82013-11-13 19:09:13 +000059///////////////////////////////////////////////////////////////////////////////
60
61/**
62 * Describes how to interpret the components of a pixel.
reed6c6ddb82014-06-30 12:44:03 -070063 *
64 * kN32_SkColorType is an alias for whichever 32bit ARGB format is the "native"
65 * form for skia's blitters. Use this if you don't have a swizzle preference
66 * for 32bit pixels.
reed@google.com3443fd82013-11-13 19:09:13 +000067 */
68enum SkColorType {
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +000069 kUnknown_SkColorType,
reed@google.com3443fd82013-11-13 19:09:13 +000070 kAlpha_8_SkColorType,
71 kRGB_565_SkColorType,
reed@google.com9230ea22013-12-09 22:01:03 +000072 kARGB_4444_SkColorType,
reed@google.com3443fd82013-11-13 19:09:13 +000073 kRGBA_8888_SkColorType,
74 kBGRA_8888_SkColorType,
reed@google.com9230ea22013-12-09 22:01:03 +000075 kIndex_8_SkColorType,
reed0c9b1a82015-03-17 17:44:06 -070076 kGray_8_SkColorType,
reed3601f282016-02-05 11:18:39 -080077 kRGBA_F16_SkColorType,
reed@google.com3443fd82013-11-13 19:09:13 +000078
reed3601f282016-02-05 11:18:39 -080079 kLastEnum_SkColorType = kRGBA_F16_SkColorType,
reed@google.com3443fd82013-11-13 19:09:13 +000080
81#if SK_PMCOLOR_BYTE_ORDER(B,G,R,A)
commit-bot@chromium.org28fcae22014-04-11 17:15:40 +000082 kN32_SkColorType = kBGRA_8888_SkColorType,
reed@google.com3443fd82013-11-13 19:09:13 +000083#elif SK_PMCOLOR_BYTE_ORDER(R,G,B,A)
commit-bot@chromium.org28fcae22014-04-11 17:15:40 +000084 kN32_SkColorType = kRGBA_8888_SkColorType,
reed@google.com3443fd82013-11-13 19:09:13 +000085#else
reed3b40ee62014-12-21 14:29:04 -080086 #error "SK_*32_SHFIT values must correspond to BGRA or RGBA byte order"
commit-bot@chromium.org28fcae22014-04-11 17:15:40 +000087#endif
reed@google.com3443fd82013-11-13 19:09:13 +000088};
89
reed6d7cd1f2016-04-15 10:03:03 -070090static int SkColorTypeBytesPerPixel(SkColorType ct) {
reed9d453b02016-04-16 12:24:09 -070091 static const uint8_t gSize[] = {
92 0, // Unknown
93 1, // Alpha_8
94 2, // RGB_565
95 2, // ARGB_4444
96 4, // RGBA_8888
97 4, // BGRA_8888
98 1, // kIndex_8
99 1, // kGray_8
100 8, // kRGBA_F16
101 };
102 static_assert(SK_ARRAY_COUNT(gSize) == (size_t)(kLastEnum_SkColorType + 1),
103 "size_mismatch_with_SkColorType_enum");
104
105 SkASSERT((size_t)ct < SK_ARRAY_COUNT(gSize));
106 return gSize[ct];
reed@google.com3443fd82013-11-13 19:09:13 +0000107}
108
reed8c3fd4f2016-04-15 06:59:38 -0700109static int SkColorTypeShiftPerPixel(SkColorType ct) {
reed9d453b02016-04-16 12:24:09 -0700110 static const uint8_t gShift[] = {
111 0, // Unknown
112 0, // Alpha_8
113 1, // RGB_565
114 1, // ARGB_4444
115 2, // RGBA_8888
116 2, // BGRA_8888
117 0, // kIndex_8
118 0, // kGray_8
119 3, // kRGBA_F16
120 };
121 static_assert(SK_ARRAY_COUNT(gShift) == (size_t)(kLastEnum_SkColorType + 1),
122 "size_mismatch_with_SkColorType_enum");
123
124 SkASSERT((size_t)ct < SK_ARRAY_COUNT(gShift));
125 return gShift[ct];
reed8c3fd4f2016-04-15 06:59:38 -0700126}
127
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +0000128static inline size_t SkColorTypeMinRowBytes(SkColorType ct, int width) {
129 return width * SkColorTypeBytesPerPixel(ct);
130}
131
132static inline bool SkColorTypeIsValid(unsigned value) {
133 return value <= kLastEnum_SkColorType;
134}
135
reed92fc2ae2015-05-22 08:06:21 -0700136static inline size_t SkColorTypeComputeOffset(SkColorType ct, int x, int y, size_t rowBytes) {
reed8c3fd4f2016-04-15 06:59:38 -0700137 if (kUnknown_SkColorType == ct) {
138 return 0;
reed92fc2ae2015-05-22 08:06:21 -0700139 }
reed8c3fd4f2016-04-15 06:59:38 -0700140 return y * rowBytes + (x << SkColorTypeShiftPerPixel(ct));
reed92fc2ae2015-05-22 08:06:21 -0700141}
142
reed@google.com3443fd82013-11-13 19:09:13 +0000143///////////////////////////////////////////////////////////////////////////////
144
145/**
scroggo2fd0d142014-07-01 07:08:19 -0700146 * Return true if alphaType is supported by colorType. If there is a canonical
147 * alphaType for this colorType, return it in canonical.
148 */
149bool SkColorTypeValidateAlphaType(SkColorType colorType, SkAlphaType alphaType,
150 SkAlphaType* canonical = NULL);
151
152///////////////////////////////////////////////////////////////////////////////
153
154/**
rileyaabaef862014-09-12 17:45:58 -0700155 * Describes the color space a YUV pixel.
156 */
157enum SkYUVColorSpace {
158 /** Standard JPEG color space. */
159 kJPEG_SkYUVColorSpace,
160 /** SDTV standard Rec. 601 color space. Uses "studio swing" [16, 235] color
161 range. See http://en.wikipedia.org/wiki/Rec._601 for details. */
162 kRec601_SkYUVColorSpace,
rileya13400392015-07-20 15:00:03 -0700163 /** HDTV standard Rec. 709 color space. Uses "studio swing" [16, 235] color
164 range. See http://en.wikipedia.org/wiki/Rec._709 for details. */
165 kRec709_SkYUVColorSpace,
rileyaabaef862014-09-12 17:45:58 -0700166
rileya13400392015-07-20 15:00:03 -0700167 kLastEnum_SkYUVColorSpace = kRec709_SkYUVColorSpace
rileyaabaef862014-09-12 17:45:58 -0700168};
169
170///////////////////////////////////////////////////////////////////////////////
171
reed7c748852014-11-10 08:57:21 -0800172enum SkColorProfileType {
173 kLinear_SkColorProfileType,
174 kSRGB_SkColorProfileType,
175
176 kLastEnum_SkColorProfileType = kSRGB_SkColorProfileType
177};
178
brianosman982eb7f2016-06-06 13:10:58 -0700179enum class SkSourceGammaTreatment {
180 kRespect,
181 kIgnore,
182};
183
rileyaabaef862014-09-12 17:45:58 -0700184/**
reed@google.com3443fd82013-11-13 19:09:13 +0000185 * Describe an image's dimensions and pixel type.
reed7c748852014-11-10 08:57:21 -0800186 * Used for both src images and render-targets (surfaces).
reed@google.com3443fd82013-11-13 19:09:13 +0000187 */
reed41814922015-01-30 14:54:38 -0800188struct SK_API SkImageInfo {
reede5ea5002014-09-03 11:54:58 -0700189public:
190 SkImageInfo()
msarett23c51102016-05-27 07:39:02 -0700191 : fColorSpace(nullptr)
192 , fWidth(0)
reede5ea5002014-09-03 11:54:58 -0700193 , fHeight(0)
194 , fColorType(kUnknown_SkColorType)
reed44977482015-02-27 10:23:00 -0800195 , fAlphaType(kUnknown_SkAlphaType)
reed7c748852014-11-10 08:57:21 -0800196 , fProfileType(kLinear_SkColorProfileType)
reede5ea5002014-09-03 11:54:58 -0700197 {}
reed@google.com3443fd82013-11-13 19:09:13 +0000198
reed7c748852014-11-10 08:57:21 -0800199 static SkImageInfo Make(int width, int height, SkColorType ct, SkAlphaType at,
200 SkColorProfileType pt = kLinear_SkColorProfileType) {
msarett23c51102016-05-27 07:39:02 -0700201 return SkImageInfo(width, height, ct, at, pt, nullptr);
commit-bot@chromium.org32678d92014-01-15 02:38:22 +0000202 }
203
msarett23c51102016-05-27 07:39:02 -0700204 static SkImageInfo Make(int width, int height, SkColorType ct, SkAlphaType at,
205 sk_sp<SkColorSpace> cs);
206
commit-bot@chromium.org32678d92014-01-15 02:38:22 +0000207 /**
208 * Sets colortype to the native ARGB32 type.
209 */
reed7c748852014-11-10 08:57:21 -0800210 static SkImageInfo MakeN32(int width, int height, SkAlphaType at,
211 SkColorProfileType pt = kLinear_SkColorProfileType) {
msarett23c51102016-05-27 07:39:02 -0700212 return SkImageInfo(width, height, kN32_SkColorType, at, pt, nullptr);
commit-bot@chromium.org32678d92014-01-15 02:38:22 +0000213 }
214
215 /**
216 * Sets colortype to the native ARGB32 type, and the alphatype to premul.
217 */
reed7c748852014-11-10 08:57:21 -0800218 static SkImageInfo MakeN32Premul(int width, int height,
219 SkColorProfileType pt = kLinear_SkColorProfileType) {
msarett23c51102016-05-27 07:39:02 -0700220 return SkImageInfo(width, height, kN32_SkColorType, kPremul_SkAlphaType, pt, nullptr);
commit-bot@chromium.org32678d92014-01-15 02:38:22 +0000221 }
skia.committer@gmail.comd77b3ec2014-01-15 07:01:43 +0000222
commit-bot@chromium.org32678d92014-01-15 02:38:22 +0000223 /**
224 * Sets colortype to the native ARGB32 type, and the alphatype to premul.
225 */
reed7c748852014-11-10 08:57:21 -0800226 static SkImageInfo MakeN32Premul(const SkISize& size,
227 SkColorProfileType pt = kLinear_SkColorProfileType) {
228 return MakeN32Premul(size.width(), size.height(), pt);
commit-bot@chromium.org32678d92014-01-15 02:38:22 +0000229 }
skia.committer@gmail.comd77b3ec2014-01-15 07:01:43 +0000230
reedfbce71f2016-06-02 12:40:22 -0700231 /**
232 * Create an ImageInfo marked as SRGB with N32 swizzle.
233 */
234 static SkImageInfo MakeS32(int width, int height, SkAlphaType at);
235
commit-bot@chromium.org32678d92014-01-15 02:38:22 +0000236 static SkImageInfo MakeA8(int width, int height) {
reed7c748852014-11-10 08:57:21 -0800237 return SkImageInfo(width, height, kAlpha_8_SkColorType, kPremul_SkAlphaType,
msarett23c51102016-05-27 07:39:02 -0700238 kLinear_SkColorProfileType, nullptr);
commit-bot@chromium.org32678d92014-01-15 02:38:22 +0000239 }
skia.committer@gmail.comd77b3ec2014-01-15 07:01:43 +0000240
reed@google.com900ecf22014-02-20 20:55:37 +0000241 static SkImageInfo MakeUnknown(int width, int height) {
reed44977482015-02-27 10:23:00 -0800242 return SkImageInfo(width, height, kUnknown_SkColorType, kUnknown_SkAlphaType,
msarett23c51102016-05-27 07:39:02 -0700243 kLinear_SkColorProfileType, nullptr);
reed@google.com900ecf22014-02-20 20:55:37 +0000244 }
245
reedf252f642014-06-14 04:24:56 -0700246 static SkImageInfo MakeUnknown() {
reede5ea5002014-09-03 11:54:58 -0700247 return SkImageInfo();
reedf252f642014-06-14 04:24:56 -0700248 }
249
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +0000250 int width() const { return fWidth; }
251 int height() const { return fHeight; }
252 SkColorType colorType() const { return fColorType; }
253 SkAlphaType alphaType() const { return fAlphaType; }
msarett23c51102016-05-27 07:39:02 -0700254 SkColorSpace* colorSpace() const { return fColorSpace.get(); }
255
256 // Deprecated
reed7c748852014-11-10 08:57:21 -0800257 SkColorProfileType profileType() const { return fProfileType; }
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +0000258
259 bool isEmpty() const { return fWidth <= 0 || fHeight <= 0; }
260
reed@google.com3443fd82013-11-13 19:09:13 +0000261 bool isOpaque() const {
262 return SkAlphaTypeIsOpaque(fAlphaType);
263 }
264
msarett23c51102016-05-27 07:39:02 -0700265 // Deprecated
reed7c748852014-11-10 08:57:21 -0800266 bool isLinear() const { return kLinear_SkColorProfileType == fProfileType; }
267 bool isSRGB() const { return kSRGB_SkColorProfileType == fProfileType; }
268
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +0000269 SkISize dimensions() const { return SkISize::Make(fWidth, fHeight); }
reed7c748852014-11-10 08:57:21 -0800270 SkIRect bounds() const { return SkIRect::MakeWH(fWidth, fHeight); }
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +0000271
reedc77392e2014-06-02 13:07:26 -0700272 /**
273 * Return a new ImageInfo with the same colortype and alphatype as this info,
274 * but with the specified width and height.
275 */
276 SkImageInfo makeWH(int newWidth, int newHeight) const {
msarett23c51102016-05-27 07:39:02 -0700277 return SkImageInfo(newWidth, newHeight, fColorType, fAlphaType, fProfileType, fColorSpace);
reedc77392e2014-06-02 13:07:26 -0700278 }
279
reede5ea5002014-09-03 11:54:58 -0700280 SkImageInfo makeAlphaType(SkAlphaType newAlphaType) const {
msarett23c51102016-05-27 07:39:02 -0700281 return SkImageInfo(fWidth, fHeight, fColorType, newAlphaType, fProfileType, fColorSpace);
reede5ea5002014-09-03 11:54:58 -0700282 }
283
284 SkImageInfo makeColorType(SkColorType newColorType) const {
msarett23c51102016-05-27 07:39:02 -0700285 return SkImageInfo(fWidth, fHeight, newColorType, fAlphaType, fProfileType, fColorSpace);
reede5ea5002014-09-03 11:54:58 -0700286 }
reed7c748852014-11-10 08:57:21 -0800287
msarett804b4612016-06-09 11:03:45 -0700288 SkImageInfo makeColorSpace(sk_sp<SkColorSpace> cs) const {
289 return SkImageInfo::Make(fWidth, fHeight, fColorType, fAlphaType, std::move(cs));
290 }
291
reed8c3fd4f2016-04-15 06:59:38 -0700292 int bytesPerPixel() const { return SkColorTypeBytesPerPixel(fColorType); }
293
294 int shiftPerPixel() const { return SkColorTypeShiftPerPixel(fColorType); }
reed@google.com3443fd82013-11-13 19:09:13 +0000295
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +0000296 uint64_t minRowBytes64() const {
297 return sk_64_mul(fWidth, this->bytesPerPixel());
298 }
299
mike@reedtribe.org169a0ed2014-02-11 02:20:17 +0000300 size_t minRowBytes() const {
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +0000301 return (size_t)this->minRowBytes64();
mike@reedtribe.org169a0ed2014-02-11 02:20:17 +0000302 }
skia.committer@gmail.com8ed64432014-02-11 03:02:13 +0000303
reed92fc2ae2015-05-22 08:06:21 -0700304 size_t computeOffset(int x, int y, size_t rowBytes) const {
305 SkASSERT((unsigned)x < (unsigned)fWidth);
306 SkASSERT((unsigned)y < (unsigned)fHeight);
307 return SkColorTypeComputeOffset(fColorType, x, y, rowBytes);
308 }
309
reed@google.com3443fd82013-11-13 19:09:13 +0000310 bool operator==(const SkImageInfo& other) const {
msarett23c51102016-05-27 07:39:02 -0700311 return fWidth == other.fWidth && fHeight == other.fHeight &&
312 fColorType == other.fColorType && fAlphaType == other.fAlphaType &&
313 fProfileType == other.fProfileType && fColorSpace == other.fColorSpace;
reed@google.com3443fd82013-11-13 19:09:13 +0000314 }
315 bool operator!=(const SkImageInfo& other) const {
msarett23c51102016-05-27 07:39:02 -0700316 return fWidth != other.fWidth || fHeight != other.fHeight ||
317 fColorType != other.fColorType || fAlphaType != other.fAlphaType ||
318 fProfileType != other.fProfileType || fColorSpace != other.fColorSpace;
reed@google.com3443fd82013-11-13 19:09:13 +0000319 }
reed@google.com9230ea22013-12-09 22:01:03 +0000320
commit-bot@chromium.org8b0e8ac2014-01-30 18:58:24 +0000321 void unflatten(SkReadBuffer&);
322 void flatten(SkWriteBuffer&) const;
reed@google.com9230ea22013-12-09 22:01:03 +0000323
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +0000324 int64_t getSafeSize64(size_t rowBytes) const {
reed@google.com9230ea22013-12-09 22:01:03 +0000325 if (0 == fHeight) {
326 return 0;
327 }
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +0000328 return sk_64_mul(fHeight - 1, rowBytes) + fWidth * this->bytesPerPixel();
reed@google.com9230ea22013-12-09 22:01:03 +0000329 }
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +0000330
331 size_t getSafeSize(size_t rowBytes) const {
reedde499882015-06-18 13:41:40 -0700332 int64_t size = this->getSafeSize64(rowBytes);
333 if (!sk_64_isS32(size)) {
334 return 0;
335 }
336 return sk_64_asS32(size);
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +0000337 }
338
339 bool validRowBytes(size_t rowBytes) const {
340 uint64_t rb = sk_64_mul(fWidth, this->bytesPerPixel());
341 return rowBytes >= rb;
342 }
343
msarett23c51102016-05-27 07:39:02 -0700344 void reset() {
345 fWidth = 0;
346 fHeight = 0;
347 fColorType = kUnknown_SkColorType;
348 fAlphaType = kUnknown_SkAlphaType;
349 fProfileType = kLinear_SkColorProfileType;
350 fColorSpace = nullptr;
351 }
352
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +0000353 SkDEBUGCODE(void validate() const;)
reede5ea5002014-09-03 11:54:58 -0700354
reede5ea5002014-09-03 11:54:58 -0700355private:
msarett23c51102016-05-27 07:39:02 -0700356 sk_sp<SkColorSpace> fColorSpace;
reed7c748852014-11-10 08:57:21 -0800357 int fWidth;
358 int fHeight;
359 SkColorType fColorType;
360 SkAlphaType fAlphaType;
reedc2e68272015-08-01 07:03:20 -0700361 SkColorProfileType fProfileType;
reede5ea5002014-09-03 11:54:58 -0700362
msarett23c51102016-05-27 07:39:02 -0700363 SkImageInfo(int width, int height, SkColorType ct, SkAlphaType at, SkColorProfileType pt,
364 sk_sp<SkColorSpace> cs)
365 : fColorSpace(std::move(cs))
366 , fWidth(width)
reede5ea5002014-09-03 11:54:58 -0700367 , fHeight(height)
368 , fColorType(ct)
369 , fAlphaType(at)
reed7c748852014-11-10 08:57:21 -0800370 , fProfileType(pt)
reede5ea5002014-09-03 11:54:58 -0700371 {}
reed@google.com3443fd82013-11-13 19:09:13 +0000372};
373
brianosman898235c2016-04-06 07:38:23 -0700374///////////////////////////////////////////////////////////////////////////////
375
376static inline bool SkColorAndProfileAreGammaCorrect(SkColorType ct, SkColorProfileType pt) {
377 return kSRGB_SkColorProfileType == pt || kRGBA_F16_SkColorType == ct;
378}
379
380static inline bool SkImageInfoIsGammaCorrect(const SkImageInfo& info) {
381 return SkColorAndProfileAreGammaCorrect(info.colorType(), info.profileType());
382}
383
reed@google.com3443fd82013-11-13 19:09:13 +0000384#endif