#include "cc/tiles/gpu_image_decode_cache.h"
#include <algorithm>
#include <limits>
#include <map>
#include <memory>
#include <string>
#include <tuple>
#include <vector>
#include "base/command_line.h"
#include "base/compiler_specific.h"
#include "base/containers/contains.h"
#include "base/feature_list.h"
#include "base/memory/raw_ptr.h"
#include "base/test/scoped_feature_list.h"
#include "base/test/simple_test_tick_clock.h"
#include "base/test/test_mock_time_task_runner.h"
#include "cc/base/features.h"
#include "cc/base/switches.h"
#include "cc/paint/color_filter.h"
#include "cc/paint/draw_image.h"
#include "cc/paint/image_transfer_cache_entry.h"
#include "cc/paint/paint_image.h"
#include "cc/paint/paint_image_builder.h"
#include "cc/paint/paint_op_writer.h"
#include "cc/test/fake_paint_image_generator.h"
#include "cc/test/skia_common.h"
#include "cc/test/test_tile_task_runner.h"
#include "cc/test/transfer_cache_test_helper.h"
#include "cc/tiles/raster_dark_mode_filter.h"
#include "components/viz/test/test_context_provider.h"
#include "components/viz/test/test_gles2_interface.h"
#include "components/viz/test/test_raster_interface.h"
#include "gpu/command_buffer/common/command_buffer_id.h"
#include "gpu/command_buffer/common/constants.h"
#include "gpu/config/gpu_finch_features.h"
#include "gpu/skia_bindings/gl_bindings_skia_cmd_buffer.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "third_party/skia/include/core/SkBitmap.h"
#include "third_party/skia/include/core/SkColorFilter.h"
#include "third_party/skia/include/core/SkColorSpace.h"
#include "third_party/skia/include/core/SkImage.h"
#include "third_party/skia/include/core/SkImageGenerator.h"
#include "third_party/skia/include/core/SkImageInfo.h"
#include "third_party/skia/include/core/SkM44.h"
#include "third_party/skia/include/core/SkRect.h"
#include "third_party/skia/include/core/SkRefCnt.h"
#include "third_party/skia/include/core/SkSize.h"
#include "third_party/skia/include/core/SkYUVAPixmaps.h"
#include "third_party/skia/include/effects/SkHighContrastFilter.h"
#include "third_party/skia/include/gpu/GpuTypes.h"
#include "third_party/skia/include/gpu/ganesh/GrBackendSurface.h"
#include "third_party/skia/include/gpu/ganesh/GrDirectContext.h"
#include "third_party/skia/include/gpu/ganesh/SkImageGanesh.h"
#include "third_party/skia/include/gpu/ganesh/gl/GrGLDirectContext.h"
#include "third_party/skia/include/gpu/ganesh/gl/GrGLInterface.h"
using testing::_;
using testing::StrictMock;
namespace cc {
namespace {
class FakeGPUImageDecodeTestGLES2Interface : public viz::TestGLES2Interface,
public viz::TestContextSupport {
public:
explicit FakeGPUImageDecodeTestGLES2Interface(
TransferCacheTestHelper* transfer_cache_helper)
: extension_string_(
"GL_EXT_texture_format_BGRA8888 GL_OES_rgb8_rgba8 "
"GL_OES_texture_npot GL_EXT_texture_rg "
"GL_OES_texture_half_float GL_OES_texture_half_float_linear "
"GL_EXT_texture_norm16"),
transfer_cache_helper_(transfer_cache_helper) {}
~FakeGPUImageDecodeTestGLES2Interface() override {
EXPECT_EQ(0u, NumTextures());
EXPECT_EQ(0u, NumFramebuffers());
EXPECT_EQ(0u, NumRenderbuffers());
}
base::span<uint8_t> MapTransferCacheEntry(uint32_t serialized_size) override {
mapped_entry_size_ = serialized_size;
auto buffer =
PaintOpWriter::AllocateAlignedBuffer<uint8_t>(serialized_size);
mapped_entry_.swap(buffer);
return UNSAFE_TODO(base::span(mapped_entry_.get(), mapped_entry_size_));
}
void UnmapAndCreateTransferCacheEntry(uint32_t type, uint32_t id) override {
transfer_cache_helper_->CreateEntryDirect(
MakeEntryKey(type, id),
UNSAFE_TODO(base::span(mapped_entry_.get(), mapped_entry_size_)));
mapped_entry_ = nullptr;
mapped_entry_size_ = 0;
}
bool ThreadsafeLockTransferCacheEntry(uint32_t type, uint32_t id) override {
return transfer_cache_helper_->LockEntryDirect(MakeEntryKey(type, id));
}
void UnlockTransferCacheEntries(
const std::vector<std::pair<uint32_t, uint32_t>>& entries) override {
std::vector<std::pair<TransferCacheEntryType, uint32_t>> keys;
keys.reserve(entries.size());
for (const auto& e : entries)
keys.emplace_back(MakeEntryKey(e.first, e.second));
transfer_cache_helper_->UnlockEntriesDirect(keys);
}
void DeleteTransferCacheEntry(uint32_t type, uint32_t id) override {
transfer_cache_helper_->DeleteEntryDirect(MakeEntryKey(type, id));
}
std::pair<TransferCacheEntryType, uint32_t> MakeEntryKey(uint32_t type,
uint32_t id) {
DCHECK_LE(type, static_cast<uint32_t>(TransferCacheEntryType::kLast));
return std::make_pair(static_cast<TransferCacheEntryType>(type), id);
}
const GLubyte* GetString(GLenum name) override {
switch (name) {
case GL_EXTENSIONS:
return reinterpret_cast<const GLubyte*>(extension_string_.c_str());
case GL_VERSION:
return reinterpret_cast<const GLubyte*>("4.0 Null GL");
case GL_SHADING_LANGUAGE_VERSION:
return reinterpret_cast<const GLubyte*>("4.20.8 Null GLSL");
case GL_VENDOR:
return reinterpret_cast<const GLubyte*>("Null Vendor");
case GL_RENDERER:
return reinterpret_cast<const GLubyte*>("The Null (Non-)Renderer");
}
return nullptr;
}
void GetIntegerv(GLenum name, GLint* params) override {
switch (name) {
case GL_MAX_TEXTURE_IMAGE_UNITS:
*params = 8;
return;
case GL_MAX_COMBINED_TEXTURE_IMAGE_UNITS:
*params = 8;
return;
case GL_MAX_RENDERBUFFER_SIZE:
*params = 2048;
return;
case GL_MAX_VERTEX_ATTRIBS:
*params = 8;
return;
default:
break;
}
TestGLES2Interface::GetIntegerv(name, params);
}
private:
const std::string extension_string_;
raw_ptr<TransferCacheTestHelper> transfer_cache_helper_;
size_t mapped_entry_size_ = 0;
std::unique_ptr<uint8_t, base::AlignedFreeDeleter> mapped_entry_;
};
class GPUImageDecodeTestMockContextProvider : public viz::TestContextProvider {
public:
static scoped_refptr<GPUImageDecodeTestMockContextProvider> Create(
TransferCacheTestHelper* transfer_cache_helper) {
auto support = std::make_unique<FakeGPUImageDecodeTestGLES2Interface>(
transfer_cache_helper);
auto gl = std::make_unique<FakeGPUImageDecodeTestGLES2Interface>(
transfer_cache_helper);
auto raster = std::make_unique<viz::TestRasterInterface>();
return new GPUImageDecodeTestMockContextProvider(
std::move(support), std::move(gl), std::move(raster));
}
void SetContextCapabilitiesOverride(std::optional<gpu::Capabilities> caps) {
capabilities_override_ = caps;
}
const gpu::Capabilities& ContextCapabilities() const override {
if (capabilities_override_.has_value())
return *capabilities_override_;
return viz::TestContextProvider::ContextCapabilities();
}
gpu::raster::RasterInterface* RasterInterface() override {
return raster_interface_gles_.get();
}
private:
~GPUImageDecodeTestMockContextProvider() override = default;
GPUImageDecodeTestMockContextProvider(
std::unique_ptr<viz::TestContextSupport> support,
std::unique_ptr<viz::TestGLES2Interface> gl,
std::unique_ptr<gpu::raster::RasterInterface> raster)
: TestContextProvider(std::move(support),
std::move(gl),
nullptr ,
true),
raster_interface_gles_(std::move(raster)) {}
std::unique_ptr<gpu::raster::RasterInterface> raster_interface_gles_;
std::optional<gpu::Capabilities> capabilities_override_;
};
class FakeRasterDarkModeFilter : public RasterDarkModeFilter {
public:
FakeRasterDarkModeFilter() {
SkHighContrastConfig config;
config.fInvertStyle = SkHighContrastConfig::InvertStyle::kInvertLightness;
color_filter_ = ColorFilter::MakeHighContrast(config);
}
sk_sp<ColorFilter> ApplyToImage(const SkPixmap& pixmap,
const SkIRect& src) const override {
return color_filter_;
}
const sk_sp<ColorFilter> GetFilter() const { return color_filter_; }
private:
sk_sp<ColorFilter> color_filter_;
};
SkM44 CreateMatrix(const SkSize& scale) {
return SkM44::Scale(scale.width(), scale.height());
}
constexpr size_t kGpuMemoryLimitBytes = 96 * 1024 * 1024;
class GpuImageDecodeCacheTest
: public ::testing::TestWithParam<
std::tuple<SkColorType,
bool /* do_yuv_decode */,
bool /* enable_clipped_image_scaling */,
bool /* no_discardable_memory */>> {
public:
void SetUp() override {
std::vector<base::test::FeatureRef> enabled_features;
no_discardable_memory_ = std::get<3>(GetParam());
if (no_discardable_memory_)
enabled_features.push_back(
features::kNoDiscardableMemoryForGpuDecodePath);
feature_list_.InitWithFeatures(enabled_features,
{} );
enable_clipped_image_scaling_ = std::get<2>(GetParam());
if (enable_clipped_image_scaling_) {
auto* command_line = base::CommandLine::ForCurrentProcess();
ASSERT_TRUE(command_line != nullptr);
command_line->AppendSwitch(switches::kEnableClippedImageScaling);
}
context_provider_ =
GPUImageDecodeTestMockContextProvider::Create(&transfer_cache_helper_);
context_provider_->BindToCurrentSequence();
sk_sp<const GrGLInterface> gr_interface =
skia_bindings::CreateGLES2InterfaceBindings(
context_provider_->UnboundTestContextGL(),
context_provider_->ContextSupport());
gr_context_ = GrDirectContexts::MakeGL(std::move(gr_interface));
ASSERT_TRUE(!!gr_context_);
{
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider_.get());
transfer_cache_helper_.SetGrContext(gr_context_.get());
max_texture_size_ =
context_provider_->ContextCapabilities().max_texture_size;
}
color_type_ = std::get<0>(GetParam());
do_yuv_decode_ = std::get<1>(GetParam());
}
void TearDown() override {
transfer_cache_helper_.SetGrContext(nullptr);
}
std::unique_ptr<GpuImageDecodeCache> CreateCache(
size_t memory_limit_bytes = kGpuMemoryLimitBytes,
RasterDarkModeFilter* const dark_mode_filter = nullptr) {
return std::make_unique<GpuImageDecodeCache>(
context_provider_.get(), color_type_, memory_limit_bytes,
max_texture_size_, dark_mode_filter);
}
gfx::Size GetLargeImageSize() const {
return gfx::Size(1, max_texture_size_ + 1);
}
gfx::Size GetNormalImageSize() const {
int dimension = std::min(100, max_texture_size_ - 1);
return gfx::Size(dimension, dimension);
}
PaintImage CreatePaintImageInternal(
const gfx::Size& size,
sk_sp<SkColorSpace> color_space = nullptr,
PaintImage::Id id = PaintImage::kInvalidId) {
const bool allocate_encoded_memory = true;
if (do_yuv_decode_) {
return CreateDiscardablePaintImage(
size, color_space, allocate_encoded_memory, id, color_type_,
yuv_format_, yuv_data_type_);
}
return CreateDiscardablePaintImage(
size, color_space, allocate_encoded_memory, id, color_type_);
}
sk_sp<FakePaintImageGenerator> CreateFakePaintImageGenerator(
const gfx::Size& size) {
constexpr bool allocate_encoded_memory = true;
SkImageInfo info =
SkImageInfo::Make(size.width(), size.height(), color_type_,
kPremul_SkAlphaType, SkColorSpace::MakeSRGB());
if (do_yuv_decode_) {
SkYUVAPixmapInfo yuva_pixmap_info =
GetYUVAPixmapInfo(size, yuv_format_, yuv_data_type_);
return sk_make_sp<FakePaintImageGenerator>(
info, yuva_pixmap_info, std::vector<FrameMetadata>{FrameMetadata()},
allocate_encoded_memory);
} else {
return sk_make_sp<FakePaintImageGenerator>(
info, std::vector<FrameMetadata>{FrameMetadata()},
allocate_encoded_memory);
}
}
PaintImage CreateLargePaintImageForSoftwareFallback(
sk_sp<SkColorSpace> image_color_space = SkColorSpace::MakeSRGB()) {
return CreatePaintImageForFallbackToRGB(GetLargeImageSize(),
image_color_space);
}
PaintImage CreatePaintImageForFallbackToRGB(
const gfx::Size test_image_size,
sk_sp<SkColorSpace> image_color_space = SkColorSpace::MakeSRGB()) {
SkImageInfo info =
SkImageInfo::Make(test_image_size.width(), test_image_size.height(),
color_type_, kPremul_SkAlphaType, image_color_space);
sk_sp<FakePaintImageGenerator> generator;
if (do_yuv_decode_) {
SkYUVAPixmapInfo yuva_pixmap_info =
GetYUVAPixmapInfo(test_image_size, yuv_format_, yuv_data_type_);
generator = sk_make_sp<FakePaintImageGenerator>(info, yuva_pixmap_info);
generator->SetExpectFallbackToRGB();
} else {
generator = sk_make_sp<FakePaintImageGenerator>(info);
}
PaintImage image = PaintImageBuilder::WithDefault()
.set_id(PaintImage::GetNextId())
.set_paint_image_generator(generator)
.TakePaintImage();
return image;
}
PaintImage CreateBitmapImageInternal(const gfx::Size& size) {
return CreateBitmapImage(size, color_type_);
}
gfx::ColorSpace DefaultColorSpace() {
if (color_type_ != kRGBA_F16_SkColorType)
return gfx::ColorSpace::CreateSRGB();
return gfx::ColorSpace(gfx::ColorSpace::PrimaryID::P3,
gfx::ColorSpace::TransferID::LINEAR);
}
TargetColorParams DefaultTargetColorParams() {
return TargetColorParams(DefaultColorSpace());
}
DrawImage CreateDrawImageInternal(
const PaintImage& paint_image,
const SkM44& matrix = SkM44(),
gfx::ColorSpace* color_space = nullptr,
PaintFlags::FilterQuality filter_quality =
PaintFlags::FilterQuality::kMedium,
SkIRect* src_rect = nullptr,
size_t frame_index = PaintImage::kDefaultFrameIndex,
float sdr_white_level = gfx::ColorSpace::kDefaultSDRWhiteLevel,
bool use_dark_mode = false) {
SkIRect src_rectangle;
if (!src_rect) {
src_rectangle =
SkIRect::MakeWH(paint_image.width(), paint_image.height());
src_rect = &src_rectangle;
}
TargetColorParams target_color_params = DefaultTargetColorParams();
if (color_space)
target_color_params.color_space = *color_space;
return DrawImage(paint_image, use_dark_mode, *src_rect, filter_quality,
matrix, frame_index, target_color_params);
}
DrawImage CreateDrawImageWithDarkModeInternal(
const PaintImage& paint_image,
const SkM44& matrix = SkM44(),
gfx::ColorSpace* color_space = nullptr,
PaintFlags::FilterQuality filter_quality =
PaintFlags::FilterQuality::kMedium,
SkIRect* src_rect = nullptr,
size_t frame_index = PaintImage::kDefaultFrameIndex,
float sdr_white_level = gfx::ColorSpace::kDefaultSDRWhiteLevel) {
return CreateDrawImageInternal(paint_image, matrix, color_space,
filter_quality, src_rect, frame_index,
sdr_white_level, true);
}
void GetImageAndDrawFinishedForDarkMode(
GpuImageDecodeCache* cache,
const DrawImage& draw_image,
FakeRasterDarkModeFilter* dark_mode_filter) {
DCHECK(cache);
DCHECK(dark_mode_filter);
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
cache->GetDecodedImageForDraw(draw_image);
EXPECT_EQ(decoded_draw_image.dark_mode_color_filter(),
dark_mode_filter->GetFilter());
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
}
GPUImageDecodeTestMockContextProvider* context_provider() {
return context_provider_.get();
}
size_t GetBytesNeededForSingleImage(gfx::Size image_dimensions) {
if (do_yuv_decode_) {
SkYUVAPixmapInfo yuva_pixmap_info =
GetYUVAPixmapInfo(image_dimensions, yuv_format_, yuv_data_type_);
return yuva_pixmap_info.computeTotalBytes();
}
const size_t test_image_area_bytes =
base::checked_cast<size_t>(image_dimensions.GetArea());
base::CheckedNumeric<size_t> bytes_for_rgb_image_safe(
test_image_area_bytes);
bytes_for_rgb_image_safe *= SkColorTypeBytesPerPixel(color_type_);
return bytes_for_rgb_image_safe.ValueOrDie();
}
void SetCachedTexturesLimit(size_t limit) {
transfer_cache_helper_.SetCachedItemsLimit(limit);
}
DecodedDrawImage EnsureImageBacked(DecodedDrawImage&& draw_image) {
if (draw_image.transfer_cache_entry_id()) {
auto* image_entry =
transfer_cache_helper_.GetEntryAs<ServiceImageTransferCacheEntry>(
*draw_image.transfer_cache_entry_id());
if (draw_image.transfer_cache_entry_needs_mips())
image_entry->EnsureMips();
DecodedDrawImage new_draw_image(
image_entry->image(), draw_image.dark_mode_color_filter(),
draw_image.src_rect_offset(), draw_image.scale_adjustment(),
draw_image.filter_quality(), draw_image.is_budgeted());
return new_draw_image;
}
return std::move(draw_image);
}
ServiceImageTransferCacheEntry* GetLastTransferredCacheEntry() {
auto& key = transfer_cache_helper_.GetLastAddedEntry();
ServiceTransferCacheEntry* entry =
transfer_cache_helper_.GetEntryInternal(key.first, key.second);
if (!entry)
return nullptr;
CHECK_EQ(TransferCacheEntryType::kImage, entry->Type());
return static_cast<ServiceImageTransferCacheEntry*>(entry);
}
sk_sp<SkImage> GetLastTransferredImage() {
auto* entry = GetLastTransferredCacheEntry();
return entry ? entry->image() : nullptr;
}
void CompareAllPlanesToMippedVersions(
GpuImageDecodeCache* cache,
const DrawImage& draw_image,
const std::optional<uint32_t> transfer_cache_id,
bool should_have_mips) {
for (size_t i = 0; i < kNumYUVPlanes; ++i) {
sk_sp<SkImage> original_uploaded_plane;
DCHECK(transfer_cache_id.has_value());
const uint32_t id = transfer_cache_id.value();
auto* image_entry =
transfer_cache_helper_.GetEntryAs<ServiceImageTransferCacheEntry>(id);
original_uploaded_plane = image_entry->GetPlaneImage(i);
ASSERT_TRUE(original_uploaded_plane);
if (original_uploaded_plane->colorType() == kA16_unorm_SkColorType ||
original_uploaded_plane->colorType() == kA16_float_SkColorType) {
break;
}
EXPECT_EQ(original_uploaded_plane->hasMipmaps(), should_have_mips);
}
}
void VerifyUploadedPlaneSizes(
GpuImageDecodeCache* cache,
const DrawImage& draw_image,
const std::optional<uint32_t> transfer_cache_id,
const SkISize plane_sizes[SkYUVAInfo::kMaxPlanes],
SkYUVAPixmapInfo::DataType expected_type =
SkYUVAPixmapInfo::DataType::kUnorm8,
const SkColorSpace* expected_cs = nullptr) {
SkColorType expected_color_type =
SkYUVAPixmapInfo::DefaultColorTypeForDataType(expected_type, 1);
for (size_t i = 0; i < kNumYUVPlanes; ++i) {
sk_sp<SkImage> uploaded_plane;
DCHECK(transfer_cache_id.has_value());
const uint32_t id = transfer_cache_id.value();
auto* image_entry =
transfer_cache_helper_.GetEntryAs<ServiceImageTransferCacheEntry>(id);
uploaded_plane = image_entry->GetPlaneImage(i);
ASSERT_TRUE(uploaded_plane);
UNSAFE_TODO(EXPECT_EQ(plane_sizes[i], uploaded_plane->dimensions()));
EXPECT_EQ(expected_color_type, uploaded_plane->colorType());
if (expected_cs) {
EXPECT_TRUE(
SkColorSpace::Equals(expected_cs, uploaded_plane->colorSpace()));
}
}
}
protected:
base::test::ScopedFeatureList feature_list_;
TransferCacheTestHelper transfer_cache_helper_;
scoped_refptr<GPUImageDecodeTestMockContextProvider> context_provider_;
sk_sp<GrDirectContext> gr_context_;
SkYUVAPixmapInfo::DataType yuv_data_type_ =
SkYUVAPixmapInfo::DataType::kUnorm8;
YUVSubsampling yuv_format_ = YUVSubsampling::k420;
SkColorType color_type_;
bool do_yuv_decode_;
bool enable_clipped_image_scaling_;
bool no_discardable_memory_;
int max_texture_size_ = 0;
};
TEST_P(GpuImageDecodeCacheTest, GetTaskForImageSameImage) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(1.5f, 1.5f)));
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
DrawImage another_draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(1.5f, 1.5f)));
ImageDecodeCache::TaskResult another_result = cache->GetTaskForImageAndRef(
client_id, another_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(another_result.need_unref);
EXPECT_TRUE(result.task.get() == another_result.task.get());
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
cache->UnrefImage(draw_image);
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest, GetRasterTaskBeforeStandAloneTaskSameImage) {
if (!base::FeatureList::IsEnabled(features::kPreventDuplicateImageDecodes)) {
return;
}
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(1.5f, 1.5f)));
ImageDecodeCache::TaskResult raster_result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(raster_result.need_unref);
EXPECT_TRUE(raster_result.task);
EXPECT_EQ(raster_result.task->dependencies().size(), 1u);
TileTask* raster_decode_task = raster_result.task->dependencies()[0].get();
ImageDecodeCache::TaskResult stand_alone_result =
cache->GetOutOfRasterDecodeTaskForImageAndRef(client_id, draw_image,
false);
EXPECT_TRUE(stand_alone_result.need_unref);
EXPECT_EQ(stand_alone_result.task->dependencies().size(), 1u);
EXPECT_EQ(stand_alone_result.task->dependencies()[0].get(),
raster_decode_task);
EXPECT_EQ(raster_decode_task->external_dependent().get(),
stand_alone_result.task.get());
TestTileTaskRunner::ProcessTask(raster_decode_task);
EXPECT_TRUE(stand_alone_result.task->dependencies().empty());
TestTileTaskRunner::ProcessTask(raster_result.task.get());
TestTileTaskRunner::ProcessTask(stand_alone_result.task.get());
cache->UnrefImage(draw_image);
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest, GetStandAloneTaskBeforeRasterTaskSameImage) {
if (!base::FeatureList::IsEnabled(features::kPreventDuplicateImageDecodes)) {
return;
}
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(1.5f, 1.5f)));
ImageDecodeCache::TaskResult stand_alone_result =
cache->GetOutOfRasterDecodeTaskForImageAndRef(client_id, draw_image,
false);
EXPECT_TRUE(stand_alone_result.need_unref);
EXPECT_TRUE(stand_alone_result.task);
ImageDecodeCache::TaskResult raster_result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(raster_result.need_unref);
EXPECT_TRUE(raster_result.task);
EXPECT_EQ(raster_result.task->dependencies().size(), 1u);
TileTask* raster_decode_task = raster_result.task->dependencies()[0].get();
EXPECT_EQ(stand_alone_result.task->dependencies().size(), 1u);
EXPECT_EQ(stand_alone_result.task->dependencies()[0].get(),
raster_decode_task);
EXPECT_EQ(raster_decode_task->external_dependent().get(),
stand_alone_result.task.get());
TestTileTaskRunner::ProcessTask(raster_decode_task);
EXPECT_TRUE(stand_alone_result.task->dependencies().empty());
TestTileTaskRunner::ProcessTask(raster_result.task.get());
TestTileTaskRunner::ProcessTask(stand_alone_result.task.get());
cache->UnrefImage(draw_image);
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest,
StandAloneTaskStartedBeforeRasterTaskSameImage) {
if (!base::FeatureList::IsEnabled(features::kPreventDuplicateImageDecodes)) {
return;
}
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(1.5f, 1.5f)));
ImageDecodeCache::TaskResult stand_alone_result =
cache->GetOutOfRasterDecodeTaskForImageAndRef(client_id, draw_image,
false);
EXPECT_TRUE(stand_alone_result.need_unref);
EXPECT_TRUE(stand_alone_result.task);
TileTask* stand_alone_decode_task = stand_alone_result.task.get();
stand_alone_decode_task->state().DidSchedule();
stand_alone_decode_task->state().DidStart();
ImageDecodeCache::TaskResult raster_result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(raster_result.need_unref);
EXPECT_TRUE(raster_result.task);
EXPECT_EQ(raster_result.task->dependencies().size(), 1u);
TileTask* raster_decode_task = raster_result.task->dependencies()[0].get();
EXPECT_EQ(raster_decode_task->dependencies().size(), 1u);
EXPECT_EQ(raster_decode_task->dependencies()[0].get(),
stand_alone_decode_task);
EXPECT_EQ(stand_alone_decode_task->external_dependent().get(),
raster_decode_task);
stand_alone_decode_task->RunOnWorkerThread();
stand_alone_decode_task->state().DidFinish();
TestTileTaskRunner::CompleteTask(stand_alone_decode_task);
EXPECT_TRUE(raster_decode_task->dependencies().empty());
TestTileTaskRunner::ProcessTask(raster_decode_task);
TestTileTaskRunner::ProcessTask(raster_result.task.get());
cache->UnrefImage(draw_image);
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest, ExternalDependentRasterTaskCanceled) {
if (!base::FeatureList::IsEnabled(features::kPreventDuplicateImageDecodes)) {
return;
}
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(1.5f, 1.5f)));
ImageDecodeCache::TaskResult stand_alone_result =
cache->GetOutOfRasterDecodeTaskForImageAndRef(client_id, draw_image,
false);
EXPECT_TRUE(stand_alone_result.need_unref);
EXPECT_TRUE(stand_alone_result.task);
TileTask* stand_alone_decode_task = stand_alone_result.task.get();
stand_alone_decode_task->state().DidSchedule();
stand_alone_decode_task->state().DidStart();
ImageDecodeCache::TaskResult raster_result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(raster_result.need_unref);
EXPECT_TRUE(raster_result.task);
EXPECT_EQ(raster_result.task->dependencies().size(), 1u);
TileTask* raster_upload_task = raster_result.task.get();
TileTask* raster_decode_task = raster_upload_task->dependencies()[0].get();
EXPECT_EQ(raster_decode_task->dependencies().size(), 1u);
EXPECT_EQ(raster_decode_task->dependencies()[0].get(),
stand_alone_decode_task);
EXPECT_EQ(stand_alone_decode_task->external_dependent().get(),
raster_decode_task);
TestTileTaskRunner::CancelTask(raster_decode_task);
TestTileTaskRunner::CancelTask(raster_upload_task);
TestTileTaskRunner::CompleteTask(raster_decode_task);
TestTileTaskRunner::CompleteTask(raster_upload_task);
raster_result = cache->GetTaskForImageAndRef(client_id, draw_image,
ImageDecodeCache::TracingInfo());
EXPECT_TRUE(raster_result.need_unref);
EXPECT_TRUE(raster_result.task);
EXPECT_NE(raster_result.task.get(), raster_upload_task);
raster_upload_task = raster_result.task.get();
EXPECT_EQ(raster_upload_task->dependencies().size(), 1u);
EXPECT_NE(raster_upload_task->dependencies()[0].get(), raster_decode_task);
raster_decode_task = raster_upload_task->dependencies()[0].get();
EXPECT_EQ(raster_decode_task->dependencies().size(), 1u);
EXPECT_EQ(raster_decode_task->dependencies()[0].get(),
stand_alone_decode_task);
EXPECT_EQ(stand_alone_decode_task->external_dependent().get(),
raster_decode_task);
stand_alone_decode_task->RunOnWorkerThread();
stand_alone_decode_task->state().DidFinish();
TestTileTaskRunner::CompleteTask(stand_alone_decode_task);
EXPECT_TRUE(raster_decode_task->dependencies().empty());
TestTileTaskRunner::ProcessTask(raster_decode_task);
TestTileTaskRunner::ProcessTask(raster_upload_task);
cache->UnrefImage(draw_image);
cache->UnrefImage(draw_image);
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest, GetTaskForImageSameImageDifferentClients) {
auto cache = CreateCache();
const uint32_t kClientId1 = cache->GenerateClientId();
const uint32_t kClientId2 = cache->GenerateClientId();
for (size_t order = 1; order <= 4; ++order) {
sk_sp<FakePaintImageGenerator> generator =
CreateFakePaintImageGenerator(GetNormalImageSize());
PaintImage image =
PaintImageBuilder::WithDefault()
.set_id(PaintImage::GetNextId())
.set_paint_image_generator(generator)
.set_decoding_mode(PaintImage::DecodingMode::kUnspecified)
.TakePaintImage();
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(1.5f, 1.5f)));
EXPECT_EQ(draw_image.frame_index(), PaintImage::kDefaultFrameIndex);
ImageDecodeCache::TaskResult result1 = cache->GetTaskForImageAndRef(
kClientId1, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result1.need_unref);
EXPECT_TRUE(result1.task);
ImageDecodeCache::TaskResult result2 = cache->GetTaskForImageAndRef(
kClientId2, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result2.need_unref);
EXPECT_TRUE(result2.task);
EXPECT_NE(result1.task, result2.task);
DrawImage draw_image2 =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(1.5f, 1.5f)));
EXPECT_EQ(draw_image2.frame_index(), PaintImage::kDefaultFrameIndex);
ImageDecodeCache::TaskResult another_result = cache->GetTaskForImageAndRef(
kClientId1, draw_image2, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(another_result.need_unref);
EXPECT_TRUE(result1.task.get() == another_result.task.get());
DrawImage draw_image3 =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(1.5f, 1.5f)));
EXPECT_EQ(draw_image3.frame_index(), PaintImage::kDefaultFrameIndex);
ImageDecodeCache::TaskResult another_result2 = cache->GetTaskForImageAndRef(
kClientId2, draw_image3, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(another_result2.need_unref);
EXPECT_TRUE(result2.task.get() == another_result2.task.get());
testing::InSequence s;
if (order == 1u) {
TestTileTaskRunner::ProcessTask(result1.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result1.task.get());
TestTileTaskRunner::ProcessTask(result2.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result2.task.get());
} else if (order == 2u) {
TestTileTaskRunner::ProcessTask(result2.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result1.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result1.task.get());
TestTileTaskRunner::ProcessTask(result2.task.get());
} else if (order == 3u) {
TestTileTaskRunner::ProcessTask(result2.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result1.task.get());
TestTileTaskRunner::ProcessTask(result1.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result2.task.get());
} else {
DCHECK_EQ(order, 4u);
TestTileTaskRunner::ProcessTask(result2.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result2.task.get());
TestTileTaskRunner::ProcessTask(result1.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result1.task.get());
}
EXPECT_EQ(generator->frames_decoded().size(), 1u);
EXPECT_EQ(generator->frames_decoded().count(PaintImage::kDefaultFrameIndex),
1u);
EXPECT_EQ(transfer_cache_helper_.num_of_entries(), 1u);
EXPECT_TRUE(cache->IsInInUseCacheForTesting(draw_image));
EXPECT_TRUE(cache->IsInInUseCacheForTesting(draw_image2));
EXPECT_TRUE(cache->IsInInUseCacheForTesting(draw_image3));
cache->UnrefImage(draw_image);
EXPECT_TRUE(cache->IsInInUseCacheForTesting(draw_image));
cache->UnrefImage(draw_image);
EXPECT_TRUE(cache->IsInInUseCacheForTesting(draw_image));
cache->UnrefImage(draw_image);
EXPECT_TRUE(cache->IsInInUseCacheForTesting(draw_image));
cache->UnrefImage(draw_image);
EXPECT_FALSE(cache->IsInInUseCacheForTesting(draw_image));
EXPECT_FALSE(cache->IsInInUseCacheForTesting(draw_image2));
EXPECT_FALSE(cache->IsInInUseCacheForTesting(draw_image3));
cache->ClearCache();
}
}
TEST_P(GpuImageDecodeCacheTest, DoesNotCreateATaskForAlreadyUploadedImage) {
auto cache = CreateCache();
const uint32_t kClientId1 = cache->GenerateClientId();
const uint32_t kClientId2 = cache->GenerateClientId();
sk_sp<FakePaintImageGenerator> generator =
CreateFakePaintImageGenerator(GetNormalImageSize());
PaintImage image =
PaintImageBuilder::WithDefault()
.set_id(PaintImage::GetNextId())
.set_paint_image_generator(generator)
.set_decoding_mode(PaintImage::DecodingMode::kUnspecified)
.TakePaintImage();
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(1.5f, 1.5f)));
EXPECT_EQ(draw_image.frame_index(), PaintImage::kDefaultFrameIndex);
ImageDecodeCache::TaskResult result1 = cache->GetTaskForImageAndRef(
kClientId1, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result1.need_unref);
EXPECT_TRUE(result1.task);
TestTileTaskRunner::ProcessTask(result1.task->dependencies()[0].get());
TestTileTaskRunner::ScheduleTask(result1.task.get());
TestTileTaskRunner::RunTask(result1.task.get());
DrawImage another_draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(1.5f, 1.5f)));
EXPECT_EQ(another_draw_image.frame_index(), PaintImage::kDefaultFrameIndex);
ImageDecodeCache::TaskResult result2 = cache->GetTaskForImageAndRef(
kClientId2, another_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result2.need_unref);
EXPECT_FALSE(result2.task);
TestTileTaskRunner::CompleteTask(result1.task.get());
cache->UnrefImage(draw_image);
cache->UnrefImage(another_draw_image);
cache->ClearCache();
}
TEST_P(GpuImageDecodeCacheTest, DoesNotCreateATaskForAlreadyUploadedImage2) {
auto cache = CreateCache();
const uint32_t kClientId1 = cache->GenerateClientId();
sk_sp<FakePaintImageGenerator> generator =
CreateFakePaintImageGenerator(GetNormalImageSize());
PaintImage image =
PaintImageBuilder::WithDefault()
.set_id(PaintImage::GetNextId())
.set_paint_image_generator(generator)
.set_decoding_mode(PaintImage::DecodingMode::kUnspecified)
.TakePaintImage();
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(1.5f, 1.5f)));
EXPECT_EQ(draw_image.frame_index(), PaintImage::kDefaultFrameIndex);
ImageDecodeCache::TaskResult result1 = cache->GetTaskForImageAndRef(
kClientId1, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result1.need_unref);
EXPECT_TRUE(result1.task);
EXPECT_EQ(result1.task->dependencies().size(), 1u);
DrawImage another_draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(1.5f, 1.5f)));
EXPECT_EQ(another_draw_image.frame_index(), PaintImage::kDefaultFrameIndex);
ImageDecodeCache::TaskResult result2 =
cache->GetOutOfRasterDecodeTaskForImageAndRef(
kClientId1, another_draw_image, false);
EXPECT_TRUE(result2.need_unref);
EXPECT_TRUE(result2.task);
if (base::FeatureList::IsEnabled(features::kPreventDuplicateImageDecodes)) {
EXPECT_EQ(result2.task->dependencies().size(), 1u);
EXPECT_EQ(result1.task->dependencies()[0]->external_dependent().get(),
result2.task.get());
}
TestTileTaskRunner::ProcessTask(result1.task->dependencies()[0].get());
TestTileTaskRunner::ScheduleTask(result1.task.get());
TestTileTaskRunner::RunTask(result1.task.get());
DrawImage yet_another_draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(1.5f, 1.5f)));
EXPECT_EQ(yet_another_draw_image.frame_index(),
PaintImage::kDefaultFrameIndex);
ImageDecodeCache::TaskResult result3 =
cache->GetOutOfRasterDecodeTaskForImageAndRef(
kClientId1, yet_another_draw_image, false);
EXPECT_TRUE(result3.need_unref);
EXPECT_FALSE(result3.task);
TestTileTaskRunner::CompleteTask(result1.task.get());
TestTileTaskRunner::ProcessTask(result2.task.get());
cache->UnrefImage(draw_image);
cache->UnrefImage(another_draw_image);
cache->UnrefImage(yet_another_draw_image);
cache->ClearCache();
}
TEST_P(GpuImageDecodeCacheTest, GetTaskForImageSmallerScale) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(1.5f, 1.5f)));
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
EXPECT_EQ(result.task->dependencies().size(), 1u);
EXPECT_TRUE(result.task->dependencies()[0]);
DrawImage another_draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(0.5f, 0.5f)));
ImageDecodeCache::TaskResult another_result = cache->GetTaskForImageAndRef(
client_id, another_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_EQ(another_result.task->dependencies().size(), 1u);
EXPECT_TRUE(another_result.task->dependencies()[0]);
EXPECT_TRUE(another_result.need_unref);
EXPECT_TRUE(result.task.get() == another_result.task.get());
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
cache->UnrefImage(draw_image);
cache->UnrefImage(another_draw_image);
}
TEST_P(GpuImageDecodeCacheTest, GetTaskForImageLowerQuality) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
SkM44 matrix = CreateMatrix(SkSize::Make(0.4f, 0.4f));
DrawImage draw_image = CreateDrawImageInternal(image, matrix);
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
DrawImage another_draw_image =
CreateDrawImageInternal(image, matrix, nullptr ,
PaintFlags::FilterQuality::kLow);
ImageDecodeCache::TaskResult another_result = cache->GetTaskForImageAndRef(
client_id, another_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(another_result.need_unref);
EXPECT_TRUE(result.task.get() == another_result.task.get());
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
cache->UnrefImage(draw_image);
cache->UnrefImage(another_draw_image);
}
TEST_P(GpuImageDecodeCacheTest, GetTaskForImageDifferentImage) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage first_image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage first_draw_image = CreateDrawImageInternal(
first_image, CreateMatrix(SkSize::Make(0.5f, 0.5f)));
ImageDecodeCache::TaskResult first_result = cache->GetTaskForImageAndRef(
client_id, first_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(first_result.need_unref);
EXPECT_TRUE(first_result.task);
PaintImage second_image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage second_draw_image = CreateDrawImageInternal(
second_image, CreateMatrix(SkSize::Make(0.25f, 0.25f)));
ImageDecodeCache::TaskResult second_result = cache->GetTaskForImageAndRef(
client_id, second_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(second_result.need_unref);
EXPECT_TRUE(second_result.task);
EXPECT_TRUE(first_result.task.get() != second_result.task.get());
TestTileTaskRunner::ProcessTask(first_result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(first_result.task.get());
TestTileTaskRunner::ProcessTask(second_result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(second_result.task.get());
cache->UnrefImage(first_draw_image);
cache->UnrefImage(second_draw_image);
}
TEST_P(GpuImageDecodeCacheTest, GetTaskForImageLargerScale) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage first_image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage first_draw_image = CreateDrawImageInternal(
first_image, CreateMatrix(SkSize::Make(0.5f, 0.5f)));
ImageDecodeCache::TaskResult first_result = cache->GetTaskForImageAndRef(
client_id, first_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(first_result.need_unref);
EXPECT_TRUE(first_result.task);
TestTileTaskRunner::ProcessTask(first_result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(first_result.task.get());
cache->UnrefImage(first_draw_image);
DrawImage second_draw_image = CreateDrawImageInternal(first_image);
ImageDecodeCache::TaskResult second_result = cache->GetTaskForImageAndRef(
client_id, second_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(second_result.need_unref);
EXPECT_TRUE(second_result.task);
EXPECT_TRUE(first_result.task.get() != second_result.task.get());
DrawImage third_draw_image = CreateDrawImageInternal(
first_image, CreateMatrix(SkSize::Make(0.5f, 0.5f)));
ImageDecodeCache::TaskResult third_result = cache->GetTaskForImageAndRef(
client_id, third_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(third_result.need_unref);
EXPECT_TRUE(third_result.task.get() == second_result.task.get());
TestTileTaskRunner::ProcessTask(second_result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(second_result.task.get());
cache->UnrefImage(second_draw_image);
cache->UnrefImage(third_draw_image);
}
TEST_P(GpuImageDecodeCacheTest, GetTaskForImageLargerScaleNoReuse) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage first_image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage first_draw_image = CreateDrawImageInternal(
first_image, CreateMatrix(SkSize::Make(0.5f, 0.5f)));
ImageDecodeCache::TaskResult first_result = cache->GetTaskForImageAndRef(
client_id, first_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(first_result.need_unref);
EXPECT_TRUE(first_result.task);
DrawImage second_draw_image = CreateDrawImageInternal(first_image);
ImageDecodeCache::TaskResult second_result = cache->GetTaskForImageAndRef(
client_id, second_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(second_result.need_unref);
EXPECT_TRUE(second_result.task);
EXPECT_TRUE(first_result.task.get() != second_result.task.get());
DrawImage third_draw_image = CreateDrawImageInternal(
first_image, CreateMatrix(SkSize::Make(0.5f, 0.5f)));
ImageDecodeCache::TaskResult third_result = cache->GetTaskForImageAndRef(
client_id, third_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(third_result.need_unref);
EXPECT_TRUE(third_result.task.get() == first_result.task.get());
TestTileTaskRunner::ProcessTask(first_result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(first_result.task.get());
TestTileTaskRunner::ProcessTask(second_result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(second_result.task.get());
cache->UnrefImage(first_draw_image);
cache->UnrefImage(second_draw_image);
cache->UnrefImage(third_draw_image);
}
TEST_P(GpuImageDecodeCacheTest, GetTaskForImageHigherQuality) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
SkM44 matrix = CreateMatrix(SkSize::Make(0.4f, 0.4f));
PaintImage first_image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage first_draw_image =
CreateDrawImageInternal(first_image, matrix, nullptr ,
PaintFlags::FilterQuality::kLow);
ImageDecodeCache::TaskResult first_result = cache->GetTaskForImageAndRef(
client_id, first_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(first_result.need_unref);
EXPECT_TRUE(first_result.task);
TestTileTaskRunner::ProcessTask(first_result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(first_result.task.get());
cache->UnrefImage(first_draw_image);
DrawImage second_draw_image =
CreateDrawImageInternal(first_image, matrix, nullptr ,
PaintFlags::FilterQuality::kMedium);
ImageDecodeCache::TaskResult second_result = cache->GetTaskForImageAndRef(
client_id, second_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(second_result.need_unref);
EXPECT_TRUE(second_result.task);
EXPECT_TRUE(first_result.task.get() != second_result.task.get());
TestTileTaskRunner::ProcessTask(second_result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(second_result.task.get());
cache->UnrefImage(second_draw_image);
}
TEST_P(GpuImageDecodeCacheTest, GetTaskForImageAlreadyDecodedAndLocked) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(0.5f, 0.5f)));
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
EXPECT_EQ(result.task->dependencies().size(), 1u);
EXPECT_TRUE(result.task->dependencies()[0]);
TestTileTaskRunner::ScheduleTask(result.task->dependencies()[0].get());
TestTileTaskRunner::RunTask(result.task->dependencies()[0].get());
TestTileTaskRunner::CancelTask(result.task.get());
TestTileTaskRunner::CompleteTask(result.task.get());
ImageDecodeCache::TaskResult another_result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(another_result.need_unref);
EXPECT_TRUE(another_result.task);
EXPECT_EQ(another_result.task->dependencies().size(), 0u);
TestTileTaskRunner::ProcessTask(another_result.task.get());
TestTileTaskRunner::CompleteTask(result.task->dependencies()[0].get());
cache->UnrefImage(draw_image);
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest, GetTaskForImageAlreadyDecodedNotLocked) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(0.5f, 0.5f)));
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
EXPECT_EQ(result.task->dependencies().size(), 1u);
EXPECT_TRUE(result.task->dependencies()[0]);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::CancelTask(result.task.get());
TestTileTaskRunner::CompleteTask(result.task.get());
cache->UnrefImage(draw_image);
ImageDecodeCache::TaskResult another_result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(another_result.need_unref);
EXPECT_TRUE(another_result.task);
EXPECT_EQ(another_result.task->dependencies().size(), 1u);
EXPECT_TRUE(result.task->dependencies()[0]);
TestTileTaskRunner::ProcessTask(another_result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(another_result.task.get());
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest, GetTaskForImageAlreadyUploaded) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(0.5f, 0.5f)));
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
EXPECT_EQ(result.task->dependencies().size(), 1u);
EXPECT_TRUE(result.task->dependencies()[0]);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ScheduleTask(result.task.get());
TestTileTaskRunner::RunTask(result.task.get());
TestTileTaskRunner::CompleteTask(result.task.get());
ImageDecodeCache::TaskResult another_result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(another_result.need_unref);
EXPECT_FALSE(another_result.task);
cache->UnrefImage(draw_image);
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest, GetTaskForImageCanceledGetsNewTask) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(0.5f, 0.5f)));
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
ImageDecodeCache::TaskResult another_result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(another_result.task.get() == result.task.get());
TestTileTaskRunner::CancelTask(result.task.get());
TestTileTaskRunner::CompleteTask(result.task.get());
cache->UnrefImage(draw_image);
cache->UnrefImage(draw_image);
ImageDecodeCache::TaskResult third_result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(third_result.need_unref);
EXPECT_TRUE(third_result.task);
EXPECT_FALSE(third_result.task.get() == result.task.get());
TestTileTaskRunner::ProcessTask(third_result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(third_result.task.get());
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest, GetTaskForImageCanceledWhileReffedGetsNewTask) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(0.5f, 0.5f)));
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
ASSERT_GT(result.task->dependencies().size(), 0u);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
ImageDecodeCache::TaskResult another_result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(another_result.need_unref);
EXPECT_TRUE(another_result.task.get() == result.task.get());
TestTileTaskRunner::CancelTask(result.task.get());
TestTileTaskRunner::CompleteTask(result.task.get());
cache->UnrefImage(draw_image);
cache->UnrefImage(draw_image);
ImageDecodeCache::TaskResult third_result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(third_result.need_unref);
EXPECT_TRUE(third_result.task);
EXPECT_FALSE(third_result.task.get() == result.task.get());
ASSERT_GT(third_result.task->dependencies().size(), 0u);
TestTileTaskRunner::ProcessTask(third_result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(third_result.task.get());
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest, GetTaskForImageUploadCanceledButDecodeRun) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(0.5f, 0.5f)));
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
EXPECT_EQ(result.task->dependencies().size(), 1u);
EXPECT_TRUE(result.task->dependencies()[0]);
TestTileTaskRunner::CancelTask(result.task.get());
TestTileTaskRunner::CompleteTask(result.task.get());
cache->UnrefImage(draw_image);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
}
TEST_P(GpuImageDecodeCacheTest, NoTaskForImageAlreadyFailedDecoding) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(0.5f, 0.5f)));
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::CancelTask(result.task.get());
TestTileTaskRunner::CompleteTask(result.task.get());
cache->SetImageDecodingFailedForTesting(draw_image);
ImageDecodeCache::TaskResult another_result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_FALSE(another_result.need_unref);
EXPECT_EQ(another_result.task.get(), nullptr);
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest, GetDecodedImageForDraw) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(0.5f, 0.5f)));
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
EXPECT_EQ(result.task->dependencies().size(), 1u);
EXPECT_TRUE(result.task->dependencies()[0]);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
EXPECT_TRUE(decoded_draw_image.image());
EXPECT_TRUE(decoded_draw_image.image()->isTextureBacked());
EXPECT_TRUE(decoded_draw_image.is_budgeted());
EXPECT_FALSE(cache->DiscardableIsLockedForTesting(draw_image));
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest, GetHdrDecodedImageForDrawToHdr) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
auto color_space = gfx::ColorSpace::CreateHDR10();
auto size = GetNormalImageSize();
auto info =
SkImageInfo::Make(size.width(), size.height(), kRGBA_F16_SkColorType,
kPremul_SkAlphaType, color_space.ToSkColorSpace());
SkBitmap bitmap;
bitmap.allocPixels(info);
PaintImage image = PaintImageBuilder::WithDefault()
.set_id(PaintImage::kInvalidId)
.set_is_high_bit_depth(true)
.set_image(SkImages::RasterFromBitmap(bitmap),
PaintImage::GetNextContentId())
.TakePaintImage();
constexpr float kCustomWhiteLevel = 200.f;
DrawImage draw_image = CreateDrawImageInternal(
image, CreateMatrix(SkSize::Make(0.5f, 0.5f)), &color_space,
PaintFlags::FilterQuality::kMedium, nullptr,
PaintImage::kDefaultFrameIndex, kCustomWhiteLevel);
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_EQ(draw_image.target_color_space(), color_space);
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
EXPECT_EQ(result.task->dependencies().size(), 1u);
EXPECT_TRUE(result.task->dependencies()[0]);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
EXPECT_TRUE(decoded_draw_image.image());
EXPECT_TRUE(decoded_draw_image.image()->isTextureBacked());
EXPECT_TRUE(decoded_draw_image.is_budgeted());
EXPECT_TRUE(decoded_draw_image.image()->colorType() ==
kRGBA_F16_SkColorType ||
decoded_draw_image.image()->colorType() == kN32_SkColorType);
EXPECT_FALSE(cache->DiscardableIsLockedForTesting(draw_image));
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest, GetHdrDecodedImageForDrawToSdr) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
auto image_color_space = gfx::ColorSpace::CreateHDR10();
auto size = GetNormalImageSize();
auto info = SkImageInfo::Make(size.width(), size.height(),
kRGBA_F16_SkColorType, kPremul_SkAlphaType,
image_color_space.ToSkColorSpace());
SkBitmap bitmap;
bitmap.allocPixels(info);
PaintImage image = PaintImageBuilder::WithDefault()
.set_id(PaintImage::kInvalidId)
.set_is_high_bit_depth(true)
.set_image(SkImages::RasterFromBitmap(bitmap),
PaintImage::GetNextContentId())
.TakePaintImage();
auto raster_color_space = gfx::ColorSpace::CreateSRGB();
DrawImage draw_image = CreateDrawImageInternal(
image, CreateMatrix(SkSize::Make(0.5f, 0.5f)), &raster_color_space);
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_EQ(draw_image.target_color_space(), raster_color_space);
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
EXPECT_EQ(result.task->dependencies().size(), 1u);
EXPECT_TRUE(result.task->dependencies()[0]);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
EXPECT_TRUE(decoded_draw_image.image());
EXPECT_TRUE(decoded_draw_image.image()->isTextureBacked());
EXPECT_TRUE(decoded_draw_image.is_budgeted());
EXPECT_TRUE(decoded_draw_image.image()->colorType() ==
kRGBA_F16_SkColorType ||
decoded_draw_image.image()->colorType() == kN32_SkColorType);
EXPECT_FALSE(cache->DiscardableIsLockedForTesting(draw_image));
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest, GetLargeDecodedImageForDraw) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreateLargePaintImageForSoftwareFallback();
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(1.0f, 1.0f)));
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
EXPECT_TRUE(decoded_draw_image.image());
EXPECT_TRUE(decoded_draw_image.is_budgeted());
EXPECT_FALSE(decoded_draw_image.image()->isTextureBacked());
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
cache->UnrefImage(draw_image);
EXPECT_FALSE(cache->DiscardableIsLockedForTesting(draw_image));
}
TEST_P(GpuImageDecodeCacheTest, GetDecodedImageForDrawAtRasterDecode) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
cache->SetWorkingSetLimitsForTesting(0 , 0 );
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(1.0f, 1.0f)));
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_FALSE(result.need_unref);
EXPECT_FALSE(result.task);
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
EXPECT_TRUE(decoded_draw_image.image());
EXPECT_FALSE(decoded_draw_image.is_budgeted());
EXPECT_TRUE(decoded_draw_image.image()->isTextureBacked());
EXPECT_FALSE(decoded_draw_image.is_budgeted());
EXPECT_FALSE(cache->DiscardableIsLockedForTesting(draw_image));
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
EXPECT_EQ(cache->GetNumCacheEntriesForTesting(), 1u);
}
TEST_P(GpuImageDecodeCacheTest, GetDecodedImageForDrawLargerScale) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image = CreateDrawImageInternal(
image, CreateMatrix(SkSize::Make(0.5f, 0.5f)), nullptr ,
PaintFlags::FilterQuality::kLow);
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
DrawImage larger_draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(1.5f, 1.5f)));
ImageDecodeCache::TaskResult larger_result = cache->GetTaskForImageAndRef(
client_id, larger_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(larger_result.need_unref);
EXPECT_TRUE(larger_result.task);
TestTileTaskRunner::ProcessTask(larger_result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(larger_result.task.get());
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
EXPECT_TRUE(decoded_draw_image.is_budgeted());
EXPECT_TRUE(decoded_draw_image.image());
EXPECT_TRUE(decoded_draw_image.image()->isTextureBacked());
EXPECT_FALSE(cache->DiscardableIsLockedForTesting(draw_image));
DecodedDrawImage larger_decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(larger_draw_image));
EXPECT_TRUE(larger_decoded_draw_image.image());
EXPECT_TRUE(larger_decoded_draw_image.is_budgeted());
EXPECT_TRUE(larger_decoded_draw_image.image()->isTextureBacked());
EXPECT_FALSE(cache->DiscardableIsLockedForTesting(draw_image));
EXPECT_FALSE(decoded_draw_image.image() == larger_decoded_draw_image.image());
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
cache->UnrefImage(draw_image);
cache->DrawWithImageFinished(larger_draw_image, larger_decoded_draw_image);
cache->UnrefImage(larger_draw_image);
}
TEST_P(GpuImageDecodeCacheTest, GetDecodedImageForDrawHigherQuality) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
SkM44 matrix = CreateMatrix(SkSize::Make(0.5f, 0.5f));
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image =
CreateDrawImageInternal(image, matrix, nullptr ,
PaintFlags::FilterQuality::kLow);
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
DrawImage higher_quality_draw_image = CreateDrawImageInternal(image, matrix);
ImageDecodeCache::TaskResult hq_result = cache->GetTaskForImageAndRef(
client_id, higher_quality_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(hq_result.need_unref);
EXPECT_TRUE(hq_result.task);
TestTileTaskRunner::ProcessTask(hq_result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(hq_result.task.get());
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
EXPECT_TRUE(decoded_draw_image.image());
EXPECT_TRUE(decoded_draw_image.is_budgeted());
EXPECT_TRUE(decoded_draw_image.image()->isTextureBacked());
EXPECT_FALSE(cache->DiscardableIsLockedForTesting(draw_image));
DecodedDrawImage larger_decoded_draw_image = EnsureImageBacked(
cache->GetDecodedImageForDraw(higher_quality_draw_image));
EXPECT_TRUE(larger_decoded_draw_image.image());
EXPECT_TRUE(larger_decoded_draw_image.image()->isTextureBacked());
EXPECT_FALSE(cache->DiscardableIsLockedForTesting(draw_image));
EXPECT_FALSE(decoded_draw_image.image() == larger_decoded_draw_image.image());
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
cache->UnrefImage(draw_image);
cache->DrawWithImageFinished(higher_quality_draw_image,
larger_decoded_draw_image);
cache->UnrefImage(higher_quality_draw_image);
}
TEST_P(GpuImageDecodeCacheTest, GetDecodedImageForDrawNegative) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(-0.5f, 0.5f)));
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
EXPECT_TRUE(decoded_draw_image.image());
EXPECT_TRUE(decoded_draw_image.is_budgeted());
const int expected_width =
image.width() * std::abs(draw_image.scale().width());
const int expected_height =
image.height() * std::abs(draw_image.scale().height());
EXPECT_EQ(decoded_draw_image.image()->width(), expected_width);
EXPECT_EQ(decoded_draw_image.image()->height(), expected_height);
EXPECT_TRUE(decoded_draw_image.image()->isTextureBacked());
EXPECT_FALSE(cache->DiscardableIsLockedForTesting(draw_image));
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest, GetLargeScaledDecodedImageForDraw) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageForFallbackToRGB(
gfx::Size(GetLargeImageSize().width(), GetLargeImageSize().height() * 2));
DrawImage draw_image = CreateDrawImageInternal(
image, CreateMatrix(SkSize::Make(0.5f, 0.5f)), nullptr ,
PaintFlags::FilterQuality::kHigh);
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
EXPECT_TRUE(decoded_draw_image.image());
EXPECT_TRUE(decoded_draw_image.is_budgeted());
EXPECT_EQ(GetLargeImageSize().width(), decoded_draw_image.image()->width());
EXPECT_EQ(GetLargeImageSize().height(), decoded_draw_image.image()->height());
EXPECT_EQ(decoded_draw_image.filter_quality(),
PaintFlags::FilterQuality::kMedium);
EXPECT_FALSE(decoded_draw_image.image()->isTextureBacked());
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
cache->UnrefImage(draw_image);
EXPECT_FALSE(cache->DiscardableIsLockedForTesting(draw_image));
}
TEST_P(GpuImageDecodeCacheTest, AtRasterUsedDirectlyIfSpaceAllows) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
const gfx::Size test_image_size = GetNormalImageSize();
cache->SetWorkingSetLimitsForTesting(0 , 0 );
PaintImage image = CreatePaintImageInternal(test_image_size);
DrawImage draw_image = CreateDrawImageInternal(image);
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_FALSE(result.need_unref);
EXPECT_FALSE(result.task);
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
EXPECT_TRUE(decoded_draw_image.image());
EXPECT_TRUE(decoded_draw_image.image()->isTextureBacked());
EXPECT_FALSE(decoded_draw_image.is_budgeted());
EXPECT_FALSE(cache->DiscardableIsLockedForTesting(draw_image));
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
const size_t bytes_for_test_image =
GetBytesNeededForSingleImage(test_image_size);
cache->SetWorkingSetLimitsForTesting(bytes_for_test_image ,
256 );
ImageDecodeCache::TaskResult another_result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(another_result.need_unref);
EXPECT_FALSE(another_result.task);
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest,
GetDecodedImageForDrawAtRasterDecodeMultipleTimes) {
auto cache = CreateCache();
cache->SetWorkingSetLimitsForTesting(0 , 0 );
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(0.5f, 0.5f)));
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
EXPECT_TRUE(decoded_draw_image.image());
EXPECT_TRUE(decoded_draw_image.image()->isTextureBacked());
EXPECT_FALSE(decoded_draw_image.is_budgeted());
EXPECT_FALSE(cache->DiscardableIsLockedForTesting(draw_image));
DecodedDrawImage another_decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
EXPECT_FALSE(another_decoded_draw_image.is_budgeted());
EXPECT_EQ(decoded_draw_image.image()->uniqueID(),
another_decoded_draw_image.image()->uniqueID());
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
cache->DrawWithImageFinished(draw_image, another_decoded_draw_image);
EXPECT_EQ(cache->GetNumCacheEntriesForTesting(), 1u);
}
TEST_P(GpuImageDecodeCacheTest,
GetLargeDecodedImageForDrawAtRasterDecodeMultipleTimes) {
auto cache = CreateCache();
cache->SetWorkingSetLimitsForTesting(0 , 0 );
PaintImage image = CreateLargePaintImageForSoftwareFallback();
DrawImage draw_image = CreateDrawImageInternal(image);
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
EXPECT_TRUE(decoded_draw_image.image());
EXPECT_FALSE(decoded_draw_image.is_budgeted());
EXPECT_FALSE(decoded_draw_image.image()->isTextureBacked());
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
EXPECT_FALSE(cache->DiscardableIsLockedForTesting(draw_image));
DecodedDrawImage second_decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
EXPECT_TRUE(second_decoded_draw_image.image());
EXPECT_FALSE(decoded_draw_image.is_budgeted());
EXPECT_FALSE(second_decoded_draw_image.image()->isTextureBacked());
cache->DrawWithImageFinished(draw_image, second_decoded_draw_image);
EXPECT_FALSE(cache->DiscardableIsLockedForTesting(draw_image));
}
TEST_P(GpuImageDecodeCacheTest, ZeroSizedImagesAreSkipped) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(0.f, 0.f)));
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_FALSE(result.task);
EXPECT_FALSE(result.need_unref);
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
EXPECT_FALSE(decoded_draw_image.image());
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
}
TEST_P(GpuImageDecodeCacheTest, NonOverlappingSrcRectImagesAreSkipped) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image(
image, false,
SkIRect::MakeXYWH(image.width() + 1, image.height() + 1, image.width(),
image.height()),
PaintFlags::FilterQuality::kMedium, CreateMatrix(SkSize::Make(1.f, 1.f)),
PaintImage::kDefaultFrameIndex, DefaultTargetColorParams());
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_FALSE(result.task);
EXPECT_FALSE(result.need_unref);
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
EXPECT_FALSE(decoded_draw_image.image());
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
}
TEST_P(GpuImageDecodeCacheTest, CanceledTasksDoNotCountAgainstBudget) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
SkIRect src_rect = SkIRect::MakeXYWH(0, 0, image.width(), image.height());
DrawImage draw_image = CreateDrawImageInternal(
image, CreateMatrix(SkSize::Make(1.f, 1.f)), nullptr ,
PaintFlags::FilterQuality::kMedium, &src_rect);
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_NE(0u, cache->GetNumCacheEntriesForTesting());
EXPECT_TRUE(result.task);
EXPECT_TRUE(result.need_unref);
TestTileTaskRunner::CancelTask(result.task->dependencies()[0].get());
TestTileTaskRunner::CompleteTask(result.task->dependencies()[0].get());
TestTileTaskRunner::CancelTask(result.task.get());
TestTileTaskRunner::CompleteTask(result.task.get());
cache->UnrefImage(draw_image);
EXPECT_EQ(0u, cache->GetWorkingSetBytesForTesting());
}
TEST_P(GpuImageDecodeCacheTest, ShouldAggressivelyFreeResources) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(0.5f, 0.5f)));
{
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
cache->UnrefImage(draw_image);
EXPECT_GT(cache->GetNumCacheEntriesForTesting(), 0u);
cache->SetShouldAggressivelyFreeResources(true);
EXPECT_EQ(0u, cache->GetNumCacheEntriesForTesting());
}
{
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
cache->UnrefImage(draw_image);
EXPECT_EQ(cache->GetNumCacheEntriesForTesting(), 0u);
}
cache->SetShouldAggressivelyFreeResources(false);
{
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
cache->UnrefImage(draw_image);
EXPECT_GT(cache->GetNumCacheEntriesForTesting(), 0u);
}
}
TEST_P(GpuImageDecodeCacheTest, OrphanedImagesFreeOnReachingZeroRefs) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage first_image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage first_draw_image = CreateDrawImageInternal(
first_image, CreateMatrix(SkSize::Make(0.5f, 0.5f)));
ImageDecodeCache::TaskResult first_result = cache->GetTaskForImageAndRef(
client_id, first_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(first_result.need_unref);
EXPECT_TRUE(first_result.task);
EXPECT_EQ(cache->GetWorkingSetBytesForTesting(),
cache->GetDrawImageSizeForTesting(first_draw_image));
DrawImage second_draw_image = CreateDrawImageInternal(
first_image, CreateMatrix(SkSize::Make(1.0f, 1.0f)));
ImageDecodeCache::TaskResult second_result = cache->GetTaskForImageAndRef(
client_id, second_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(second_result.need_unref);
EXPECT_TRUE(second_result.task);
EXPECT_TRUE(first_result.task.get() != second_result.task.get());
TestTileTaskRunner::ProcessTask(second_result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(second_result.task.get());
cache->UnrefImage(second_draw_image);
TestTileTaskRunner::ProcessTask(first_result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(first_result.task.get());
cache->UnrefImage(first_draw_image);
EXPECT_EQ(1u, cache->GetNumCacheEntriesForTesting());
EXPECT_EQ(0u, cache->GetInUseCacheEntriesForTesting());
}
TEST_P(GpuImageDecodeCacheTest, OrphanedZeroRefImagesImmediatelyDeleted) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage first_image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage first_draw_image = CreateDrawImageInternal(
first_image, CreateMatrix(SkSize::Make(0.5f, 0.5f)));
ImageDecodeCache::TaskResult first_result = cache->GetTaskForImageAndRef(
client_id, first_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(first_result.need_unref);
EXPECT_TRUE(first_result.task);
TestTileTaskRunner::ProcessTask(first_result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(first_result.task.get());
cache->UnrefImage(first_draw_image);
EXPECT_EQ(cache->GetNumCacheEntriesForTesting(), 1u);
EXPECT_EQ(cache->GetInUseCacheEntriesForTesting(), 0u);
DrawImage second_draw_image = CreateDrawImageInternal(first_image);
ImageDecodeCache::TaskResult second_result = cache->GetTaskForImageAndRef(
client_id, second_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(second_result.need_unref);
EXPECT_TRUE(second_result.task);
EXPECT_TRUE(first_result.task.get() != second_result.task.get());
TestTileTaskRunner::ProcessTask(second_result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(second_result.task.get());
cache->UnrefImage(second_draw_image);
EXPECT_EQ(cache->GetNumCacheEntriesForTesting(), 1u);
EXPECT_EQ(cache->GetInUseCacheEntriesForTesting(), 0u);
}
TEST_P(GpuImageDecodeCacheTest, QualityCappedAtMedium) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
SkM44 matrix = CreateMatrix(SkSize::Make(0.4f, 0.4f));
DrawImage low_draw_image =
CreateDrawImageInternal(image, matrix, nullptr ,
PaintFlags::FilterQuality::kLow);
ImageDecodeCache::TaskResult low_result = cache->GetTaskForImageAndRef(
client_id, low_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(low_result.need_unref);
EXPECT_TRUE(low_result.task);
DrawImage medium_draw_image = CreateDrawImageInternal(image);
ImageDecodeCache::TaskResult medium_result = cache->GetTaskForImageAndRef(
client_id, medium_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(medium_result.need_unref);
EXPECT_TRUE(medium_result.task.get());
EXPECT_FALSE(low_result.task.get() == medium_result.task.get());
DrawImage high_quality_draw_image =
CreateDrawImageInternal(image, matrix, nullptr ,
PaintFlags::FilterQuality::kHigh);
ImageDecodeCache::TaskResult high_quality_result =
cache->GetTaskForImageAndRef(client_id, high_quality_draw_image,
ImageDecodeCache::TracingInfo());
EXPECT_TRUE(high_quality_result.need_unref);
EXPECT_TRUE(medium_result.task.get() == high_quality_result.task.get());
TestTileTaskRunner::ProcessTask(low_result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(low_result.task.get());
TestTileTaskRunner::ProcessTask(medium_result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(medium_result.task.get());
cache->UnrefImage(low_draw_image);
cache->UnrefImage(medium_draw_image);
cache->UnrefImage(high_quality_draw_image);
}
TEST_P(GpuImageDecodeCacheTest, GetDecodedImageForDrawMipUsageChange) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image = CreateDrawImageInternal(image);
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::CancelTask(result.task->dependencies()[0].get());
TestTileTaskRunner::CompleteTask(result.task->dependencies()[0].get());
TestTileTaskRunner::CancelTask(result.task.get());
TestTileTaskRunner::CompleteTask(result.task.get());
cache->UnrefImage(draw_image);
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DrawImage draw_image_mips =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(0.6f, 0.6f)));
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image_mips));
cache->DrawWithImageFinished(draw_image_mips, decoded_draw_image);
}
TEST_P(GpuImageDecodeCacheTest, OutOfRasterDecodeTask) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
SkM44 matrix = CreateMatrix(SkSize::Make(1.0f, 1.0f));
DrawImage draw_image =
CreateDrawImageInternal(image, matrix, nullptr ,
PaintFlags::FilterQuality::kLow);
ImageDecodeCache::TaskResult result =
cache->GetOutOfRasterDecodeTaskForImageAndRef(client_id, draw_image,
false);
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
EXPECT_TRUE(cache->IsInInUseCacheForTesting(draw_image));
TestTileTaskRunner::ProcessTask(result.task.get());
EXPECT_TRUE(cache->IsInInUseCacheForTesting(draw_image));
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest, OutOfRasterDecodeTaskMultipleClients) {
auto cache = CreateCache();
const uint32_t kClientId1 = cache->GenerateClientId();
const uint32_t kClientId2 = cache->GenerateClientId();
for (size_t order = 1; order <= 2; ++order) {
sk_sp<FakePaintImageGenerator> generator =
CreateFakePaintImageGenerator(GetNormalImageSize());
PaintImage image =
PaintImageBuilder::WithDefault()
.set_id(PaintImage::GetNextId())
.set_paint_image_generator(generator)
.set_decoding_mode(PaintImage::DecodingMode::kUnspecified)
.TakePaintImage();
SkM44 matrix = CreateMatrix(SkSize::Make(1.0f, 1.0f));
DrawImage draw_image =
CreateDrawImageInternal(image, matrix, nullptr ,
PaintFlags::FilterQuality::kLow);
ImageDecodeCache::TaskResult result1 =
cache->GetOutOfRasterDecodeTaskForImageAndRef(kClientId1, draw_image,
false);
EXPECT_TRUE(result1.need_unref);
EXPECT_TRUE(result1.task);
EXPECT_TRUE(cache->IsInInUseCacheForTesting(draw_image));
DrawImage draw_image2 =
CreateDrawImageInternal(image, matrix, nullptr ,
PaintFlags::FilterQuality::kLow);
ImageDecodeCache::TaskResult result2 =
cache->GetOutOfRasterDecodeTaskForImageAndRef(kClientId2, draw_image,
false);
EXPECT_TRUE(result2.need_unref);
EXPECT_TRUE(result2.task);
EXPECT_TRUE(cache->IsInInUseCacheForTesting(draw_image2));
if (order == 1u) {
TestTileTaskRunner::ProcessTask(result1.task.get());
TestTileTaskRunner::ProcessTask(result2.task.get());
} else {
DCHECK_EQ(order, 2u);
TestTileTaskRunner::ProcessTask(result2.task.get());
TestTileTaskRunner::ProcessTask(result1.task.get());
}
EXPECT_EQ(generator->frames_decoded().size(), 1u);
EXPECT_EQ(generator->frames_decoded().count(PaintImage::kDefaultFrameIndex),
1u);
EXPECT_TRUE(cache->IsInInUseCacheForTesting(draw_image));
EXPECT_TRUE(cache->IsInInUseCacheForTesting(draw_image2));
cache->UnrefImage(draw_image);
cache->UnrefImage(draw_image2);
EXPECT_FALSE(cache->IsInInUseCacheForTesting(draw_image));
EXPECT_FALSE(cache->IsInInUseCacheForTesting(draw_image2));
}
}
TEST_P(GpuImageDecodeCacheTest,
DoesNotCreateOutOfRasterDecodeTaskForNonCompletedTask) {
auto cache = CreateCache();
const uint32_t kClientId1 = cache->GenerateClientId();
const uint32_t kClientId2 = cache->GenerateClientId();
sk_sp<FakePaintImageGenerator> generator =
CreateFakePaintImageGenerator(GetNormalImageSize());
PaintImage image =
PaintImageBuilder::WithDefault()
.set_id(PaintImage::GetNextId())
.set_paint_image_generator(generator)
.set_decoding_mode(PaintImage::DecodingMode::kUnspecified)
.TakePaintImage();
SkM44 matrix = CreateMatrix(SkSize::Make(1.0f, 1.0f));
DrawImage draw_image =
CreateDrawImageInternal(image, matrix, nullptr ,
PaintFlags::FilterQuality::kLow);
ImageDecodeCache::TaskResult result1 =
cache->GetOutOfRasterDecodeTaskForImageAndRef(kClientId1, draw_image,
false);
EXPECT_TRUE(result1.need_unref);
EXPECT_TRUE(result1.task);
EXPECT_TRUE(cache->IsInInUseCacheForTesting(draw_image));
TestTileTaskRunner::ScheduleTask(result1.task.get());
TestTileTaskRunner::RunTask(result1.task.get());
DrawImage draw_image2 =
CreateDrawImageInternal(image, matrix, nullptr ,
PaintFlags::FilterQuality::kLow);
ImageDecodeCache::TaskResult result2 =
cache->GetOutOfRasterDecodeTaskForImageAndRef(kClientId2, draw_image,
false);
EXPECT_TRUE(result2.need_unref);
EXPECT_FALSE(result2.task);
EXPECT_TRUE(cache->IsInInUseCacheForTesting(draw_image2));
TestTileTaskRunner::CompleteTask(result1.task.get());
EXPECT_TRUE(cache->IsInInUseCacheForTesting(draw_image));
EXPECT_TRUE(cache->IsInInUseCacheForTesting(draw_image2));
cache->UnrefImage(draw_image);
cache->UnrefImage(draw_image2);
EXPECT_FALSE(cache->IsInInUseCacheForTesting(draw_image));
EXPECT_FALSE(cache->IsInInUseCacheForTesting(draw_image2));
}
TEST_P(GpuImageDecodeCacheTest, ZeroCacheNormalWorkingSet) {
SetCachedTexturesLimit(0);
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image = CreateDrawImageInternal(image);
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
EXPECT_EQ(result.task->dependencies().size(), 1u);
EXPECT_TRUE(result.task->dependencies()[0]);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
ImageDecodeCache::TaskResult second_result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(second_result.need_unref);
EXPECT_FALSE(second_result.task);
cache->UnrefImage(draw_image);
cache->UnrefImage(draw_image);
cache->ReduceCacheUsage();
ImageDecodeCache::TaskResult third_result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(third_result.need_unref);
EXPECT_TRUE(third_result.task);
EXPECT_EQ(third_result.task->dependencies().size(), 1u);
EXPECT_TRUE(third_result.task->dependencies()[0]);
TestTileTaskRunner::ProcessTask(third_result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(third_result.task.get());
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest, SmallCacheNormalWorkingSet) {
SetCachedTexturesLimit(1);
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image = CreateDrawImageInternal(image);
PaintImage image2 = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image2(
image2, false, SkIRect::MakeWH(image2.width(), image2.height()),
PaintFlags::FilterQuality::kMedium,
CreateMatrix(SkSize::Make(1.0f, 1.0f)), PaintImage::kDefaultFrameIndex,
DefaultTargetColorParams());
{
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
EXPECT_EQ(result.task->dependencies().size(), 1u);
EXPECT_TRUE(result.task->dependencies()[0]);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
cache->UnrefImage(draw_image);
}
{
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_FALSE(result.task);
cache->UnrefImage(draw_image);
}
{
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image2, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
EXPECT_EQ(result.task->dependencies().size(), 1u);
EXPECT_TRUE(result.task->dependencies()[0]);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
cache->UnrefImage(draw_image2);
}
{
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image2, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_FALSE(result.task);
cache->UnrefImage(draw_image2);
}
{
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
EXPECT_EQ(result.task->dependencies().size(), 1u);
EXPECT_TRUE(result.task->dependencies()[0]);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
cache->UnrefImage(draw_image);
}
}
TEST_P(GpuImageDecodeCacheTest, ClearCache) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
for (int i = 0; i < 10; ++i) {
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image = CreateDrawImageInternal(image);
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
cache->UnrefImage(draw_image);
}
EXPECT_EQ(cache->GetNumCacheEntriesForTesting(), 10u);
cache->ClearCache();
EXPECT_EQ(cache->GetNumCacheEntriesForTesting(), 0u);
}
TEST_P(GpuImageDecodeCacheTest, ClearCacheInUse) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image = CreateDrawImageInternal(image);
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
EXPECT_GT(cache->GetWorkingSetBytesForTesting(), 0u);
EXPECT_EQ(cache->GetNumCacheEntriesForTesting(), 1u);
cache->ClearCache();
EXPECT_GT(cache->GetWorkingSetBytesForTesting(), 0u);
EXPECT_EQ(cache->GetNumCacheEntriesForTesting(), 0u);
cache->UnrefImage(draw_image);
EXPECT_EQ(cache->GetWorkingSetBytesForTesting(), 0u);
EXPECT_EQ(cache->GetNumCacheEntriesForTesting(), 0u);
}
TEST_P(GpuImageDecodeCacheTest, GetTaskForImageDifferentColorSpace) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
gfx::ColorSpace color_space_a = gfx::ColorSpace::CreateSRGB();
gfx::ColorSpace color_space_b = gfx::ColorSpace::CreateXYZD50();
PaintImage first_image = CreatePaintImageInternal(gfx::Size(100, 100));
DrawImage first_draw_image = CreateDrawImageInternal(
first_image, CreateMatrix(SkSize::Make(1.0f, 1.0f)), &color_space_a);
ImageDecodeCache::TaskResult first_result = cache->GetTaskForImageAndRef(
client_id, first_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(first_result.need_unref);
EXPECT_TRUE(first_result.task);
DrawImage second_draw_image = CreateDrawImageInternal(
first_image, CreateMatrix(SkSize::Make(1.0f, 1.0f)), &color_space_b);
ImageDecodeCache::TaskResult second_result = cache->GetTaskForImageAndRef(
client_id, second_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(second_result.need_unref);
EXPECT_TRUE(second_result.task);
EXPECT_TRUE(first_result.task.get() != second_result.task.get());
DrawImage third_draw_image = CreateDrawImageInternal(
first_image, CreateMatrix(SkSize::Make(1.0f, 1.0f)), &color_space_a);
ImageDecodeCache::TaskResult third_result = cache->GetTaskForImageAndRef(
client_id, third_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(third_result.need_unref);
EXPECT_TRUE(third_result.task.get() == first_result.task.get());
TestTileTaskRunner::ProcessTask(first_result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(first_result.task.get());
TestTileTaskRunner::ProcessTask(second_result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(second_result.task.get());
cache->UnrefImage(first_draw_image);
cache->UnrefImage(second_draw_image);
cache->UnrefImage(third_draw_image);
}
TEST_P(GpuImageDecodeCacheTest, GetTaskForLargeImageNonSRGBColorSpace) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
gfx::ColorSpace color_space = gfx::ColorSpace::CreateXYZD50();
PaintImage image = CreateLargePaintImageForSoftwareFallback();
DrawImage draw_image = CreateDrawImageInternal(
image, CreateMatrix(SkSize::Make(1.0f, 1.0f)), &color_space);
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest, CacheDecodesExpectedFrames) {
auto cache = CreateCache();
std::vector<FrameMetadata> frames = {
FrameMetadata(true, base::Milliseconds(2)),
FrameMetadata(true, base::Milliseconds(3)),
FrameMetadata(true, base::Milliseconds(4)),
FrameMetadata(true, base::Milliseconds(5)),
};
const gfx::Size test_image_size = GetNormalImageSize();
SkImageInfo info =
SkImageInfo::Make(test_image_size.width(), test_image_size.height(),
color_type_, kPremul_SkAlphaType);
sk_sp<FakePaintImageGenerator> generator;
if (do_yuv_decode_) {
SkYUVAPixmapInfo yuva_pixmap_info =
GetYUVAPixmapInfo(test_image_size, yuv_format_, yuv_data_type_);
generator =
sk_make_sp<FakePaintImageGenerator>(info, yuva_pixmap_info, frames);
} else {
generator = sk_make_sp<FakePaintImageGenerator>(info, frames);
}
PaintImage image = PaintImageBuilder::WithDefault()
.set_id(PaintImage::GetNextId())
.set_paint_image_generator(generator)
.TakePaintImage();
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
PaintFlags::FilterQuality quality = PaintFlags::FilterQuality::kMedium;
DrawImage draw_image(
image, false, SkIRect::MakeWH(image.width(), image.height()), quality,
CreateMatrix(SkSize::Make(1.0f, 1.0f)), 1u, DefaultTargetColorParams());
auto decoded_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
ASSERT_TRUE(decoded_image.image());
ASSERT_EQ(generator->frames_decoded().size(), 1u);
EXPECT_EQ(generator->frames_decoded().count(1u), 1u);
generator->reset_frames_decoded();
cache->DrawWithImageFinished(draw_image, decoded_image);
TargetColorParams target_color_params;
target_color_params.color_space = draw_image.target_color_space();
DrawImage scaled_draw_image(draw_image, 0.5f, 2u, target_color_params);
decoded_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(scaled_draw_image));
ASSERT_TRUE(decoded_image.image());
ASSERT_EQ(generator->frames_decoded().size(), 1u);
EXPECT_EQ(generator->frames_decoded().count(2u), 1u);
generator->reset_frames_decoded();
cache->DrawWithImageFinished(scaled_draw_image, decoded_image);
const int32_t subset_width = 5;
const int32_t subset_height = 5;
ASSERT_LT(subset_width, test_image_size.width());
ASSERT_LT(subset_height, test_image_size.height());
DrawImage subset_draw_image(
image, false, SkIRect::MakeWH(subset_width, subset_height), quality,
CreateMatrix(SkSize::Make(1.0f, 1.0f)), 3u, DefaultTargetColorParams());
decoded_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(subset_draw_image));
ASSERT_TRUE(decoded_image.image());
ASSERT_EQ(generator->frames_decoded().size(), 1u);
EXPECT_EQ(generator->frames_decoded().count(3u), 1u);
generator->reset_frames_decoded();
cache->DrawWithImageFinished(subset_draw_image, decoded_image);
}
TEST_P(GpuImageDecodeCacheTest, OrphanedDataCancelledWhileReplaced) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage first_image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage first_draw_image = CreateDrawImageInternal(
first_image, CreateMatrix(SkSize::Make(0.5f, 0.5f)));
ImageDecodeCache::TaskResult first_result = cache->GetTaskForImageAndRef(
client_id, first_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(first_result.need_unref);
EXPECT_TRUE(first_result.task);
EXPECT_EQ(1u, cache->GetNumCacheEntriesForTesting());
DrawImage second_draw_image = CreateDrawImageInternal(first_image);
ImageDecodeCache::TaskResult second_result = cache->GetTaskForImageAndRef(
client_id, second_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(second_result.need_unref);
EXPECT_TRUE(second_result.task);
EXPECT_TRUE(first_result.task.get() != second_result.task.get());
EXPECT_EQ(1u, cache->GetNumCacheEntriesForTesting());
TestTileTaskRunner::CancelTask(first_result.task->dependencies()[0].get());
TestTileTaskRunner::CompleteTask(first_result.task->dependencies()[0].get());
TestTileTaskRunner::CancelTask(first_result.task.get());
TestTileTaskRunner::CompleteTask(first_result.task.get());
cache->UnrefImage(first_draw_image);
EXPECT_EQ(1u, cache->GetNumCacheEntriesForTesting());
TestTileTaskRunner::ProcessTask(second_result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(second_result.task.get());
cache->UnrefImage(second_draw_image);
EXPECT_EQ(1u, cache->GetNumCacheEntriesForTesting());
EXPECT_EQ(0u, cache->GetInUseCacheEntriesForTesting());
}
TEST_P(GpuImageDecodeCacheTest, AlreadyBudgetedImagesAreNotAtRaster) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
const gfx::Size test_image_size = GetNormalImageSize();
PaintImage image = CreatePaintImageInternal(test_image_size);
DrawImage draw_image = CreateDrawImageInternal(image);
const size_t bytes_for_test_image =
GetBytesNeededForSingleImage(test_image_size);
cache->SetWorkingSetLimitsForTesting(bytes_for_test_image,
1u );
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
result = cache->GetTaskForImageAndRef(client_id, draw_image,
ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
EXPECT_TRUE(decoded_draw_image.is_budgeted());
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
cache->UnrefImage(draw_image);
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest, ImageBudgetingByCount) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
const gfx::Size test_image_size = GetNormalImageSize();
const size_t bytes_for_test_image =
GetBytesNeededForSingleImage(test_image_size);
cache->SetWorkingSetLimitsForTesting(
bytes_for_test_image * 100 , 1u );
PaintImage image = CreatePaintImageInternal(test_image_size);
DrawImage draw_image = CreateDrawImageInternal(image);
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
EXPECT_TRUE(decoded_draw_image.image());
EXPECT_TRUE(decoded_draw_image.is_budgeted());
PaintImage second_paint_image =
CreatePaintImageInternal(GetNormalImageSize());
DrawImage second_draw_image = CreateDrawImageInternal(second_paint_image);
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, second_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_FALSE(result.need_unref);
EXPECT_FALSE(result.task);
DecodedDrawImage second_decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(second_draw_image));
EXPECT_TRUE(second_decoded_draw_image.image());
EXPECT_FALSE(second_decoded_draw_image.is_budgeted());
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
cache->DrawWithImageFinished(second_draw_image, second_decoded_draw_image);
}
TEST_P(GpuImageDecodeCacheTest, ImageBudgetingBySize) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
const gfx::Size test_image_size = GetNormalImageSize();
PaintImage image = CreatePaintImageInternal(test_image_size);
DrawImage draw_image = CreateDrawImageInternal(image);
const size_t bytes_for_test_image =
GetBytesNeededForSingleImage(test_image_size);
cache->SetWorkingSetLimitsForTesting(bytes_for_test_image,
256 );
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
EXPECT_TRUE(decoded_draw_image.image());
EXPECT_TRUE(decoded_draw_image.is_budgeted());
PaintImage test_paint_image = CreatePaintImageInternal(test_image_size);
DrawImage second_draw_image = CreateDrawImageInternal(test_paint_image);
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, second_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_FALSE(result.need_unref);
EXPECT_FALSE(result.task);
DecodedDrawImage second_decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(second_draw_image));
EXPECT_TRUE(second_decoded_draw_image.image());
EXPECT_FALSE(second_decoded_draw_image.is_budgeted());
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
cache->DrawWithImageFinished(second_draw_image, second_decoded_draw_image);
}
TEST_P(GpuImageDecodeCacheTest,
ColorConversionDuringDecodeForLargeImageNonSRGBColorSpace) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
sk_sp<SkColorSpace> image_color_space =
gfx::ColorSpace::CreateDisplayP3D65().ToSkColorSpace();
gfx::ColorSpace color_space = gfx::ColorSpace::CreateXYZD50();
PaintImage image =
CreateLargePaintImageForSoftwareFallback(image_color_space);
DrawImage draw_image = CreateDrawImageInternal(
image, CreateMatrix(SkSize::Make(1.0f, 1.0f)), &color_space);
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
cache->GetDecodedImageForDraw(draw_image);
sk_sp<SkColorSpace> target_color_space = cache->SupportsColorSpaceConversion()
? color_space.ToSkColorSpace()
: nullptr;
sk_sp<SkImage> service_image = GetLastTransferredImage();
ASSERT_TRUE(image);
EXPECT_FALSE(service_image->isTextureBacked());
EXPECT_EQ(image.width(), service_image->width());
EXPECT_EQ(image.height(), service_image->height());
EXPECT_TRUE(SkColorSpace::Equals(service_image->colorSpace(),
target_color_space.get()));
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest,
ColorConversionDuringUploadForSmallImageNonSRGBColorSpace) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
gfx::ColorSpace color_space = gfx::ColorSpace::CreateDisplayP3D65();
PaintImage image = CreatePaintImageInternal(gfx::Size(11, 12));
DrawImage draw_image = CreateDrawImageInternal(
image, CreateMatrix(SkSize::Make(1.0f, 1.0f)), &color_space);
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
cache->GetDecodedImageForDraw(draw_image);
sk_sp<SkColorSpace> target_color_space = cache->SupportsColorSpaceConversion()
? color_space.ToSkColorSpace()
: nullptr;
sk_sp<SkImage> service_image = GetLastTransferredImage();
ASSERT_TRUE(image);
EXPECT_TRUE(service_image->isTextureBacked());
EXPECT_EQ(image.width(), service_image->width());
EXPECT_EQ(image.height(), service_image->height());
if (!do_yuv_decode_) {
EXPECT_TRUE(SkColorSpace::Equals(service_image->colorSpace(),
target_color_space.get()));
}
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest, NonLazyImageUploadNoScale) {
if (do_yuv_decode_) {
return;
}
auto cache = CreateCache();
PaintImage image = CreateBitmapImageInternal(GetNormalImageSize());
DrawImage draw_image = CreateDrawImageInternal(image);
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
EXPECT_TRUE(decoded_draw_image.image());
EXPECT_TRUE(decoded_draw_image.is_budgeted());
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
EXPECT_FALSE(cache->GetSWImageDecodeForTesting(draw_image));
}
TEST_P(GpuImageDecodeCacheTest, NonLazyImageUploadTaskHasNoDeps) {
if (do_yuv_decode_) {
return;
}
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreateBitmapImageInternal(GetNormalImageSize());
DrawImage draw_image = CreateDrawImageInternal(image);
auto result = cache->GetTaskForImageAndRef(client_id, draw_image,
ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
EXPECT_TRUE(result.task->dependencies().empty());
TestTileTaskRunner::ProcessTask(result.task.get());
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest, NonLazyImageUploadTaskCancelled) {
if (do_yuv_decode_) {
return;
}
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreateBitmapImageInternal(GetNormalImageSize());
DrawImage draw_image = CreateDrawImageInternal(image);
auto result = cache->GetTaskForImageAndRef(client_id, draw_image,
ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
EXPECT_TRUE(result.task->dependencies().empty());
TestTileTaskRunner::CancelTask(result.task.get());
TestTileTaskRunner::CompleteTask(result.task.get());
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest,
NonLazyImageUploadTaskCancelledMultipleClients) {
if (do_yuv_decode_) {
return;
}
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
const uint32_t client_id2 = cache->GenerateClientId();
PaintImage image = CreateBitmapImageInternal(GetNormalImageSize());
DrawImage draw_image = CreateDrawImageInternal(image);
auto result = cache->GetTaskForImageAndRef(client_id, draw_image,
ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
EXPECT_TRUE(result.task->dependencies().empty());
DrawImage draw_image2 = CreateDrawImageInternal(image);
auto result2 = cache->GetTaskForImageAndRef(client_id2, draw_image2,
ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result2.need_unref);
EXPECT_TRUE(result2.task);
EXPECT_TRUE(result2.task->dependencies().empty());
TestTileTaskRunner::CancelTask(result.task.get());
TestTileTaskRunner::CompleteTask(result.task.get());
TestTileTaskRunner::CancelTask(result2.task.get());
TestTileTaskRunner::CompleteTask(result2.task.get());
cache->UnrefImage(draw_image);
cache->UnrefImage(draw_image2);
}
TEST_P(GpuImageDecodeCacheTest, NonLazyImageLargeImageNotColorConverted) {
if (do_yuv_decode_) {
return;
}
auto cache = CreateCache();
PaintImage image = CreateBitmapImageInternal(GetLargeImageSize());
gfx::ColorSpace target_color_space = gfx::ColorSpace::CreateDisplayP3D65();
DrawImage draw_image = CreateDrawImageInternal(
image, CreateMatrix(SkSize::Make(1.0f, 1.0f)), &target_color_space);
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
EXPECT_TRUE(decoded_draw_image.image());
EXPECT_TRUE(decoded_draw_image.is_budgeted());
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
auto sw_image = cache->GetSWImageDecodeForTesting(draw_image);
ASSERT_EQ(!!sw_image, false);
}
TEST_P(GpuImageDecodeCacheTest, NonLazyImageUploadDownscaled) {
if (do_yuv_decode_) {
return;
}
auto cache = CreateCache();
PaintImage image = CreateBitmapImageInternal(GetNormalImageSize());
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(0.5f, 0.5f)));
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
EXPECT_TRUE(decoded_draw_image.image());
EXPECT_TRUE(decoded_draw_image.is_budgeted());
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
}
TEST_P(GpuImageDecodeCacheTest, KeepOnlyLast2ContentIds) {
auto cache = CreateCache();
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
const PaintImage::Id paint_image_id = PaintImage::GetNextId();
std::vector<DrawImage> draw_images;
std::vector<DecodedDrawImage> decoded_draw_images;
for (int i = 0; i < 10; ++i) {
PaintImage image = CreatePaintImageInternal(
GetNormalImageSize(), SkColorSpace::MakeSRGB(), paint_image_id);
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(0.5f, 0.5f)));
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
draw_images.push_back(draw_image);
decoded_draw_images.push_back(decoded_draw_image);
if (i == 0)
continue;
EXPECT_EQ(cache->GetNumCacheEntriesForTesting(), 2u);
EXPECT_EQ(cache->GetInUseCacheEntriesForTesting(), i + 1u);
EXPECT_TRUE(cache->IsInPersistentCacheForTesting(draw_images[i]));
EXPECT_TRUE(cache->IsInPersistentCacheForTesting(draw_images[i - 1]));
}
for (int i = 0; i < 10; ++i) {
cache->DrawWithImageFinished(draw_images[i], decoded_draw_images[i]);
}
EXPECT_EQ(cache->paint_image_entries_count_for_testing(), 1u);
cache->OnMemoryPressure(base::MEMORY_PRESSURE_LEVEL_CRITICAL);
EXPECT_EQ(cache->paint_image_entries_count_for_testing(), 0u);
}
TEST_P(GpuImageDecodeCacheTest, DecodeToScale) {
if (do_yuv_decode_) {
return;
}
auto cache = CreateCache();
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
const SkISize full_size = SkISize::Make(100, 100);
const std::vector<SkISize> supported_sizes = {SkISize::Make(25, 25),
SkISize::Make(50, 50)};
const std::vector<FrameMetadata> frames = {FrameMetadata()};
const SkImageInfo info =
SkImageInfo::MakeN32Premul(full_size.width(), full_size.height(),
DefaultColorSpace().ToSkColorSpace());
sk_sp<FakePaintImageGenerator> generator =
sk_make_sp<FakePaintImageGenerator>(info, frames, true, supported_sizes);
PaintImage paint_image = PaintImageBuilder::WithDefault()
.set_id(PaintImage::GetNextId())
.set_paint_image_generator(generator)
.TakePaintImage();
DrawImage draw_image = CreateDrawImageInternal(
paint_image, CreateMatrix(SkSize::Make(0.5, 0.5)));
DecodedDrawImage decoded_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
const int expected_width =
paint_image.width() * std::abs(draw_image.scale().width());
const int expected_height =
paint_image.height() * std::abs(draw_image.scale().height());
ASSERT_TRUE(decoded_image.image());
EXPECT_EQ(decoded_image.image()->width(), expected_width);
EXPECT_EQ(decoded_image.image()->height(), expected_height);
ASSERT_EQ(generator->decode_infos().size(), 1u);
EXPECT_EQ(generator->decode_infos().at(0).width(), expected_width);
EXPECT_EQ(generator->decode_infos().at(0).height(), expected_height);
cache->DrawWithImageFinished(draw_image, decoded_image);
}
TEST_P(GpuImageDecodeCacheTest, DecodeToScaleNoneQuality) {
if (do_yuv_decode_) {
return;
}
auto cache = CreateCache();
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
SkISize full_size = SkISize::Make(100, 100);
std::vector<SkISize> supported_sizes = {SkISize::Make(25, 25),
SkISize::Make(50, 50)};
std::vector<FrameMetadata> frames = {FrameMetadata()};
sk_sp<FakePaintImageGenerator> generator =
sk_make_sp<FakePaintImageGenerator>(
SkImageInfo::MakeN32Premul(full_size.width(), full_size.height(),
DefaultColorSpace().ToSkColorSpace()),
frames, true, supported_sizes);
PaintImage paint_image = PaintImageBuilder::WithDefault()
.set_id(PaintImage::GetNextId())
.set_paint_image_generator(generator)
.TakePaintImage();
DrawImage draw_image = CreateDrawImageInternal(
paint_image, CreateMatrix(SkSize::Make(0.5, 0.5)),
nullptr , PaintFlags::FilterQuality::kNone);
DecodedDrawImage decoded_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
ASSERT_TRUE(decoded_image.image());
const int expected_drawn_width =
paint_image.width() * std::abs(draw_image.scale().width());
const int expected_drawn_height =
paint_image.height() * std::abs(draw_image.scale().height());
EXPECT_EQ(decoded_image.image()->width(), expected_drawn_width);
EXPECT_EQ(decoded_image.image()->height(), expected_drawn_height);
ASSERT_EQ(generator->decode_infos().size(), 1u);
EXPECT_EQ(generator->decode_infos().at(0).width(), full_size.width());
EXPECT_EQ(generator->decode_infos().at(0).height(), full_size.height());
cache->DrawWithImageFinished(draw_image, decoded_image);
}
TEST_P(GpuImageDecodeCacheTest, BasicMips) {
auto decode_and_check_mips = [this](PaintFlags::FilterQuality filter_quality,
SkSize scale, gfx::ColorSpace color_space,
bool should_have_mips) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image = CreateDrawImageInternal(
image, CreateMatrix(scale), &color_space, filter_quality);
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage serialized_decoded_draw_image =
cache->GetDecodedImageForDraw(draw_image);
const std::optional<uint32_t> transfer_cache_entry_id =
serialized_decoded_draw_image.transfer_cache_entry_id();
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(std::move(serialized_decoded_draw_image));
EXPECT_TRUE(decoded_draw_image.image());
EXPECT_TRUE(decoded_draw_image.image()->isTextureBacked());
EXPECT_EQ(should_have_mips, decoded_draw_image.image()->hasMipmaps());
if (do_yuv_decode_) {
CompareAllPlanesToMippedVersions(
cache.get(), draw_image, transfer_cache_entry_id, should_have_mips);
} else {
EXPECT_EQ(decoded_draw_image.image()->hasMipmaps(), should_have_mips);
}
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
cache->UnrefImage(draw_image);
};
decode_and_check_mips(PaintFlags::FilterQuality::kMedium,
SkSize::Make(1.0f, 1.0f), DefaultColorSpace(), false);
decode_and_check_mips(PaintFlags::FilterQuality::kMedium,
SkSize::Make(0.5f, 0.5f), DefaultColorSpace(), false);
decode_and_check_mips(PaintFlags::FilterQuality::kLow,
SkSize::Make(0.6f, 0.6f), DefaultColorSpace(), false);
decode_and_check_mips(PaintFlags::FilterQuality::kNone,
SkSize::Make(0.6f, 0.6f), DefaultColorSpace(), false);
decode_and_check_mips(PaintFlags::FilterQuality::kMedium,
SkSize::Make(0.6f, 0.6f), DefaultColorSpace(), true);
decode_and_check_mips(PaintFlags::FilterQuality::kMedium,
SkSize::Make(0.6f, 0.6f),
gfx::ColorSpace::CreateXYZD50(), true);
}
TEST_P(GpuImageDecodeCacheTest, MipsAddedSubsequentDraw) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
{
DrawImage draw_image = CreateDrawImageInternal(image);
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage serialized_decoded_draw_image =
cache->GetDecodedImageForDraw(draw_image);
const std::optional<uint32_t> transfer_cache_entry_id =
serialized_decoded_draw_image.transfer_cache_entry_id();
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(std::move(serialized_decoded_draw_image));
EXPECT_TRUE(decoded_draw_image.image());
EXPECT_TRUE(decoded_draw_image.image()->isTextureBacked());
if (do_yuv_decode_) {
CompareAllPlanesToMippedVersions(cache.get(), draw_image,
transfer_cache_entry_id,
false );
} else {
EXPECT_FALSE(decoded_draw_image.image()->hasMipmaps());
}
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
cache->UnrefImage(draw_image);
}
cache->ReduceCacheUsage();
{
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(0.6f, 0.6f)));
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_FALSE(result.task);
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage serialized_decoded_draw_image =
cache->GetDecodedImageForDraw(draw_image);
const std::optional<uint32_t> transfer_cache_entry_id =
serialized_decoded_draw_image.transfer_cache_entry_id();
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(std::move(serialized_decoded_draw_image));
EXPECT_TRUE(decoded_draw_image.image());
EXPECT_TRUE(decoded_draw_image.image()->isTextureBacked());
if (do_yuv_decode_) {
CompareAllPlanesToMippedVersions(cache.get(), draw_image,
transfer_cache_entry_id,
true );
} else {
EXPECT_TRUE(decoded_draw_image.image()->hasMipmaps());
}
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
cache->UnrefImage(draw_image);
}
}
TEST_P(GpuImageDecodeCacheTest, MipsAddedWhileOriginalInUse) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
struct Decode {
DrawImage image;
DecodedDrawImage decoded_image;
};
std::vector<Decode> images_to_unlock;
{
DrawImage draw_image = CreateDrawImageInternal(image);
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage serialized_decoded_draw_image =
cache->GetDecodedImageForDraw(draw_image);
const std::optional<uint32_t> transfer_cache_entry_id =
serialized_decoded_draw_image.transfer_cache_entry_id();
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(std::move(serialized_decoded_draw_image));
ASSERT_TRUE(decoded_draw_image.image());
ASSERT_TRUE(decoded_draw_image.image()->isTextureBacked());
if (do_yuv_decode_) {
CompareAllPlanesToMippedVersions(cache.get(), draw_image,
transfer_cache_entry_id,
false );
} else {
EXPECT_FALSE(decoded_draw_image.image()->hasMipmaps());
}
images_to_unlock.push_back({draw_image, decoded_draw_image});
}
{
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(0.6f, 0.6f)));
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_FALSE(result.task);
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage serialized_decoded_draw_image =
cache->GetDecodedImageForDraw(draw_image);
const std::optional<uint32_t> transfer_cache_entry_id =
serialized_decoded_draw_image.transfer_cache_entry_id();
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(std::move(serialized_decoded_draw_image));
ASSERT_TRUE(decoded_draw_image.image());
ASSERT_TRUE(decoded_draw_image.image()->isTextureBacked());
if (do_yuv_decode_) {
CompareAllPlanesToMippedVersions(cache.get(), draw_image,
transfer_cache_entry_id,
true );
} else {
EXPECT_TRUE(decoded_draw_image.image()->hasMipmaps());
}
images_to_unlock.push_back({draw_image, decoded_draw_image});
}
cache->ReduceCacheUsage();
{
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
for (const auto& draw_and_decoded_draw_image : images_to_unlock) {
cache->DrawWithImageFinished(draw_and_decoded_draw_image.image,
draw_and_decoded_draw_image.decoded_image);
cache->UnrefImage(draw_and_decoded_draw_image.image);
}
}
}
TEST_P(GpuImageDecodeCacheTest,
OriginalYUVDecodeScaledDrawCorrectlyMipsPlanes) {
if (!do_yuv_decode_) {
return;
}
auto owned_cache = CreateCache();
const uint32_t owned_cache_client_id = owned_cache->GenerateClientId();
auto decode_and_check_plane_sizes = [this, cache = owned_cache.get(),
client_id = owned_cache_client_id]() {
PaintFlags::FilterQuality filter_quality =
PaintFlags::FilterQuality::kMedium;
SkSize requires_decode_at_original_scale = SkSize::Make(0.8f, 0.8f);
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image(
image, false, SkIRect::MakeWH(image.width(), image.height()),
filter_quality, CreateMatrix(requires_decode_at_original_scale),
PaintImage::kDefaultFrameIndex, DefaultTargetColorParams());
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage serialized_decoded_draw_image =
cache->GetDecodedImageForDraw(draw_image);
const std::optional<uint32_t> transfer_cache_entry_id =
serialized_decoded_draw_image.transfer_cache_entry_id();
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(std::move(serialized_decoded_draw_image));
EXPECT_TRUE(decoded_draw_image.image());
EXPECT_TRUE(decoded_draw_image.image()->isTextureBacked());
CompareAllPlanesToMippedVersions(cache, draw_image, transfer_cache_entry_id,
true );
SkYUVAPixmapInfo yuva_pixmap_info =
GetYUVAPixmapInfo(GetNormalImageSize(), yuv_format_, yuv_data_type_);
SkISize plane_sizes[SkYUVAInfo::kMaxPlanes];
yuva_pixmap_info.yuvaInfo().planeDimensions(plane_sizes);
VerifyUploadedPlaneSizes(cache, draw_image, transfer_cache_entry_id,
plane_sizes);
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
cache->UnrefImage(draw_image);
};
yuv_format_ = YUVSubsampling::k420;
decode_and_check_plane_sizes();
yuv_format_ = YUVSubsampling::k422;
decode_and_check_plane_sizes();
yuv_format_ = YUVSubsampling::k444;
decode_and_check_plane_sizes();
}
TEST_P(GpuImageDecodeCacheTest, HighBitDepthYUVDecoding) {
if (!do_yuv_decode_) {
return;
}
auto decode_and_check_plane_sizes = [this](
GpuImageDecodeCache* cache,
uint32_t client_id,
bool decodes_to_yuv,
SkYUVAPixmapInfo::DataType
yuv_data_type = SkYUVAPixmapInfo::
DataType::kUnorm8,
gfx::ColorSpace target_cs =
gfx::ColorSpace::CreateSRGB()) {
PaintFlags::FilterQuality filter_quality =
PaintFlags::FilterQuality::kMedium;
SkSize requires_decode_at_original_scale = SkSize::Make(0.8f, 0.8f);
gfx::ColorSpace decoded_cs;
if (target_cs.IsHDR())
decoded_cs = gfx::ColorSpace::CreateHDR10();
auto sk_decoded_cs = cache->SupportsColorSpaceConversion()
? decoded_cs.ToSkColorSpace()
: nullptr;
PaintImage image =
decodes_to_yuv ? CreatePaintImageInternal(GetNormalImageSize(),
decoded_cs.ToSkColorSpace())
: CreatePaintImageForFallbackToRGB(GetNormalImageSize());
TargetColorParams target_color_params;
target_color_params.color_space = target_cs;
DrawImage draw_image(
image, false, SkIRect::MakeWH(image.width(), image.height()),
filter_quality, CreateMatrix(requires_decode_at_original_scale),
PaintImage::kDefaultFrameIndex, target_color_params);
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage serialized_decoded_draw_image =
cache->GetDecodedImageForDraw(draw_image);
const std::optional<uint32_t> transfer_cache_entry_id =
serialized_decoded_draw_image.transfer_cache_entry_id();
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(std::move(serialized_decoded_draw_image));
EXPECT_TRUE(decoded_draw_image.image());
EXPECT_TRUE(decoded_draw_image.image()->isTextureBacked());
if (decodes_to_yuv) {
CompareAllPlanesToMippedVersions(cache, draw_image,
transfer_cache_entry_id,
true );
SkYUVAPixmapInfo yuva_pixmap_info =
GetYUVAPixmapInfo(GetNormalImageSize(), yuv_format_, yuv_data_type_);
SkISize plane_sizes[SkYUVAInfo::kMaxPlanes];
yuva_pixmap_info.yuvaInfo().planeDimensions(plane_sizes);
VerifyUploadedPlaneSizes(cache, draw_image, transfer_cache_entry_id,
plane_sizes, yuv_data_type, sk_decoded_cs.get());
auto expected_image_cs =
cache->SupportsColorSpaceConversion() && sk_decoded_cs
? target_color_params.color_space.ToSkColorSpace()
: nullptr;
if (expected_image_cs) {
EXPECT_TRUE(SkColorSpace::Equals(
expected_image_cs.get(), decoded_draw_image.image()->colorSpace()));
}
} else {
EXPECT_FALSE(transfer_cache_helper_
.GetEntryAs<ServiceImageTransferCacheEntry>(
*transfer_cache_entry_id)
->is_yuv());
}
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
cache->UnrefImage(draw_image);
};
gpu::Capabilities original_caps;
{
viz::RasterContextProvider::ScopedRasterContextLock auto_lock(
context_provider_.get());
original_caps = context_provider_->ContextCapabilities();
}
const auto hdr_cs = gfx::ColorSpace::CreateHDR10();
{
auto r16_caps = original_caps;
r16_caps.texture_norm16 = true;
r16_caps.texture_half_float_linear = true;
context_provider_->SetContextCapabilitiesOverride(r16_caps);
auto r16_cache = CreateCache();
const uint32_t client_id = r16_cache->GenerateClientId();
yuv_data_type_ = SkYUVAPixmapInfo::DataType::kUnorm16;
yuv_format_ = YUVSubsampling::k420;
decode_and_check_plane_sizes(r16_cache.get(), client_id, true,
SkYUVAPixmapInfo::DataType::kUnorm16,
DefaultColorSpace());
yuv_format_ = YUVSubsampling::k422;
decode_and_check_plane_sizes(r16_cache.get(), client_id, true,
SkYUVAPixmapInfo::DataType::kUnorm16,
DefaultColorSpace());
yuv_format_ = YUVSubsampling::k444;
decode_and_check_plane_sizes(r16_cache.get(), client_id, true,
SkYUVAPixmapInfo::DataType::kUnorm16,
DefaultColorSpace());
yuv_format_ = YUVSubsampling::k420;
decode_and_check_plane_sizes(r16_cache.get(), client_id, true,
SkYUVAPixmapInfo::DataType::kUnorm16, hdr_cs);
yuv_format_ = YUVSubsampling::k422;
decode_and_check_plane_sizes(r16_cache.get(), client_id, true,
SkYUVAPixmapInfo::DataType::kUnorm16, hdr_cs);
yuv_format_ = YUVSubsampling::k444;
decode_and_check_plane_sizes(r16_cache.get(), client_id, true,
SkYUVAPixmapInfo::DataType::kUnorm16, hdr_cs);
}
{
auto f16_caps = original_caps;
f16_caps.texture_norm16 = false;
f16_caps.texture_half_float_linear = true;
context_provider_->SetContextCapabilitiesOverride(f16_caps);
auto f16_cache = CreateCache();
const uint32_t client_id = f16_cache->GenerateClientId();
yuv_data_type_ = SkYUVAPixmapInfo::DataType::kFloat16;
yuv_format_ = YUVSubsampling::k420;
decode_and_check_plane_sizes(f16_cache.get(), client_id, true,
SkYUVAPixmapInfo::DataType::kFloat16,
DefaultColorSpace());
yuv_format_ = YUVSubsampling::k422;
decode_and_check_plane_sizes(f16_cache.get(), client_id, true,
SkYUVAPixmapInfo::DataType::kFloat16,
DefaultColorSpace());
yuv_format_ = YUVSubsampling::k444;
decode_and_check_plane_sizes(f16_cache.get(), client_id, true,
SkYUVAPixmapInfo::DataType::kFloat16,
DefaultColorSpace());
yuv_format_ = YUVSubsampling::k420;
decode_and_check_plane_sizes(f16_cache.get(), client_id, true,
SkYUVAPixmapInfo::DataType::kFloat16, hdr_cs);
yuv_format_ = YUVSubsampling::k422;
decode_and_check_plane_sizes(f16_cache.get(), client_id, true,
SkYUVAPixmapInfo::DataType::kFloat16, hdr_cs);
yuv_format_ = YUVSubsampling::k444;
decode_and_check_plane_sizes(f16_cache.get(), client_id, true,
SkYUVAPixmapInfo::DataType::kFloat16, hdr_cs);
}
{
auto no_yuv16_caps = original_caps;
no_yuv16_caps.texture_norm16 = false;
no_yuv16_caps.texture_half_float_linear = false;
context_provider_->SetContextCapabilitiesOverride(no_yuv16_caps);
auto no_yuv16_cache = CreateCache();
const uint32_t client_id = no_yuv16_cache->GenerateClientId();
yuv_data_type_ = SkYUVAPixmapInfo::DataType::kUnorm16;
yuv_format_ = YUVSubsampling::k420;
decode_and_check_plane_sizes(no_yuv16_cache.get(), client_id, false);
yuv_format_ = YUVSubsampling::k422;
decode_and_check_plane_sizes(no_yuv16_cache.get(), client_id, false);
yuv_format_ = YUVSubsampling::k444;
decode_and_check_plane_sizes(no_yuv16_cache.get(), client_id, false);
yuv_data_type_ = SkYUVAPixmapInfo::DataType::kFloat16;
yuv_format_ = YUVSubsampling::k420;
decode_and_check_plane_sizes(no_yuv16_cache.get(), client_id, false);
yuv_format_ = YUVSubsampling::k422;
decode_and_check_plane_sizes(no_yuv16_cache.get(), client_id, false);
yuv_format_ = YUVSubsampling::k444;
decode_and_check_plane_sizes(no_yuv16_cache.get(), client_id, false);
}
}
TEST_P(GpuImageDecodeCacheTest, ScaledYUVDecodeScaledDrawCorrectlyMipsPlanes) {
if (!do_yuv_decode_) {
return;
}
auto owned_cache = CreateCache();
const uint32_t owned_cache_client_id = owned_cache->GenerateClientId();
auto decode_and_check_plane_sizes =
[this, cache = owned_cache.get(), client_id = owned_cache_client_id](
SkSize scaled_size,
const SkISize mipped_plane_sizes[SkYUVAInfo::kMaxPlanes]) {
PaintFlags::FilterQuality filter_quality =
PaintFlags::FilterQuality::kMedium;
gfx::Size image_size = GetNormalImageSize();
PaintImage image = CreatePaintImageInternal(image_size);
DrawImage draw_image(
image, false, SkIRect::MakeWH(image.width(), image.height()),
filter_quality, CreateMatrix(scaled_size),
PaintImage::kDefaultFrameIndex, DefaultTargetColorParams());
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage serialized_decoded_draw_image =
cache->GetDecodedImageForDraw(draw_image);
const std::optional<uint32_t> transfer_cache_entry_id =
serialized_decoded_draw_image.transfer_cache_entry_id();
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(std::move(serialized_decoded_draw_image));
EXPECT_TRUE(decoded_draw_image.image());
EXPECT_TRUE(decoded_draw_image.image()->isTextureBacked());
CompareAllPlanesToMippedVersions(cache, draw_image,
transfer_cache_entry_id,
true );
VerifyUploadedPlaneSizes(cache, draw_image, transfer_cache_entry_id,
mipped_plane_sizes);
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
cache->UnrefImage(draw_image);
};
gfx::Size image_size = GetNormalImageSize();
SkISize mipped_plane_sizes[kNumYUVPlanes];
SkSize less_than_half_scale = SkSize::Make(0.45f, 0.45f);
mipped_plane_sizes[static_cast<size_t>(YUVIndex::kY)] = SkISize::Make(
(image_size.width() + 1) / 2, (image_size.height() + 1) / 2);
mipped_plane_sizes[static_cast<size_t>(YUVIndex::kU)] =
mipped_plane_sizes[static_cast<size_t>(YUVIndex::kY)];
mipped_plane_sizes[static_cast<size_t>(YUVIndex::kV)] =
mipped_plane_sizes[static_cast<size_t>(YUVIndex::kY)];
yuv_format_ = YUVSubsampling::k420;
decode_and_check_plane_sizes(less_than_half_scale, mipped_plane_sizes);
yuv_format_ = YUVSubsampling::k422;
decode_and_check_plane_sizes(less_than_half_scale, mipped_plane_sizes);
yuv_format_ = YUVSubsampling::k444;
decode_and_check_plane_sizes(less_than_half_scale, mipped_plane_sizes);
SkSize one_quarter_scale = SkSize::Make(0.20f, 0.20f);
mipped_plane_sizes[static_cast<size_t>(YUVIndex::kY)] = SkISize::Make(
(image_size.width() + 1) / 4, (image_size.height() + 1) / 4);
mipped_plane_sizes[static_cast<size_t>(YUVIndex::kU)] =
mipped_plane_sizes[static_cast<size_t>(YUVIndex::kY)];
mipped_plane_sizes[static_cast<size_t>(YUVIndex::kV)] =
mipped_plane_sizes[static_cast<size_t>(YUVIndex::kY)];
yuv_format_ = YUVSubsampling::k420;
decode_and_check_plane_sizes(one_quarter_scale, mipped_plane_sizes);
yuv_format_ = YUVSubsampling::k422;
decode_and_check_plane_sizes(one_quarter_scale, mipped_plane_sizes);
yuv_format_ = YUVSubsampling::k444;
decode_and_check_plane_sizes(one_quarter_scale, mipped_plane_sizes);
}
TEST_P(GpuImageDecodeCacheTest, GetBorderlineLargeDecodedImageForDraw) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage almost_too_large_image =
CreatePaintImageInternal(gfx::Size(max_texture_size_, max_texture_size_));
DrawImage draw_image = CreateDrawImageInternal(almost_too_large_image);
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
ASSERT_TRUE(result.task);
ASSERT_EQ(result.task->dependencies().size(), 1u);
ASSERT_TRUE(result.task->dependencies()[0]);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
EXPECT_TRUE(decoded_draw_image.image());
EXPECT_TRUE(decoded_draw_image.image()->isTextureBacked());
EXPECT_TRUE(decoded_draw_image.is_budgeted());
EXPECT_FALSE(cache->DiscardableIsLockedForTesting(draw_image));
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest, OutOfRasterDecodeForBitmaps) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreateBitmapImageInternal(GetNormalImageSize());
DrawImage draw_image = CreateDrawImageInternal(image);
ImageDecodeCache::TaskResult result =
cache->GetOutOfRasterDecodeTaskForImageAndRef(client_id, draw_image,
false);
EXPECT_TRUE(result.need_unref);
EXPECT_FALSE(result.task);
EXPECT_FALSE(result.is_at_raster_decode);
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest, DarkModeDecodedDrawImage) {
if (do_yuv_decode_)
return;
std::unique_ptr<FakeRasterDarkModeFilter> dark_mode_filter =
std::make_unique<FakeRasterDarkModeFilter>();
auto cache = CreateCache(kGpuMemoryLimitBytes, dark_mode_filter.get());
const uint32_t client_id = cache->GenerateClientId();
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image = CreateDrawImageWithDarkModeInternal(image);
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
GetImageAndDrawFinishedForDarkMode(cache.get(), draw_image,
dark_mode_filter.get());
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest, DarkModeImageCacheSize) {
if (do_yuv_decode_)
return;
std::unique_ptr<FakeRasterDarkModeFilter> dark_mode_filter =
std::make_unique<FakeRasterDarkModeFilter>();
auto cache = CreateCache(kGpuMemoryLimitBytes, dark_mode_filter.get());
const uint32_t client_id = cache->GenerateClientId();
PaintImage image1 = CreatePaintImageInternal(GetNormalImageSize());
PaintImage image2 = CreatePaintImageInternal(gfx::Size(50, 50));
DrawImage draw_image11 = CreateDrawImageWithDarkModeInternal(image1);
EXPECT_EQ(cache->GetDarkModeImageCacheSizeForTesting(draw_image11), 0u);
ImageDecodeCache::TaskResult result11 = cache->GetTaskForImageAndRef(
client_id, draw_image11, ImageDecodeCache::TracingInfo());
TestTileTaskRunner::ProcessTask(result11.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result11.task.get());
GetImageAndDrawFinishedForDarkMode(cache.get(), draw_image11,
dark_mode_filter.get());
EXPECT_EQ(cache->GetDarkModeImageCacheSizeForTesting(draw_image11), 1u);
GetImageAndDrawFinishedForDarkMode(cache.get(), draw_image11,
dark_mode_filter.get());
EXPECT_EQ(cache->GetDarkModeImageCacheSizeForTesting(draw_image11), 1u);
SkIRect src = SkIRect::MakeWH(10, 10);
DrawImage draw_image12 = CreateDrawImageWithDarkModeInternal(
image1, SkM44(), nullptr, PaintFlags::FilterQuality::kMedium, &src);
ImageDecodeCache::TaskResult result12 = cache->GetTaskForImageAndRef(
client_id, draw_image12, ImageDecodeCache::TracingInfo());
GetImageAndDrawFinishedForDarkMode(cache.get(), draw_image12,
dark_mode_filter.get());
EXPECT_EQ(cache->GetDarkModeImageCacheSizeForTesting(draw_image12), 2u);
DrawImage draw_image13 = CreateDrawImageWithDarkModeInternal(image1);
ImageDecodeCache::TaskResult result13 = cache->GetTaskForImageAndRef(
client_id, draw_image13, ImageDecodeCache::TracingInfo());
GetImageAndDrawFinishedForDarkMode(cache.get(), draw_image13,
dark_mode_filter.get());
EXPECT_EQ(cache->GetDarkModeImageCacheSizeForTesting(draw_image13), 2u);
DrawImage draw_image21 = CreateDrawImageWithDarkModeInternal(image2);
EXPECT_EQ(cache->GetDarkModeImageCacheSizeForTesting(draw_image21), 0u);
ImageDecodeCache::TaskResult result21 = cache->GetTaskForImageAndRef(
client_id, draw_image21, ImageDecodeCache::TracingInfo());
TestTileTaskRunner::ProcessTask(result21.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result21.task.get());
GetImageAndDrawFinishedForDarkMode(cache.get(), draw_image21,
dark_mode_filter.get());
EXPECT_EQ(cache->GetDarkModeImageCacheSizeForTesting(draw_image21), 1u);
EXPECT_EQ(cache->GetDarkModeImageCacheSizeForTesting(draw_image13), 2u);
cache->UnrefImage(draw_image11);
cache->UnrefImage(draw_image12);
cache->UnrefImage(draw_image13);
cache->UnrefImage(draw_image21);
}
TEST_P(GpuImageDecodeCacheTest, DarkModeNeedsDarkModeFilter) {
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image_without_dark_mode = CreateDrawImageInternal(image);
DrawImage draw_image_with_dark_mode =
CreateDrawImageWithDarkModeInternal(image);
std::unique_ptr<FakeRasterDarkModeFilter> dark_mode_filter =
std::make_unique<FakeRasterDarkModeFilter>();
auto cache = CreateCache(kGpuMemoryLimitBytes, dark_mode_filter.get());
const uint32_t client_id = cache->GenerateClientId();
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image_with_dark_mode, ImageDecodeCache::TracingInfo());
EXPECT_FALSE(
cache->NeedsDarkModeFilterForTesting(draw_image_without_dark_mode));
if (do_yuv_decode_) {
EXPECT_FALSE(
cache->NeedsDarkModeFilterForTesting(draw_image_with_dark_mode));
} else {
EXPECT_TRUE(
cache->NeedsDarkModeFilterForTesting(draw_image_with_dark_mode));
}
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
EXPECT_FALSE(cache->NeedsDarkModeFilterForTesting(draw_image_with_dark_mode));
cache->UnrefImage(draw_image_with_dark_mode);
}
TEST_P(GpuImageDecodeCacheTest, ClippedAndScaledDrawImageRemovesCacheEntry) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
cache->SetWorkingSetLimitsForTesting(0 , 0 );
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image =
CreateDrawImageInternal(image, CreateMatrix(SkSize::Make(0.5f, 0.5f)));
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
EnsureImageBacked(cache->GetDecodedImageForDraw(draw_image));
EXPECT_TRUE(decoded_draw_image.image());
EXPECT_TRUE(decoded_draw_image.image()->isTextureBacked());
EXPECT_FALSE(cache->DiscardableIsLockedForTesting(draw_image));
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
EXPECT_EQ(cache->GetNumCacheEntriesForTesting(), 1u);
auto clipped_rect = SkIRect::MakeWH(image.width() * 0.9f, image.height());
DrawImage clipped_draw_image = CreateDrawImageInternal(
image, CreateMatrix(SkSize::Make(0.5f, 0.5f)), nullptr,
PaintFlags::FilterQuality::kMedium, &clipped_rect);
ImageDecodeCache::TaskResult clipped_result = cache->GetTaskForImageAndRef(
client_id, clipped_draw_image, ImageDecodeCache::TracingInfo());
EXPECT_EQ(cache->GetNumCacheEntriesForTesting(),
enable_clipped_image_scaling_ ? 1u : 0u);
}
SkColorType test_color_types[] = {kN32_SkColorType, kARGB_4444_SkColorType,
kRGBA_F16_SkColorType};
INSTANTIATE_TEST_SUITE_P(
GpuImageDecodeCacheTests,
GpuImageDecodeCacheTest,
testing::Combine(testing::ValuesIn(test_color_types),
testing::Bool() ,
testing::Values(false) ,
testing::Values(false) ));
class GpuImageDecodeCachePurgeOnTimerTest : public GpuImageDecodeCacheTest {
public:
static GpuImageDecodeCachePurgeOnTimerTest* last_setup_test_;
void SetUp() override {
GpuImageDecodeCacheTest::SetUp();
feature_list_enable_purge_.InitAndDisableFeature(
features::kPruneOldTransferCacheEntries);
task_runner_ = base::MakeRefCounted<base::TestMockTimeTaskRunner>();
current_default_handle_ = std::make_unique<
base::SingleThreadTaskRunner::CurrentHandleOverrideForTesting>(
task_runner_);
cache_ = CreateCache();
client_id_ = cache_->GenerateClientId();
last_setup_test_ = this;
time_override_ = std::make_unique<base::subtle::ScopedTimeClockOverrides>(
nullptr,
[]() {
return last_setup_test_->task_runner_->GetMockTickClock()->NowTicks();
},
nullptr);
}
void TearDown() override {
last_setup_test_ = nullptr;
GpuImageDecodeCacheTest::TearDown();
}
void FastForwardBy(base::TimeDelta t) { task_runner_->FastForwardBy(t); }
void CreateAndUnrefImage(unsigned n = 1) {
while (n--) {
PaintImage image = CreatePaintImageInternal(GetNormalImageSize());
DrawImage draw_image = CreateDrawImageInternal(image);
ImageDecodeCache::TaskResult result = cache_->GetTaskForImageAndRef(
client_id_, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
cache_->TouchCacheEntryForTesting(draw_image);
cache_->UnrefImage(draw_image);
}
}
base::test::ScopedFeatureList feature_list_enable_purge_;
std::unique_ptr<base::SingleThreadTaskRunner::CurrentHandleOverrideForTesting>
current_default_handle_ = nullptr;
std::unique_ptr<GpuImageDecodeCache> cache_ = nullptr;
scoped_refptr<base::TestMockTimeTaskRunner> task_runner_;
uint32_t client_id_;
std::unique_ptr<base::subtle::ScopedTimeClockOverrides> time_override_;
};
GpuImageDecodeCachePurgeOnTimerTest*
GpuImageDecodeCachePurgeOnTimerTest::last_setup_test_ = nullptr;
TEST_P(GpuImageDecodeCachePurgeOnTimerTest, SimplePurgeOneImage) {
ASSERT_EQ(cache_->GetNumCacheEntriesForTesting(), 0u);
ASSERT_FALSE(cache_->HasPendingPurgeTaskForTesting());
ASSERT_EQ(cache_->ids_pending_deletion_count_for_testing(), 0u);
CreateAndUnrefImage();
ASSERT_EQ(cache_->GetNumCacheEntriesForTesting(), 1u);
ASSERT_TRUE(cache_->HasPendingPurgeTaskForTesting());
ASSERT_EQ(cache_->ids_pending_deletion_count_for_testing(), 0u);
FastForwardBy(GpuImageDecodeCache::get_purge_interval() / 2);
EXPECT_EQ(cache_->GetNumCacheEntriesForTesting(), 1u);
EXPECT_TRUE(cache_->HasPendingPurgeTaskForTesting());
EXPECT_EQ(cache_->ids_pending_deletion_count_for_testing(), 0u);
FastForwardBy(GpuImageDecodeCache::get_purge_interval());
EXPECT_EQ(cache_->GetNumCacheEntriesForTesting(), 0u);
EXPECT_FALSE(cache_->HasPendingPurgeTaskForTesting());
EXPECT_EQ(cache_->ids_pending_deletion_count_for_testing(), 0u);
}
TEST_P(GpuImageDecodeCachePurgeOnTimerTest, SimplePurgeMultipleImages) {
ASSERT_EQ(cache_->GetNumCacheEntriesForTesting(), 0u);
ASSERT_FALSE(cache_->HasPendingPurgeTaskForTesting());
ASSERT_EQ(cache_->ids_pending_deletion_count_for_testing(), 0u);
CreateAndUnrefImage(3);
ASSERT_EQ(cache_->GetNumCacheEntriesForTesting(), 3u);
ASSERT_TRUE(cache_->HasPendingPurgeTaskForTesting());
ASSERT_EQ(cache_->ids_pending_deletion_count_for_testing(), 0u);
FastForwardBy(GpuImageDecodeCache::get_purge_interval() / 2);
EXPECT_EQ(cache_->GetNumCacheEntriesForTesting(), 3u);
EXPECT_TRUE(cache_->HasPendingPurgeTaskForTesting());
EXPECT_EQ(cache_->ids_pending_deletion_count_for_testing(), 0u);
FastForwardBy(GpuImageDecodeCache::get_purge_interval());
EXPECT_EQ(cache_->GetNumCacheEntriesForTesting(), 0u);
EXPECT_FALSE(cache_->HasPendingPurgeTaskForTesting());
EXPECT_EQ(cache_->ids_pending_deletion_count_for_testing(), 0u);
}
TEST_P(GpuImageDecodeCachePurgeOnTimerTest, MultipleImagesWithDelay) {
ASSERT_EQ(cache_->GetNumCacheEntriesForTesting(), 0u);
ASSERT_FALSE(cache_->HasPendingPurgeTaskForTesting());
ASSERT_EQ(cache_->ids_pending_deletion_count_for_testing(), 0u);
CreateAndUnrefImage(3);
ASSERT_EQ(cache_->GetNumCacheEntriesForTesting(), 3u);
ASSERT_TRUE(cache_->HasPendingPurgeTaskForTesting());
ASSERT_EQ(cache_->ids_pending_deletion_count_for_testing(), 0u);
FastForwardBy(GpuImageDecodeCache::get_purge_interval() / 2);
CreateAndUnrefImage(4);
ASSERT_EQ(cache_->GetNumCacheEntriesForTesting(), 7u);
ASSERT_TRUE(cache_->HasPendingPurgeTaskForTesting());
ASSERT_EQ(cache_->ids_pending_deletion_count_for_testing(), 0u);
FastForwardBy(GpuImageDecodeCache::get_purge_interval() / 2);
EXPECT_EQ(cache_->GetNumCacheEntriesForTesting(), 4u);
EXPECT_TRUE(cache_->HasPendingPurgeTaskForTesting());
EXPECT_EQ(cache_->ids_pending_deletion_count_for_testing(), 0u);
FastForwardBy(GpuImageDecodeCache::get_purge_interval() / 2);
EXPECT_EQ(cache_->GetNumCacheEntriesForTesting(), 4u);
EXPECT_TRUE(cache_->HasPendingPurgeTaskForTesting());
EXPECT_EQ(cache_->ids_pending_deletion_count_for_testing(), 0u);
FastForwardBy(GpuImageDecodeCache::get_purge_interval() / 2);
EXPECT_EQ(cache_->GetNumCacheEntriesForTesting(), 0u);
EXPECT_FALSE(cache_->HasPendingPurgeTaskForTesting());
EXPECT_EQ(cache_->ids_pending_deletion_count_for_testing(), 0u);
}
TEST_P(GpuImageDecodeCachePurgeOnTimerTest, MultipleImagesWithTimeGap) {
ASSERT_EQ(cache_->GetNumCacheEntriesForTesting(), 0u);
ASSERT_FALSE(cache_->HasPendingPurgeTaskForTesting());
ASSERT_EQ(cache_->ids_pending_deletion_count_for_testing(), 0u);
CreateAndUnrefImage(3);
ASSERT_EQ(cache_->GetNumCacheEntriesForTesting(), 3u);
ASSERT_TRUE(cache_->HasPendingPurgeTaskForTesting());
ASSERT_EQ(cache_->ids_pending_deletion_count_for_testing(), 0u);
FastForwardBy(GpuImageDecodeCache::get_purge_interval());
ASSERT_EQ(cache_->GetNumCacheEntriesForTesting(), 0u);
ASSERT_FALSE(cache_->HasPendingPurgeTaskForTesting());
ASSERT_EQ(cache_->ids_pending_deletion_count_for_testing(), 0u);
CreateAndUnrefImage(4);
ASSERT_EQ(cache_->GetNumCacheEntriesForTesting(), 4u);
ASSERT_TRUE(cache_->HasPendingPurgeTaskForTesting());
ASSERT_EQ(cache_->ids_pending_deletion_count_for_testing(), 0u);
FastForwardBy(GpuImageDecodeCache::get_purge_interval());
EXPECT_EQ(cache_->GetNumCacheEntriesForTesting(), 0u);
EXPECT_FALSE(cache_->HasPendingPurgeTaskForTesting());
EXPECT_EQ(cache_->ids_pending_deletion_count_for_testing(), 0u);
}
TEST_P(GpuImageDecodeCachePurgeOnTimerTest, NoDeadlock) {
ASSERT_EQ(cache_->GetNumCacheEntriesForTesting(), 0u);
ASSERT_FALSE(cache_->HasPendingPurgeTaskForTesting());
ASSERT_EQ(cache_->ids_pending_deletion_count_for_testing(), 0u);
CreateAndUnrefImage(2);
ASSERT_EQ(cache_->GetNumCacheEntriesForTesting(), 2u);
ASSERT_TRUE(cache_->HasPendingPurgeTaskForTesting());
ASSERT_EQ(cache_->ids_pending_deletion_count_for_testing(), 0u);
ASSERT_TRUE(cache_->AcquireContextLockForTesting());
FastForwardBy(GpuImageDecodeCache::get_purge_interval());
ASSERT_EQ(cache_->GetNumCacheEntriesForTesting(), 0u);
ASSERT_TRUE(cache_->HasPendingPurgeTaskForTesting());
ASSERT_EQ(cache_->ids_pending_deletion_count_for_testing(), 2u);
FastForwardBy(GpuImageDecodeCache::get_purge_interval());
ASSERT_EQ(cache_->GetNumCacheEntriesForTesting(), 0u);
ASSERT_TRUE(cache_->HasPendingPurgeTaskForTesting());
ASSERT_EQ(cache_->ids_pending_deletion_count_for_testing(), 2u);
cache_->ReleaseContextLockForTesting();
FastForwardBy(GpuImageDecodeCache::get_purge_interval());
EXPECT_EQ(cache_->GetNumCacheEntriesForTesting(), 0u);
EXPECT_FALSE(cache_->HasPendingPurgeTaskForTesting());
EXPECT_EQ(cache_->ids_pending_deletion_count_for_testing(), 0u);
}
TEST_P(GpuImageDecodeCachePurgeOnTimerTest, NoCache) {
const uint32_t client_id = cache_->GenerateClientId();
PaintImage image_no_cache =
PaintImageBuilder::WithCopy(
CreatePaintImageInternal(GetNormalImageSize()))
.set_no_cache(true)
.TakePaintImage();
DrawImage draw_image = CreateDrawImageInternal(image_no_cache);
ImageDecodeCache::TaskResult result = cache_->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
EXPECT_GT(cache_->GetWorkingSetBytesForTesting(), 0u);
EXPECT_EQ(cache_->GetNumCacheEntriesForTesting(), 0u);
cache_->UnrefImage(draw_image);
EXPECT_EQ(cache_->GetWorkingSetBytesForTesting(), 0u);
EXPECT_EQ(cache_->GetNumCacheEntriesForTesting(), 0u);
}
INSTANTIATE_TEST_SUITE_P(
GpuImageDecodeCacheTests,
GpuImageDecodeCachePurgeOnTimerTest,
testing::Combine(testing::Values(kN32_SkColorType),
testing::Bool() ,
testing::Values(false) ,
testing::Bool() ));
TEST_P(GpuImageDecodeCacheTest, GainmapImage) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
SkImageInfo base_info = SkImageInfo::Make(
16, 32, kN32_SkColorType, kPremul_SkAlphaType, SkColorSpace::MakeSRGB());
SkImageInfo gain_info = SkImageInfo::Make(
8, 16, kN32_SkColorType, kPremul_SkAlphaType, SkColorSpace::MakeSRGB());
SkGainmapInfo gainmap_info;
PaintImage paint_image;
{
const bool allocate_encoded_memory = true;
auto base_generator = sk_make_sp<FakePaintImageGenerator>(
base_info, std::vector<FrameMetadata>{FrameMetadata()},
allocate_encoded_memory);
auto gain_generator = sk_make_sp<FakePaintImageGenerator>(
gain_info, std::vector<FrameMetadata>{FrameMetadata()},
allocate_encoded_memory);
paint_image =
PaintImageBuilder::WithDefault()
.set_id(PaintImage::GetNextId())
.set_paint_image_generator(base_generator)
.set_gainmap_paint_image_generator(gain_generator, gainmap_info)
.set_decoding_mode(PaintImage::DecodingMode::kUnspecified)
.TakePaintImage();
}
DrawImage draw_image = CreateDrawImageInternal(paint_image);
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
cache->GetDecodedImageForDraw(draw_image);
auto* entry = GetLastTransferredCacheEntry();
auto service_base_image = entry->image();
auto service_gain_image = entry->gainmap_image();
sk_sp<SkImage> service_image = GetLastTransferredImage();
ASSERT_TRUE(service_base_image);
ASSERT_TRUE(service_gain_image);
EXPECT_TRUE(service_base_image->isTextureBacked());
EXPECT_TRUE(service_gain_image->isTextureBacked());
EXPECT_EQ(base_info.width(), service_base_image->width());
EXPECT_EQ(base_info.height(), service_base_image->height());
EXPECT_EQ(gain_info.width(), service_gain_image->width());
EXPECT_EQ(gain_info.height(), service_gain_image->height());
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
cache->UnrefImage(draw_image);
}
TEST_P(GpuImageDecodeCacheTest, GainmapImageFailsDecode) {
auto cache = CreateCache();
const uint32_t client_id = cache->GenerateClientId();
for (int i = 0; i < 2; ++i) {
SkImageInfo base_info =
SkImageInfo::Make(16, 32, kN32_SkColorType, kPremul_SkAlphaType,
SkColorSpace::MakeSRGB());
SkImageInfo gain_info = SkImageInfo::Make(
8, 16, kN32_SkColorType, kPremul_SkAlphaType, SkColorSpace::MakeSRGB());
SkGainmapInfo gainmap_info;
PaintImage paint_image;
{
const bool allocate_encoded_memory = true;
auto base_generator = sk_make_sp<FakePaintImageGenerator>(
base_info, std::vector<FrameMetadata>{FrameMetadata()},
allocate_encoded_memory);
auto gain_generator = sk_make_sp<FakePaintImageGenerator>(
gain_info, std::vector<FrameMetadata>{FrameMetadata()},
allocate_encoded_memory);
if (i == 1) {
gain_generator->SetForceFailDecode();
}
paint_image =
PaintImageBuilder::WithDefault()
.set_id(PaintImage::GetNextId())
.set_paint_image_generator(base_generator)
.set_gainmap_paint_image_generator(gain_generator, gainmap_info)
.set_decoding_mode(PaintImage::DecodingMode::kUnspecified)
.TakePaintImage();
}
DrawImage draw_image = CreateDrawImageInternal(paint_image);
ImageDecodeCache::TaskResult result = cache->GetTaskForImageAndRef(
client_id, draw_image, ImageDecodeCache::TracingInfo());
EXPECT_TRUE(result.need_unref);
EXPECT_TRUE(result.task);
TestTileTaskRunner::ProcessTask(result.task->dependencies()[0].get());
TestTileTaskRunner::ProcessTask(result.task.get());
viz::RasterContextProvider::ScopedRasterContextLock context_lock(
context_provider());
DecodedDrawImage decoded_draw_image =
cache->GetDecodedImageForDraw(draw_image);
auto* entry = GetLastTransferredCacheEntry();
auto service_base_image = entry->image();
auto service_gain_image = entry->gainmap_image();
ASSERT_TRUE(service_base_image);
EXPECT_TRUE(service_base_image->isTextureBacked());
EXPECT_EQ(base_info.width(), service_base_image->width());
EXPECT_EQ(base_info.height(), service_base_image->height());
if (i == 0) {
ASSERT_TRUE(service_gain_image);
EXPECT_TRUE(service_gain_image->isTextureBacked());
EXPECT_EQ(gain_info.width(), service_gain_image->width());
EXPECT_EQ(gain_info.height(), service_gain_image->height());
} else {
EXPECT_FALSE(service_gain_image);
}
cache->DrawWithImageFinished(draw_image, decoded_draw_image);
cache->UnrefImage(draw_image);
}
}
}
}