255 lines
8.1 KiB
C++
255 lines
8.1 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 "GPUVideoTextureHost.h"
|
|
|
|
#include "ImageContainer.h"
|
|
#include "mozilla/RemoteDecoderManagerParent.h"
|
|
#include "mozilla/layers/ImageBridgeParent.h"
|
|
#include "mozilla/layers/VideoBridgeParent.h"
|
|
#include "mozilla/webrender/RenderTextureHostWrapper.h"
|
|
#include "mozilla/webrender/RenderThread.h"
|
|
|
|
namespace mozilla {
|
|
namespace layers {
|
|
|
|
GPUVideoTextureHost::GPUVideoTextureHost(
|
|
const dom::ContentParentId& aContentId, TextureFlags aFlags,
|
|
const SurfaceDescriptorGPUVideo& aDescriptor)
|
|
: TextureHost(TextureHostType::Unknown, aFlags),
|
|
mContentId(aContentId),
|
|
mDescriptor(aDescriptor) {
|
|
MOZ_COUNT_CTOR(GPUVideoTextureHost);
|
|
}
|
|
|
|
GPUVideoTextureHost::~GPUVideoTextureHost() {
|
|
MOZ_COUNT_DTOR(GPUVideoTextureHost);
|
|
}
|
|
|
|
GPUVideoTextureHost* GPUVideoTextureHost::CreateFromDescriptor(
|
|
const dom::ContentParentId& aContentId, TextureFlags aFlags,
|
|
const SurfaceDescriptorGPUVideo& aDescriptor) {
|
|
return new GPUVideoTextureHost(aContentId, aFlags, aDescriptor);
|
|
}
|
|
|
|
TextureHost* GPUVideoTextureHost::EnsureWrappedTextureHost() {
|
|
if (mWrappedTextureHost) {
|
|
return mWrappedTextureHost;
|
|
}
|
|
|
|
const auto& sd =
|
|
static_cast<const SurfaceDescriptorRemoteDecoder&>(mDescriptor);
|
|
RefPtr<VideoBridgeParent> parent =
|
|
VideoBridgeParent::GetSingleton(sd.source());
|
|
if (!parent) {
|
|
// The VideoBridge went away. This can happen if the RDD process
|
|
// crashes.
|
|
return nullptr;
|
|
}
|
|
|
|
mWrappedTextureHost = parent->LookupTextureAsync(mContentId, sd.handle());
|
|
|
|
if (!mWrappedTextureHost) {
|
|
// The TextureHost hasn't been registered yet. This is due to a race
|
|
// between the ImageBridge (content) and the VideoBridge (RDD) and the
|
|
// ImageBridge won. See bug
|
|
// https://bugzilla.mozilla.org/show_bug.cgi?id=1630733#c14 for more
|
|
// details.
|
|
return nullptr;
|
|
}
|
|
|
|
if (mExternalImageId.isSome()) {
|
|
// External image id is allocated by mWrappedTextureHost.
|
|
mWrappedTextureHost->EnsureRenderTexture(Nothing());
|
|
MOZ_ASSERT(mWrappedTextureHost->mExternalImageId.isSome());
|
|
auto wrappedId = mWrappedTextureHost->mExternalImageId.ref();
|
|
|
|
RefPtr<wr::RenderTextureHost> texture =
|
|
new wr::RenderTextureHostWrapper(wrappedId);
|
|
wr::RenderThread::Get()->RegisterExternalImage(mExternalImageId.ref(),
|
|
texture.forget());
|
|
}
|
|
|
|
return mWrappedTextureHost;
|
|
}
|
|
|
|
bool GPUVideoTextureHost::IsValid() { return !!EnsureWrappedTextureHost(); }
|
|
|
|
gfx::YUVColorSpace GPUVideoTextureHost::GetYUVColorSpace() const {
|
|
MOZ_ASSERT(mWrappedTextureHost, "Image isn't valid yet");
|
|
if (!mWrappedTextureHost) {
|
|
return TextureHost::GetYUVColorSpace();
|
|
}
|
|
return mWrappedTextureHost->GetYUVColorSpace();
|
|
}
|
|
|
|
gfx::ColorDepth GPUVideoTextureHost::GetColorDepth() const {
|
|
MOZ_ASSERT(mWrappedTextureHost, "Image isn't valid yet");
|
|
if (!mWrappedTextureHost) {
|
|
return TextureHost::GetColorDepth();
|
|
}
|
|
return mWrappedTextureHost->GetColorDepth();
|
|
}
|
|
|
|
gfx::ColorRange GPUVideoTextureHost::GetColorRange() const {
|
|
MOZ_ASSERT(mWrappedTextureHost, "Image isn't valid yet");
|
|
if (!mWrappedTextureHost) {
|
|
return TextureHost::GetColorRange();
|
|
}
|
|
return mWrappedTextureHost->GetColorRange();
|
|
}
|
|
|
|
gfx::IntSize GPUVideoTextureHost::GetSize() const {
|
|
MOZ_ASSERT(mWrappedTextureHost, "Image isn't valid yet");
|
|
if (!mWrappedTextureHost) {
|
|
return gfx::IntSize();
|
|
}
|
|
return mWrappedTextureHost->GetSize();
|
|
}
|
|
|
|
gfx::SurfaceFormat GPUVideoTextureHost::GetFormat() const {
|
|
MOZ_ASSERT(mWrappedTextureHost, "Image isn't valid yet");
|
|
if (!mWrappedTextureHost) {
|
|
return gfx::SurfaceFormat::UNKNOWN;
|
|
}
|
|
return mWrappedTextureHost->GetFormat();
|
|
}
|
|
|
|
void GPUVideoTextureHost::CreateRenderTexture(
|
|
const wr::ExternalImageId& aExternalImageId) {
|
|
MOZ_ASSERT(mExternalImageId.isSome());
|
|
|
|
// When mWrappedTextureHost already exist, call CreateRenderTexture() here.
|
|
// In other cases, EnsureWrappedTextureHost() handles CreateRenderTexture().
|
|
|
|
if (mWrappedTextureHost) {
|
|
// External image id is allocated by mWrappedTextureHost.
|
|
mWrappedTextureHost->EnsureRenderTexture(Nothing());
|
|
MOZ_ASSERT(mWrappedTextureHost->mExternalImageId.isSome());
|
|
auto wrappedId = mWrappedTextureHost->mExternalImageId.ref();
|
|
|
|
RefPtr<wr::RenderTextureHost> texture =
|
|
new wr::RenderTextureHostWrapper(wrappedId);
|
|
wr::RenderThread::Get()->RegisterExternalImage(mExternalImageId.ref(),
|
|
texture.forget());
|
|
return;
|
|
}
|
|
|
|
EnsureWrappedTextureHost();
|
|
}
|
|
|
|
void GPUVideoTextureHost::MaybeDestroyRenderTexture() {
|
|
if (mExternalImageId.isNothing()) {
|
|
// RenderTextureHost was not created
|
|
return;
|
|
}
|
|
|
|
if (mExternalImageId.isSome() && mWrappedTextureHost) {
|
|
// When GPUVideoTextureHost created RenderTextureHost, delete it here.
|
|
TextureHost::DestroyRenderTexture(mExternalImageId.ref());
|
|
}
|
|
mExternalImageId = Nothing();
|
|
}
|
|
|
|
uint32_t GPUVideoTextureHost::NumSubTextures() {
|
|
if (!EnsureWrappedTextureHost()) {
|
|
return 0;
|
|
}
|
|
return EnsureWrappedTextureHost()->NumSubTextures();
|
|
}
|
|
|
|
void GPUVideoTextureHost::PushResourceUpdates(
|
|
wr::TransactionBuilder& aResources, ResourceUpdateOp aOp,
|
|
const Range<wr::ImageKey>& aImageKeys, const wr::ExternalImageId& aExtID) {
|
|
MOZ_ASSERT(EnsureWrappedTextureHost(), "Image isn't valid yet");
|
|
if (!EnsureWrappedTextureHost()) {
|
|
return;
|
|
}
|
|
EnsureWrappedTextureHost()->PushResourceUpdates(aResources, aOp, aImageKeys,
|
|
aExtID);
|
|
}
|
|
|
|
void GPUVideoTextureHost::PushDisplayItems(
|
|
wr::DisplayListBuilder& aBuilder, const wr::LayoutRect& aBounds,
|
|
const wr::LayoutRect& aClip, wr::ImageRendering aFilter,
|
|
const Range<wr::ImageKey>& aImageKeys, PushDisplayItemFlagSet aFlags) {
|
|
MOZ_ASSERT(EnsureWrappedTextureHost(), "Image isn't valid yet");
|
|
MOZ_ASSERT(aImageKeys.length() > 0);
|
|
if (!EnsureWrappedTextureHost()) {
|
|
return;
|
|
}
|
|
|
|
EnsureWrappedTextureHost()->PushDisplayItems(aBuilder, aBounds, aClip,
|
|
aFilter, aImageKeys, aFlags);
|
|
}
|
|
|
|
bool GPUVideoTextureHost::SupportsExternalCompositing(
|
|
WebRenderBackend aBackend) {
|
|
if (!EnsureWrappedTextureHost()) {
|
|
return false;
|
|
}
|
|
return EnsureWrappedTextureHost()->SupportsExternalCompositing(aBackend);
|
|
}
|
|
|
|
void GPUVideoTextureHost::UnbindTextureSource() {
|
|
if (EnsureWrappedTextureHost()) {
|
|
EnsureWrappedTextureHost()->UnbindTextureSource();
|
|
}
|
|
// Handle read unlock
|
|
TextureHost::UnbindTextureSource();
|
|
}
|
|
|
|
void GPUVideoTextureHost::NotifyNotUsed() {
|
|
if (EnsureWrappedTextureHost()) {
|
|
EnsureWrappedTextureHost()->NotifyNotUsed();
|
|
}
|
|
TextureHost::NotifyNotUsed();
|
|
}
|
|
|
|
BufferTextureHost* GPUVideoTextureHost::AsBufferTextureHost() {
|
|
if (EnsureWrappedTextureHost()) {
|
|
return EnsureWrappedTextureHost()->AsBufferTextureHost();
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
DXGITextureHostD3D11* GPUVideoTextureHost::AsDXGITextureHostD3D11() {
|
|
if (EnsureWrappedTextureHost()) {
|
|
return EnsureWrappedTextureHost()->AsDXGITextureHostD3D11();
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
DXGIYCbCrTextureHostD3D11* GPUVideoTextureHost::AsDXGIYCbCrTextureHostD3D11() {
|
|
if (EnsureWrappedTextureHost()) {
|
|
return mWrappedTextureHost->AsDXGIYCbCrTextureHostD3D11();
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
bool GPUVideoTextureHost::IsWrappingSurfaceTextureHost() {
|
|
if (EnsureWrappedTextureHost()) {
|
|
return EnsureWrappedTextureHost()->IsWrappingSurfaceTextureHost();
|
|
}
|
|
return false;
|
|
}
|
|
|
|
TextureHostType GPUVideoTextureHost::GetTextureHostType() {
|
|
if (!mWrappedTextureHost) {
|
|
return TextureHostType::Unknown;
|
|
}
|
|
return mWrappedTextureHost->GetTextureHostType();
|
|
}
|
|
|
|
bool GPUVideoTextureHost::NeedsDeferredDeletion() const {
|
|
if (!mWrappedTextureHost) {
|
|
return TextureHost::NeedsDeferredDeletion();
|
|
}
|
|
return mWrappedTextureHost->NeedsDeferredDeletion();
|
|
}
|
|
|
|
} // namespace layers
|
|
} // namespace mozilla
|