blob: 607bec2e3f81176f34ca6e97a3decf90d946c7d4 [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/SurfaceControl.h>
Ady Abraham2cb8b622019-12-02 18:55:33 -080033
34#undef LOG_TAG
35#define LOG_TAG "RefreshRateOverlay"
36
Ady Abraham03b02dd2019-03-21 15:40:11 -070037namespace android {
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070038namespace {
Ady Abraham03b02dd2019-03-21 15:40:11 -070039
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070040constexpr int kDigitWidth = 64;
41constexpr int kDigitHeight = 100;
42constexpr int kDigitSpace = 16;
43
Ady Abraham0aa373a2022-11-22 13:56:50 -080044constexpr int kMaxDigits = /*displayFps*/ 3 + /*renderFps*/ 3 + /*spinner*/ 1;
45constexpr int kBufferWidth = kMaxDigits * kDigitWidth + (kMaxDigits - 1) * kDigitSpace;
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070046constexpr int kBufferHeight = kDigitHeight;
47
48} // namespace
49
Ady Abraham3649ea82022-07-08 16:04:02 -070050SurfaceControlHolder::~SurfaceControlHolder() {
51 // Hand the sp<SurfaceControl> to the helper thread to release the last
52 // reference. This makes sure that the SurfaceControl is destructed without
53 // SurfaceFlinger::mStateLock held.
54 BackgroundExecutor::getInstance().sendCallbacks(
55 {[sc = std::move(mSurfaceControl)]() mutable { sc.clear(); }});
56}
57
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070058void RefreshRateOverlay::SevenSegmentDrawer::drawSegment(Segment segment, int left, SkColor color,
rnlee756005b2021-05-27 10:46:36 -070059 SkCanvas& canvas) {
60 const SkRect rect = [&]() {
Ady Abraham2cb8b622019-12-02 18:55:33 -080061 switch (segment) {
62 case Segment::Upper:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070063 return SkRect::MakeLTRB(left, 0, left + kDigitWidth, kDigitSpace);
Ady Abraham2cb8b622019-12-02 18:55:33 -080064 case Segment::UpperLeft:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070065 return SkRect::MakeLTRB(left, 0, left + kDigitSpace, kDigitHeight / 2);
Ady Abraham2cb8b622019-12-02 18:55:33 -080066 case Segment::UpperRight:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070067 return SkRect::MakeLTRB(left + kDigitWidth - kDigitSpace, 0, left + kDigitWidth,
68 kDigitHeight / 2);
Ady Abraham2cb8b622019-12-02 18:55:33 -080069 case Segment::Middle:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070070 return SkRect::MakeLTRB(left, kDigitHeight / 2 - kDigitSpace / 2,
71 left + kDigitWidth, kDigitHeight / 2 + kDigitSpace / 2);
Ady Abraham2cb8b622019-12-02 18:55:33 -080072 case Segment::LowerLeft:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070073 return SkRect::MakeLTRB(left, kDigitHeight / 2, left + kDigitSpace, kDigitHeight);
Ady Abraham2cb8b622019-12-02 18:55:33 -080074 case Segment::LowerRight:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070075 return SkRect::MakeLTRB(left + kDigitWidth - kDigitSpace, kDigitHeight / 2,
76 left + kDigitWidth, kDigitHeight);
rnlee756005b2021-05-27 10:46:36 -070077 case Segment::Bottom:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070078 return SkRect::MakeLTRB(left, kDigitHeight - kDigitSpace, left + kDigitWidth,
79 kDigitHeight);
Ady Abraham2cb8b622019-12-02 18:55:33 -080080 }
81 }();
82
rnlee756005b2021-05-27 10:46:36 -070083 SkPaint paint;
84 paint.setColor(color);
85 paint.setBlendMode(SkBlendMode::kSrc);
86 canvas.drawRect(rect, paint);
Ady Abraham2cb8b622019-12-02 18:55:33 -080087}
88
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070089void RefreshRateOverlay::SevenSegmentDrawer::drawDigit(int digit, int left, SkColor color,
rnlee756005b2021-05-27 10:46:36 -070090 SkCanvas& canvas) {
Ady Abraham2cb8b622019-12-02 18:55:33 -080091 if (digit < 0 || digit > 9) return;
92
93 if (digit == 0 || digit == 2 || digit == 3 || digit == 5 || digit == 6 || digit == 7 ||
94 digit == 8 || digit == 9)
rnlee756005b2021-05-27 10:46:36 -070095 drawSegment(Segment::Upper, left, color, canvas);
Ady Abraham2cb8b622019-12-02 18:55:33 -080096 if (digit == 0 || digit == 4 || digit == 5 || digit == 6 || digit == 8 || digit == 9)
rnlee756005b2021-05-27 10:46:36 -070097 drawSegment(Segment::UpperLeft, left, color, canvas);
Ady Abraham2cb8b622019-12-02 18:55:33 -080098 if (digit == 0 || digit == 1 || digit == 2 || digit == 3 || digit == 4 || digit == 7 ||
99 digit == 8 || digit == 9)
rnlee756005b2021-05-27 10:46:36 -0700100 drawSegment(Segment::UpperRight, left, color, canvas);
Ady Abraham2cb8b622019-12-02 18:55:33 -0800101 if (digit == 2 || digit == 3 || digit == 4 || digit == 5 || digit == 6 || digit == 8 ||
102 digit == 9)
rnlee756005b2021-05-27 10:46:36 -0700103 drawSegment(Segment::Middle, left, color, canvas);
Ady Abraham2cb8b622019-12-02 18:55:33 -0800104 if (digit == 0 || digit == 2 || digit == 6 || digit == 8)
rnlee756005b2021-05-27 10:46:36 -0700105 drawSegment(Segment::LowerLeft, left, color, canvas);
Ady Abraham2cb8b622019-12-02 18:55:33 -0800106 if (digit == 0 || digit == 1 || digit == 3 || digit == 4 || digit == 5 || digit == 6 ||
107 digit == 7 || digit == 8 || digit == 9)
rnlee756005b2021-05-27 10:46:36 -0700108 drawSegment(Segment::LowerRight, left, color, canvas);
Ady Abraham2cb8b622019-12-02 18:55:33 -0800109 if (digit == 0 || digit == 2 || digit == 3 || digit == 5 || digit == 6 || digit == 8 ||
110 digit == 9)
rnlee756005b2021-05-27 10:46:36 -0700111 drawSegment(Segment::Bottom, left, color, canvas);
Ady Abraham2cb8b622019-12-02 18:55:33 -0800112}
113
Ady Abraham0aa373a2022-11-22 13:56:50 -0800114auto RefreshRateOverlay::SevenSegmentDrawer::draw(int displayFps, int renderFps, SkColor color,
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700115 ui::Transform::RotationFlags rotation,
Ady Abraham0aa373a2022-11-22 13:56:50 -0800116 ftl::Flags<Features> features) -> Buffers {
117 const size_t loopCount = features.test(Features::Spinner) ? 6 : 1;
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700118
119 Buffers buffers;
120 buffers.reserve(loopCount);
121
122 for (size_t i = 0; i < loopCount; i++) {
rnlee756005b2021-05-27 10:46:36 -0700123 // Pre-rotate the buffer before it reaches SurfaceFlinger.
124 SkMatrix canvasTransform = SkMatrix();
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700125 const auto [bufferWidth, bufferHeight] = [&]() -> std::pair<int, int> {
rnlee756005b2021-05-27 10:46:36 -0700126 switch (rotation) {
127 case ui::Transform::ROT_90:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700128 canvasTransform.setTranslate(kBufferHeight, 0);
129 canvasTransform.preRotate(90.f);
130 return {kBufferHeight, kBufferWidth};
rnlee756005b2021-05-27 10:46:36 -0700131 case ui::Transform::ROT_270:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700132 canvasTransform.setRotate(270.f, kBufferWidth / 2.f, kBufferWidth / 2.f);
133 return {kBufferHeight, kBufferWidth};
rnlee756005b2021-05-27 10:46:36 -0700134 default:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700135 return {kBufferWidth, kBufferHeight};
rnlee756005b2021-05-27 10:46:36 -0700136 }
137 }();
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700138
Ady Abrahamd11bade2022-08-01 16:18:03 -0700139 const auto kUsageFlags =
140 static_cast<uint64_t>(GRALLOC_USAGE_SW_WRITE_RARELY | GRALLOC_USAGE_HW_COMPOSER |
141 GRALLOC_USAGE_HW_TEXTURE);
142 sp<GraphicBuffer> buffer = sp<GraphicBuffer>::make(static_cast<uint32_t>(bufferWidth),
143 static_cast<uint32_t>(bufferHeight),
144 HAL_PIXEL_FORMAT_RGBA_8888, 1u,
145 kUsageFlags, "RefreshRateOverlayBuffer");
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700146
Alec Mouri7013b6f2021-02-12 11:16:54 -0800147 const status_t bufferStatus = buffer->initCheck();
148 LOG_ALWAYS_FATAL_IF(bufferStatus != OK, "RefreshRateOverlay: Buffer failed to allocate: %d",
149 bufferStatus);
rnlee756005b2021-05-27 10:46:36 -0700150
Kevin Lubick00bec722023-05-12 19:26:03 +0000151 sk_sp<SkSurface> surface = SkSurfaces::Raster(
152 SkImageInfo::MakeN32Premul(bufferWidth, bufferHeight));
rnlee756005b2021-05-27 10:46:36 -0700153 SkCanvas* canvas = surface->getCanvas();
154 canvas->setMatrix(canvasTransform);
155
Ady Abraham29d0da32020-07-16 18:39:33 -0700156 int left = 0;
Ady Abraham0aa373a2022-11-22 13:56:50 -0800157 drawNumber(displayFps, left, color, *canvas);
158 left += 3 * (kDigitWidth + kDigitSpace);
159 if (features.test(Features::Spinner)) {
Ady Abraham29d0da32020-07-16 18:39:33 -0700160 switch (i) {
161 case 0:
rnlee756005b2021-05-27 10:46:36 -0700162 drawSegment(Segment::Upper, left, color, *canvas);
Ady Abraham29d0da32020-07-16 18:39:33 -0700163 break;
164 case 1:
rnlee756005b2021-05-27 10:46:36 -0700165 drawSegment(Segment::UpperRight, left, color, *canvas);
Ady Abraham29d0da32020-07-16 18:39:33 -0700166 break;
167 case 2:
rnlee756005b2021-05-27 10:46:36 -0700168 drawSegment(Segment::LowerRight, left, color, *canvas);
Ady Abraham29d0da32020-07-16 18:39:33 -0700169 break;
170 case 3:
rnlee756005b2021-05-27 10:46:36 -0700171 drawSegment(Segment::Bottom, left, color, *canvas);
Ady Abraham29d0da32020-07-16 18:39:33 -0700172 break;
173 case 4:
rnlee756005b2021-05-27 10:46:36 -0700174 drawSegment(Segment::LowerLeft, left, color, *canvas);
Ady Abraham29d0da32020-07-16 18:39:33 -0700175 break;
176 case 5:
rnlee756005b2021-05-27 10:46:36 -0700177 drawSegment(Segment::UpperLeft, left, color, *canvas);
Ady Abraham29d0da32020-07-16 18:39:33 -0700178 break;
179 }
180 }
181
Ady Abraham0aa373a2022-11-22 13:56:50 -0800182 left += kDigitWidth + kDigitSpace;
183
184 if (features.test(Features::RenderRate)) {
185 drawNumber(renderFps, left, color, *canvas);
186 }
187 left += 3 * (kDigitWidth + kDigitSpace);
188
rnlee756005b2021-05-27 10:46:36 -0700189 void* pixels = nullptr;
190 buffer->lock(GRALLOC_USAGE_SW_WRITE_RARELY, reinterpret_cast<void**>(&pixels));
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700191
rnlee756005b2021-05-27 10:46:36 -0700192 const SkImageInfo& imageInfo = surface->imageInfo();
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700193 const size_t dstRowBytes =
194 buffer->getStride() * static_cast<size_t>(imageInfo.bytesPerPixel());
195
rnlee756005b2021-05-27 10:46:36 -0700196 canvas->readPixels(imageInfo, pixels, dstRowBytes, 0, 0);
Ady Abraham29d0da32020-07-16 18:39:33 -0700197 buffer->unlock();
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700198 buffers.push_back(std::move(buffer));
Ady Abraham29d0da32020-07-16 18:39:33 -0700199 }
200 return buffers;
Ady Abraham2cb8b622019-12-02 18:55:33 -0800201}
202
Ady Abraham0aa373a2022-11-22 13:56:50 -0800203void RefreshRateOverlay::SevenSegmentDrawer::drawNumber(int number, int left, SkColor color,
204 SkCanvas& canvas) {
205 if (number < 0 || number >= 1000) return;
206
207 if (number >= 100) {
208 drawDigit(number / 100, left, color, canvas);
209 }
210 left += kDigitWidth + kDigitSpace;
211
212 if (number >= 10) {
213 drawDigit((number / 10) % 10, left, color, canvas);
214 }
215 left += kDigitWidth + kDigitSpace;
216
217 drawDigit(number % 10, left, color, canvas);
218}
219
Ady Abraham3649ea82022-07-08 16:04:02 -0700220std::unique_ptr<SurfaceControlHolder> createSurfaceControlHolder() {
221 sp<SurfaceControl> surfaceControl =
222 SurfaceComposerClient::getDefault()
223 ->createSurface(String8("RefreshRateOverlay"), kBufferWidth, kBufferHeight,
224 PIXEL_FORMAT_RGBA_8888,
225 ISurfaceComposerClient::eFXSurfaceBufferState);
226 return std::make_unique<SurfaceControlHolder>(std::move(surfaceControl));
227}
228
Ady Abraham0aa373a2022-11-22 13:56:50 -0800229RefreshRateOverlay::RefreshRateOverlay(FpsRange fpsRange, ftl::Flags<Features> features)
230 : mFpsRange(fpsRange), mFeatures(features), mSurfaceControl(createSurfaceControlHolder()) {
chaviw70aa7572021-09-22 12:27:57 -0500231 if (!mSurfaceControl) {
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700232 ALOGE("%s: Failed to create buffer state layer", __func__);
233 return;
Ady Abraham03b02dd2019-03-21 15:40:11 -0700234 }
235
ramindanib2158ee2023-02-13 20:29:59 -0800236 createTransaction()
Ady Abraham3649ea82022-07-08 16:04:02 -0700237 .setLayer(mSurfaceControl->get(), INT32_MAX - 2)
238 .setTrustedOverlay(mSurfaceControl->get(), true)
Ady Abraham78c4a242021-11-30 17:49:44 -0800239 .apply();
Ady Abraham03b02dd2019-03-21 15:40:11 -0700240}
241
Ady Abraham0aa373a2022-11-22 13:56:50 -0800242auto RefreshRateOverlay::getOrCreateBuffers(Fps displayFps, Fps renderFps) -> const Buffers& {
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700243 static const Buffers kNoBuffers;
244 if (!mSurfaceControl) return kNoBuffers;
245
Ady Abraham0aa373a2022-11-22 13:56:50 -0800246 // avoid caching different render rates if RenderRate is anyway not visible
247 if (!mFeatures.test(Features::RenderRate)) {
248 renderFps = 0_Hz;
249 }
250
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700251 const auto transformHint =
Ady Abraham3649ea82022-07-08 16:04:02 -0700252 static_cast<ui::Transform::RotationFlags>(mSurfaceControl->get()->getTransformHint());
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700253
rnlee756005b2021-05-27 10:46:36 -0700254 // Tell SurfaceFlinger about the pre-rotation on the buffer.
255 const auto transform = [&] {
256 switch (transformHint) {
257 case ui::Transform::ROT_90:
258 return ui::Transform::ROT_270;
259 case ui::Transform::ROT_270:
260 return ui::Transform::ROT_90;
261 default:
262 return ui::Transform::ROT_0;
263 }
264 }();
chaviw70aa7572021-09-22 12:27:57 -0500265
ramindanib2158ee2023-02-13 20:29:59 -0800266 createTransaction().setTransform(mSurfaceControl->get(), transform).apply();
rnlee756005b2021-05-27 10:46:36 -0700267
Ady Abraham0aa373a2022-11-22 13:56:50 -0800268 BufferCache::const_iterator it =
269 mBufferCache.find({displayFps.getIntValue(), renderFps.getIntValue(), transformHint});
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700270 if (it == mBufferCache.end()) {
ramindanib2158ee2023-02-13 20:29:59 -0800271 // HWC minFps is not known by the framework in order
272 // to consider lower rates we set minFps to 0.
273 const int minFps = isSetByHwc() ? 0 : mFpsRange.min.getIntValue();
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700274 const int maxFps = mFpsRange.max.getIntValue();
275
Ady Abraham0aa373a2022-11-22 13:56:50 -0800276 // Clamp to the range. The current displayFps may be outside of this range if the display
277 // has changed its set of supported refresh rates.
278 const int displayIntFps = std::clamp(displayFps.getIntValue(), minFps, maxFps);
279 const int renderIntFps = renderFps.getIntValue();
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700280
281 // Ensure non-zero range to avoid division by zero.
Ady Abraham0aa373a2022-11-22 13:56:50 -0800282 const float fpsScale =
283 static_cast<float>(displayIntFps - minFps) / std::max(1, maxFps - minFps);
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700284
285 constexpr SkColor kMinFpsColor = SK_ColorRED;
286 constexpr SkColor kMaxFpsColor = SK_ColorGREEN;
287 constexpr float kAlpha = 0.8f;
288
289 SkColor4f colorBase = SkColor4f::FromColor(kMaxFpsColor) * fpsScale;
290 const SkColor4f minFpsColor = SkColor4f::FromColor(kMinFpsColor) * (1 - fpsScale);
291
292 colorBase.fR = colorBase.fR + minFpsColor.fR;
293 colorBase.fG = colorBase.fG + minFpsColor.fG;
294 colorBase.fB = colorBase.fB + minFpsColor.fB;
295 colorBase.fA = kAlpha;
296
297 const SkColor color = colorBase.toSkColor();
298
Ady Abraham0aa373a2022-11-22 13:56:50 -0800299 auto buffers = SevenSegmentDrawer::draw(displayIntFps, renderIntFps, color, transformHint,
300 mFeatures);
301 it = mBufferCache
302 .try_emplace({displayIntFps, renderIntFps, transformHint}, std::move(buffers))
303 .first;
Ady Abraham2cb8b622019-12-02 18:55:33 -0800304 }
Marin Shalamanovf7f6b3c2020-12-09 13:19:38 +0100305
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700306 return it->second;
Ady Abraham2cb8b622019-12-02 18:55:33 -0800307}
308
Dominik Laskowski20134642020-04-20 22:36:44 -0700309void RefreshRateOverlay::setViewport(ui::Size viewport) {
Ady Abrahamcba8d6c2021-06-03 18:05:04 -0700310 constexpr int32_t kMaxWidth = 1000;
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700311 const auto width = std::min({kMaxWidth, viewport.width, viewport.height});
Ady Abrahamcba8d6c2021-06-03 18:05:04 -0700312 const auto height = 2 * width;
Ady Abraham0aa373a2022-11-22 13:56:50 -0800313 Rect frame((5 * width) >> 4, height >> 5);
Yifei Zhangcfb7bb32023-01-12 16:17:14 -0800314
315 if (!mFeatures.test(Features::ShowInMiddle)) {
316 frame.offsetBy(width >> 5, height >> 4);
317 } else {
318 frame.offsetBy(width >> 1, height >> 4);
319 }
Ady Abraham2cb8b622019-12-02 18:55:33 -0800320
ramindanib2158ee2023-02-13 20:29:59 -0800321 createTransaction()
Ady Abraham3649ea82022-07-08 16:04:02 -0700322 .setMatrix(mSurfaceControl->get(), frame.getWidth() / static_cast<float>(kBufferWidth),
323 0, 0, frame.getHeight() / static_cast<float>(kBufferHeight))
324 .setPosition(mSurfaceControl->get(), frame.left, frame.top)
Dominik Laskowski1f6fc702022-03-21 08:34:50 -0700325 .apply();
Ady Abraham03b02dd2019-03-21 15:40:11 -0700326}
327
Dominik Laskowski29fa1462021-04-27 15:51:50 -0700328void RefreshRateOverlay::setLayerStack(ui::LayerStack stack) {
ramindanib2158ee2023-02-13 20:29:59 -0800329 createTransaction().setLayerStack(mSurfaceControl->get(), stack).apply();
Ady Abraham1b11bc62021-06-03 19:51:19 -0700330}
331
Ady Abraham0aa373a2022-11-22 13:56:50 -0800332void RefreshRateOverlay::changeRefreshRate(Fps displayFps, Fps renderFps) {
333 mDisplayFps = displayFps;
334 mRenderFps = renderFps;
335 const auto buffer = getOrCreateBuffers(displayFps, renderFps)[mFrame];
ramindanib2158ee2023-02-13 20:29:59 -0800336 createTransaction().setBuffer(mSurfaceControl->get(), buffer).apply();
Ady Abraham29d0da32020-07-16 18:39:33 -0700337}
338
Dominik Laskowskie0e0cde2021-07-30 10:42:05 -0700339void RefreshRateOverlay::animate() {
Ady Abraham0aa373a2022-11-22 13:56:50 -0800340 if (!mFeatures.test(Features::Spinner) || !mDisplayFps) return;
Ady Abraham29d0da32020-07-16 18:39:33 -0700341
Ady Abraham0aa373a2022-11-22 13:56:50 -0800342 const auto& buffers = getOrCreateBuffers(*mDisplayFps, *mRenderFps);
Ady Abraham29d0da32020-07-16 18:39:33 -0700343 mFrame = (mFrame + 1) % buffers.size();
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700344 const auto buffer = buffers[mFrame];
ramindanib2158ee2023-02-13 20:29:59 -0800345 createTransaction().setBuffer(mSurfaceControl->get(), buffer).apply();
346}
347
348SurfaceComposerClient::Transaction RefreshRateOverlay::createTransaction() const {
349 constexpr float kFrameRate = 0.f;
350 constexpr int8_t kCompatibility = ANATIVEWINDOW_FRAME_RATE_NO_VOTE;
351 constexpr int8_t kSeamlessness = ANATIVEWINDOW_CHANGE_FRAME_RATE_ONLY_IF_SEAMLESS;
352
353 const sp<SurfaceControl>& surface = mSurfaceControl->get();
354
355 SurfaceComposerClient::Transaction transaction;
356 if (isSetByHwc()) {
357 transaction.setFlags(surface, layer_state_t::eLayerIsRefreshRateIndicator,
358 layer_state_t::eLayerIsRefreshRateIndicator);
ramindania3816ab2023-04-25 18:41:16 -0700359 // Disable overlay layer caching when refresh rate is updated by the HWC.
360 transaction.setCachingHint(surface, gui::CachingHint::Disabled);
ramindanib2158ee2023-02-13 20:29:59 -0800361 }
362 transaction.setFrameRate(surface, kFrameRate, kCompatibility, kSeamlessness);
363 return transaction;
Dominik Laskowski20134642020-04-20 22:36:44 -0700364}
365
366} // namespace android