gecko-dev/gfx/layers/wr/WebRenderPaintedLayer.cpp

192 строки
5.8 KiB
C++

/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
#include "WebRenderPaintedLayer.h"
#include "LayersLogging.h"
#include "mozilla/ArrayUtils.h"
#include "mozilla/layers/ImageClient.h"
#include "mozilla/layers/ScrollingLayersHelper.h"
#include "mozilla/layers/StackingContextHelper.h"
#include "mozilla/layers/WebRenderBridgeChild.h"
#include "mozilla/layers/UpdateImageHelper.h"
#include "mozilla/webrender/WebRenderTypes.h"
#include "gfxPrefs.h"
#include "gfxUtils.h"
namespace mozilla {
namespace layers {
using namespace mozilla::gfx;
bool
WebRenderPaintedLayer::SetupExternalImages()
{
// XXX We won't keep using ContentClient for WebRenderPaintedLayer in the future and
// there is a crash problem for ContentClient on MacOS. So replace ContentClient with
// ImageClient. See bug 1341001.
if (!mImageContainer) {
mImageContainer = LayerManager::CreateImageContainer();
}
if (!mImageClient) {
mImageClient = ImageClient::CreateImageClient(CompositableType::IMAGE,
WrBridge(),
TextureFlags::DEFAULT);
if (!mImageClient) {
return false;
}
mImageClient->Connect();
}
if (mExternalImageId.isNothing()) {
mExternalImageId = Some(WrBridge()->AllocExternalImageIdForCompositable(mImageClient));
}
return true;
}
bool
WebRenderPaintedLayer::UpdateImageClient()
{
MOZ_ASSERT(WrManager()->GetPaintedLayerCallback());
nsIntRegion visibleRegion = GetVisibleRegion().ToUnknownRegion();
IntRect bounds = visibleRegion.GetBounds();
IntSize imageSize = bounds.Size();
UpdateImageHelper helper(mImageContainer, mImageClient, imageSize);
{
RefPtr<DrawTarget> target = helper.GetDrawTarget();
if (!target) {
return false;
}
target->ClearRect(Rect(0, 0, imageSize.width, imageSize.height));
target->SetTransform(Matrix().PreTranslate(-bounds.x, -bounds.y));
RefPtr<gfxContext> ctx =
gfxContext::CreatePreservingTransformOrNull(target);
MOZ_ASSERT(ctx); // already checked the target above
WrManager()->GetPaintedLayerCallback()(this,
ctx,
visibleRegion, visibleRegion,
DrawRegionClip::DRAW, nsIntRegion(), WrManager()->GetPaintedLayerCallbackData());
if (gfxPrefs::WebRenderHighlightPaintedLayers()) {
target->SetTransform(Matrix());
target->FillRect(Rect(0, 0, imageSize.width, imageSize.height), ColorPattern(Color(1.0, 0.0, 0.0, 0.5)));
}
}
if (!helper.UpdateImage()) {
return false;
}
return true;
}
void
WebRenderPaintedLayer::CreateWebRenderDisplayList(wr::DisplayListBuilder& aBuilder,
const StackingContextHelper& aSc)
{
ScrollingLayersHelper scroller(this, aBuilder, aSc);
StackingContextHelper sc(aSc, aBuilder, this);
LayerRect rect = Bounds();
DumpLayerInfo("PaintedLayer", rect);
wr::WrImageKey key = GetImageKey();
WrBridge()->AddWebRenderParentCommand(OpAddExternalImage(mExternalImageId.value(), key));
WrManager()->AddImageKeyForDiscard(key);
wr::LayoutRect r = sc.ToRelativeLayoutRect(rect);
aBuilder.PushImage(r, r, wr::ImageRendering::Auto, key);
}
void
WebRenderPaintedLayer::RenderLayer(wr::DisplayListBuilder& aBuilder,
const StackingContextHelper& aSc)
{
if (!SetupExternalImages()) {
return;
}
if (GetVisibleRegion().IsEmpty()) {
if (gfxPrefs::LayersDump()) {
printf_stderr("PaintedLayer %p skipping\n", this->GetLayer());
}
return;
}
bool hasSomethingToPaint = true;
LayerIntRect visibleBounds = mVisibleRegion.GetBounds();
nsIntRegion visibleRegion = mVisibleRegion.ToUnknownRegion();
if (visibleBounds == mPaintedRect) {
// If the visible bounds haven't changed, there is a chance that the visible region
// might be entirely valid. If there is anything to paint, though, we'll repaint
// the entire visible region.
nsIntRegion regionToPaint = visibleRegion;
regionToPaint.SubOut(GetValidRegion());
if (regionToPaint.IsEmpty()) {
hasSomethingToPaint = false; // yay!
}
}
// We have something to paint but can't. This usually happens only in
// empty transactions
if (hasSomethingToPaint && !WrManager()->GetPaintedLayerCallback()) {
WrManager()->SetTransactionIncomplete();
return;
}
if (hasSomethingToPaint && WrManager()->GetPaintedLayerCallback()) {
// In UpdateImageClient we throw away the previous buffer and paint everything in
// a new one, which amounts to losing the valid region.
ClearValidRegion();
if (!UpdateImageClient()) {
mPaintedRect = LayerIntRect();
return;
}
mPaintedRect = visibleBounds;
SetValidRegion(visibleRegion);
} else {
// We have an empty transaction, just reuse the old image we had before.
MOZ_ASSERT(mExternalImageId);
MOZ_ASSERT(mImageContainer->HasCurrentImage());
}
CreateWebRenderDisplayList(aBuilder, aSc);
}
void
WebRenderPaintedLayer::ClearCachedResources()
{
ClearWrResources();
if (mImageClient) {
mImageClient->FlushAllImages();
mImageClient->ClearCachedResources();
}
if (mImageContainer) {
mImageContainer->ClearAllImages();
mImageContainer->ClearCachedResources();
}
ClearValidRegion();
}
void
WebRenderPaintedLayer::ClearWrResources()
{
if (mExternalImageId.isSome()) {
WrBridge()->DeallocExternalImageId(mExternalImageId.ref());
mExternalImageId = Nothing();
}
}
} // namespace layers
} // namespace mozilla