Bug 1833753 [wpt PR 40065] - Allow newly-added test to also pass when mutation events...
[gecko.git] / gfx / webrender_bindings / RenderBufferTextureHost.cpp
blob920556c7efd1c398c20e57a341912a1225111107
1 /* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
2 /* vim: set ts=8 sts=2 et sw=2 tw=80: */
3 /* This Source Code Form is subject to the terms of the Mozilla Public
4 * License, v. 2.0. If a copy of the MPL was not distributed with this
5 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
7 #include "RenderBufferTextureHost.h"
9 #include "mozilla/gfx/Logging.h"
10 #include "mozilla/layers/ImageDataSerializer.h"
12 namespace mozilla {
13 namespace wr {
15 RenderBufferTextureHost::RenderBufferTextureHost(
16 uint8_t* aBuffer, const layers::BufferDescriptor& aDescriptor)
17 : mBuffer(aBuffer),
18 mDescriptor(aDescriptor),
19 mMap(),
20 mYMap(),
21 mCbMap(),
22 mCrMap(),
23 mLocked(false) {
24 MOZ_COUNT_CTOR_INHERITED(RenderBufferTextureHost, RenderTextureHost);
26 switch (mDescriptor.type()) {
27 case layers::BufferDescriptor::TYCbCrDescriptor: {
28 const layers::YCbCrDescriptor& ycbcr = mDescriptor.get_YCbCrDescriptor();
29 mSize = ycbcr.display().Size();
30 mFormat = gfx::SurfaceFormat::YUV;
31 break;
33 case layers::BufferDescriptor::TRGBDescriptor: {
34 const layers::RGBDescriptor& rgb = mDescriptor.get_RGBDescriptor();
35 mSize = rgb.size();
36 mFormat = rgb.format();
37 break;
39 default:
40 gfxCriticalError() << "Bad buffer host descriptor "
41 << (int)mDescriptor.type();
42 MOZ_CRASH("GFX: Bad descriptor");
46 RenderBufferTextureHost::~RenderBufferTextureHost() {
47 MOZ_COUNT_DTOR_INHERITED(RenderBufferTextureHost, RenderTextureHost);
50 wr::WrExternalImage RenderBufferTextureHost::Lock(uint8_t aChannelIndex,
51 gl::GLContext* aGL) {
52 if (!mLocked) {
53 if (!GetBuffer()) {
54 if (!mDestroyed) {
55 // We hit some problems to get the shmem.
56 gfxCriticalNote << "GetBuffer Failed";
58 return InvalidToWrExternalImage();
60 if (mFormat != gfx::SurfaceFormat::YUV) {
61 mSurface = gfx::Factory::CreateWrappingDataSourceSurface(
62 GetBuffer(),
63 layers::ImageDataSerializer::GetRGBStride(
64 mDescriptor.get_RGBDescriptor()),
65 mSize, mFormat);
66 if (NS_WARN_IF(!mSurface)) {
67 gfxCriticalNote << "DataSourceSurface is null";
68 return InvalidToWrExternalImage();
70 if (NS_WARN_IF(
71 !mSurface->Map(gfx::DataSourceSurface::MapType::READ, &mMap))) {
72 mSurface = nullptr;
73 gfxCriticalNote << "Failed to map Surface";
74 return InvalidToWrExternalImage();
76 } else {
77 const layers::YCbCrDescriptor& desc = mDescriptor.get_YCbCrDescriptor();
78 auto cbcrSize = layers::ImageDataSerializer::GetCroppedCbCrSize(desc);
80 mYSurface = gfx::Factory::CreateWrappingDataSourceSurface(
81 layers::ImageDataSerializer::GetYChannel(GetBuffer(), desc),
82 desc.yStride(), desc.display().Size(), gfx::SurfaceFormat::A8);
83 mCbSurface = gfx::Factory::CreateWrappingDataSourceSurface(
84 layers::ImageDataSerializer::GetCbChannel(GetBuffer(), desc),
85 desc.cbCrStride(), cbcrSize, gfx::SurfaceFormat::A8);
86 mCrSurface = gfx::Factory::CreateWrappingDataSourceSurface(
87 layers::ImageDataSerializer::GetCrChannel(GetBuffer(), desc),
88 desc.cbCrStride(), cbcrSize, gfx::SurfaceFormat::A8);
89 if (NS_WARN_IF(!mYSurface || !mCbSurface || !mCrSurface)) {
90 mYSurface = mCbSurface = mCrSurface = nullptr;
91 gfxCriticalNote << "YCbCr Surface is null";
92 return InvalidToWrExternalImage();
94 if (NS_WARN_IF(
95 !mYSurface->Map(gfx::DataSourceSurface::MapType::READ, &mYMap) ||
96 !mCbSurface->Map(gfx::DataSourceSurface::MapType::READ,
97 &mCbMap) ||
98 !mCrSurface->Map(gfx::DataSourceSurface::MapType::READ,
99 &mCrMap))) {
100 mYSurface = mCbSurface = mCrSurface = nullptr;
101 gfxCriticalNote << "Failed to map YCbCr Surface";
102 return InvalidToWrExternalImage();
105 mLocked = true;
108 RenderBufferData data = GetBufferDataForRender(aChannelIndex);
109 return RawDataToWrExternalImage(data.mData, data.mBufferSize);
112 void RenderBufferTextureHost::Unlock() {
113 if (mLocked) {
114 if (mSurface) {
115 mSurface->Unmap();
116 mSurface = nullptr;
117 } else if (mYSurface) {
118 mYSurface->Unmap();
119 mCbSurface->Unmap();
120 mCrSurface->Unmap();
121 mYSurface = mCbSurface = mCrSurface = nullptr;
123 mLocked = false;
127 RenderBufferTextureHost::RenderBufferData
128 RenderBufferTextureHost::GetBufferDataForRender(uint8_t aChannelIndex) {
129 MOZ_ASSERT(mFormat != gfx::SurfaceFormat::YUV || aChannelIndex < 3);
130 MOZ_ASSERT(mFormat == gfx::SurfaceFormat::YUV || aChannelIndex < 1);
131 MOZ_ASSERT(mLocked);
133 if (mFormat != gfx::SurfaceFormat::YUV) {
134 MOZ_ASSERT(mSurface);
136 return RenderBufferData(mMap.mData,
137 mMap.mStride * mSurface->GetSize().height);
138 } else {
139 MOZ_ASSERT(mYSurface && mCbSurface && mCrSurface);
141 switch (aChannelIndex) {
142 case 0:
143 return RenderBufferData(mYMap.mData,
144 mYMap.mStride * mYSurface->GetSize().height);
145 break;
146 case 1:
147 return RenderBufferData(mCbMap.mData,
148 mCbMap.mStride * mCbSurface->GetSize().height);
149 break;
150 case 2:
151 return RenderBufferData(mCrMap.mData,
152 mCrMap.mStride * mCrSurface->GetSize().height);
153 break;
154 default:
155 MOZ_ASSERT_UNREACHABLE("unexpected to be called");
156 return RenderBufferData(nullptr, 0);
161 size_t RenderBufferTextureHost::GetPlaneCount() const {
162 switch (mDescriptor.type()) {
163 case layers::BufferDescriptor::TYCbCrDescriptor:
164 return 3;
165 default:
166 return 1;
170 gfx::SurfaceFormat RenderBufferTextureHost::GetFormat() const {
171 switch (mDescriptor.type()) {
172 case layers::BufferDescriptor::TYCbCrDescriptor:
173 return gfx::SurfaceFormat::YUV;
174 default:
175 return mDescriptor.get_RGBDescriptor().format();
179 gfx::ColorDepth RenderBufferTextureHost::GetColorDepth() const {
180 switch (mDescriptor.type()) {
181 case layers::BufferDescriptor::TYCbCrDescriptor:
182 return mDescriptor.get_YCbCrDescriptor().colorDepth();
183 default:
184 return gfx::ColorDepth::COLOR_8;
188 gfx::YUVRangedColorSpace RenderBufferTextureHost::GetYUVColorSpace() const {
189 switch (mDescriptor.type()) {
190 case layers::BufferDescriptor::TYCbCrDescriptor:
191 return gfx::GetYUVRangedColorSpace(mDescriptor.get_YCbCrDescriptor());
192 default:
193 return gfx::YUVRangedColorSpace::Default;
197 bool RenderBufferTextureHost::MapPlane(RenderCompositor* aCompositor,
198 uint8_t aChannelIndex,
199 PlaneInfo& aPlaneInfo) {
200 if (!mBuffer) {
201 if (!mDestroyed) {
202 // We hit some problems to get the shmem.
203 gfxCriticalNote << "GetBuffer Failed";
205 return false;
208 switch (mDescriptor.type()) {
209 case layers::BufferDescriptor::TYCbCrDescriptor: {
210 const layers::YCbCrDescriptor& desc = mDescriptor.get_YCbCrDescriptor();
211 switch (aChannelIndex) {
212 case 0:
213 aPlaneInfo.mData =
214 layers::ImageDataSerializer::GetYChannel(mBuffer, desc);
215 aPlaneInfo.mStride = desc.yStride();
216 aPlaneInfo.mSize = desc.display().Size();
217 break;
218 case 1:
219 aPlaneInfo.mData =
220 layers::ImageDataSerializer::GetCbChannel(mBuffer, desc);
221 aPlaneInfo.mStride = desc.cbCrStride();
222 aPlaneInfo.mSize =
223 layers::ImageDataSerializer::GetCroppedCbCrSize(desc);
224 break;
225 case 2:
226 aPlaneInfo.mData =
227 layers::ImageDataSerializer::GetCrChannel(mBuffer, desc);
228 aPlaneInfo.mStride = desc.cbCrStride();
229 aPlaneInfo.mSize =
230 layers::ImageDataSerializer::GetCroppedCbCrSize(desc);
231 break;
233 break;
235 default: {
236 const layers::RGBDescriptor& desc = mDescriptor.get_RGBDescriptor();
237 aPlaneInfo.mData = mBuffer;
238 aPlaneInfo.mStride = layers::ImageDataSerializer::GetRGBStride(desc);
239 aPlaneInfo.mSize = desc.size();
240 break;
243 return true;
246 void RenderBufferTextureHost::UnmapPlanes() {}
248 void RenderBufferTextureHost::Destroy() {
249 mBuffer = nullptr;
250 mDestroyed = true;
253 } // namespace wr
254 } // namespace mozilla