blob: d126c27728b75997d33ebc210760916006c7b83d [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) {
75 AHardwareBuffer_Desc desc;
76 AHardwareBuffer_describe(buffer, &desc);
77 bool createProtectedImage = 0 != (desc.usage & AHARDWAREBUFFER_USAGE_PROTECTED_CONTENT);
78 GrBackendFormat backendFormat =
79 GrAHardwareBufferUtils::GetBackendFormat(context, buffer, desc.format, false);
80 mBackendTexture =
81 GrAHardwareBufferUtils::MakeBackendTexture(context, buffer, desc.width, desc.height,
82 &mDeleteProc, &mUpdateProc, &mImageCtx,
83 createProtectedImage, backendFormat,
84 isRender);
85 mColorType = GrAHardwareBufferUtils::GetSkColorTypeFromBufferFormat(desc.format);
86}
87
88void AutoBackendTexture::unref(bool releaseLocalResources) {
89 if (releaseLocalResources) {
90 mSurface = nullptr;
91 mImage = nullptr;
92 }
93
94 mUsageCount--;
95 if (mUsageCount <= 0) {
96 if (mBackendTexture.isValid()) {
97 mDeleteProc(mImageCtx);
98 mBackendTexture = {};
99 }
100 delete this;
101 }
102}
103
104// releaseSurfaceProc is invoked by SkSurface, when the texture is no longer in use.
105// "releaseContext" contains an "AutoBackendTexture*".
106void AutoBackendTexture::releaseSurfaceProc(SkSurface::ReleaseContext releaseContext) {
107 AutoBackendTexture* textureRelease = reinterpret_cast<AutoBackendTexture*>(releaseContext);
108 textureRelease->unref(false);
109}
110
111// releaseImageProc is invoked by SkImage, when the texture is no longer in use.
112// "releaseContext" contains an "AutoBackendTexture*".
113void AutoBackendTexture::releaseImageProc(SkImage::ReleaseContext releaseContext) {
114 AutoBackendTexture* textureRelease = reinterpret_cast<AutoBackendTexture*>(releaseContext);
115 textureRelease->unref(false);
116}
117
118sk_sp<SkImage> AutoBackendTexture::makeImage(ui::Dataspace dataspace, SkAlphaType alphaType,
119 GrDirectContext* context) {
120 ATRACE_CALL();
121
122 if (mBackendTexture.isValid()) {
123 mUpdateProc(mImageCtx, context);
124 }
125
126 sk_sp<SkImage> image =
127 SkImage::MakeFromTexture(context, mBackendTexture, kTopLeft_GrSurfaceOrigin, mColorType,
128 alphaType, toSkColorSpace(dataspace), releaseImageProc, this);
129 if (image.get()) {
130 // The following ref will be counteracted by releaseProc, when SkImage is discarded.
131 ref();
132 }
133
134 mImage = image;
135 mDataspace = dataspace;
136 LOG_ALWAYS_FATAL_IF(mImage == nullptr, "Unable to generate SkImage from buffer");
137 return mImage;
138}
139
140sk_sp<SkSurface> AutoBackendTexture::getOrCreateSurface(ui::Dataspace dataspace,
141 GrDirectContext* context) {
142 ATRACE_CALL();
143 if (!mSurface.get() || mDataspace != dataspace) {
144 sk_sp<SkSurface> surface =
145 SkSurface::MakeFromBackendTexture(context, mBackendTexture,
146 kTopLeft_GrSurfaceOrigin, 0, mColorType,
147 toSkColorSpace(dataspace), nullptr,
148 releaseSurfaceProc, this);
149 if (surface.get()) {
150 // The following ref will be counteracted by releaseProc, when SkSurface is discarded.
151 ref();
152 }
153 mSurface = surface;
154 }
155
156 mDataspace = dataspace;
157 LOG_ALWAYS_FATAL_IF(mSurface == nullptr, "Unable to generate SkSurface");
158 return mSurface;
159}
160
161} // namespace skia
162} // namespace renderengine
163} // namespace android