Bug 1685225 - Use state bits to determine the color for non-native theme meter chunks...
[gecko.git] / gfx / webrender_bindings / RenderBufferTextureHost.cpp
blob7f8e45463318aad69247440de0aa9625cb8c3db6
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(
51 uint8_t aChannelIndex, gl::GLContext* aGL, wr::ImageRendering aRendering) {
52 if (!mLocked) {
53 if (!GetBuffer()) {
54 // We hit some problems to get the shmem.
55 gfxCriticalNote << "GetBuffer Failed";
56 return InvalidToWrExternalImage();
58 if (mFormat != gfx::SurfaceFormat::YUV) {
59 mSurface = gfx::Factory::CreateWrappingDataSourceSurface(
60 GetBuffer(),
61 layers::ImageDataSerializer::GetRGBStride(
62 mDescriptor.get_RGBDescriptor()),
63 mSize, mFormat);
64 if (NS_WARN_IF(!mSurface)) {
65 gfxCriticalNote << "DataSourceSurface is null";
66 return InvalidToWrExternalImage();
68 if (NS_WARN_IF(!mSurface->Map(gfx::DataSourceSurface::MapType::READ_WRITE,
69 &mMap))) {
70 mSurface = nullptr;
71 gfxCriticalNote << "Failed to map Surface";
72 return InvalidToWrExternalImage();
74 } else {
75 const layers::YCbCrDescriptor& desc = mDescriptor.get_YCbCrDescriptor();
77 mYSurface = gfx::Factory::CreateWrappingDataSourceSurface(
78 layers::ImageDataSerializer::GetYChannel(GetBuffer(), desc),
79 desc.yStride(), desc.ySize(), gfx::SurfaceFormat::A8);
80 mCbSurface = gfx::Factory::CreateWrappingDataSourceSurface(
81 layers::ImageDataSerializer::GetCbChannel(GetBuffer(), desc),
82 desc.cbCrStride(), desc.cbCrSize(), gfx::SurfaceFormat::A8);
83 mCrSurface = gfx::Factory::CreateWrappingDataSourceSurface(
84 layers::ImageDataSerializer::GetCrChannel(GetBuffer(), desc),
85 desc.cbCrStride(), desc.cbCrSize(), gfx::SurfaceFormat::A8);
86 if (NS_WARN_IF(!mYSurface || !mCbSurface || !mCrSurface)) {
87 mYSurface = mCbSurface = mCrSurface = nullptr;
88 gfxCriticalNote << "YCbCr Surface is null";
89 return InvalidToWrExternalImage();
91 if (NS_WARN_IF(
92 !mYSurface->Map(gfx::DataSourceSurface::MapType::READ_WRITE,
93 &mYMap) ||
94 !mCbSurface->Map(gfx::DataSourceSurface::MapType::READ_WRITE,
95 &mCbMap) ||
96 !mCrSurface->Map(gfx::DataSourceSurface::MapType::READ_WRITE,
97 &mCrMap))) {
98 mYSurface = mCbSurface = mCrSurface = nullptr;
99 gfxCriticalNote << "Failed to map YCbCr Surface";
100 return InvalidToWrExternalImage();
103 mLocked = true;
106 RenderBufferData data = GetBufferDataForRender(aChannelIndex);
107 return RawDataToWrExternalImage(data.mData, data.mBufferSize);
110 void RenderBufferTextureHost::Unlock() {
111 if (mLocked) {
112 if (mSurface) {
113 mSurface->Unmap();
114 mSurface = nullptr;
115 } else if (mYSurface) {
116 mYSurface->Unmap();
117 mCbSurface->Unmap();
118 mCrSurface->Unmap();
119 mYSurface = mCbSurface = mCrSurface = nullptr;
121 mLocked = false;
125 RenderBufferTextureHost::RenderBufferData
126 RenderBufferTextureHost::GetBufferDataForRender(uint8_t aChannelIndex) {
127 MOZ_ASSERT(mFormat != gfx::SurfaceFormat::YUV || aChannelIndex < 3);
128 MOZ_ASSERT(mFormat == gfx::SurfaceFormat::YUV || aChannelIndex < 1);
129 MOZ_ASSERT(mLocked);
131 if (mFormat != gfx::SurfaceFormat::YUV) {
132 MOZ_ASSERT(mSurface);
134 return RenderBufferData(mMap.mData,
135 mMap.mStride * mSurface->GetSize().height);
136 } else {
137 MOZ_ASSERT(mYSurface && mCbSurface && mCrSurface);
139 switch (aChannelIndex) {
140 case 0:
141 return RenderBufferData(mYMap.mData,
142 mYMap.mStride * mYSurface->GetSize().height);
143 break;
144 case 1:
145 return RenderBufferData(mCbMap.mData,
146 mCbMap.mStride * mCbSurface->GetSize().height);
147 break;
148 case 2:
149 return RenderBufferData(mCrMap.mData,
150 mCrMap.mStride * mCrSurface->GetSize().height);
151 break;
152 default:
153 MOZ_ASSERT_UNREACHABLE("unexpected to be called");
154 return RenderBufferData(nullptr, 0);
159 } // namespace wr
160 } // namespace mozilla