reed@google.com | f309dbc | 2013-12-09 22:09:41 +0000 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2010 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 "SkImageInfo.h" |
halcanary | 484b3d0 | 2016-04-25 10:32:23 -0700 | [diff] [blame] | 9 | #include "SkImageInfoPriv.h" |
commit-bot@chromium.org | 8b0e8ac | 2014-01-30 18:58:24 +0000 | [diff] [blame] | 10 | #include "SkReadBuffer.h" |
| 11 | #include "SkWriteBuffer.h" |
reed@google.com | f309dbc | 2013-12-09 22:09:41 +0000 | [diff] [blame] | 12 | |
reed | 960b2d6 | 2016-06-17 09:26:41 -0700 | [diff] [blame^] | 13 | /* |
| 14 | * We store this as a byte in the ImageInfo flatten buffer. |
| 15 | */ |
| 16 | enum class SkFlattenColorSpaceEnum { |
| 17 | kUnspecified, |
| 18 | kSRGB, |
| 19 | kAdobe1998, |
| 20 | // ... add more here |
| 21 | kLastEnum = kAdobe1998, |
| 22 | // final value means the actual profile data follows the info |
| 23 | kICCProfile = 0xFF, |
| 24 | }; |
| 25 | |
| 26 | static sk_sp<SkColorSpace> make_from_enum(SkFlattenColorSpaceEnum value) { |
| 27 | switch (value) { |
| 28 | case SkFlattenColorSpaceEnum::kSRGB: |
| 29 | return SkColorSpace::NewNamed(SkColorSpace::kSRGB_Named); |
| 30 | case SkFlattenColorSpaceEnum::kAdobe1998: |
| 31 | return SkColorSpace::NewNamed(SkColorSpace::kAdobeRGB_Named); |
| 32 | default: |
| 33 | return nullptr; |
| 34 | } |
| 35 | } |
| 36 | |
| 37 | SkColorSpace::Named sk_deduce_named_from_colorspace(SkColorSpace* cs) { |
| 38 | return cs->fNamed; |
| 39 | } |
| 40 | |
| 41 | static SkFlattenColorSpaceEnum deduce_from_colorspace(SkColorSpace* cs) { |
| 42 | if (!cs) { |
| 43 | return SkFlattenColorSpaceEnum::kUnspecified; |
| 44 | } |
| 45 | switch (sk_deduce_named_from_colorspace(cs)) { |
| 46 | case SkColorSpace::kSRGB_Named: |
| 47 | return SkFlattenColorSpaceEnum::kSRGB; |
| 48 | case SkColorSpace::kAdobeRGB_Named: |
| 49 | return SkFlattenColorSpaceEnum::kAdobe1998; |
| 50 | default: |
| 51 | return SkFlattenColorSpaceEnum::kICCProfile; |
| 52 | } |
| 53 | } |
| 54 | |
| 55 | /////////////////////////////////////////////////////////////////////////////////////////////////// |
| 56 | |
| 57 | #ifdef SK_SUPPORT_LEGACY_COLORPROFILETYPE |
| 58 | SkColorProfileType SkImageInfo::profileType() const { |
| 59 | return fColorSpace && fColorSpace->gammaCloseToSRGB() |
| 60 | ? kSRGB_SkColorProfileType : kLinear_SkColorProfileType; |
| 61 | } |
| 62 | #endif |
| 63 | |
halcanary | 484b3d0 | 2016-04-25 10:32:23 -0700 | [diff] [blame] | 64 | // Indicate how images and gradients should interpret colors by default. |
herb | 835acc7 | 2016-05-17 06:50:15 -0700 | [diff] [blame] | 65 | bool gDefaultProfileIsSRGB; |
halcanary | 484b3d0 | 2016-04-25 10:32:23 -0700 | [diff] [blame] | 66 | |
| 67 | SkColorProfileType SkDefaultColorProfile() { |
| 68 | return gDefaultProfileIsSRGB ? kSRGB_SkColorProfileType |
| 69 | : kLinear_SkColorProfileType; |
| 70 | } |
herb | 9bc2235 | 2016-04-20 15:07:32 -0700 | [diff] [blame] | 71 | |
reed | 2bdf1f5 | 2014-09-03 05:48:56 -0700 | [diff] [blame] | 72 | static bool alpha_type_is_valid(SkAlphaType alphaType) { |
| 73 | return (alphaType >= 0) && (alphaType <= kLastEnum_SkAlphaType); |
commit-bot@chromium.org | ef74fa1 | 2013-12-17 20:49:46 +0000 | [diff] [blame] | 74 | } |
| 75 | |
reed | 2bdf1f5 | 2014-09-03 05:48:56 -0700 | [diff] [blame] | 76 | static bool color_type_is_valid(SkColorType colorType) { |
| 77 | return (colorType >= 0) && (colorType <= kLastEnum_SkColorType); |
commit-bot@chromium.org | ef74fa1 | 2013-12-17 20:49:46 +0000 | [diff] [blame] | 78 | } |
| 79 | |
reed | fbce71f | 2016-06-02 12:40:22 -0700 | [diff] [blame] | 80 | SkImageInfo SkImageInfo::MakeS32(int width, int height, SkAlphaType at) { |
reed | 960b2d6 | 2016-06-17 09:26:41 -0700 | [diff] [blame^] | 81 | return SkImageInfo(width, height, kN32_SkColorType, at, |
reed | fbce71f | 2016-06-02 12:40:22 -0700 | [diff] [blame] | 82 | SkColorSpace::NewNamed(SkColorSpace::kSRGB_Named)); |
| 83 | } |
| 84 | |
reed | 2bdf1f5 | 2014-09-03 05:48:56 -0700 | [diff] [blame] | 85 | void SkImageInfo::unflatten(SkReadBuffer& buffer) { |
| 86 | fWidth = buffer.read32(); |
| 87 | fHeight = buffer.read32(); |
| 88 | |
| 89 | uint32_t packed = buffer.read32(); |
reed | 7c74885 | 2014-11-10 08:57:21 -0800 | [diff] [blame] | 90 | SkASSERT(0 == (packed >> 24)); |
reed | 2bdf1f5 | 2014-09-03 05:48:56 -0700 | [diff] [blame] | 91 | fColorType = (SkColorType)((packed >> 0) & 0xFF); |
reed | 960b2d6 | 2016-06-17 09:26:41 -0700 | [diff] [blame^] | 92 | fAlphaType = (SkAlphaType)((packed >> 8) & 0xFF); |
| 93 | SkFlattenColorSpaceEnum csenum = (SkFlattenColorSpaceEnum)((packed >> 16) & 0xFF); |
| 94 | buffer.validate(alpha_type_is_valid(fAlphaType) && color_type_is_valid(fColorType)); |
| 95 | |
| 96 | if (SkFlattenColorSpaceEnum::kICCProfile == csenum) { |
| 97 | SkASSERT(false); // we shouldn't hit this yet, as we don't write these yet |
| 98 | fColorSpace.reset(); |
| 99 | } else { |
| 100 | if (csenum > SkFlattenColorSpaceEnum::kLastEnum) { |
| 101 | csenum = SkFlattenColorSpaceEnum::kUnspecified; |
| 102 | } |
| 103 | fColorSpace = make_from_enum(csenum); |
| 104 | } |
reed@google.com | f309dbc | 2013-12-09 22:09:41 +0000 | [diff] [blame] | 105 | } |
| 106 | |
reed | 2bdf1f5 | 2014-09-03 05:48:56 -0700 | [diff] [blame] | 107 | void SkImageInfo::flatten(SkWriteBuffer& buffer) const { |
| 108 | buffer.write32(fWidth); |
| 109 | buffer.write32(fHeight); |
reed@google.com | f309dbc | 2013-12-09 22:09:41 +0000 | [diff] [blame] | 110 | |
reed | 960b2d6 | 2016-06-17 09:26:41 -0700 | [diff] [blame^] | 111 | SkFlattenColorSpaceEnum csenum = deduce_from_colorspace(fColorSpace.get()); |
| 112 | |
| 113 | // TODO: when we actually support flattening the colorspace to a profile blob, remove this |
| 114 | // hack (and write the blob after we write packed. |
| 115 | if (SkFlattenColorSpaceEnum::kICCProfile == csenum) { |
| 116 | csenum = SkFlattenColorSpaceEnum::kUnspecified; |
| 117 | } |
| 118 | |
| 119 | SkASSERT(0 == ((int)csenum & ~0xFF)); |
reed | 2bdf1f5 | 2014-09-03 05:48:56 -0700 | [diff] [blame] | 120 | SkASSERT(0 == (fAlphaType & ~0xFF)); |
| 121 | SkASSERT(0 == (fColorType & ~0xFF)); |
reed | 960b2d6 | 2016-06-17 09:26:41 -0700 | [diff] [blame^] | 122 | uint32_t packed = ((int)csenum << 16) | (fAlphaType << 8) | fColorType; |
reed | 2bdf1f5 | 2014-09-03 05:48:56 -0700 | [diff] [blame] | 123 | buffer.write32(packed); |
reed | 960b2d6 | 2016-06-17 09:26:41 -0700 | [diff] [blame^] | 124 | |
| 125 | if (SkFlattenColorSpaceEnum::kICCProfile == csenum) { |
| 126 | // TODO: write the ICCProfile blob |
| 127 | } |
reed@google.com | f309dbc | 2013-12-09 22:09:41 +0000 | [diff] [blame] | 128 | } |
scroggo | 2fd0d14 | 2014-07-01 07:08:19 -0700 | [diff] [blame] | 129 | |
| 130 | bool SkColorTypeValidateAlphaType(SkColorType colorType, SkAlphaType alphaType, |
| 131 | SkAlphaType* canonical) { |
| 132 | switch (colorType) { |
| 133 | case kUnknown_SkColorType: |
reed | 4497748 | 2015-02-27 10:23:00 -0800 | [diff] [blame] | 134 | alphaType = kUnknown_SkAlphaType; |
scroggo | 2fd0d14 | 2014-07-01 07:08:19 -0700 | [diff] [blame] | 135 | break; |
| 136 | case kAlpha_8_SkColorType: |
| 137 | if (kUnpremul_SkAlphaType == alphaType) { |
| 138 | alphaType = kPremul_SkAlphaType; |
| 139 | } |
| 140 | // fall-through |
| 141 | case kIndex_8_SkColorType: |
| 142 | case kARGB_4444_SkColorType: |
| 143 | case kRGBA_8888_SkColorType: |
| 144 | case kBGRA_8888_SkColorType: |
reed | 9e01ac7 | 2016-02-06 20:38:45 -0800 | [diff] [blame] | 145 | case kRGBA_F16_SkColorType: |
reed | 4497748 | 2015-02-27 10:23:00 -0800 | [diff] [blame] | 146 | if (kUnknown_SkAlphaType == alphaType) { |
scroggo | 2fd0d14 | 2014-07-01 07:08:19 -0700 | [diff] [blame] | 147 | return false; |
| 148 | } |
| 149 | break; |
| 150 | case kRGB_565_SkColorType: |
reed | 0c9b1a8 | 2015-03-17 17:44:06 -0700 | [diff] [blame] | 151 | case kGray_8_SkColorType: |
scroggo | 2fd0d14 | 2014-07-01 07:08:19 -0700 | [diff] [blame] | 152 | alphaType = kOpaque_SkAlphaType; |
| 153 | break; |
| 154 | default: |
| 155 | return false; |
| 156 | } |
| 157 | if (canonical) { |
| 158 | *canonical = alphaType; |
| 159 | } |
| 160 | return true; |
| 161 | } |
reed | 96472de | 2014-12-10 09:53:42 -0800 | [diff] [blame] | 162 | |
| 163 | /////////////////////////////////////////////////////////////////////////////////////////////////// |
| 164 | |
| 165 | #include "SkReadPixelsRec.h" |
| 166 | |
| 167 | bool SkReadPixelsRec::trim(int srcWidth, int srcHeight) { |
| 168 | switch (fInfo.colorType()) { |
| 169 | case kUnknown_SkColorType: |
| 170 | case kIndex_8_SkColorType: |
| 171 | return false; |
| 172 | default: |
| 173 | break; |
| 174 | } |
halcanary | 96fcdcc | 2015-08-27 07:41:13 -0700 | [diff] [blame] | 175 | if (nullptr == fPixels || fRowBytes < fInfo.minRowBytes()) { |
reed | 96472de | 2014-12-10 09:53:42 -0800 | [diff] [blame] | 176 | return false; |
| 177 | } |
| 178 | if (0 == fInfo.width() || 0 == fInfo.height()) { |
| 179 | return false; |
| 180 | } |
| 181 | |
| 182 | int x = fX; |
| 183 | int y = fY; |
| 184 | SkIRect srcR = SkIRect::MakeXYWH(x, y, fInfo.width(), fInfo.height()); |
| 185 | if (!srcR.intersect(0, 0, srcWidth, srcHeight)) { |
| 186 | return false; |
| 187 | } |
| 188 | |
| 189 | // if x or y are negative, then we have to adjust pixels |
| 190 | if (x > 0) { |
| 191 | x = 0; |
| 192 | } |
| 193 | if (y > 0) { |
| 194 | y = 0; |
| 195 | } |
| 196 | // here x,y are either 0 or negative |
| 197 | fPixels = ((char*)fPixels - y * fRowBytes - x * fInfo.bytesPerPixel()); |
| 198 | // the intersect may have shrunk info's logical size |
| 199 | fInfo = fInfo.makeWH(srcR.width(), srcR.height()); |
| 200 | fX = srcR.x(); |
| 201 | fY = srcR.y(); |
| 202 | |
| 203 | return true; |
| 204 | } |