| /* |
| * Copyright 2017 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #ifndef SkImageInfoPriv_DEFINED |
| #define SkImageInfoPriv_DEFINED |
| |
| #include "SkImageInfo.h" |
| |
| enum SkColorTypeComponentFlag { |
| kRed_SkColorTypeComponentFlag = 0x1, |
| kGreen_SkColorTypeComponentFlag = 0x2, |
| kBlue_SkColorTypeComponentFlag = 0x4, |
| kAlpha_SkColorTypeComponentFlag = 0x8, |
| kGray_SkColorTypeComponentFlag = 0x10, |
| kRGB_SkColorTypeComponentFlags = kRed_SkColorTypeComponentFlag | |
| kGreen_SkColorTypeComponentFlag | |
| kBlue_SkColorTypeComponentFlag, |
| kRGBA_SkColorTypeComponentFlags = kRGB_SkColorTypeComponentFlags | |
| kAlpha_SkColorTypeComponentFlag, |
| }; |
| |
| static inline uint32_t SkColorTypeComponentFlags(SkColorType ct) { |
| switch (ct) { |
| case kUnknown_SkColorType: return 0; |
| case kAlpha_8_SkColorType: return kAlpha_SkColorTypeComponentFlag; |
| case kRGB_565_SkColorType: return kRGB_SkColorTypeComponentFlags; |
| case kARGB_4444_SkColorType: return kRGBA_SkColorTypeComponentFlags; |
| case kRGBA_8888_SkColorType: return kRGBA_SkColorTypeComponentFlags; |
| case kRGB_888x_SkColorType: return kRGB_SkColorTypeComponentFlags; |
| case kBGRA_8888_SkColorType: return kRGBA_SkColorTypeComponentFlags; |
| case kRGBA_1010102_SkColorType: return kRGBA_SkColorTypeComponentFlags; |
| case kRGB_101010x_SkColorType: return kRGB_SkColorTypeComponentFlags; |
| case kGray_8_SkColorType: return kGray_SkColorTypeComponentFlag; |
| case kRGBA_F16_SkColorType: return kRGBA_SkColorTypeComponentFlags; |
| case kRGBA_F32_SkColorType: return kRGBA_SkColorTypeComponentFlags; |
| } |
| return 0; |
| } |
| |
| static inline bool SkColorTypeIsAlphaOnly(SkColorType ct) { |
| return kAlpha_SkColorTypeComponentFlag == SkColorTypeComponentFlags(ct); |
| } |
| |
| static inline bool SkAlphaTypeIsValid(unsigned value) { |
| return value <= kLastEnum_SkAlphaType; |
| } |
| |
| static inline bool SkColorTypeIsGray(SkColorType ct) { |
| auto flags = SkColorTypeComponentFlags(ct); |
| // Currently assuming that a color type has only gray or does not have gray. |
| SkASSERT(!(kGray_SkColorTypeComponentFlag & flags) || kGray_SkColorTypeComponentFlag == flags); |
| return kGray_SkColorTypeComponentFlag == flags; |
| } |
| |
| static int SkColorTypeShiftPerPixel(SkColorType ct) { |
| switch (ct) { |
| case kUnknown_SkColorType: return 0; |
| case kAlpha_8_SkColorType: return 0; |
| case kRGB_565_SkColorType: return 1; |
| case kARGB_4444_SkColorType: return 1; |
| case kRGBA_8888_SkColorType: return 2; |
| case kRGB_888x_SkColorType: return 2; |
| case kBGRA_8888_SkColorType: return 2; |
| case kRGBA_1010102_SkColorType: return 2; |
| case kRGB_101010x_SkColorType: return 2; |
| case kGray_8_SkColorType: return 0; |
| case kRGBA_F16_SkColorType: return 3; |
| case kRGBA_F32_SkColorType: return 4; |
| } |
| return 0; |
| } |
| |
| static inline size_t SkColorTypeMinRowBytes(SkColorType ct, int width) { |
| return width * SkColorTypeBytesPerPixel(ct); |
| } |
| |
| static inline bool SkColorTypeIsValid(unsigned value) { |
| return value <= kLastEnum_SkColorType; |
| } |
| |
| static inline size_t SkColorTypeComputeOffset(SkColorType ct, int x, int y, size_t rowBytes) { |
| if (kUnknown_SkColorType == ct) { |
| return 0; |
| } |
| return y * rowBytes + (x << SkColorTypeShiftPerPixel(ct)); |
| } |
| |
| /** |
| * Returns true if |info| contains a valid combination of width, height, colorType, and alphaType. |
| */ |
| static inline bool SkImageInfoIsValid(const SkImageInfo& info) { |
| if (info.width() <= 0 || info.height() <= 0) { |
| return false; |
| } |
| |
| const int kMaxDimension = SK_MaxS32 >> 2; |
| if (info.width() > kMaxDimension || info.height() > kMaxDimension) { |
| return false; |
| } |
| |
| if (kUnknown_SkColorType == info.colorType() || kUnknown_SkAlphaType == info.alphaType()) { |
| return false; |
| } |
| |
| if (kOpaque_SkAlphaType != info.alphaType() && |
| (kRGB_565_SkColorType == info.colorType() || kGray_8_SkColorType == info.colorType())) { |
| return false; |
| } |
| |
| return true; |
| } |
| |
| /** |
| * Returns true if Skia has defined a pixel conversion from the |src| to the |dst|. |
| * Returns false otherwise. Some discussion of false cases: |
| * We do not convert to kGray8 when the |src| is not kGray8 in the same color space. |
| * We may add this feature - it just requires some work to convert to luminance while |
| * handling color spaces correctly. Currently no one is asking for this. |
| * We will not convert from kAlpha8 when the |dst| is not kAlpha8. This would require |
| * inventing color information. |
| * We will not convert to kOpaque when the |src| is not kOpaque. This could be |
| * implemented to set all the alpha values to 1, but there is still some ambiguity - |
| * should we use kPremul or kUnpremul color values with the opaque alphas? Or should |
| * we just use whatever the |src| alpha is? In the future, we could choose to clearly |
| * define this, but currently no one is asking for this feature. |
| * We will not convert to a particular color space if |src| is nullptr. The color space |
| * conversion is not well-defined. |
| */ |
| static inline bool SkImageInfoValidConversion(const SkImageInfo& dst, const SkImageInfo& src) { |
| if (!SkImageInfoIsValid(dst) || !SkImageInfoIsValid(src)) { |
| return false; |
| } |
| |
| if (SkColorTypeIsGray(dst.colorType())) { |
| if (!SkColorTypeIsGray(src.colorType())) { |
| return false; |
| } |
| |
| if (dst.colorSpace() && !SkColorSpace::Equals(dst.colorSpace(), src.colorSpace())) { |
| return false; |
| } |
| } |
| |
| if (!SkColorTypeIsAlphaOnly(dst.colorType()) && SkColorTypeIsAlphaOnly(src.colorType())) { |
| return false; |
| } |
| |
| if (kOpaque_SkAlphaType == dst.alphaType() && kOpaque_SkAlphaType != src.alphaType()) { |
| return false; |
| } |
| |
| if (dst.colorSpace() && !src.colorSpace()) { |
| return false; |
| } |
| |
| return true; |
| } |
| #endif // SkImageInfoPriv_DEFINED |