blob: 80aa07231f5148be3c72bc6f6abfa63d3a2da943 [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 Abraham03b02dd2019-03-21 15:40:11 -070019#include "RefreshRateOverlay.h"
20#include "Client.h"
21#include "Layer.h"
22
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070023#pragma clang diagnostic push
24#pragma clang diagnostic ignored "-Wconversion"
25#include <SkCanvas.h>
rnlee756005b2021-05-27 10:46:36 -070026#include <SkPaint.h>
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070027#pragma clang diagnostic pop
28#include <SkBlendMode.h>
rnlee756005b2021-05-27 10:46:36 -070029#include <SkRect.h>
30#include <SkSurface.h>
chaviw70aa7572021-09-22 12:27:57 -050031#include <gui/SurfaceComposerClient.h>
32#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
44// Layout is digit, space, digit, space, digit, space, spinner.
45constexpr int kBufferWidth = 4 * kDigitWidth + 3 * kDigitSpace;
46constexpr int kBufferHeight = kDigitHeight;
47
48} // namespace
49
50void RefreshRateOverlay::SevenSegmentDrawer::drawSegment(Segment segment, int left, SkColor color,
rnlee756005b2021-05-27 10:46:36 -070051 SkCanvas& canvas) {
52 const SkRect rect = [&]() {
Ady Abraham2cb8b622019-12-02 18:55:33 -080053 switch (segment) {
54 case Segment::Upper:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070055 return SkRect::MakeLTRB(left, 0, left + kDigitWidth, kDigitSpace);
Ady Abraham2cb8b622019-12-02 18:55:33 -080056 case Segment::UpperLeft:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070057 return SkRect::MakeLTRB(left, 0, left + kDigitSpace, kDigitHeight / 2);
Ady Abraham2cb8b622019-12-02 18:55:33 -080058 case Segment::UpperRight:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070059 return SkRect::MakeLTRB(left + kDigitWidth - kDigitSpace, 0, left + kDigitWidth,
60 kDigitHeight / 2);
Ady Abraham2cb8b622019-12-02 18:55:33 -080061 case Segment::Middle:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070062 return SkRect::MakeLTRB(left, kDigitHeight / 2 - kDigitSpace / 2,
63 left + kDigitWidth, kDigitHeight / 2 + kDigitSpace / 2);
Ady Abraham2cb8b622019-12-02 18:55:33 -080064 case Segment::LowerLeft:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070065 return SkRect::MakeLTRB(left, kDigitHeight / 2, left + kDigitSpace, kDigitHeight);
Ady Abraham2cb8b622019-12-02 18:55:33 -080066 case Segment::LowerRight:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070067 return SkRect::MakeLTRB(left + kDigitWidth - kDigitSpace, kDigitHeight / 2,
68 left + kDigitWidth, kDigitHeight);
rnlee756005b2021-05-27 10:46:36 -070069 case Segment::Bottom:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070070 return SkRect::MakeLTRB(left, kDigitHeight - kDigitSpace, left + kDigitWidth,
71 kDigitHeight);
Ady Abraham2cb8b622019-12-02 18:55:33 -080072 }
73 }();
74
rnlee756005b2021-05-27 10:46:36 -070075 SkPaint paint;
76 paint.setColor(color);
77 paint.setBlendMode(SkBlendMode::kSrc);
78 canvas.drawRect(rect, paint);
Ady Abraham2cb8b622019-12-02 18:55:33 -080079}
80
Dominik Laskowski8c4356c2022-03-21 08:19:54 -070081void RefreshRateOverlay::SevenSegmentDrawer::drawDigit(int digit, int left, SkColor color,
rnlee756005b2021-05-27 10:46:36 -070082 SkCanvas& canvas) {
Ady Abraham2cb8b622019-12-02 18:55:33 -080083 if (digit < 0 || digit > 9) return;
84
85 if (digit == 0 || digit == 2 || digit == 3 || digit == 5 || digit == 6 || digit == 7 ||
86 digit == 8 || digit == 9)
rnlee756005b2021-05-27 10:46:36 -070087 drawSegment(Segment::Upper, left, color, canvas);
Ady Abraham2cb8b622019-12-02 18:55:33 -080088 if (digit == 0 || digit == 4 || digit == 5 || digit == 6 || digit == 8 || digit == 9)
rnlee756005b2021-05-27 10:46:36 -070089 drawSegment(Segment::UpperLeft, left, color, canvas);
Ady Abraham2cb8b622019-12-02 18:55:33 -080090 if (digit == 0 || digit == 1 || digit == 2 || digit == 3 || digit == 4 || digit == 7 ||
91 digit == 8 || digit == 9)
rnlee756005b2021-05-27 10:46:36 -070092 drawSegment(Segment::UpperRight, left, color, canvas);
Ady Abraham2cb8b622019-12-02 18:55:33 -080093 if (digit == 2 || digit == 3 || digit == 4 || digit == 5 || digit == 6 || digit == 8 ||
94 digit == 9)
rnlee756005b2021-05-27 10:46:36 -070095 drawSegment(Segment::Middle, left, color, canvas);
Ady Abraham2cb8b622019-12-02 18:55:33 -080096 if (digit == 0 || digit == 2 || digit == 6 || digit == 8)
rnlee756005b2021-05-27 10:46:36 -070097 drawSegment(Segment::LowerLeft, left, color, canvas);
Ady Abraham2cb8b622019-12-02 18:55:33 -080098 if (digit == 0 || digit == 1 || digit == 3 || digit == 4 || digit == 5 || digit == 6 ||
99 digit == 7 || digit == 8 || digit == 9)
rnlee756005b2021-05-27 10:46:36 -0700100 drawSegment(Segment::LowerRight, left, color, canvas);
Ady Abraham2cb8b622019-12-02 18:55:33 -0800101 if (digit == 0 || digit == 2 || digit == 3 || digit == 5 || digit == 6 || digit == 8 ||
102 digit == 9)
rnlee756005b2021-05-27 10:46:36 -0700103 drawSegment(Segment::Bottom, left, color, canvas);
Ady Abraham2cb8b622019-12-02 18:55:33 -0800104}
105
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700106auto RefreshRateOverlay::SevenSegmentDrawer::draw(int number, SkColor color,
107 ui::Transform::RotationFlags rotation,
108 bool showSpinner) -> Buffers {
Ady Abraham29d0da32020-07-16 18:39:33 -0700109 if (number < 0 || number > 1000) return {};
Ady Abraham2cb8b622019-12-02 18:55:33 -0800110
111 const auto hundreds = number / 100;
112 const auto tens = (number / 10) % 10;
113 const auto ones = number % 10;
114
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700115 const size_t loopCount = showSpinner ? 6 : 1;
116
117 Buffers buffers;
118 buffers.reserve(loopCount);
119
120 for (size_t i = 0; i < loopCount; i++) {
rnlee756005b2021-05-27 10:46:36 -0700121 // Pre-rotate the buffer before it reaches SurfaceFlinger.
122 SkMatrix canvasTransform = SkMatrix();
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700123 const auto [bufferWidth, bufferHeight] = [&]() -> std::pair<int, int> {
rnlee756005b2021-05-27 10:46:36 -0700124 switch (rotation) {
125 case ui::Transform::ROT_90:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700126 canvasTransform.setTranslate(kBufferHeight, 0);
127 canvasTransform.preRotate(90.f);
128 return {kBufferHeight, kBufferWidth};
rnlee756005b2021-05-27 10:46:36 -0700129 case ui::Transform::ROT_270:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700130 canvasTransform.setRotate(270.f, kBufferWidth / 2.f, kBufferWidth / 2.f);
131 return {kBufferHeight, kBufferWidth};
rnlee756005b2021-05-27 10:46:36 -0700132 default:
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700133 return {kBufferWidth, kBufferHeight};
rnlee756005b2021-05-27 10:46:36 -0700134 }
135 }();
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700136
Ady Abraham29d0da32020-07-16 18:39:33 -0700137 sp<GraphicBuffer> buffer =
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700138 new GraphicBuffer(static_cast<uint32_t>(bufferWidth),
139 static_cast<uint32_t>(bufferHeight), HAL_PIXEL_FORMAT_RGBA_8888,
140 1,
Ady Abraham29d0da32020-07-16 18:39:33 -0700141 GRALLOC_USAGE_SW_WRITE_RARELY | GRALLOC_USAGE_HW_COMPOSER |
142 GRALLOC_USAGE_HW_TEXTURE,
143 "RefreshRateOverlayBuffer");
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700144
Alec Mouri7013b6f2021-02-12 11:16:54 -0800145 const status_t bufferStatus = buffer->initCheck();
146 LOG_ALWAYS_FATAL_IF(bufferStatus != OK, "RefreshRateOverlay: Buffer failed to allocate: %d",
147 bufferStatus);
rnlee756005b2021-05-27 10:46:36 -0700148
149 sk_sp<SkSurface> surface = SkSurface::MakeRasterN32Premul(bufferWidth, bufferHeight);
150 SkCanvas* canvas = surface->getCanvas();
151 canvas->setMatrix(canvasTransform);
152
Ady Abraham29d0da32020-07-16 18:39:33 -0700153 int left = 0;
154 if (hundreds != 0) {
rnlee756005b2021-05-27 10:46:36 -0700155 drawDigit(hundreds, left, color, *canvas);
Ady Abraham29d0da32020-07-16 18:39:33 -0700156 }
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700157 left += kDigitWidth + kDigitSpace;
Ady Abraham2cb8b622019-12-02 18:55:33 -0800158
Ady Abraham29d0da32020-07-16 18:39:33 -0700159 if (tens != 0) {
rnlee756005b2021-05-27 10:46:36 -0700160 drawDigit(tens, left, color, *canvas);
Ady Abraham29d0da32020-07-16 18:39:33 -0700161 }
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700162 left += kDigitWidth + kDigitSpace;
Ady Abraham2cb8b622019-12-02 18:55:33 -0800163
rnlee756005b2021-05-27 10:46:36 -0700164 drawDigit(ones, left, color, *canvas);
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700165 left += kDigitWidth + kDigitSpace;
Ady Abraham29d0da32020-07-16 18:39:33 -0700166
167 if (showSpinner) {
168 switch (i) {
169 case 0:
rnlee756005b2021-05-27 10:46:36 -0700170 drawSegment(Segment::Upper, left, color, *canvas);
Ady Abraham29d0da32020-07-16 18:39:33 -0700171 break;
172 case 1:
rnlee756005b2021-05-27 10:46:36 -0700173 drawSegment(Segment::UpperRight, left, color, *canvas);
Ady Abraham29d0da32020-07-16 18:39:33 -0700174 break;
175 case 2:
rnlee756005b2021-05-27 10:46:36 -0700176 drawSegment(Segment::LowerRight, left, color, *canvas);
Ady Abraham29d0da32020-07-16 18:39:33 -0700177 break;
178 case 3:
rnlee756005b2021-05-27 10:46:36 -0700179 drawSegment(Segment::Bottom, left, color, *canvas);
Ady Abraham29d0da32020-07-16 18:39:33 -0700180 break;
181 case 4:
rnlee756005b2021-05-27 10:46:36 -0700182 drawSegment(Segment::LowerLeft, left, color, *canvas);
Ady Abraham29d0da32020-07-16 18:39:33 -0700183 break;
184 case 5:
rnlee756005b2021-05-27 10:46:36 -0700185 drawSegment(Segment::UpperLeft, left, color, *canvas);
Ady Abraham29d0da32020-07-16 18:39:33 -0700186 break;
187 }
188 }
189
rnlee756005b2021-05-27 10:46:36 -0700190 void* pixels = nullptr;
191 buffer->lock(GRALLOC_USAGE_SW_WRITE_RARELY, reinterpret_cast<void**>(&pixels));
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700192
rnlee756005b2021-05-27 10:46:36 -0700193 const SkImageInfo& imageInfo = surface->imageInfo();
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700194 const size_t dstRowBytes =
195 buffer->getStride() * static_cast<size_t>(imageInfo.bytesPerPixel());
196
rnlee756005b2021-05-27 10:46:36 -0700197 canvas->readPixels(imageInfo, pixels, dstRowBytes, 0, 0);
Ady Abraham29d0da32020-07-16 18:39:33 -0700198 buffer->unlock();
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700199 buffers.push_back(std::move(buffer));
Ady Abraham29d0da32020-07-16 18:39:33 -0700200 }
201 return buffers;
Ady Abraham2cb8b622019-12-02 18:55:33 -0800202}
203
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700204RefreshRateOverlay::RefreshRateOverlay(FpsRange fpsRange, bool showSpinner)
205 : mFpsRange(fpsRange),
Ady Abraham1b11bc62021-06-03 19:51:19 -0700206 mShowSpinner(showSpinner),
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700207 mSurfaceControl(SurfaceComposerClient::getDefault()
208 ->createSurface(String8("RefreshRateOverlay"), kBufferWidth,
209 kBufferHeight, PIXEL_FORMAT_RGBA_8888,
210 ISurfaceComposerClient::eFXSurfaceBufferState)) {
chaviw70aa7572021-09-22 12:27:57 -0500211 if (!mSurfaceControl) {
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700212 ALOGE("%s: Failed to create buffer state layer", __func__);
213 return;
Ady Abraham03b02dd2019-03-21 15:40:11 -0700214 }
215
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700216 constexpr float kFrameRate = 0.f;
217 constexpr int8_t kCompatibility = static_cast<int8_t>(Layer::FrameRateCompatibility::NoVote);
218 constexpr int8_t kSeamlessness = ANATIVEWINDOW_CHANGE_FRAME_RATE_ONLY_IF_SEAMLESS;
219
Ady Abraham78c4a242021-11-30 17:49:44 -0800220 SurfaceComposerClient::Transaction()
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700221 .setFrameRate(mSurfaceControl, kFrameRate, kCompatibility, kSeamlessness)
Ady Abraham78c4a242021-11-30 17:49:44 -0800222 .setLayer(mSurfaceControl, INT32_MAX - 2)
223 .setTrustedOverlay(mSurfaceControl, true)
224 .apply();
Ady Abraham03b02dd2019-03-21 15:40:11 -0700225}
226
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700227auto RefreshRateOverlay::getOrCreateBuffers(Fps fps) -> const Buffers& {
228 static const Buffers kNoBuffers;
229 if (!mSurfaceControl) return kNoBuffers;
230
231 const auto transformHint =
chaviw70aa7572021-09-22 12:27:57 -0500232 static_cast<ui::Transform::RotationFlags>(mSurfaceControl->getTransformHint());
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700233
rnlee756005b2021-05-27 10:46:36 -0700234 // Tell SurfaceFlinger about the pre-rotation on the buffer.
235 const auto transform = [&] {
236 switch (transformHint) {
237 case ui::Transform::ROT_90:
238 return ui::Transform::ROT_270;
239 case ui::Transform::ROT_270:
240 return ui::Transform::ROT_90;
241 default:
242 return ui::Transform::ROT_0;
243 }
244 }();
chaviw70aa7572021-09-22 12:27:57 -0500245
246 SurfaceComposerClient::Transaction t;
247 t.setTransform(mSurfaceControl, transform);
248 t.apply();
rnlee756005b2021-05-27 10:46:36 -0700249
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700250 BufferCache::const_iterator it = mBufferCache.find({fps.getIntValue(), transformHint});
251 if (it == mBufferCache.end()) {
252 const int minFps = mFpsRange.min.getIntValue();
253 const int maxFps = mFpsRange.max.getIntValue();
254
255 // Clamp to the range. The current fps may be outside of this range if the display has
256 // changed its set of supported refresh rates.
257 const int intFps = std::clamp(fps.getIntValue(), minFps, maxFps);
258
259 // Ensure non-zero range to avoid division by zero.
260 const float fpsScale = static_cast<float>(intFps - minFps) / std::max(1, maxFps - minFps);
261
262 constexpr SkColor kMinFpsColor = SK_ColorRED;
263 constexpr SkColor kMaxFpsColor = SK_ColorGREEN;
264 constexpr float kAlpha = 0.8f;
265
266 SkColor4f colorBase = SkColor4f::FromColor(kMaxFpsColor) * fpsScale;
267 const SkColor4f minFpsColor = SkColor4f::FromColor(kMinFpsColor) * (1 - fpsScale);
268
269 colorBase.fR = colorBase.fR + minFpsColor.fR;
270 colorBase.fG = colorBase.fG + minFpsColor.fG;
271 colorBase.fB = colorBase.fB + minFpsColor.fB;
272 colorBase.fA = kAlpha;
273
274 const SkColor color = colorBase.toSkColor();
275
276 auto buffers = SevenSegmentDrawer::draw(intFps, color, transformHint, mShowSpinner);
277 it = mBufferCache.try_emplace({intFps, transformHint}, std::move(buffers)).first;
Ady Abraham2cb8b622019-12-02 18:55:33 -0800278 }
Marin Shalamanovf7f6b3c2020-12-09 13:19:38 +0100279
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700280 return it->second;
Ady Abraham2cb8b622019-12-02 18:55:33 -0800281}
282
Dominik Laskowski20134642020-04-20 22:36:44 -0700283void RefreshRateOverlay::setViewport(ui::Size viewport) {
Ady Abrahamcba8d6c2021-06-03 18:05:04 -0700284 constexpr int32_t kMaxWidth = 1000;
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700285 const auto width = std::min({kMaxWidth, viewport.width, viewport.height});
Ady Abrahamcba8d6c2021-06-03 18:05:04 -0700286 const auto height = 2 * width;
287 Rect frame((3 * width) >> 4, height >> 5);
288 frame.offsetBy(width >> 5, height >> 4);
Ady Abraham2cb8b622019-12-02 18:55:33 -0800289
chaviw70aa7572021-09-22 12:27:57 -0500290 SurfaceComposerClient::Transaction t;
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700291 t.setMatrix(mSurfaceControl, frame.getWidth() / static_cast<float>(kBufferWidth), 0, 0,
292 frame.getHeight() / static_cast<float>(kBufferHeight));
chaviw70aa7572021-09-22 12:27:57 -0500293 t.setPosition(mSurfaceControl, frame.left, frame.top);
294 t.apply();
Ady Abraham03b02dd2019-03-21 15:40:11 -0700295}
296
Dominik Laskowski29fa1462021-04-27 15:51:50 -0700297void RefreshRateOverlay::setLayerStack(ui::LayerStack stack) {
chaviw70aa7572021-09-22 12:27:57 -0500298 SurfaceComposerClient::Transaction t;
299 t.setLayerStack(mSurfaceControl, stack);
300 t.apply();
Ady Abraham1b11bc62021-06-03 19:51:19 -0700301}
302
Dominik Laskowskif6b4ba62021-11-09 12:46:10 -0800303void RefreshRateOverlay::changeRefreshRate(Fps fps) {
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700304 mCurrentFps = fps;
305 const auto buffer = getOrCreateBuffers(fps)[mFrame];
chaviw70aa7572021-09-22 12:27:57 -0500306 SurfaceComposerClient::Transaction t;
307 t.setBuffer(mSurfaceControl, buffer);
308 t.apply();
Ady Abraham29d0da32020-07-16 18:39:33 -0700309}
310
Dominik Laskowskie0e0cde2021-07-30 10:42:05 -0700311void RefreshRateOverlay::animate() {
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700312 if (!mShowSpinner || !mCurrentFps) return;
Ady Abraham29d0da32020-07-16 18:39:33 -0700313
Marin Shalamanovf7f6b3c2020-12-09 13:19:38 +0100314 const auto& buffers = getOrCreateBuffers(*mCurrentFps);
Ady Abraham29d0da32020-07-16 18:39:33 -0700315 mFrame = (mFrame + 1) % buffers.size();
Dominik Laskowski8c4356c2022-03-21 08:19:54 -0700316 const auto buffer = buffers[mFrame];
chaviw70aa7572021-09-22 12:27:57 -0500317 SurfaceComposerClient::Transaction t;
318 t.setBuffer(mSurfaceControl, buffer);
319 t.apply();
Dominik Laskowski20134642020-04-20 22:36:44 -0700320}
321
322} // namespace android