/* * Copyright 2020 Google LLC * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include "src/ports/SkNDKConversions.h" namespace { static const struct { SkColorType colorType; AndroidBitmapFormat format; } gColorTypeTable[] = { { kRGBA_8888_SkColorType, ANDROID_BITMAP_FORMAT_RGBA_8888 }, { kRGBA_F16_SkColorType, ANDROID_BITMAP_FORMAT_RGBA_F16 }, { kRGB_565_SkColorType, ANDROID_BITMAP_FORMAT_RGB_565 }, // Android allows using its alpha 8 format to get 8 bit gray pixels. { kGray_8_SkColorType, ANDROID_BITMAP_FORMAT_A_8 }, }; } // anonymous namespace namespace SkNDKConversions { AndroidBitmapFormat toAndroidBitmapFormat(SkColorType colorType) { for (const auto& entry : gColorTypeTable) { if (entry.colorType == colorType) { return entry.format; } } return ANDROID_BITMAP_FORMAT_NONE; } SkColorType toColorType(AndroidBitmapFormat format) { for (const auto& entry : gColorTypeTable) { if (entry.format == format) { return entry.colorType; } } return kUnknown_SkColorType; } } // SkNDKConversions static constexpr skcms_TransferFunction k2Dot6 = {2.6f, 1.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f}; static constexpr skcms_Matrix3x3 kDCIP3 = {{ {0.486143, 0.323835, 0.154234}, {0.226676, 0.710327, 0.0629966}, {0.000800549, 0.0432385, 0.78275}, }}; namespace { static const struct { ADataSpace dataSpace; skcms_TransferFunction transferFunction; skcms_Matrix3x3 gamut; } gColorSpaceTable[] = { { ADATASPACE_SRGB, SkNamedTransferFn::kSRGB, SkNamedGamut::kSRGB }, { ADATASPACE_SCRGB, SkNamedTransferFn::kSRGB, SkNamedGamut::kSRGB }, { ADATASPACE_SCRGB_LINEAR, SkNamedTransferFn::kLinear, SkNamedGamut::kSRGB }, { ADATASPACE_SRGB_LINEAR, SkNamedTransferFn::kLinear, SkNamedGamut::kSRGB }, { ADATASPACE_ADOBE_RGB, SkNamedTransferFn::k2Dot2, SkNamedGamut::kAdobeRGB }, { ADATASPACE_DISPLAY_P3, SkNamedTransferFn::kSRGB, SkNamedGamut::kDisplayP3 }, { ADATASPACE_BT2020, SkNamedTransferFn::kRec2020, SkNamedGamut::kRec2020 }, { ADATASPACE_BT709, SkNamedTransferFn::kRec2020, SkNamedGamut::kSRGB }, { ADATASPACE_DCI_P3, k2Dot6, kDCIP3 }, }; } // anonymous namespace static bool nearly_equal(float a, float b) { return fabs(a - b) < .002f; } static bool nearly_equal(const skcms_TransferFunction& x, const skcms_TransferFunction& y) { return nearly_equal(x.g, y.g) && nearly_equal(x.a, y.a) && nearly_equal(x.b, y.b) && nearly_equal(x.c, y.c) && nearly_equal(x.d, y.d) && nearly_equal(x.e, y.e) && nearly_equal(x.f, y.f); } static bool nearly_equal(const skcms_Matrix3x3& a, const skcms_Matrix3x3& b) { for (int i = 0; i < 3; i++) for (int j = 0; j < 3; j++) { if (!nearly_equal(a.vals[i][j], b.vals[i][j])) return false; } return true; } namespace SkNDKConversions { ADataSpace toDataSpace(SkColorSpace* cs) { if (!cs) return ADATASPACE_SRGB; skcms_TransferFunction fn; skcms_Matrix3x3 gamut; if (cs->isNumericalTransferFn(&fn) && cs->toXYZD50(&gamut)) { for (const auto& entry : gColorSpaceTable) { if (nearly_equal(gamut, entry.gamut) && nearly_equal(fn, entry.transferFunction)) { return entry.dataSpace; } } } return ADATASPACE_UNKNOWN; } sk_sp toColorSpace(ADataSpace dataSpace) { for (const auto& entry : gColorSpaceTable) { if (entry.dataSpace == dataSpace) { return SkColorSpace::MakeRGB(entry.transferFunction, entry.gamut); } } return nullptr; } }