blob: 05ab1190b7a7a27a628216a2943af3e1ef19546a [file] [log] [blame]
// Copyright 2010 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include <stdint.h>
#include "base/memory/platform_shared_memory_region.h"
#include "build/build_config.h"
#include "skia/ext/platform_canvas.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "ui/gfx/blit.h"
#include "ui/gfx/geometry/point.h"
#include "ui/gfx/geometry/rect.h"
namespace {
// A TestPixelMap is a thin shim layer that allows writing a square canvas of
// 32-bit pixels as a flat array of 8-bit value, which makes the tests a bit
// less fiddly to read and maintain. At construction time, each 8-bit value is
// used to fill in each of the ARGB channels of the 32-bit pixels.
struct TestPixelMap {
static constexpr size_t kWidth = 5;
static constexpr size_t kHeight = 5;
static uint32_t ExpandByte(uint8_t b) {
return (b << 24) | (b << 16) | (b << 8) | b;
}
TestPixelMap(std::initializer_list<uint8_t> values) {
std::transform(values.begin(), values.end(), pixels.begin(),
&TestPixelMap::ExpandByte);
}
uint32_t AtXY(size_t x, size_t y) const { return pixels[y * kWidth + x]; }
std::array<uint32_t, kWidth * kHeight> pixels;
};
// Fills the given canvas with the values by duplicating the values into each
// color channel for the corresponding pixel.
//
// Example values = {{0x0, 0x01}, {0x12, 0xFF}} would give a canvas with:
// 0x00000000 0x01010101
// 0x12121212 0xFFFFFFFF
void SetToCanvas(SkCanvas* canvas, const TestPixelMap& values) {
ASSERT_EQ(TestPixelMap::kHeight,
base::checked_cast<size_t>(canvas->imageInfo().height()));
ASSERT_EQ(TestPixelMap::kWidth,
base::checked_cast<size_t>(canvas->imageInfo().width()));
SkImageInfo info =
SkImageInfo::MakeN32Premul(TestPixelMap::kWidth, TestPixelMap::kHeight);
canvas->writePixels(info, values.pixels.data(), TestPixelMap::kWidth * 4, 0,
0);
}
// Checks each pixel in the given canvas and see if it is made up of the given
// values, where each value has been duplicated into each channel of the given
// bitmap (see SetToCanvas above).
void VerifyCanvasValues(SkCanvas* canvas, const TestPixelMap& values) {
SkBitmap bitmap = skia::ReadPixels(canvas);
ASSERT_EQ(TestPixelMap::kHeight, base::checked_cast<size_t>(bitmap.height()));
ASSERT_EQ(TestPixelMap::kWidth, base::checked_cast<size_t>(bitmap.width()));
for (size_t y = 0; y < TestPixelMap::kHeight; y++) {
for (size_t x = 0; x < TestPixelMap::kWidth; x++) {
ASSERT_EQ(values.AtXY(x, y), *bitmap.getAddr32(x, y));
}
}
}
} // namespace
TEST(Blit, ScrollCanvas) {
const int kCanvasWidth = 5;
const int kCanvasHeight = 5;
std::unique_ptr<SkCanvas> canvas =
skia::CreatePlatformCanvas(kCanvasWidth, kCanvasHeight, false);
const TestPixelMap initial_values({0x00, 0x01, 0x02, 0x03, 0x04, 0x10, 0x11,
0x12, 0x13, 0x14, 0x20, 0x21, 0x22, 0x23,
0x24, 0x30, 0x31, 0x32, 0x33, 0x34, 0x40,
0x41, 0x42, 0x43, 0x44});
SetToCanvas(canvas.get(), initial_values);
VerifyCanvasValues(canvas.get(), initial_values);
// Scroll none and make sure it's a NOP.
gfx::ScrollCanvas(canvas.get(),
gfx::Rect(0, 0, kCanvasWidth, kCanvasHeight),
gfx::Vector2d(0, 0));
VerifyCanvasValues(canvas.get(), initial_values);
// Scroll with a empty clip and make sure it's a NOP.
gfx::Rect empty_clip(1, 1, 0, 0);
gfx::ScrollCanvas(canvas.get(), empty_clip, gfx::Vector2d(0, 1));
VerifyCanvasValues(canvas.get(), initial_values);
// Scroll the center 3 pixels up one.
gfx::Rect center_three(1, 1, 3, 3);
gfx::ScrollCanvas(canvas.get(), center_three, gfx::Vector2d(0, -1));
const TestPixelMap scroll_up_expected(
{0x00, 0x01, 0x02, 0x03, 0x04, 0x10, 0x21, 0x22, 0x23,
0x14, 0x20, 0x31, 0x32, 0x33, 0x24, 0x30, 0x31, 0x32,
0x33, 0x34, 0x40, 0x41, 0x42, 0x43, 0x44});
VerifyCanvasValues(canvas.get(), scroll_up_expected);
// Reset and scroll the center 3 pixels down one.
SetToCanvas(canvas.get(), initial_values);
gfx::ScrollCanvas(canvas.get(), center_three, gfx::Vector2d(0, 1));
const TestPixelMap scroll_down_expected(
{0x00, 0x01, 0x02, 0x03, 0x04, 0x10, 0x11, 0x12, 0x13,
0x14, 0x20, 0x11, 0x12, 0x13, 0x24, 0x30, 0x21, 0x22,
0x23, 0x34, 0x40, 0x41, 0x42, 0x43, 0x44});
VerifyCanvasValues(canvas.get(), scroll_down_expected);
// Reset and scroll the center 3 pixels right one.
SetToCanvas(canvas.get(), initial_values);
gfx::ScrollCanvas(canvas.get(), center_three, gfx::Vector2d(1, 0));
const TestPixelMap scroll_right_expected(
{0x00, 0x01, 0x02, 0x03, 0x04, 0x10, 0x11, 0x11, 0x12,
0x14, 0x20, 0x21, 0x21, 0x22, 0x24, 0x30, 0x31, 0x31,
0x32, 0x34, 0x40, 0x41, 0x42, 0x43, 0x44});
VerifyCanvasValues(canvas.get(), scroll_right_expected);
// Reset and scroll the center 3 pixels left one.
SetToCanvas(canvas.get(), initial_values);
gfx::ScrollCanvas(canvas.get(), center_three, gfx::Vector2d(-1, 0));
const TestPixelMap scroll_left_expected(
{0x00, 0x01, 0x02, 0x03, 0x04, 0x10, 0x12, 0x13, 0x13,
0x14, 0x20, 0x22, 0x23, 0x23, 0x24, 0x30, 0x32, 0x33,
0x33, 0x34, 0x40, 0x41, 0x42, 0x43, 0x44});
VerifyCanvasValues(canvas.get(), scroll_left_expected);
// Diagonal scroll.
SetToCanvas(canvas.get(), initial_values);
gfx::ScrollCanvas(canvas.get(), center_three, gfx::Vector2d(2, 2));
const TestPixelMap scroll_diagonal_expected(
{0x00, 0x01, 0x02, 0x03, 0x04, 0x10, 0x11, 0x12, 0x13,
0x14, 0x20, 0x21, 0x22, 0x23, 0x24, 0x30, 0x31, 0x32,
0x11, 0x34, 0x40, 0x41, 0x42, 0x43, 0x44});
VerifyCanvasValues(canvas.get(), scroll_diagonal_expected);
}
#if BUILDFLAG(IS_WIN)
TEST(Blit, WithSharedMemory) {
const int kCanvasWidth = 5;
const int kCanvasHeight = 5;
base::subtle::PlatformSharedMemoryRegion section =
base::subtle::PlatformSharedMemoryRegion::CreateWritable(kCanvasWidth *
kCanvasHeight);
ASSERT_TRUE(section.IsValid());
std::unique_ptr<SkCanvas> canvas =
skia::CreatePlatformCanvasWithSharedSection(
kCanvasWidth, kCanvasHeight, false, section.GetPlatformHandle(),
skia::RETURN_NULL_ON_FAILURE);
ASSERT_TRUE(canvas);
// Closes a HANDLE associated with |section|, |canvas| must remain valid.
section = base::subtle::PlatformSharedMemoryRegion();
const TestPixelMap initial_values({0x00, 0x01, 0x02, 0x03, 0x04, 0x10, 0x11,
0x12, 0x13, 0x14, 0x20, 0x21, 0x22, 0x23,
0x24, 0x30, 0x31, 0x32, 0x33, 0x34, 0x40,
0x41, 0x42, 0x43, 0x44});
SetToCanvas(canvas.get(), initial_values);
VerifyCanvasValues(canvas.get(), initial_values);
}
#endif