blob: 9b19afbdacac2af7504c4809e20a5c1df50e44ee [file] [log] [blame]
Ady Abraham03b02dd2019-03-21 15:40:11 -07001/*
2 * Copyright 2019 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
Marin Shalamanovf7f6b3c2020-12-09 13:19:38 +010017#include <algorithm>
18
Ady Abraham3649ea82022-07-08 16:04:02 -070019#include "BackgroundExecutor.h"
Ady Abraham03b02dd2019-03-21 15:40:11 -070020#include "Client.h"
21#include "Layer.h"
Ady Abraham3649ea82022-07-08 16:04:02 -070022#include "RefreshRateOverlay.h"
Ady Abraham03b02dd2019-03-21 15:40:11 -070023
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070024#pragma clang diagnostic push
25#pragma clang diagnostic ignored "-Wconversion"
26#include <SkCanvas.h>
rnlee756005b2021-05-27 10:46:36 -070027#include <SkPaint.h>
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070028#pragma clang diagnostic pop
29#include <SkBlendMode.h>
rnlee756005b2021-05-27 10:46:36 -070030#include <SkRect.h>
31#include <SkSurface.h>
chaviw70aa7572021-09-22 12:27:57 -050032#include <gui/SurfaceComposerClient.h>
33#include <gui/SurfaceControl.h>
Ady Abraham2cb8b622019-12-02 18:55:33 -080034
35#undef LOG_TAG
36#define LOG_TAG "RefreshRateOverlay"
37
Ady Abraham03b02dd2019-03-21 15:40:11 -070038namespace android {
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070039namespace {
Ady Abraham03b02dd2019-03-21 15:40:11 -070040
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070041constexpr int kDigitWidth = 64;
42constexpr int kDigitHeight = 100;
43constexpr int kDigitSpace = 16;
44
45// Layout is digit, space, digit, space, digit, space, spinner.
46constexpr int kBufferWidth = 4 * kDigitWidth + 3 * kDigitSpace;
47constexpr int kBufferHeight = kDigitHeight;
48
Dominik Laskowski1f6fc702022-03-21 08:34:50 -070049SurfaceComposerClient::Transaction createTransaction(const sp<SurfaceControl>& surface) {
50 constexpr float kFrameRate = 0.f;
51 constexpr int8_t kCompatibility = ANATIVEWINDOW_FRAME_RATE_NO_VOTE;
52 constexpr int8_t kSeamlessness = ANATIVEWINDOW_CHANGE_FRAME_RATE_ONLY_IF_SEAMLESS;
53
54 return SurfaceComposerClient::Transaction().setFrameRate(surface, kFrameRate, kCompatibility,
55 kSeamlessness);
56}
57
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070058} // namespace
59
Ady Abraham3649ea82022-07-08 16:04:02 -070060SurfaceControlHolder::~SurfaceControlHolder() {
61 // Hand the sp<SurfaceControl> to the helper thread to release the last
62 // reference. This makes sure that the SurfaceControl is destructed without
63 // SurfaceFlinger::mStateLock held.
64 BackgroundExecutor::getInstance().sendCallbacks(
65 {[sc = std::move(mSurfaceControl)]() mutable { sc.clear(); }});
66}
67
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070068void RefreshRateOverlay::SevenSegmentDrawer::drawSegment(Segment segment, int left, SkColor color,
rnlee756005b2021-05-27 10:46:36 -070069 SkCanvas& canvas) {
70 const SkRect rect = [&]() {
Ady Abraham2cb8b622019-12-02 18:55:33 -080071 switch (segment) {
72 case Segment::Upper:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070073 return SkRect::MakeLTRB(left, 0, left + kDigitWidth, kDigitSpace);
Ady Abraham2cb8b622019-12-02 18:55:33 -080074 case Segment::UpperLeft:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070075 return SkRect::MakeLTRB(left, 0, left + kDigitSpace, kDigitHeight / 2);
Ady Abraham2cb8b622019-12-02 18:55:33 -080076 case Segment::UpperRight:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070077 return SkRect::MakeLTRB(left + kDigitWidth - kDigitSpace, 0, left + kDigitWidth,
78 kDigitHeight / 2);
Ady Abraham2cb8b622019-12-02 18:55:33 -080079 case Segment::Middle:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070080 return SkRect::MakeLTRB(left, kDigitHeight / 2 - kDigitSpace / 2,
81 left + kDigitWidth, kDigitHeight / 2 + kDigitSpace / 2);
Ady Abraham2cb8b622019-12-02 18:55:33 -080082 case Segment::LowerLeft:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070083 return SkRect::MakeLTRB(left, kDigitHeight / 2, left + kDigitSpace, kDigitHeight);
Ady Abraham2cb8b622019-12-02 18:55:33 -080084 case Segment::LowerRight:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070085 return SkRect::MakeLTRB(left + kDigitWidth - kDigitSpace, kDigitHeight / 2,
86 left + kDigitWidth, kDigitHeight);
rnlee756005b2021-05-27 10:46:36 -070087 case Segment::Bottom:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070088 return SkRect::MakeLTRB(left, kDigitHeight - kDigitSpace, left + kDigitWidth,
89 kDigitHeight);
Ady Abraham2cb8b622019-12-02 18:55:33 -080090 }
91 }();
92
rnlee756005b2021-05-27 10:46:36 -070093 SkPaint paint;
94 paint.setColor(color);
95 paint.setBlendMode(SkBlendMode::kSrc);
96 canvas.drawRect(rect, paint);
Ady Abraham2cb8b622019-12-02 18:55:33 -080097}
98
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070099void RefreshRateOverlay::SevenSegmentDrawer::drawDigit(int digit, int left, SkColor color,
rnlee756005b2021-05-27 10:46:36 -0700100 SkCanvas& canvas) {
Ady Abraham2cb8b622019-12-02 18:55:33 -0800101 if (digit < 0 || digit > 9) return;
102
103 if (digit == 0 || digit == 2 || digit == 3 || digit == 5 || digit == 6 || digit == 7 ||
104 digit == 8 || digit == 9)
rnlee756005b2021-05-27 10:46:36 -0700105 drawSegment(Segment::Upper, left, color, canvas);
Ady Abraham2cb8b622019-12-02 18:55:33 -0800106 if (digit == 0 || digit == 4 || digit == 5 || digit == 6 || digit == 8 || digit == 9)
rnlee756005b2021-05-27 10:46:36 -0700107 drawSegment(Segment::UpperLeft, left, color, canvas);
Ady Abraham2cb8b622019-12-02 18:55:33 -0800108 if (digit == 0 || digit == 1 || digit == 2 || digit == 3 || digit == 4 || digit == 7 ||
109 digit == 8 || digit == 9)
rnlee756005b2021-05-27 10:46:36 -0700110 drawSegment(Segment::UpperRight, left, color, canvas);
Ady Abraham2cb8b622019-12-02 18:55:33 -0800111 if (digit == 2 || digit == 3 || digit == 4 || digit == 5 || digit == 6 || digit == 8 ||
112 digit == 9)
rnlee756005b2021-05-27 10:46:36 -0700113 drawSegment(Segment::Middle, left, color, canvas);
Ady Abraham2cb8b622019-12-02 18:55:33 -0800114 if (digit == 0 || digit == 2 || digit == 6 || digit == 8)
rnlee756005b2021-05-27 10:46:36 -0700115 drawSegment(Segment::LowerLeft, left, color, canvas);
Ady Abraham2cb8b622019-12-02 18:55:33 -0800116 if (digit == 0 || digit == 1 || digit == 3 || digit == 4 || digit == 5 || digit == 6 ||
117 digit == 7 || digit == 8 || digit == 9)
rnlee756005b2021-05-27 10:46:36 -0700118 drawSegment(Segment::LowerRight, left, color, canvas);
Ady Abraham2cb8b622019-12-02 18:55:33 -0800119 if (digit == 0 || digit == 2 || digit == 3 || digit == 5 || digit == 6 || digit == 8 ||
120 digit == 9)
rnlee756005b2021-05-27 10:46:36 -0700121 drawSegment(Segment::Bottom, left, color, canvas);
Ady Abraham2cb8b622019-12-02 18:55:33 -0800122}
123
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700124auto RefreshRateOverlay::SevenSegmentDrawer::draw(int number, SkColor color,
125 ui::Transform::RotationFlags rotation,
126 bool showSpinner) -> Buffers {
Ady Abraham29d0da32020-07-16 18:39:33 -0700127 if (number < 0 || number > 1000) return {};
Ady Abraham2cb8b622019-12-02 18:55:33 -0800128
129 const auto hundreds = number / 100;
130 const auto tens = (number / 10) % 10;
131 const auto ones = number % 10;
132
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700133 const size_t loopCount = showSpinner ? 6 : 1;
134
135 Buffers buffers;
136 buffers.reserve(loopCount);
137
138 for (size_t i = 0; i < loopCount; i++) {
rnlee756005b2021-05-27 10:46:36 -0700139 // Pre-rotate the buffer before it reaches SurfaceFlinger.
140 SkMatrix canvasTransform = SkMatrix();
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700141 const auto [bufferWidth, bufferHeight] = [&]() -> std::pair<int, int> {
rnlee756005b2021-05-27 10:46:36 -0700142 switch (rotation) {
143 case ui::Transform::ROT_90:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700144 canvasTransform.setTranslate(kBufferHeight, 0);
145 canvasTransform.preRotate(90.f);
146 return {kBufferHeight, kBufferWidth};
rnlee756005b2021-05-27 10:46:36 -0700147 case ui::Transform::ROT_270:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700148 canvasTransform.setRotate(270.f, kBufferWidth / 2.f, kBufferWidth / 2.f);
149 return {kBufferHeight, kBufferWidth};
rnlee756005b2021-05-27 10:46:36 -0700150 default:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700151 return {kBufferWidth, kBufferHeight};
rnlee756005b2021-05-27 10:46:36 -0700152 }
153 }();
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700154
Ady Abrahamd11bade2022-08-01 16:18:03 -0700155 const auto kUsageFlags =
156 static_cast<uint64_t>(GRALLOC_USAGE_SW_WRITE_RARELY | GRALLOC_USAGE_HW_COMPOSER |
157 GRALLOC_USAGE_HW_TEXTURE);
158 sp<GraphicBuffer> buffer = sp<GraphicBuffer>::make(static_cast<uint32_t>(bufferWidth),
159 static_cast<uint32_t>(bufferHeight),
160 HAL_PIXEL_FORMAT_RGBA_8888, 1u,
161 kUsageFlags, "RefreshRateOverlayBuffer");
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700162
Alec Mouri7013b6f2021-02-12 11:16:54 -0800163 const status_t bufferStatus = buffer->initCheck();
164 LOG_ALWAYS_FATAL_IF(bufferStatus != OK, "RefreshRateOverlay: Buffer failed to allocate: %d",
165 bufferStatus);
rnlee756005b2021-05-27 10:46:36 -0700166
167 sk_sp<SkSurface> surface = SkSurface::MakeRasterN32Premul(bufferWidth, bufferHeight);
168 SkCanvas* canvas = surface->getCanvas();
169 canvas->setMatrix(canvasTransform);
170
Ady Abraham29d0da32020-07-16 18:39:33 -0700171 int left = 0;
172 if (hundreds != 0) {
rnlee756005b2021-05-27 10:46:36 -0700173 drawDigit(hundreds, left, color, *canvas);
Ady Abraham29d0da32020-07-16 18:39:33 -0700174 }
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700175 left += kDigitWidth + kDigitSpace;
Ady Abraham2cb8b622019-12-02 18:55:33 -0800176
Ady Abraham29d0da32020-07-16 18:39:33 -0700177 if (tens != 0) {
rnlee756005b2021-05-27 10:46:36 -0700178 drawDigit(tens, left, color, *canvas);
Ady Abraham29d0da32020-07-16 18:39:33 -0700179 }
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700180 left += kDigitWidth + kDigitSpace;
Ady Abraham2cb8b622019-12-02 18:55:33 -0800181
rnlee756005b2021-05-27 10:46:36 -0700182 drawDigit(ones, left, color, *canvas);
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700183 left += kDigitWidth + kDigitSpace;
Ady Abraham29d0da32020-07-16 18:39:33 -0700184
185 if (showSpinner) {
186 switch (i) {
187 case 0:
rnlee756005b2021-05-27 10:46:36 -0700188 drawSegment(Segment::Upper, left, color, *canvas);
Ady Abraham29d0da32020-07-16 18:39:33 -0700189 break;
190 case 1:
rnlee756005b2021-05-27 10:46:36 -0700191 drawSegment(Segment::UpperRight, left, color, *canvas);
Ady Abraham29d0da32020-07-16 18:39:33 -0700192 break;
193 case 2:
rnlee756005b2021-05-27 10:46:36 -0700194 drawSegment(Segment::LowerRight, left, color, *canvas);
Ady Abraham29d0da32020-07-16 18:39:33 -0700195 break;
196 case 3:
rnlee756005b2021-05-27 10:46:36 -0700197 drawSegment(Segment::Bottom, left, color, *canvas);
Ady Abraham29d0da32020-07-16 18:39:33 -0700198 break;
199 case 4:
rnlee756005b2021-05-27 10:46:36 -0700200 drawSegment(Segment::LowerLeft, left, color, *canvas);
Ady Abraham29d0da32020-07-16 18:39:33 -0700201 break;
202 case 5:
rnlee756005b2021-05-27 10:46:36 -0700203 drawSegment(Segment::UpperLeft, left, color, *canvas);
Ady Abraham29d0da32020-07-16 18:39:33 -0700204 break;
205 }
206 }
207
rnlee756005b2021-05-27 10:46:36 -0700208 void* pixels = nullptr;
209 buffer->lock(GRALLOC_USAGE_SW_WRITE_RARELY, reinterpret_cast<void**>(&pixels));
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700210
rnlee756005b2021-05-27 10:46:36 -0700211 const SkImageInfo& imageInfo = surface->imageInfo();
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700212 const size_t dstRowBytes =
213 buffer->getStride() * static_cast<size_t>(imageInfo.bytesPerPixel());
214
rnlee756005b2021-05-27 10:46:36 -0700215 canvas->readPixels(imageInfo, pixels, dstRowBytes, 0, 0);
Ady Abraham29d0da32020-07-16 18:39:33 -0700216 buffer->unlock();
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700217 buffers.push_back(std::move(buffer));
Ady Abraham29d0da32020-07-16 18:39:33 -0700218 }
219 return buffers;
Ady Abraham2cb8b622019-12-02 18:55:33 -0800220}
221
Ady Abraham3649ea82022-07-08 16:04:02 -0700222std::unique_ptr<SurfaceControlHolder> createSurfaceControlHolder() {
223 sp<SurfaceControl> surfaceControl =
224 SurfaceComposerClient::getDefault()
225 ->createSurface(String8("RefreshRateOverlay"), kBufferWidth, kBufferHeight,
226 PIXEL_FORMAT_RGBA_8888,
227 ISurfaceComposerClient::eFXSurfaceBufferState);
228 return std::make_unique<SurfaceControlHolder>(std::move(surfaceControl));
229}
230
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700231RefreshRateOverlay::RefreshRateOverlay(FpsRange fpsRange, bool showSpinner)
232 : mFpsRange(fpsRange),
Ady Abraham1b11bc62021-06-03 19:51:19 -0700233 mShowSpinner(showSpinner),
Ady Abraham3649ea82022-07-08 16:04:02 -0700234 mSurfaceControl(createSurfaceControlHolder()) {
chaviw70aa7572021-09-22 12:27:57 -0500235 if (!mSurfaceControl) {
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700236 ALOGE("%s: Failed to create buffer state layer", __func__);
237 return;
Ady Abraham03b02dd2019-03-21 15:40:11 -0700238 }
239
Ady Abraham3649ea82022-07-08 16:04:02 -0700240 createTransaction(mSurfaceControl->get())
241 .setLayer(mSurfaceControl->get(), INT32_MAX - 2)
242 .setTrustedOverlay(mSurfaceControl->get(), true)
Ady Abraham78c4a242021-11-30 17:49:44 -0800243 .apply();
Ady Abraham03b02dd2019-03-21 15:40:11 -0700244}
245
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700246auto RefreshRateOverlay::getOrCreateBuffers(Fps fps) -> const Buffers& {
247 static const Buffers kNoBuffers;
248 if (!mSurfaceControl) return kNoBuffers;
249
250 const auto transformHint =
Ady Abraham3649ea82022-07-08 16:04:02 -0700251 static_cast<ui::Transform::RotationFlags>(mSurfaceControl->get()->getTransformHint());
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700252
rnlee756005b2021-05-27 10:46:36 -0700253 // Tell SurfaceFlinger about the pre-rotation on the buffer.
254 const auto transform = [&] {
255 switch (transformHint) {
256 case ui::Transform::ROT_90:
257 return ui::Transform::ROT_270;
258 case ui::Transform::ROT_270:
259 return ui::Transform::ROT_90;
260 default:
261 return ui::Transform::ROT_0;
262 }
263 }();
chaviw70aa7572021-09-22 12:27:57 -0500264
Ady Abraham3649ea82022-07-08 16:04:02 -0700265 createTransaction(mSurfaceControl->get())
266 .setTransform(mSurfaceControl->get(), transform)
267 .apply();
rnlee756005b2021-05-27 10:46:36 -0700268
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700269 BufferCache::const_iterator it = mBufferCache.find({fps.getIntValue(), transformHint});
270 if (it == mBufferCache.end()) {
271 const int minFps = mFpsRange.min.getIntValue();
272 const int maxFps = mFpsRange.max.getIntValue();
273
274 // Clamp to the range. The current fps may be outside of this range if the display has
275 // changed its set of supported refresh rates.
276 const int intFps = std::clamp(fps.getIntValue(), minFps, maxFps);
277
278 // Ensure non-zero range to avoid division by zero.
279 const float fpsScale = static_cast<float>(intFps - minFps) / std::max(1, maxFps - minFps);
280
281 constexpr SkColor kMinFpsColor = SK_ColorRED;
282 constexpr SkColor kMaxFpsColor = SK_ColorGREEN;
283 constexpr float kAlpha = 0.8f;
284
285 SkColor4f colorBase = SkColor4f::FromColor(kMaxFpsColor) * fpsScale;
286 const SkColor4f minFpsColor = SkColor4f::FromColor(kMinFpsColor) * (1 - fpsScale);
287
288 colorBase.fR = colorBase.fR + minFpsColor.fR;
289 colorBase.fG = colorBase.fG + minFpsColor.fG;
290 colorBase.fB = colorBase.fB + minFpsColor.fB;
291 colorBase.fA = kAlpha;
292
293 const SkColor color = colorBase.toSkColor();
294
295 auto buffers = SevenSegmentDrawer::draw(intFps, color, transformHint, mShowSpinner);
296 it = mBufferCache.try_emplace({intFps, transformHint}, std::move(buffers)).first;
Ady Abraham2cb8b622019-12-02 18:55:33 -0800297 }
Marin Shalamanovf7f6b3c2020-12-09 13:19:38 +0100298
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700299 return it->second;
Ady Abraham2cb8b622019-12-02 18:55:33 -0800300}
301
Dominik Laskowski20134642020-04-20 22:36:44 -0700302void RefreshRateOverlay::setViewport(ui::Size viewport) {
Ady Abrahamcba8d6c2021-06-03 18:05:04 -0700303 constexpr int32_t kMaxWidth = 1000;
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700304 const auto width = std::min({kMaxWidth, viewport.width, viewport.height});
Ady Abrahamcba8d6c2021-06-03 18:05:04 -0700305 const auto height = 2 * width;
306 Rect frame((3 * width) >> 4, height >> 5);
307 frame.offsetBy(width >> 5, height >> 4);
Ady Abraham2cb8b622019-12-02 18:55:33 -0800308
Ady Abraham3649ea82022-07-08 16:04:02 -0700309 createTransaction(mSurfaceControl->get())
310 .setMatrix(mSurfaceControl->get(), frame.getWidth() / static_cast<float>(kBufferWidth),
311 0, 0, frame.getHeight() / static_cast<float>(kBufferHeight))
312 .setPosition(mSurfaceControl->get(), frame.left, frame.top)
Dominik Laskowski1f6fc702022-03-21 08:34:50 -0700313 .apply();
Ady Abraham03b02dd2019-03-21 15:40:11 -0700314}
315
Dominik Laskowski29fa1462021-04-27 15:51:50 -0700316void RefreshRateOverlay::setLayerStack(ui::LayerStack stack) {
Ady Abraham3649ea82022-07-08 16:04:02 -0700317 createTransaction(mSurfaceControl->get()).setLayerStack(mSurfaceControl->get(), stack).apply();
Ady Abraham1b11bc62021-06-03 19:51:19 -0700318}
319
Dominik Laskowskif6b4ba62021-11-09 12:46:10 -0800320void RefreshRateOverlay::changeRefreshRate(Fps fps) {
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700321 mCurrentFps = fps;
322 const auto buffer = getOrCreateBuffers(fps)[mFrame];
Ady Abraham3649ea82022-07-08 16:04:02 -0700323 createTransaction(mSurfaceControl->get()).setBuffer(mSurfaceControl->get(), buffer).apply();
Ady Abraham29d0da32020-07-16 18:39:33 -0700324}
325
Dominik Laskowskie0e0cde2021-07-30 10:42:05 -0700326void RefreshRateOverlay::animate() {
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700327 if (!mShowSpinner || !mCurrentFps) return;
Ady Abraham29d0da32020-07-16 18:39:33 -0700328
Marin Shalamanovf7f6b3c2020-12-09 13:19:38 +0100329 const auto& buffers = getOrCreateBuffers(*mCurrentFps);
Ady Abraham29d0da32020-07-16 18:39:33 -0700330 mFrame = (mFrame + 1) % buffers.size();
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700331 const auto buffer = buffers[mFrame];
Ady Abraham3649ea82022-07-08 16:04:02 -0700332 createTransaction(mSurfaceControl->get()).setBuffer(mSurfaceControl->get(), buffer).apply();
Dominik Laskowski20134642020-04-20 22:36:44 -0700333}
334
335} // namespace android