2017-01-17 15:48:53 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2017 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
2017-02-07 19:09:38 +00:00
|
|
|
#ifndef SkImageInfoPriv_DEFINED
|
|
|
|
#define SkImageInfoPriv_DEFINED
|
|
|
|
|
|
|
|
#include "SkImageInfo.h"
|
|
|
|
|
2018-03-22 14:01:16 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline bool SkColorTypeIsAlphaOnly(SkColorType ct) {
|
|
|
|
return kAlpha_SkColorTypeComponentFlag == SkColorTypeComponentFlags(ct);
|
|
|
|
}
|
|
|
|
|
2018-02-09 18:26:46 +00:00
|
|
|
static inline bool SkAlphaTypeIsValid(unsigned value) {
|
|
|
|
return value <= kLastEnum_SkAlphaType;
|
|
|
|
}
|
|
|
|
|
2018-03-22 14:01:16 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2018-02-09 18:26:46 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
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));
|
|
|
|
}
|
|
|
|
|
2017-01-17 15:48:53 +00:00
|
|
|
/**
|
2018-06-04 13:21:17 +00:00
|
|
|
* Returns true if |info| contains a valid combination of width, height, colorType, and alphaType.
|
2017-01-17 15:48:53 +00:00
|
|
|
*/
|
2018-06-04 13:21:17 +00:00
|
|
|
static inline bool SkImageInfoIsValid(const SkImageInfo& info) {
|
2017-01-17 15:48:53 +00:00
|
|
|
if (info.width() <= 0 || info.height() <= 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-02-14 16:21:02 +00:00
|
|
|
const int kMaxDimension = SK_MaxS32 >> 2;
|
|
|
|
if (info.width() > kMaxDimension || info.height() > kMaxDimension) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-01-17 15:48:53 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2017-05-02 20:04:56 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-01-17 15:48:53 +00:00
|
|
|
/**
|
|
|
|
* Returns true if Skia has defined a pixel conversion from the |src| to the |dst|.
|
|
|
|
* Returns false otherwise. Some discussion of false cases:
|
2017-02-09 18:50:45 +00:00
|
|
|
* 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.
|
2017-01-17 15:48:53 +00:00
|
|
|
* 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.
|
2017-02-07 21:52:07 +00:00
|
|
|
* We will not convert to a particular color space if |src| is nullptr. The color space
|
|
|
|
* conversion is not well-defined.
|
2017-01-17 15:48:53 +00:00
|
|
|
*/
|
|
|
|
static inline bool SkImageInfoValidConversion(const SkImageInfo& dst, const SkImageInfo& src) {
|
2018-06-04 13:21:17 +00:00
|
|
|
if (!SkImageInfoIsValid(dst) || !SkImageInfoIsValid(src)) {
|
2017-01-17 15:48:53 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-03-22 14:01:16 +00:00
|
|
|
if (SkColorTypeIsGray(dst.colorType())) {
|
|
|
|
if (!SkColorTypeIsGray(src.colorType())) {
|
2017-02-14 16:21:02 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dst.colorSpace() && !SkColorSpace::Equals(dst.colorSpace(), src.colorSpace())) {
|
|
|
|
return false;
|
|
|
|
}
|
2017-01-17 15:48:53 +00:00
|
|
|
}
|
|
|
|
|
2018-03-22 14:01:16 +00:00
|
|
|
if (!SkColorTypeIsAlphaOnly(dst.colorType()) && SkColorTypeIsAlphaOnly(src.colorType())) {
|
2017-01-17 15:48:53 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-02-06 18:28:49 +00:00
|
|
|
if (kOpaque_SkAlphaType == dst.alphaType() && kOpaque_SkAlphaType != src.alphaType()) {
|
|
|
|
return false;
|
|
|
|
}
|
2017-01-17 15:48:53 +00:00
|
|
|
|
2017-02-07 21:52:07 +00:00
|
|
|
if (dst.colorSpace() && !src.colorSpace()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-01-17 15:48:53 +00:00
|
|
|
return true;
|
|
|
|
}
|
2017-02-07 19:09:38 +00:00
|
|
|
#endif // SkImageInfoPriv_DEFINED
|