blob: 2ffb547514b1b77db381567d0183daa1e56ee2d2 [file] [log] [blame]
Alec Mouric7f6c8b2020-11-09 18:35:20 -08001/*
2 * Copyright 2020 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#include "AutoBackendTexture.h"
18
19#undef LOG_TAG
20#define LOG_TAG "RenderEngine"
21#define ATRACE_TAG ATRACE_TAG_GRAPHICS
22
23#include <utils/Trace.h>
24
25#include "log/log_main.h"
26#include "utils/Trace.h"
27
28namespace android {
29namespace renderengine {
30namespace skia {
31
32// Converts an android dataspace to a supported SkColorSpace
33// Supported dataspaces are
34// 1. sRGB
35// 2. Display P3
36// 3. BT2020 PQ
37// 4. BT2020 HLG
38// Unknown primaries are mapped to BT709, and unknown transfer functions
39// are mapped to sRGB.
40static sk_sp<SkColorSpace> toSkColorSpace(ui::Dataspace dataspace) {
41 skcms_Matrix3x3 gamut;
42 switch (dataspace & HAL_DATASPACE_STANDARD_MASK) {
43 case HAL_DATASPACE_STANDARD_BT709:
44 gamut = SkNamedGamut::kSRGB;
45 break;
46 case HAL_DATASPACE_STANDARD_BT2020:
47 gamut = SkNamedGamut::kRec2020;
48 break;
49 case HAL_DATASPACE_STANDARD_DCI_P3:
50 gamut = SkNamedGamut::kDisplayP3;
51 break;
52 default:
53 ALOGV("Unsupported Gamut: %d, defaulting to sRGB", dataspace);
54 gamut = SkNamedGamut::kSRGB;
55 break;
56 }
57
58 switch (dataspace & HAL_DATASPACE_TRANSFER_MASK) {
59 case HAL_DATASPACE_TRANSFER_LINEAR:
60 return SkColorSpace::MakeRGB(SkNamedTransferFn::kLinear, gamut);
61 case HAL_DATASPACE_TRANSFER_SRGB:
62 return SkColorSpace::MakeRGB(SkNamedTransferFn::kSRGB, gamut);
63 case HAL_DATASPACE_TRANSFER_ST2084:
64 return SkColorSpace::MakeRGB(SkNamedTransferFn::kPQ, gamut);
65 case HAL_DATASPACE_TRANSFER_HLG:
66 return SkColorSpace::MakeRGB(SkNamedTransferFn::kHLG, gamut);
67 default:
68 ALOGV("Unsupported Gamma: %d, defaulting to sRGB transfer", dataspace);
69 return SkColorSpace::MakeRGB(SkNamedTransferFn::kSRGB, gamut);
70 }
71}
72
73AutoBackendTexture::AutoBackendTexture(GrDirectContext* context, AHardwareBuffer* buffer,
74 bool isRender) {
Alec Mourie1f81982021-01-11 10:24:27 -080075 ATRACE_CALL();
Alec Mouric7f6c8b2020-11-09 18:35:20 -080076 AHardwareBuffer_Desc desc;
77 AHardwareBuffer_describe(buffer, &desc);
78 bool createProtectedImage = 0 != (desc.usage & AHARDWAREBUFFER_USAGE_PROTECTED_CONTENT);
79 GrBackendFormat backendFormat =
80 GrAHardwareBufferUtils::GetBackendFormat(context, buffer, desc.format, false);
81 mBackendTexture =
82 GrAHardwareBufferUtils::MakeBackendTexture(context, buffer, desc.width, desc.height,
83 &mDeleteProc, &mUpdateProc, &mImageCtx,
84 createProtectedImage, backendFormat,
85 isRender);
86 mColorType = GrAHardwareBufferUtils::GetSkColorTypeFromBufferFormat(desc.format);
87}
88
89void AutoBackendTexture::unref(bool releaseLocalResources) {
90 if (releaseLocalResources) {
91 mSurface = nullptr;
92 mImage = nullptr;
93 }
94
95 mUsageCount--;
96 if (mUsageCount <= 0) {
97 if (mBackendTexture.isValid()) {
98 mDeleteProc(mImageCtx);
99 mBackendTexture = {};
100 }
101 delete this;
102 }
103}
104
105// releaseSurfaceProc is invoked by SkSurface, when the texture is no longer in use.
106// "releaseContext" contains an "AutoBackendTexture*".
107void AutoBackendTexture::releaseSurfaceProc(SkSurface::ReleaseContext releaseContext) {
108 AutoBackendTexture* textureRelease = reinterpret_cast<AutoBackendTexture*>(releaseContext);
109 textureRelease->unref(false);
110}
111
112// releaseImageProc is invoked by SkImage, when the texture is no longer in use.
113// "releaseContext" contains an "AutoBackendTexture*".
114void AutoBackendTexture::releaseImageProc(SkImage::ReleaseContext releaseContext) {
115 AutoBackendTexture* textureRelease = reinterpret_cast<AutoBackendTexture*>(releaseContext);
116 textureRelease->unref(false);
117}
118
119sk_sp<SkImage> AutoBackendTexture::makeImage(ui::Dataspace dataspace, SkAlphaType alphaType,
120 GrDirectContext* context) {
121 ATRACE_CALL();
122
123 if (mBackendTexture.isValid()) {
124 mUpdateProc(mImageCtx, context);
125 }
126
127 sk_sp<SkImage> image =
128 SkImage::MakeFromTexture(context, mBackendTexture, kTopLeft_GrSurfaceOrigin, mColorType,
129 alphaType, toSkColorSpace(dataspace), releaseImageProc, this);
130 if (image.get()) {
131 // The following ref will be counteracted by releaseProc, when SkImage is discarded.
132 ref();
133 }
134
135 mImage = image;
136 mDataspace = dataspace;
137 LOG_ALWAYS_FATAL_IF(mImage == nullptr, "Unable to generate SkImage from buffer");
138 return mImage;
139}
140
141sk_sp<SkSurface> AutoBackendTexture::getOrCreateSurface(ui::Dataspace dataspace,
142 GrDirectContext* context) {
143 ATRACE_CALL();
144 if (!mSurface.get() || mDataspace != dataspace) {
145 sk_sp<SkSurface> surface =
146 SkSurface::MakeFromBackendTexture(context, mBackendTexture,
147 kTopLeft_GrSurfaceOrigin, 0, mColorType,
148 toSkColorSpace(dataspace), nullptr,
149 releaseSurfaceProc, this);
150 if (surface.get()) {
151 // The following ref will be counteracted by releaseProc, when SkSurface is discarded.
152 ref();
153 }
154 mSurface = surface;
155 }
156
157 mDataspace = dataspace;
158 LOG_ALWAYS_FATAL_IF(mSurface == nullptr, "Unable to generate SkSurface");
159 return mSurface;
160}
161
162} // namespace skia
163} // namespace renderengine
164} // namespace android