blob: 9ead8818a9c2645bcab63bee4a3b27277e966f6f [file] [log] [blame]
// Copyright 2013 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "cc/picture_layer_impl.h"
#include "cc/layer_tree_impl.h"
#include "cc/test/fake_content_layer_client.h"
#include "cc/test/fake_impl_proxy.h"
#include "cc/test/fake_layer_tree_host_impl.h"
#include "cc/test/fake_output_surface.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "ui/gfx/rect_conversions.h"
namespace cc {
namespace {
class TestablePictureLayerImpl : public PictureLayerImpl {
public:
static scoped_ptr<TestablePictureLayerImpl> create(
LayerTreeImpl* treeImpl,
int id,
scoped_refptr<PicturePileImpl> pile)
{
return make_scoped_ptr(new TestablePictureLayerImpl(treeImpl, id, pile));
}
PictureLayerTilingSet& tilings() { return *tilings_; }
Region& invalidation() { return invalidation_; }
using PictureLayerImpl::AddTiling;
using PictureLayerImpl::CleanUpTilingsOnActiveLayer;
private:
TestablePictureLayerImpl(
LayerTreeImpl* treeImpl,
int id,
scoped_refptr<PicturePileImpl> pile)
: PictureLayerImpl(treeImpl, id) {
pile_ = pile;
setBounds(pile_->size());
CreateTilingSet();
}
};
class TestablePicturePileImpl : public PicturePileImpl {
public:
static scoped_refptr<TestablePicturePileImpl> CreateFilledPile(
gfx::Size tile_size,
gfx::Size layer_bounds) {
scoped_refptr<TestablePicturePileImpl> pile(new TestablePicturePileImpl());
pile->tiling().SetTotalSize(layer_bounds);
pile->tiling().SetMaxTextureSize(tile_size);
for (int x = 0; x < pile->tiling().num_tiles_x(); ++x) {
for (int y = 0; y < pile->tiling().num_tiles_y(); ++y)
pile->AddRecordingAt(x, y);
}
pile->UpdateRecordedRegion();
return pile;
}
static scoped_refptr<TestablePicturePileImpl> CreateEmptyPile(
gfx::Size tile_size,
gfx::Size layer_bounds) {
scoped_refptr<TestablePicturePileImpl> pile(new TestablePicturePileImpl());
pile->tiling().SetTotalSize(layer_bounds);
pile->tiling().SetMaxTextureSize(tile_size);
pile->UpdateRecordedRegion();
return pile;
}
TilingData& tiling() { return tiling_; }
void AddRecordingAt(int x, int y) {
EXPECT_GE(x, 0);
EXPECT_GE(y, 0);
EXPECT_LT(x, tiling_.num_tiles_x());
EXPECT_LT(y, tiling_.num_tiles_y());
if (HasRecordingAt(x, y))
return;
gfx::Rect bounds(tiling().TileBounds(x, y));
scoped_refptr<Picture> picture(Picture::Create(bounds));
FakeContentLayerClient client;
picture->Record(&client, NULL);
picture_list_map_[std::pair<int, int>(x, y)].push_back(picture);
EXPECT_TRUE(HasRecordingAt(x, y));
UpdateRecordedRegion();
}
void RemoveRecordingAt(int x, int y) {
EXPECT_GE(x, 0);
EXPECT_GE(y, 0);
EXPECT_LT(x, tiling_.num_tiles_x());
EXPECT_LT(y, tiling_.num_tiles_y());
if (!HasRecordingAt(x, y))
return;
picture_list_map_.erase(std::pair<int, int>(x, y));
EXPECT_FALSE(HasRecordingAt(x, y));
UpdateRecordedRegion();
}
protected:
~TestablePicturePileImpl() {
}
};
class ImplSidePaintingSettings : public LayerTreeSettings {
public:
ImplSidePaintingSettings() {
implSidePainting = true;
}
};
class PictureLayerImplTest : public testing::Test {
public:
PictureLayerImplTest()
: host_impl_(ImplSidePaintingSettings(), &proxy_),
id_(7) {
host_impl_.initializeRenderer(createFakeOutputSurface());
}
~PictureLayerImplTest() {
}
void SetupTrees(
scoped_refptr<PicturePileImpl> pending_pile,
scoped_refptr<PicturePileImpl> active_pile) {
SetupPendingTree(active_pile);
host_impl_.activatePendingTree();
active_layer_ = static_cast<TestablePictureLayerImpl*>(
host_impl_.activeTree()->LayerById(id_));
SetupPendingTree(pending_pile);
pending_layer_ = static_cast<TestablePictureLayerImpl*>(
host_impl_.pendingTree()->LayerById(id_));
}
void AddDefaultTilingsWithInvalidation(const Region& invalidation) {
active_layer_->AddTiling(2.3f);
active_layer_->AddTiling(1.0f);
active_layer_->AddTiling(0.5f);
pending_layer_->invalidation() = invalidation;
pending_layer_->SyncFromActiveLayer();
}
void SetupPendingTree(
scoped_refptr<PicturePileImpl> pile) {
host_impl_.createPendingTree();
LayerTreeImpl* pending_tree = host_impl_.pendingTree();
// Clear recycled tree.
pending_tree->DetachLayerTree();
scoped_ptr<TestablePictureLayerImpl> pending_layer =
TestablePictureLayerImpl::create(pending_tree, id_, pile);
pending_layer->setDrawsContent(true);
pending_tree->SetRootLayer(pending_layer.PassAs<LayerImpl>());
}
static void VerifyAllTilesExistAndHavePile(
const PictureLayerTiling* tiling,
PicturePileImpl* pile) {
for (PictureLayerTiling::Iterator iter(tiling,
tiling->contents_scale(),
tiling->ContentRect());
iter; ++iter) {
EXPECT_TRUE(*iter);
EXPECT_EQ(pile, iter->picture_pile());
}
}
void SetContentsScaleOnBothLayers(float scale) {
float result_scale_x, result_scale_y;
gfx::Size result_bounds;
pending_layer_->calculateContentsScale(
scale, &result_scale_x, &result_scale_y, &result_bounds);
active_layer_->calculateContentsScale(
scale, &result_scale_x, &result_scale_y, &result_bounds);
}
protected:
FakeImplProxy proxy_;
FakeLayerTreeHostImpl host_impl_;
int id_;
TestablePictureLayerImpl* pending_layer_;
TestablePictureLayerImpl* active_layer_;
DISALLOW_COPY_AND_ASSIGN(PictureLayerImplTest);
};
TEST_F(PictureLayerImplTest, cloneNoInvalidation) {
gfx::Size tile_size(100, 100);
gfx::Size layer_bounds(400, 400);
scoped_refptr<TestablePicturePileImpl> pending_pile =
TestablePicturePileImpl::CreateFilledPile(tile_size, layer_bounds);
scoped_refptr<TestablePicturePileImpl> active_pile =
TestablePicturePileImpl::CreateFilledPile(tile_size, layer_bounds);
SetupTrees(pending_pile, active_pile);
Region invalidation;
AddDefaultTilingsWithInvalidation(invalidation);
EXPECT_EQ(pending_layer_->tilings().num_tilings(),
active_layer_->tilings().num_tilings());
const PictureLayerTilingSet& tilings = pending_layer_->tilings();
EXPECT_GT(tilings.num_tilings(), 0u);
for (size_t i = 0; i < tilings.num_tilings(); ++i)
VerifyAllTilesExistAndHavePile(tilings.tiling_at(i), active_pile.get());
}
TEST_F(PictureLayerImplTest, clonePartialInvalidation) {
gfx::Size tile_size(100, 100);
gfx::Size layer_bounds(400, 400);
gfx::Rect layer_invalidation(150, 200, 30, 180);
scoped_refptr<TestablePicturePileImpl> pending_pile =
TestablePicturePileImpl::CreateFilledPile(tile_size, layer_bounds);
scoped_refptr<TestablePicturePileImpl> active_pile =
TestablePicturePileImpl::CreateFilledPile(tile_size, layer_bounds);
SetupTrees(pending_pile, active_pile);
Region invalidation(layer_invalidation);
AddDefaultTilingsWithInvalidation(invalidation);
const PictureLayerTilingSet& tilings = pending_layer_->tilings();
EXPECT_GT(tilings.num_tilings(), 0u);
for (size_t i = 0; i < tilings.num_tilings(); ++i) {
const PictureLayerTiling* tiling = tilings.tiling_at(i);
gfx::Rect content_invalidation = gfx::ToEnclosingRect(gfx::ScaleRect(
layer_invalidation,
tiling->contents_scale()));
for (PictureLayerTiling::Iterator iter(tiling,
tiling->contents_scale(),
tiling->ContentRect());
iter; ++iter) {
EXPECT_TRUE(*iter);
EXPECT_FALSE(iter.geometry_rect().IsEmpty());
if (iter.geometry_rect().Intersects(content_invalidation))
EXPECT_EQ(pending_pile, iter->picture_pile());
else
EXPECT_EQ(active_pile, iter->picture_pile());
}
}
}
TEST_F(PictureLayerImplTest, cloneFullInvalidation) {
gfx::Size tile_size(90, 80);
gfx::Size layer_bounds(300, 500);
scoped_refptr<TestablePicturePileImpl> pending_pile =
TestablePicturePileImpl::CreateFilledPile(tile_size, layer_bounds);
scoped_refptr<TestablePicturePileImpl> active_pile =
TestablePicturePileImpl::CreateFilledPile(tile_size, layer_bounds);
SetupTrees(pending_pile, active_pile);
Region invalidation((gfx::Rect(layer_bounds)));
AddDefaultTilingsWithInvalidation(invalidation);
EXPECT_EQ(pending_layer_->tilings().num_tilings(),
active_layer_->tilings().num_tilings());
const PictureLayerTilingSet& tilings = pending_layer_->tilings();
EXPECT_GT(tilings.num_tilings(), 0u);
for (size_t i = 0; i < tilings.num_tilings(); ++i)
VerifyAllTilesExistAndHavePile(tilings.tiling_at(i), pending_pile.get());
}
TEST_F(PictureLayerImplTest, noInvalidationBoundsChange) {
gfx::Size tile_size(90, 80);
gfx::Size active_layer_bounds(300, 500);
gfx::Size pending_layer_bounds(400, 800);
scoped_refptr<TestablePicturePileImpl> pending_pile =
TestablePicturePileImpl::CreateFilledPile(tile_size,
pending_layer_bounds);
scoped_refptr<TestablePicturePileImpl> active_pile =
TestablePicturePileImpl::CreateFilledPile(tile_size, active_layer_bounds);
SetupTrees(pending_pile, active_pile);
Region invalidation;
AddDefaultTilingsWithInvalidation(invalidation);
const PictureLayerTilingSet& tilings = pending_layer_->tilings();
EXPECT_GT(tilings.num_tilings(), 0u);
for (size_t i = 0; i < tilings.num_tilings(); ++i) {
const PictureLayerTiling* tiling = tilings.tiling_at(i);
gfx::Rect active_content_bounds = gfx::ToEnclosingRect(gfx::ScaleRect(
gfx::Rect(active_layer_bounds),
tiling->contents_scale()));
for (PictureLayerTiling::Iterator iter(tiling,
tiling->contents_scale(),
tiling->ContentRect());
iter; ++iter) {
EXPECT_TRUE(*iter);
EXPECT_FALSE(iter.geometry_rect().IsEmpty());
if (iter.geometry_rect().right() >= active_content_bounds.width() ||
iter.geometry_rect().bottom() >= active_content_bounds.height()) {
EXPECT_EQ(pending_pile, iter->picture_pile());
} else {
EXPECT_EQ(active_pile, iter->picture_pile());
}
}
}
}
TEST_F(PictureLayerImplTest, addTilesFromNewRecording) {
gfx::Size tile_size(400, 400);
gfx::Size layer_bounds(1300, 1900);
scoped_refptr<TestablePicturePileImpl> pending_pile =
TestablePicturePileImpl::CreateEmptyPile(tile_size, layer_bounds);
scoped_refptr<TestablePicturePileImpl> active_pile =
TestablePicturePileImpl::CreateEmptyPile(tile_size, layer_bounds);
// Fill in some of active pile, but more of pending pile.
int hole_count = 0;
for (int x = 0; x < active_pile->tiling().num_tiles_x(); ++x) {
for (int y = 0; y < active_pile->tiling().num_tiles_y(); ++y) {
if ((x + y) % 2) {
pending_pile->AddRecordingAt(x, y);
active_pile->AddRecordingAt(x, y);
} else {
hole_count++;
if (hole_count % 2)
pending_pile->AddRecordingAt(x, y);
}
}
}
SetupTrees(pending_pile, active_pile);
Region invalidation;
AddDefaultTilingsWithInvalidation(invalidation);
const PictureLayerTilingSet& tilings = pending_layer_->tilings();
EXPECT_GT(tilings.num_tilings(), 0u);
for (size_t i = 0; i < tilings.num_tilings(); ++i) {
const PictureLayerTiling* tiling = tilings.tiling_at(i);
for (PictureLayerTiling::Iterator iter(tiling,
tiling->contents_scale(),
tiling->ContentRect());
iter; ++iter) {
EXPECT_FALSE(iter.full_tile_geometry_rect().IsEmpty());
// Ensure there is a recording for this tile.
gfx::Rect layer_rect = gfx::ToEnclosingRect(
gfx::ScaleRect(
iter.full_tile_geometry_rect(), 1.f / tiling->contents_scale()));
layer_rect.Intersect(gfx::Rect(layer_bounds));
bool in_pending = pending_pile->recorded_region().Contains(layer_rect);
bool in_active = active_pile->recorded_region().Contains(layer_rect);
if (in_pending && !in_active)
EXPECT_EQ(pending_pile, iter->picture_pile());
else if (in_active)
EXPECT_EQ(active_pile, iter->picture_pile());
else
EXPECT_FALSE(*iter);
}
}
}
TEST_F(PictureLayerImplTest, ManageTilingsWithNoRecording) {
gfx::Size tile_size(400, 400);
gfx::Size layer_bounds(1300, 1900);
scoped_refptr<TestablePicturePileImpl> pending_pile =
TestablePicturePileImpl::CreateEmptyPile(tile_size, layer_bounds);
scoped_refptr<TestablePicturePileImpl> active_pile =
TestablePicturePileImpl::CreateEmptyPile(tile_size, layer_bounds);
float result_scale_x, result_scale_y;
gfx::Size result_bounds;
SetupTrees(pending_pile, active_pile);
// These are included in the scale given to the layer.
host_impl_.setDeviceScaleFactor(1.f);
host_impl_.pendingTree()->SetPageScaleFactorAndLimits(1.f, 1.f, 1.f);
pending_layer_->calculateContentsScale(
1.f, &result_scale_x, &result_scale_y, &result_bounds);
EXPECT_EQ(0u, pending_layer_->tilings().num_tilings());
}
TEST_F(PictureLayerImplTest, ManageTilingsCreatesTilings) {
gfx::Size tile_size(400, 400);
gfx::Size layer_bounds(1300, 1900);
scoped_refptr<TestablePicturePileImpl> pending_pile =
TestablePicturePileImpl::CreateFilledPile(tile_size, layer_bounds);
scoped_refptr<TestablePicturePileImpl> active_pile =
TestablePicturePileImpl::CreateFilledPile(tile_size, layer_bounds);
float result_scale_x, result_scale_y;
gfx::Size result_bounds;
SetupTrees(pending_pile, active_pile);
EXPECT_EQ(0u, pending_layer_->tilings().num_tilings());
float low_res_factor = host_impl_.settings().lowResContentsScaleFactor;
EXPECT_LT(low_res_factor, 1.f);
// These are included in the scale given to the layer.
host_impl_.setDeviceScaleFactor(1.7f);
host_impl_.pendingTree()->SetPageScaleFactorAndLimits(3.2f, 3.2f, 3.2f);
pending_layer_->calculateContentsScale(
1.3f, &result_scale_x, &result_scale_y, &result_bounds);
ASSERT_EQ(2u, pending_layer_->tilings().num_tilings());
EXPECT_FLOAT_EQ(
1.3f,
pending_layer_->tilings().tiling_at(0)->contents_scale());
EXPECT_FLOAT_EQ(
1.3f * low_res_factor,
pending_layer_->tilings().tiling_at(1)->contents_scale());
// If we change the layer's CSS scale factor, then we should not get new
// tilings.
pending_layer_->calculateContentsScale(
1.8f, &result_scale_x, &result_scale_y, &result_bounds);
ASSERT_EQ(2u, pending_layer_->tilings().num_tilings());
EXPECT_FLOAT_EQ(
1.3f,
pending_layer_->tilings().tiling_at(0)->contents_scale());
EXPECT_FLOAT_EQ(
1.3f * low_res_factor,
pending_layer_->tilings().tiling_at(1)->contents_scale());
// If we change the page scale factor, then we should get new tilings.
host_impl_.pendingTree()->SetPageScaleFactorAndLimits(2.2f, 2.2f, 2.2f);
pending_layer_->calculateContentsScale(
1.8f, &result_scale_x, &result_scale_y, &result_bounds);
ASSERT_EQ(4u, pending_layer_->tilings().num_tilings());
EXPECT_FLOAT_EQ(
1.8f,
pending_layer_->tilings().tiling_at(0)->contents_scale());
EXPECT_FLOAT_EQ(
1.8f * low_res_factor,
pending_layer_->tilings().tiling_at(2)->contents_scale());
// If we change the device scale factor, then we should get new tilings.
host_impl_.setDeviceScaleFactor(1.4f);
pending_layer_->calculateContentsScale(
1.9f, &result_scale_x, &result_scale_y, &result_bounds);
ASSERT_EQ(6u, pending_layer_->tilings().num_tilings());
EXPECT_FLOAT_EQ(
1.9f,
pending_layer_->tilings().tiling_at(0)->contents_scale());
EXPECT_FLOAT_EQ(
1.9f * low_res_factor,
pending_layer_->tilings().tiling_at(3)->contents_scale());
// If we change the device scale factor, but end up at the same total scale
// factor somehow, then we don't get new tilings.
host_impl_.setDeviceScaleFactor(2.2f);
host_impl_.pendingTree()->SetPageScaleFactorAndLimits(1.4f, 1.4f, 1.4f);
pending_layer_->calculateContentsScale(
1.9f, &result_scale_x, &result_scale_y, &result_bounds);
ASSERT_EQ(6u, pending_layer_->tilings().num_tilings());
EXPECT_FLOAT_EQ(
1.9f,
pending_layer_->tilings().tiling_at(0)->contents_scale());
EXPECT_FLOAT_EQ(
1.9f * low_res_factor,
pending_layer_->tilings().tiling_at(3)->contents_scale());
}
TEST_F(PictureLayerImplTest, CleanUpTilings) {
gfx::Size tile_size(400, 400);
gfx::Size layer_bounds(1300, 1900);
scoped_refptr<TestablePicturePileImpl> pending_pile =
TestablePicturePileImpl::CreateFilledPile(tile_size, layer_bounds);
scoped_refptr<TestablePicturePileImpl> active_pile =
TestablePicturePileImpl::CreateFilledPile(tile_size, layer_bounds);
float result_scale_x, result_scale_y;
gfx::Size result_bounds;
std::vector<PictureLayerTiling*> used_tilings;
SetupTrees(pending_pile, active_pile);
EXPECT_EQ(0u, pending_layer_->tilings().num_tilings());
float low_res_factor = host_impl_.settings().lowResContentsScaleFactor;
EXPECT_LT(low_res_factor, 1.f);
// These are included in the scale given to the layer.
host_impl_.setDeviceScaleFactor(1.7f);
host_impl_.pendingTree()->SetPageScaleFactorAndLimits(3.2f, 3.2f, 3.2f);
host_impl_.activeTree()->SetPageScaleFactorAndLimits(3.2f, 3.2f, 3.2f);
SetContentsScaleOnBothLayers(1.f);
ASSERT_EQ(2u, active_layer_->tilings().num_tilings());
// We only have ideal tilings, so they aren't removed.
used_tilings.clear();
active_layer_->CleanUpTilingsOnActiveLayer(used_tilings);
ASSERT_EQ(2u, active_layer_->tilings().num_tilings());
// Changing the ideal but not creating new tilings.
SetContentsScaleOnBothLayers(1.5f);
ASSERT_EQ(2u, active_layer_->tilings().num_tilings());
// The tilings are still our target scale, so they aren't removed.
used_tilings.clear();
active_layer_->CleanUpTilingsOnActiveLayer(used_tilings);
ASSERT_EQ(2u, active_layer_->tilings().num_tilings());
// Create a 1.2 scale tiling. Now we have 1.0 and 1.2 tilings. Ideal = 1.2.
host_impl_.pendingTree()->SetPageScaleFactorAndLimits(1.2f, 1.2f, 1.2f);
host_impl_.activeTree()->SetPageScaleFactorAndLimits(1.2f, 1.2f, 1.2f);
SetContentsScaleOnBothLayers(1.2f);
ASSERT_EQ(4u, active_layer_->tilings().num_tilings());
EXPECT_FLOAT_EQ(
1.f,
active_layer_->tilings().tiling_at(1)->contents_scale());
EXPECT_FLOAT_EQ(
1.f * low_res_factor,
active_layer_->tilings().tiling_at(3)->contents_scale());
// Mark the non-ideal tilings as used. They won't be removed.
used_tilings.clear();
used_tilings.push_back(active_layer_->tilings().tiling_at(1));
used_tilings.push_back(active_layer_->tilings().tiling_at(3));
active_layer_->CleanUpTilingsOnActiveLayer(used_tilings);
ASSERT_EQ(4u, active_layer_->tilings().num_tilings());
// Now move the ideal scale to 0.5. Our target stays 1.2.
SetContentsScaleOnBothLayers(0.5f);
// All the tilings are between are target and the ideal, so they are not
// removed.
used_tilings.clear();
active_layer_->CleanUpTilingsOnActiveLayer(used_tilings);
ASSERT_EQ(4u, active_layer_->tilings().num_tilings());
// Now move the ideal scale to 1.0. Our target stays 1.2.
SetContentsScaleOnBothLayers(1.f);
// All the tilings are between are target and the ideal, so they are not
// removed.
used_tilings.clear();
active_layer_->CleanUpTilingsOnActiveLayer(used_tilings);
ASSERT_EQ(4u, active_layer_->tilings().num_tilings());
// Now move the ideal scale to 1.1 on the active layer. Our target stays 1.2.
active_layer_->calculateContentsScale(
1.1f, &result_scale_x, &result_scale_y, &result_bounds);
// Because the pending layer's ideal scale is still 1.0, our tilings fall
// in the range [1.0,1.2] and are kept.
used_tilings.clear();
active_layer_->CleanUpTilingsOnActiveLayer(used_tilings);
ASSERT_EQ(4u, active_layer_->tilings().num_tilings());
// Move the ideal scale on the pending layer to 1.1 as well. Our target stays
// 1.2 still.
pending_layer_->calculateContentsScale(
1.1f, &result_scale_x, &result_scale_y, &result_bounds);
// Our 1.0 tiling now falls outside the range between our ideal scale and our
// target raster scale. But it is in our used tilings set, so nothing is
// deleted.
used_tilings.clear();
used_tilings.push_back(active_layer_->tilings().tiling_at(1));
used_tilings.push_back(active_layer_->tilings().tiling_at(3));
active_layer_->CleanUpTilingsOnActiveLayer(used_tilings);
ASSERT_EQ(4u, active_layer_->tilings().num_tilings());
// If we remove it from our used tilings set, it is outside the range to keep
// so it is deleted. Try one tiling at a time.
used_tilings.clear();
used_tilings.push_back(active_layer_->tilings().tiling_at(1));
active_layer_->CleanUpTilingsOnActiveLayer(used_tilings);
ASSERT_EQ(3u, active_layer_->tilings().num_tilings());
used_tilings.clear();
active_layer_->CleanUpTilingsOnActiveLayer(used_tilings);
ASSERT_EQ(2u, active_layer_->tilings().num_tilings());
}
} // namespace
} // namespace cc