mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-23 04:41:11 +00:00
9c96712d23
GPUVideoImage::GetAsSourceSurface() is heavy weight. It seems better to prevent to call it if possible. The change handles only a case that CanvasTranslator is used, SurfaceDescriptor is RemoteDecoderVideoSubDescriptor::Tnull_t and OptimizeShadow is not used. DrawTargetRecording::DrawSurfaceDescriptor() also holds layers::Image of SurfaceDescriptor. It is necessary to keep the Image alive in CanvasDrawEventRecorder until end of its usage in CanvasTranslator. Differential Revision: https://phabricator.services.mozilla.com/D205677
196 lines
6.4 KiB
C++
196 lines
6.4 KiB
C++
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
|
|
/* 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 "DrawEventRecorder.h"
|
|
|
|
#include "mozilla/UniquePtrExtensions.h"
|
|
#include "PathRecording.h"
|
|
#include "RecordingTypes.h"
|
|
#include "RecordedEventImpl.h"
|
|
|
|
namespace mozilla {
|
|
namespace gfx {
|
|
|
|
DrawEventRecorderPrivate::DrawEventRecorderPrivate() : mExternalFonts(false) {}
|
|
|
|
DrawEventRecorderPrivate::~DrawEventRecorderPrivate() = default;
|
|
|
|
void DrawEventRecorderPrivate::SetDrawTarget(ReferencePtr aDT) {
|
|
NS_ASSERT_OWNINGTHREAD(DrawEventRecorderPrivate);
|
|
|
|
RecordEvent(RecordedSetCurrentDrawTarget(aDT));
|
|
mCurrentDT = aDT;
|
|
}
|
|
|
|
void DrawEventRecorderPrivate::StoreExternalSurfaceRecording(
|
|
SourceSurface* aSurface, uint64_t aKey) {
|
|
NS_ASSERT_OWNINGTHREAD(DrawEventRecorderPrivate);
|
|
|
|
RecordEvent(RecordedExternalSurfaceCreation(aSurface, aKey));
|
|
mExternalSurfaces.push_back({aSurface});
|
|
}
|
|
|
|
void DrawEventRecorderPrivate::StoreExternalImageRecording(
|
|
const RefPtr<layers::Image>& aImageOfSurfaceDescriptor) {
|
|
NS_ASSERT_OWNINGTHREAD(DrawEventRecorderPrivate);
|
|
|
|
mExternalImages.push_back({aImageOfSurfaceDescriptor});
|
|
}
|
|
|
|
void DrawEventRecorderPrivate::StoreSourceSurfaceRecording(
|
|
SourceSurface* aSurface, const char* aReason) {
|
|
NS_ASSERT_OWNINGTHREAD(DrawEventRecorderPrivate);
|
|
|
|
RefPtr<DataSourceSurface> dataSurf = aSurface->GetDataSurface();
|
|
IntSize surfaceSize = aSurface->GetSize();
|
|
Maybe<DataSourceSurface::ScopedMap> map;
|
|
if (dataSurf) {
|
|
map.emplace(dataSurf, DataSourceSurface::READ);
|
|
}
|
|
if (!dataSurf || !map->IsMapped() ||
|
|
!Factory::AllowedSurfaceSize(surfaceSize)) {
|
|
gfxWarning() << "Recording failed to record SourceSurface for " << aReason;
|
|
|
|
// If surface size is not allowed, replace with reasonable size.
|
|
if (!Factory::AllowedSurfaceSize(surfaceSize)) {
|
|
surfaceSize.width = std::min(surfaceSize.width, kReasonableSurfaceSize);
|
|
surfaceSize.height = std::min(surfaceSize.height, kReasonableSurfaceSize);
|
|
}
|
|
|
|
// Insert a dummy source surface.
|
|
int32_t stride = surfaceSize.width * BytesPerPixel(aSurface->GetFormat());
|
|
UniquePtr<uint8_t[]> sourceData =
|
|
MakeUniqueFallible<uint8_t[]>(stride * surfaceSize.height);
|
|
if (!sourceData) {
|
|
// If the surface is too big just create a 1 x 1 dummy.
|
|
surfaceSize.width = 1;
|
|
surfaceSize.height = 1;
|
|
stride = surfaceSize.width * BytesPerPixel(aSurface->GetFormat());
|
|
sourceData = MakeUnique<uint8_t[]>(stride * surfaceSize.height);
|
|
}
|
|
|
|
RecordEvent(RecordedSourceSurfaceCreation(aSurface, sourceData.get(),
|
|
stride, surfaceSize,
|
|
aSurface->GetFormat()));
|
|
return;
|
|
}
|
|
|
|
RecordEvent(RecordedSourceSurfaceCreation(
|
|
aSurface, map->GetData(), map->GetStride(), dataSurf->GetSize(),
|
|
dataSurf->GetFormat()));
|
|
}
|
|
|
|
void DrawEventRecorderPrivate::RecordSourceSurfaceDestruction(void* aSurface) {
|
|
NS_ASSERT_OWNINGTHREAD(DrawEventRecorderPrivate);
|
|
|
|
RemoveSourceSurface(static_cast<SourceSurface*>(aSurface));
|
|
RemoveStoredObject(aSurface);
|
|
RecordEvent(RecordedSourceSurfaceDestruction(ReferencePtr(aSurface)));
|
|
}
|
|
|
|
void DrawEventRecorderPrivate::DecrementUnscaledFontRefCount(
|
|
const ReferencePtr aUnscaledFont) {
|
|
NS_ASSERT_OWNINGTHREAD(DrawEventRecorderPrivate);
|
|
|
|
auto element = mUnscaledFontRefs.Lookup(aUnscaledFont);
|
|
MOZ_DIAGNOSTIC_ASSERT(element,
|
|
"DecrementUnscaledFontRefCount calls should balance "
|
|
"with IncrementUnscaledFontRefCount calls");
|
|
if (--element.Data() <= 0) {
|
|
RecordEvent(RecordedUnscaledFontDestruction(aUnscaledFont));
|
|
element.Remove();
|
|
}
|
|
}
|
|
|
|
void DrawEventRecorderMemory::RecordEvent(const RecordedEvent& aEvent) {
|
|
NS_ASSERT_OWNINGTHREAD(DrawEventRecorderMemory);
|
|
aEvent.RecordToStream(mOutputStream);
|
|
}
|
|
|
|
void DrawEventRecorderMemory::AddDependentSurface(uint64_t aDependencyId) {
|
|
NS_ASSERT_OWNINGTHREAD(DrawEventRecorderMemory);
|
|
mDependentSurfaces.Insert(aDependencyId);
|
|
}
|
|
|
|
nsTHashSet<uint64_t>&& DrawEventRecorderMemory::TakeDependentSurfaces() {
|
|
NS_ASSERT_OWNINGTHREAD(DrawEventRecorderMemory);
|
|
return std::move(mDependentSurfaces);
|
|
}
|
|
|
|
DrawEventRecorderMemory::DrawEventRecorderMemory() {
|
|
WriteHeader(mOutputStream);
|
|
}
|
|
|
|
DrawEventRecorderMemory::DrawEventRecorderMemory(
|
|
const SerializeResourcesFn& aFn)
|
|
: mSerializeCallback(aFn) {
|
|
mExternalFonts = !!mSerializeCallback;
|
|
WriteHeader(mOutputStream);
|
|
}
|
|
|
|
void DrawEventRecorderMemory::Flush() {
|
|
NS_ASSERT_OWNINGTHREAD(DrawEventRecorderMemory);
|
|
}
|
|
|
|
void DrawEventRecorderMemory::FlushItem(IntRect aRect) {
|
|
NS_ASSERT_OWNINGTHREAD(DrawEventRecorderMemory);
|
|
|
|
MOZ_RELEASE_ASSERT(!aRect.IsEmpty());
|
|
// Detaching our existing resources will add some
|
|
// destruction events to our stream so we need to do that
|
|
// first.
|
|
DetachResources();
|
|
|
|
// See moz2d_renderer.rs for a description of the stream format
|
|
WriteElement(mIndex, mOutputStream.mLength);
|
|
|
|
// write out the fonts into the extra data section
|
|
mSerializeCallback(mOutputStream, mScaledFonts);
|
|
WriteElement(mIndex, mOutputStream.mLength);
|
|
|
|
WriteElement(mIndex, aRect.x);
|
|
WriteElement(mIndex, aRect.y);
|
|
WriteElement(mIndex, aRect.XMost());
|
|
WriteElement(mIndex, aRect.YMost());
|
|
ClearResources();
|
|
|
|
// write out a new header for the next recording in the stream
|
|
WriteHeader(mOutputStream);
|
|
}
|
|
|
|
bool DrawEventRecorderMemory::Finish() {
|
|
NS_ASSERT_OWNINGTHREAD(DrawEventRecorderMemory);
|
|
|
|
// this length might be 0, and things should still work.
|
|
// for example if there are no items in a particular area
|
|
size_t indexOffset = mOutputStream.mLength;
|
|
// write out the index
|
|
mOutputStream.write(mIndex.mData, mIndex.mLength);
|
|
bool hasItems = mIndex.mLength != 0;
|
|
mIndex.reset();
|
|
// write out the offset of the Index to the end of the output stream
|
|
WriteElement(mOutputStream, indexOffset);
|
|
ClearResources();
|
|
return hasItems;
|
|
}
|
|
|
|
size_t DrawEventRecorderMemory::RecordingSize() {
|
|
NS_ASSERT_OWNINGTHREAD(DrawEventRecorderMemory);
|
|
return mOutputStream.mLength;
|
|
}
|
|
|
|
void DrawEventRecorderMemory::WipeRecording() {
|
|
NS_ASSERT_OWNINGTHREAD(DrawEventRecorderMemory);
|
|
|
|
mOutputStream.reset();
|
|
mIndex.reset();
|
|
|
|
WriteHeader(mOutputStream);
|
|
}
|
|
|
|
} // namespace gfx
|
|
} // namespace mozilla
|