blob: 4b82ae6e2fb317cf2414cb09801d72efc26de423 [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
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +000011#include "SkMath.h"
halcanaryf622a6c2014-10-24 12:54:53 -070012#include "SkRect.h"
commit-bot@chromium.org32678d92014-01-15 02:38:22 +000013#include "SkSize.h"
reed@google.com3443fd82013-11-13 19:09:13 +000014
commit-bot@chromium.org8b0e8ac2014-01-30 18:58:24 +000015class SkReadBuffer;
halcanaryf622a6c2014-10-24 12:54:53 -070016class SkWriteBuffer;
reed@google.com9230ea22013-12-09 22:01:03 +000017
reed@google.com3443fd82013-11-13 19:09:13 +000018/**
bsalomonafa95e22015-10-12 10:39:46 -070019 * Describes how to interpret the alpha component of a pixel.
reed@google.com3443fd82013-11-13 19:09:13 +000020 */
21enum SkAlphaType {
reed44977482015-02-27 10:23:00 -080022 kUnknown_SkAlphaType,
skia.committer@gmail.com73a5d532013-11-14 07:02:31 +000023
reed@google.com3443fd82013-11-13 19:09:13 +000024 /**
25 * All pixels are stored as opaque. This differs slightly from kIgnore in
26 * that kOpaque has correct "opaque" values stored in the pixels, while
27 * kIgnore may not, but in both cases the caller should treat the pixels
28 * as opaque.
29 */
30 kOpaque_SkAlphaType,
skia.committer@gmail.com73a5d532013-11-14 07:02:31 +000031
reed@google.com3443fd82013-11-13 19:09:13 +000032 /**
33 * All pixels have their alpha premultiplied in their color components.
34 * This is the natural format for the rendering target pixels.
35 */
36 kPremul_SkAlphaType,
skia.committer@gmail.com73a5d532013-11-14 07:02:31 +000037
reed@google.com3443fd82013-11-13 19:09:13 +000038 /**
39 * All pixels have their color components stored without any regard to the
40 * alpha. e.g. this is the default configuration for PNG images.
41 *
42 * This alpha-type is ONLY supported for input images. Rendering cannot
43 * generate this on output.
44 */
45 kUnpremul_SkAlphaType,
skia.committer@gmail.com73a5d532013-11-14 07:02:31 +000046
reed@google.com3443fd82013-11-13 19:09:13 +000047 kLastEnum_SkAlphaType = kUnpremul_SkAlphaType
48};
49
50static inline bool SkAlphaTypeIsOpaque(SkAlphaType at) {
reed44977482015-02-27 10:23:00 -080051 return kOpaque_SkAlphaType == at;
reed@google.com3443fd82013-11-13 19:09:13 +000052}
53
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +000054static inline bool SkAlphaTypeIsValid(unsigned value) {
55 return value <= kLastEnum_SkAlphaType;
56}
57
reed@google.com3443fd82013-11-13 19:09:13 +000058///////////////////////////////////////////////////////////////////////////////
59
60/**
61 * Describes how to interpret the components of a pixel.
reed6c6ddb82014-06-30 12:44:03 -070062 *
63 * kN32_SkColorType is an alias for whichever 32bit ARGB format is the "native"
64 * form for skia's blitters. Use this if you don't have a swizzle preference
65 * for 32bit pixels.
reed@google.com3443fd82013-11-13 19:09:13 +000066 */
67enum SkColorType {
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +000068 kUnknown_SkColorType,
reed@google.com3443fd82013-11-13 19:09:13 +000069 kAlpha_8_SkColorType,
70 kRGB_565_SkColorType,
reed@google.com9230ea22013-12-09 22:01:03 +000071 kARGB_4444_SkColorType,
reed@google.com3443fd82013-11-13 19:09:13 +000072 kRGBA_8888_SkColorType,
73 kBGRA_8888_SkColorType,
reed@google.com9230ea22013-12-09 22:01:03 +000074 kIndex_8_SkColorType,
reed0c9b1a82015-03-17 17:44:06 -070075 kGray_8_SkColorType,
reed@google.com3443fd82013-11-13 19:09:13 +000076
reed0c9b1a82015-03-17 17:44:06 -070077 kLastEnum_SkColorType = kGray_8_SkColorType,
reed@google.com3443fd82013-11-13 19:09:13 +000078
79#if SK_PMCOLOR_BYTE_ORDER(B,G,R,A)
commit-bot@chromium.org28fcae22014-04-11 17:15:40 +000080 kN32_SkColorType = kBGRA_8888_SkColorType,
reed@google.com3443fd82013-11-13 19:09:13 +000081#elif SK_PMCOLOR_BYTE_ORDER(R,G,B,A)
commit-bot@chromium.org28fcae22014-04-11 17:15:40 +000082 kN32_SkColorType = kRGBA_8888_SkColorType,
reed@google.com3443fd82013-11-13 19:09:13 +000083#else
reed3b40ee62014-12-21 14:29:04 -080084 #error "SK_*32_SHFIT values must correspond to BGRA or RGBA byte order"
commit-bot@chromium.org28fcae22014-04-11 17:15:40 +000085#endif
reed@google.com3443fd82013-11-13 19:09:13 +000086};
87
88static int SkColorTypeBytesPerPixel(SkColorType ct) {
89 static const uint8_t gSize[] = {
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +000090 0, // Unknown
reed@google.com3443fd82013-11-13 19:09:13 +000091 1, // Alpha_8
92 2, // RGB_565
reed@google.com9230ea22013-12-09 22:01:03 +000093 2, // ARGB_4444
reed@google.com3443fd82013-11-13 19:09:13 +000094 4, // RGBA_8888
95 4, // BGRA_8888
96 1, // kIndex_8
reed0c9b1a82015-03-17 17:44:06 -070097 1, // kGray_8
reed@google.com3443fd82013-11-13 19:09:13 +000098 };
bungeman99fe8222015-08-20 07:57:51 -070099 static_assert(SK_ARRAY_COUNT(gSize) == (size_t)(kLastEnum_SkColorType + 1),
100 "size_mismatch_with_SkColorType_enum");
reed@google.com3443fd82013-11-13 19:09:13 +0000101
102 SkASSERT((size_t)ct < SK_ARRAY_COUNT(gSize));
103 return gSize[ct];
104}
105
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +0000106static inline size_t SkColorTypeMinRowBytes(SkColorType ct, int width) {
107 return width * SkColorTypeBytesPerPixel(ct);
108}
109
110static inline bool SkColorTypeIsValid(unsigned value) {
111 return value <= kLastEnum_SkColorType;
112}
113
reed92fc2ae2015-05-22 08:06:21 -0700114static inline size_t SkColorTypeComputeOffset(SkColorType ct, int x, int y, size_t rowBytes) {
115 int shift = 0;
116 switch (SkColorTypeBytesPerPixel(ct)) {
117 case 4: shift = 2; break;
118 case 2: shift = 1; break;
119 case 1: shift = 0; break;
120 default: return 0;
121 }
122 return y * rowBytes + (x << shift);
123}
124
reed@google.com3443fd82013-11-13 19:09:13 +0000125///////////////////////////////////////////////////////////////////////////////
126
127/**
scroggo2fd0d142014-07-01 07:08:19 -0700128 * Return true if alphaType is supported by colorType. If there is a canonical
129 * alphaType for this colorType, return it in canonical.
130 */
131bool SkColorTypeValidateAlphaType(SkColorType colorType, SkAlphaType alphaType,
132 SkAlphaType* canonical = NULL);
133
134///////////////////////////////////////////////////////////////////////////////
135
136/**
rileyaabaef862014-09-12 17:45:58 -0700137 * Describes the color space a YUV pixel.
138 */
139enum SkYUVColorSpace {
140 /** Standard JPEG color space. */
141 kJPEG_SkYUVColorSpace,
142 /** SDTV standard Rec. 601 color space. Uses "studio swing" [16, 235] color
143 range. See http://en.wikipedia.org/wiki/Rec._601 for details. */
144 kRec601_SkYUVColorSpace,
rileya13400392015-07-20 15:00:03 -0700145 /** HDTV standard Rec. 709 color space. Uses "studio swing" [16, 235] color
146 range. See http://en.wikipedia.org/wiki/Rec._709 for details. */
147 kRec709_SkYUVColorSpace,
rileyaabaef862014-09-12 17:45:58 -0700148
rileya13400392015-07-20 15:00:03 -0700149 kLastEnum_SkYUVColorSpace = kRec709_SkYUVColorSpace
rileyaabaef862014-09-12 17:45:58 -0700150};
151
152///////////////////////////////////////////////////////////////////////////////
153
reed7c748852014-11-10 08:57:21 -0800154enum SkColorProfileType {
155 kLinear_SkColorProfileType,
156 kSRGB_SkColorProfileType,
157
158 kLastEnum_SkColorProfileType = kSRGB_SkColorProfileType
159};
160
rileyaabaef862014-09-12 17:45:58 -0700161/**
reed@google.com3443fd82013-11-13 19:09:13 +0000162 * Describe an image's dimensions and pixel type.
reed7c748852014-11-10 08:57:21 -0800163 * Used for both src images and render-targets (surfaces).
reed@google.com3443fd82013-11-13 19:09:13 +0000164 */
reed41814922015-01-30 14:54:38 -0800165struct SK_API SkImageInfo {
reede5ea5002014-09-03 11:54:58 -0700166public:
167 SkImageInfo()
168 : fWidth(0)
169 , fHeight(0)
170 , fColorType(kUnknown_SkColorType)
reed44977482015-02-27 10:23:00 -0800171 , fAlphaType(kUnknown_SkAlphaType)
reed7c748852014-11-10 08:57:21 -0800172 , fProfileType(kLinear_SkColorProfileType)
reede5ea5002014-09-03 11:54:58 -0700173 {}
reed@google.com3443fd82013-11-13 19:09:13 +0000174
reed7c748852014-11-10 08:57:21 -0800175 static SkImageInfo Make(int width, int height, SkColorType ct, SkAlphaType at,
176 SkColorProfileType pt = kLinear_SkColorProfileType) {
177 return SkImageInfo(width, height, ct, at, pt);
commit-bot@chromium.org32678d92014-01-15 02:38:22 +0000178 }
179
180 /**
181 * Sets colortype to the native ARGB32 type.
182 */
reed7c748852014-11-10 08:57:21 -0800183 static SkImageInfo MakeN32(int width, int height, SkAlphaType at,
184 SkColorProfileType pt = kLinear_SkColorProfileType) {
185 return SkImageInfo(width, height, kN32_SkColorType, at, pt);
commit-bot@chromium.org32678d92014-01-15 02:38:22 +0000186 }
187
188 /**
189 * Sets colortype to the native ARGB32 type, and the alphatype to premul.
190 */
reed7c748852014-11-10 08:57:21 -0800191 static SkImageInfo MakeN32Premul(int width, int height,
192 SkColorProfileType pt = kLinear_SkColorProfileType) {
193 return SkImageInfo(width, height, kN32_SkColorType, kPremul_SkAlphaType, pt);
commit-bot@chromium.org32678d92014-01-15 02:38:22 +0000194 }
skia.committer@gmail.comd77b3ec2014-01-15 07:01:43 +0000195
commit-bot@chromium.org32678d92014-01-15 02:38:22 +0000196 /**
197 * Sets colortype to the native ARGB32 type, and the alphatype to premul.
198 */
reed7c748852014-11-10 08:57:21 -0800199 static SkImageInfo MakeN32Premul(const SkISize& size,
200 SkColorProfileType pt = kLinear_SkColorProfileType) {
201 return MakeN32Premul(size.width(), size.height(), pt);
commit-bot@chromium.org32678d92014-01-15 02:38:22 +0000202 }
skia.committer@gmail.comd77b3ec2014-01-15 07:01:43 +0000203
commit-bot@chromium.org32678d92014-01-15 02:38:22 +0000204 static SkImageInfo MakeA8(int width, int height) {
reed7c748852014-11-10 08:57:21 -0800205 return SkImageInfo(width, height, kAlpha_8_SkColorType, kPremul_SkAlphaType,
206 kLinear_SkColorProfileType);
commit-bot@chromium.org32678d92014-01-15 02:38:22 +0000207 }
skia.committer@gmail.comd77b3ec2014-01-15 07:01:43 +0000208
reed@google.com900ecf22014-02-20 20:55:37 +0000209 static SkImageInfo MakeUnknown(int width, int height) {
reed44977482015-02-27 10:23:00 -0800210 return SkImageInfo(width, height, kUnknown_SkColorType, kUnknown_SkAlphaType,
reed7c748852014-11-10 08:57:21 -0800211 kLinear_SkColorProfileType);
reed@google.com900ecf22014-02-20 20:55:37 +0000212 }
213
reedf252f642014-06-14 04:24:56 -0700214 static SkImageInfo MakeUnknown() {
reede5ea5002014-09-03 11:54:58 -0700215 return SkImageInfo();
reedf252f642014-06-14 04:24:56 -0700216 }
217
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +0000218 int width() const { return fWidth; }
219 int height() const { return fHeight; }
220 SkColorType colorType() const { return fColorType; }
221 SkAlphaType alphaType() const { return fAlphaType; }
reed7c748852014-11-10 08:57:21 -0800222 SkColorProfileType profileType() const { return fProfileType; }
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +0000223
224 bool isEmpty() const { return fWidth <= 0 || fHeight <= 0; }
225
reed@google.com3443fd82013-11-13 19:09:13 +0000226 bool isOpaque() const {
227 return SkAlphaTypeIsOpaque(fAlphaType);
228 }
229
reed7c748852014-11-10 08:57:21 -0800230 bool isLinear() const { return kLinear_SkColorProfileType == fProfileType; }
231 bool isSRGB() const { return kSRGB_SkColorProfileType == fProfileType; }
232
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +0000233 SkISize dimensions() const { return SkISize::Make(fWidth, fHeight); }
reed7c748852014-11-10 08:57:21 -0800234 SkIRect bounds() const { return SkIRect::MakeWH(fWidth, fHeight); }
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +0000235
reedc77392e2014-06-02 13:07:26 -0700236 /**
237 * Return a new ImageInfo with the same colortype and alphatype as this info,
238 * but with the specified width and height.
239 */
240 SkImageInfo makeWH(int newWidth, int newHeight) const {
reed7c748852014-11-10 08:57:21 -0800241 return SkImageInfo::Make(newWidth, newHeight, fColorType, fAlphaType, fProfileType);
reedc77392e2014-06-02 13:07:26 -0700242 }
243
reede5ea5002014-09-03 11:54:58 -0700244 SkImageInfo makeAlphaType(SkAlphaType newAlphaType) const {
reed7c748852014-11-10 08:57:21 -0800245 return SkImageInfo::Make(fWidth, fHeight, fColorType, newAlphaType, fProfileType);
reede5ea5002014-09-03 11:54:58 -0700246 }
247
248 SkImageInfo makeColorType(SkColorType newColorType) const {
reed7c748852014-11-10 08:57:21 -0800249 return SkImageInfo::Make(fWidth, fHeight, newColorType, fAlphaType, fProfileType);
reede5ea5002014-09-03 11:54:58 -0700250 }
reed7c748852014-11-10 08:57:21 -0800251
reed@google.com3443fd82013-11-13 19:09:13 +0000252 int bytesPerPixel() const {
253 return SkColorTypeBytesPerPixel(fColorType);
254 }
255
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +0000256 uint64_t minRowBytes64() const {
257 return sk_64_mul(fWidth, this->bytesPerPixel());
258 }
259
mike@reedtribe.org169a0ed2014-02-11 02:20:17 +0000260 size_t minRowBytes() const {
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +0000261 return (size_t)this->minRowBytes64();
mike@reedtribe.org169a0ed2014-02-11 02:20:17 +0000262 }
skia.committer@gmail.com8ed64432014-02-11 03:02:13 +0000263
reed92fc2ae2015-05-22 08:06:21 -0700264 size_t computeOffset(int x, int y, size_t rowBytes) const {
265 SkASSERT((unsigned)x < (unsigned)fWidth);
266 SkASSERT((unsigned)y < (unsigned)fHeight);
267 return SkColorTypeComputeOffset(fColorType, x, y, rowBytes);
268 }
269
reed@google.com3443fd82013-11-13 19:09:13 +0000270 bool operator==(const SkImageInfo& other) const {
271 return 0 == memcmp(this, &other, sizeof(other));
272 }
273 bool operator!=(const SkImageInfo& other) const {
274 return 0 != memcmp(this, &other, sizeof(other));
275 }
reed@google.com9230ea22013-12-09 22:01:03 +0000276
commit-bot@chromium.org8b0e8ac2014-01-30 18:58:24 +0000277 void unflatten(SkReadBuffer&);
278 void flatten(SkWriteBuffer&) const;
reed@google.com9230ea22013-12-09 22:01:03 +0000279
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +0000280 int64_t getSafeSize64(size_t rowBytes) const {
reed@google.com9230ea22013-12-09 22:01:03 +0000281 if (0 == fHeight) {
282 return 0;
283 }
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +0000284 return sk_64_mul(fHeight - 1, rowBytes) + fWidth * this->bytesPerPixel();
reed@google.com9230ea22013-12-09 22:01:03 +0000285 }
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +0000286
287 size_t getSafeSize(size_t rowBytes) const {
reedde499882015-06-18 13:41:40 -0700288 int64_t size = this->getSafeSize64(rowBytes);
289 if (!sk_64_isS32(size)) {
290 return 0;
291 }
292 return sk_64_asS32(size);
commit-bot@chromium.org61e96cd2014-02-11 18:21:45 +0000293 }
294
295 bool validRowBytes(size_t rowBytes) const {
296 uint64_t rb = sk_64_mul(fWidth, this->bytesPerPixel());
297 return rowBytes >= rb;
298 }
299
300 SkDEBUGCODE(void validate() const;)
reede5ea5002014-09-03 11:54:58 -0700301
reede5ea5002014-09-03 11:54:58 -0700302private:
reed7c748852014-11-10 08:57:21 -0800303 int fWidth;
304 int fHeight;
305 SkColorType fColorType;
306 SkAlphaType fAlphaType;
reedc2e68272015-08-01 07:03:20 -0700307 SkColorProfileType fProfileType;
reede5ea5002014-09-03 11:54:58 -0700308
reed7c748852014-11-10 08:57:21 -0800309 SkImageInfo(int width, int height, SkColorType ct, SkAlphaType at, SkColorProfileType pt)
reede5ea5002014-09-03 11:54:58 -0700310 : fWidth(width)
311 , fHeight(height)
312 , fColorType(ct)
313 , fAlphaType(at)
reed7c748852014-11-10 08:57:21 -0800314 , fProfileType(pt)
reede5ea5002014-09-03 11:54:58 -0700315 {}
reed@google.com3443fd82013-11-13 19:09:13 +0000316};
317
318#endif