1cb93a386Sopenharmony_ci/* 2cb93a386Sopenharmony_ci * Copyright 2021 Google LLC 3cb93a386Sopenharmony_ci * 4cb93a386Sopenharmony_ci * Use of this source code is governed by a BSD-style license that can be 5cb93a386Sopenharmony_ci * found in the LICENSE file. 6cb93a386Sopenharmony_ci */ 7cb93a386Sopenharmony_ci 8cb93a386Sopenharmony_ci#include "gm/gm.h" 9cb93a386Sopenharmony_ci#include "include/core/SkBitmap.h" 10cb93a386Sopenharmony_ci#include "include/core/SkCanvas.h" 11cb93a386Sopenharmony_ci#include "include/core/SkPaint.h" 12cb93a386Sopenharmony_ci#include "include/core/SkSize.h" 13cb93a386Sopenharmony_ci#include "include/core/SkSurface.h" 14cb93a386Sopenharmony_ci#include "include/effects/SkRuntimeEffect.h" 15cb93a386Sopenharmony_ci#include "src/gpu/GrCaps.h" 16cb93a386Sopenharmony_ci#include "src/gpu/GrDirectContextPriv.h" 17cb93a386Sopenharmony_ci#include "tests/Test.h" 18cb93a386Sopenharmony_ci#include "tools/Resources.h" 19cb93a386Sopenharmony_ci#include "tools/RuntimeBlendUtils.h" 20cb93a386Sopenharmony_ci#include "tools/ToolUtils.h" 21cb93a386Sopenharmony_ci 22cb93a386Sopenharmony_cistatic bool nearly_equal(const SkColor& x, const SkColor& y) { 23cb93a386Sopenharmony_ci const int kTolerance = 1; 24cb93a386Sopenharmony_ci return abs((int)SkColorGetA(x) - (int)SkColorGetA(y)) <= kTolerance && 25cb93a386Sopenharmony_ci abs((int)SkColorGetR(x) - (int)SkColorGetR(y)) <= kTolerance && 26cb93a386Sopenharmony_ci abs((int)SkColorGetG(x) - (int)SkColorGetG(y)) <= kTolerance && 27cb93a386Sopenharmony_ci abs((int)SkColorGetB(x) - (int)SkColorGetB(y)) <= kTolerance; 28cb93a386Sopenharmony_ci} 29cb93a386Sopenharmony_ci 30cb93a386Sopenharmony_cistatic void test_blend(skiatest::Reporter* r, SkSurface* surface) { 31cb93a386Sopenharmony_ci SkBitmap bitmap; 32cb93a386Sopenharmony_ci REPORTER_ASSERT(r, bitmap.tryAllocPixels(surface->imageInfo())); 33cb93a386Sopenharmony_ci 34cb93a386Sopenharmony_ci for (int m = 0; m <= (int)SkBlendMode::kLastMode; ++m) { 35cb93a386Sopenharmony_ci SkBlendMode mode = (SkBlendMode)m; 36cb93a386Sopenharmony_ci for (int alpha : {0x80, 0xFF}) { 37cb93a386Sopenharmony_ci std::vector<SkColor> colors; 38cb93a386Sopenharmony_ci for (bool useRuntimeBlend : {false, true}) { 39cb93a386Sopenharmony_ci // Draw a solid red pixel. 40cb93a386Sopenharmony_ci SkPaint paint; 41cb93a386Sopenharmony_ci paint.setColor(SK_ColorRED); 42cb93a386Sopenharmony_ci paint.setBlendMode(SkBlendMode::kSrc); 43cb93a386Sopenharmony_ci surface->getCanvas()->drawRect(SkRect::MakeWH(1, 1), paint); 44cb93a386Sopenharmony_ci 45cb93a386Sopenharmony_ci // Draw a blue pixel on top of it, using the passed-in blend mode. 46cb93a386Sopenharmony_ci paint.setColor(SkColorSetARGB(alpha, 0x00, 0x00, 0xFF)); 47cb93a386Sopenharmony_ci if (useRuntimeBlend) { 48cb93a386Sopenharmony_ci paint.setBlender(GetRuntimeBlendForBlendMode(mode)); 49cb93a386Sopenharmony_ci } else { 50cb93a386Sopenharmony_ci paint.setBlendMode(mode); 51cb93a386Sopenharmony_ci } 52cb93a386Sopenharmony_ci surface->getCanvas()->drawRect(SkRect::MakeWH(1, 1), paint); 53cb93a386Sopenharmony_ci 54cb93a386Sopenharmony_ci // Read back the red/blue blended pixel. 55cb93a386Sopenharmony_ci REPORTER_ASSERT(r, surface->readPixels(bitmap.info(), bitmap.getPixels(), 56cb93a386Sopenharmony_ci bitmap.rowBytes(), /*srcX=*/0, /*srcY=*/0)); 57cb93a386Sopenharmony_ci colors.push_back(bitmap.getColor(/*x=*/0, /*y=*/0)); 58cb93a386Sopenharmony_ci } 59cb93a386Sopenharmony_ci 60cb93a386Sopenharmony_ci REPORTER_ASSERT(r, nearly_equal(colors[0], colors[1]), 61cb93a386Sopenharmony_ci "Expected: %s %s blend matches. Actual: Built-in " 62cb93a386Sopenharmony_ci "A=%02X R=%02X G=%02X B=%02X, Runtime A=%02X R=%02X G=%02X B=%02X", 63cb93a386Sopenharmony_ci SkBlendMode_Name(mode), 64cb93a386Sopenharmony_ci (alpha == 0xFF) ? "solid" : "transparent", 65cb93a386Sopenharmony_ci SkColorGetA(colors[0]), 66cb93a386Sopenharmony_ci SkColorGetR(colors[0]), 67cb93a386Sopenharmony_ci SkColorGetG(colors[0]), 68cb93a386Sopenharmony_ci SkColorGetB(colors[0]), 69cb93a386Sopenharmony_ci SkColorGetA(colors[1]), 70cb93a386Sopenharmony_ci SkColorGetR(colors[1]), 71cb93a386Sopenharmony_ci SkColorGetG(colors[1]), 72cb93a386Sopenharmony_ci SkColorGetB(colors[1])); 73cb93a386Sopenharmony_ci } 74cb93a386Sopenharmony_ci } 75cb93a386Sopenharmony_ci} 76cb93a386Sopenharmony_ci 77cb93a386Sopenharmony_ciDEF_TEST(SkRuntimeBlender_CPU, r) { 78cb93a386Sopenharmony_ci const SkImageInfo info = SkImageInfo::MakeN32Premul(/*width=*/1, /*height=*/1); 79cb93a386Sopenharmony_ci sk_sp<SkSurface> surface(SkSurface::MakeRaster(info)); 80cb93a386Sopenharmony_ci 81cb93a386Sopenharmony_ci test_blend(r, surface.get()); 82cb93a386Sopenharmony_ci} 83cb93a386Sopenharmony_ci 84cb93a386Sopenharmony_ciDEF_GPUTEST_FOR_RENDERING_CONTEXTS(SkRuntimeBlender_GPU, r, ctxInfo) { 85cb93a386Sopenharmony_ci const SkImageInfo info = SkImageInfo::MakeN32Premul(/*width=*/1, /*height=*/1); 86cb93a386Sopenharmony_ci sk_sp<SkSurface> surface(SkSurface::MakeRenderTarget(ctxInfo.directContext(), 87cb93a386Sopenharmony_ci SkBudgeted::kNo, info)); 88cb93a386Sopenharmony_ci test_blend(r, surface.get()); 89cb93a386Sopenharmony_ci} 90