1 /*
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
17 #include "VsyncConfiguration.h"
18
19 #include <chrono>
20 #include <cinttypes>
21 #include <optional>
22
23 #include <cutils/properties.h>
24 #include <log/log.h>
25
26 #include "SurfaceFlingerProperties.h"
27
28 namespace {
29
30 using namespace std::chrono_literals;
31
getProperty(const char * name)32 std::optional<nsecs_t> getProperty(const char* name) {
33 char value[PROPERTY_VALUE_MAX];
34 property_get(name, value, "-1");
35 if (const int i = atoi(value); i != -1) return i;
36 return std::nullopt;
37 }
38
39 } // namespace
40
41 namespace android::scheduler::impl {
42
VsyncConfiguration(Fps currentFps)43 VsyncConfiguration::VsyncConfiguration(Fps currentFps) : mRefreshRateFps(currentFps) {}
44
getConfigsForRefreshRate(Fps fps) const45 VsyncConfigSet VsyncConfiguration::getConfigsForRefreshRate(Fps fps) const {
46 std::lock_guard lock(mLock);
47 return getConfigsForRefreshRateLocked(fps);
48 }
49
getConfigsForRefreshRateLocked(Fps fps) const50 VsyncConfigSet VsyncConfiguration::getConfigsForRefreshRateLocked(Fps fps) const {
51 if (const auto offsets = mOffsetsCache.get(fps)) {
52 return offsets->get();
53 }
54
55 const auto [it, _] = mOffsetsCache.try_emplace(fps, constructOffsets(fps.getPeriodNsecs()));
56 return it->second;
57 }
58
dump(std::string & result) const59 void VsyncConfiguration::dump(std::string& result) const {
60 const auto [early, earlyGpu, late, hwcMinWorkDuration] = getCurrentConfigs();
61 using base::StringAppendF;
62 StringAppendF(&result,
63 " app phase: %9" PRId64 " ns\t SF phase: %9" PRId64
64 " ns\n"
65 " app duration: %9lld ns\t SF duration: %9lld ns\n"
66 " early app phase: %9" PRId64 " ns\t early SF phase: %9" PRId64
67 " ns\n"
68 " early app duration: %9lld ns\t early SF duration: %9lld ns\n"
69 " GL early app phase: %9" PRId64 " ns\tGL early SF phase: %9" PRId64
70 " ns\n"
71 " GL early app duration: %9lld ns\tGL early SF duration: %9lld ns\n"
72 " HWC min duration: %9lld ns\n",
73 late.appOffset, late.sfOffset,
74
75 late.appWorkDuration.count(), late.sfWorkDuration.count(),
76
77 early.appOffset, early.sfOffset,
78
79 early.appWorkDuration.count(), early.sfWorkDuration.count(),
80
81 earlyGpu.appOffset, earlyGpu.sfOffset,
82
83 earlyGpu.appWorkDuration.count(), earlyGpu.sfWorkDuration.count(),
84
85 hwcMinWorkDuration.count());
86 }
87
PhaseOffsets(Fps currentRefreshRate)88 PhaseOffsets::PhaseOffsets(Fps currentRefreshRate)
89 : PhaseOffsets(currentRefreshRate, sysprop::vsync_event_phase_offset_ns(1000000),
90 sysprop::vsync_sf_event_phase_offset_ns(1000000),
91 getProperty("debug.sf.early_phase_offset_ns"),
92 getProperty("debug.sf.early_gl_phase_offset_ns"),
93 getProperty("debug.sf.early_app_phase_offset_ns"),
94 getProperty("debug.sf.early_gl_app_phase_offset_ns"),
95 getProperty("debug.sf.high_fps_late_app_phase_offset_ns").value_or(2000000),
96 getProperty("debug.sf.high_fps_late_sf_phase_offset_ns").value_or(1000000),
97 getProperty("debug.sf.high_fps_early_phase_offset_ns"),
98 getProperty("debug.sf.high_fps_early_gl_phase_offset_ns"),
99 getProperty("debug.sf.high_fps_early_app_phase_offset_ns"),
100 getProperty("debug.sf.high_fps_early_gl_app_phase_offset_ns"),
101 // Below defines the threshold when an offset is considered to be negative,
102 // i.e. targeting for the N+2 vsync instead of N+1. This means that: For offset
103 // < threshold, SF wake up (vsync_duration - offset) before HW vsync. For
104 // offset >= threshold, SF wake up (2 * vsync_duration - offset) before HW
105 // vsync.
106 getProperty("debug.sf.phase_offset_threshold_for_next_vsync_ns")
107 .value_or(std::numeric_limits<nsecs_t>::max()),
108 getProperty("debug.sf.hwc.min.duration").value_or(0)) {}
109
PhaseOffsets(Fps currentFps,nsecs_t vsyncPhaseOffsetNs,nsecs_t sfVSyncPhaseOffsetNs,std::optional<nsecs_t> earlySfOffsetNs,std::optional<nsecs_t> earlyGpuSfOffsetNs,std::optional<nsecs_t> earlyAppOffsetNs,std::optional<nsecs_t> earlyGpuAppOffsetNs,nsecs_t highFpsVsyncPhaseOffsetNs,nsecs_t highFpsSfVSyncPhaseOffsetNs,std::optional<nsecs_t> highFpsEarlySfOffsetNs,std::optional<nsecs_t> highFpsEarlyGpuSfOffsetNs,std::optional<nsecs_t> highFpsEarlyAppOffsetNs,std::optional<nsecs_t> highFpsEarlyGpuAppOffsetNs,nsecs_t thresholdForNextVsync,nsecs_t hwcMinWorkDuration)110 PhaseOffsets::PhaseOffsets(Fps currentFps, nsecs_t vsyncPhaseOffsetNs, nsecs_t sfVSyncPhaseOffsetNs,
111 std::optional<nsecs_t> earlySfOffsetNs,
112 std::optional<nsecs_t> earlyGpuSfOffsetNs,
113 std::optional<nsecs_t> earlyAppOffsetNs,
114 std::optional<nsecs_t> earlyGpuAppOffsetNs,
115 nsecs_t highFpsVsyncPhaseOffsetNs, nsecs_t highFpsSfVSyncPhaseOffsetNs,
116 std::optional<nsecs_t> highFpsEarlySfOffsetNs,
117 std::optional<nsecs_t> highFpsEarlyGpuSfOffsetNs,
118 std::optional<nsecs_t> highFpsEarlyAppOffsetNs,
119 std::optional<nsecs_t> highFpsEarlyGpuAppOffsetNs,
120 nsecs_t thresholdForNextVsync, nsecs_t hwcMinWorkDuration)
121 : VsyncConfiguration(currentFps),
122 mVSyncPhaseOffsetNs(vsyncPhaseOffsetNs),
123 mSfVSyncPhaseOffsetNs(sfVSyncPhaseOffsetNs),
124 mEarlySfOffsetNs(earlySfOffsetNs),
125 mEarlyGpuSfOffsetNs(earlyGpuSfOffsetNs),
126 mEarlyAppOffsetNs(earlyAppOffsetNs),
127 mEarlyGpuAppOffsetNs(earlyGpuAppOffsetNs),
128 mHighFpsVSyncPhaseOffsetNs(highFpsVsyncPhaseOffsetNs),
129 mHighFpsSfVSyncPhaseOffsetNs(highFpsSfVSyncPhaseOffsetNs),
130 mHighFpsEarlySfOffsetNs(highFpsEarlySfOffsetNs),
131 mHighFpsEarlyGpuSfOffsetNs(highFpsEarlyGpuSfOffsetNs),
132 mHighFpsEarlyAppOffsetNs(highFpsEarlyAppOffsetNs),
133 mHighFpsEarlyGpuAppOffsetNs(highFpsEarlyGpuAppOffsetNs),
134 mThresholdForNextVsync(thresholdForNextVsync),
135 mHwcMinWorkDuration(hwcMinWorkDuration) {}
136
constructOffsets(nsecs_t vsyncDuration) const137 VsyncConfigSet PhaseOffsets::constructOffsets(nsecs_t vsyncDuration) const {
138 if (vsyncDuration < std::chrono::nanoseconds(15ms).count()) {
139 return getHighFpsOffsets(vsyncDuration);
140 } else {
141 return getDefaultOffsets(vsyncDuration);
142 }
143 }
144
145 namespace {
sfOffsetToDuration(nsecs_t sfOffset,nsecs_t vsyncDuration)146 std::chrono::nanoseconds sfOffsetToDuration(nsecs_t sfOffset, nsecs_t vsyncDuration) {
147 return std::chrono::nanoseconds(vsyncDuration - sfOffset);
148 }
149
appOffsetToDuration(nsecs_t appOffset,nsecs_t sfOffset,nsecs_t vsyncDuration)150 std::chrono::nanoseconds appOffsetToDuration(nsecs_t appOffset, nsecs_t sfOffset,
151 nsecs_t vsyncDuration) {
152 auto duration = vsyncDuration + (sfOffset - appOffset);
153 if (duration < vsyncDuration) {
154 duration += vsyncDuration;
155 }
156
157 return std::chrono::nanoseconds(duration);
158 }
159 } // namespace
160
getDefaultOffsets(nsecs_t vsyncDuration) const161 VsyncConfigSet PhaseOffsets::getDefaultOffsets(nsecs_t vsyncDuration) const {
162 const auto earlySfOffset =
163 mEarlySfOffsetNs.value_or(mSfVSyncPhaseOffsetNs) < mThresholdForNextVsync
164
165 ? mEarlySfOffsetNs.value_or(mSfVSyncPhaseOffsetNs)
166 : mEarlySfOffsetNs.value_or(mSfVSyncPhaseOffsetNs) - vsyncDuration;
167 const auto earlyAppOffset = mEarlyAppOffsetNs.value_or(mVSyncPhaseOffsetNs);
168 const auto earlyGpuSfOffset =
169 mEarlyGpuSfOffsetNs.value_or(mSfVSyncPhaseOffsetNs) < mThresholdForNextVsync
170
171 ? mEarlyGpuSfOffsetNs.value_or(mSfVSyncPhaseOffsetNs)
172 : mEarlyGpuSfOffsetNs.value_or(mSfVSyncPhaseOffsetNs) - vsyncDuration;
173 const auto earlyGpuAppOffset = mEarlyGpuAppOffsetNs.value_or(mVSyncPhaseOffsetNs);
174 const auto lateSfOffset = mSfVSyncPhaseOffsetNs < mThresholdForNextVsync
175 ? mSfVSyncPhaseOffsetNs
176 : mSfVSyncPhaseOffsetNs - vsyncDuration;
177 const auto lateAppOffset = mVSyncPhaseOffsetNs;
178
179 return {
180 .early = {.sfOffset = earlySfOffset,
181 .appOffset = earlyAppOffset,
182 .sfWorkDuration = sfOffsetToDuration(earlySfOffset, vsyncDuration),
183 .appWorkDuration =
184 appOffsetToDuration(earlyAppOffset, earlySfOffset, vsyncDuration)},
185 .earlyGpu = {.sfOffset = earlyGpuSfOffset,
186 .appOffset = earlyGpuAppOffset,
187 .sfWorkDuration = sfOffsetToDuration(earlyGpuSfOffset, vsyncDuration),
188 .appWorkDuration = appOffsetToDuration(earlyGpuAppOffset, earlyGpuSfOffset,
189 vsyncDuration)},
190 .late = {.sfOffset = lateSfOffset,
191 .appOffset = lateAppOffset,
192 .sfWorkDuration = sfOffsetToDuration(lateSfOffset, vsyncDuration),
193 .appWorkDuration =
194 appOffsetToDuration(lateAppOffset, lateSfOffset, vsyncDuration)},
195 .hwcMinWorkDuration = std::chrono::nanoseconds(mHwcMinWorkDuration),
196 };
197 }
198
getHighFpsOffsets(nsecs_t vsyncDuration) const199 VsyncConfigSet PhaseOffsets::getHighFpsOffsets(nsecs_t vsyncDuration) const {
200 const auto earlySfOffset =
201 mHighFpsEarlySfOffsetNs.value_or(mHighFpsSfVSyncPhaseOffsetNs) < mThresholdForNextVsync
202 ? mHighFpsEarlySfOffsetNs.value_or(mHighFpsSfVSyncPhaseOffsetNs)
203 : mHighFpsEarlySfOffsetNs.value_or(mHighFpsSfVSyncPhaseOffsetNs) - vsyncDuration;
204 const auto earlyAppOffset = mHighFpsEarlyAppOffsetNs.value_or(mHighFpsVSyncPhaseOffsetNs);
205 const auto earlyGpuSfOffset = mHighFpsEarlyGpuSfOffsetNs.value_or(
206 mHighFpsSfVSyncPhaseOffsetNs) < mThresholdForNextVsync
207
208 ? mHighFpsEarlyGpuSfOffsetNs.value_or(mHighFpsSfVSyncPhaseOffsetNs)
209 : mHighFpsEarlyGpuSfOffsetNs.value_or(mHighFpsSfVSyncPhaseOffsetNs) - vsyncDuration;
210 const auto earlyGpuAppOffset = mHighFpsEarlyGpuAppOffsetNs.value_or(mHighFpsVSyncPhaseOffsetNs);
211 const auto lateSfOffset = mHighFpsSfVSyncPhaseOffsetNs < mThresholdForNextVsync
212 ? mHighFpsSfVSyncPhaseOffsetNs
213 : mHighFpsSfVSyncPhaseOffsetNs - vsyncDuration;
214 const auto lateAppOffset = mHighFpsVSyncPhaseOffsetNs;
215
216 return {
217 .early =
218 {
219 .sfOffset = earlySfOffset,
220 .appOffset = earlyAppOffset,
221 .sfWorkDuration = sfOffsetToDuration(earlySfOffset, vsyncDuration),
222 .appWorkDuration = appOffsetToDuration(earlyAppOffset, earlySfOffset,
223 vsyncDuration),
224 },
225 .earlyGpu =
226 {
227 .sfOffset = earlyGpuSfOffset,
228 .appOffset = earlyGpuAppOffset,
229 .sfWorkDuration = sfOffsetToDuration(earlyGpuSfOffset, vsyncDuration),
230 .appWorkDuration = appOffsetToDuration(earlyGpuAppOffset,
231 earlyGpuSfOffset, vsyncDuration),
232 },
233 .late =
234 {
235 .sfOffset = lateSfOffset,
236 .appOffset = lateAppOffset,
237 .sfWorkDuration = sfOffsetToDuration(lateSfOffset, vsyncDuration),
238 .appWorkDuration =
239 appOffsetToDuration(lateAppOffset, lateSfOffset, vsyncDuration),
240 },
241 .hwcMinWorkDuration = std::chrono::nanoseconds(mHwcMinWorkDuration),
242 };
243 }
244
validateSysprops()245 static void validateSysprops() {
246 const auto validatePropertyBool = [](const char* prop) {
247 LOG_ALWAYS_FATAL_IF(!property_get_bool(prop, false), "%s is false", prop);
248 };
249
250 validatePropertyBool("debug.sf.use_phase_offsets_as_durations");
251
252 LOG_ALWAYS_FATAL_IF(sysprop::vsync_event_phase_offset_ns(-1) != -1,
253 "ro.surface_flinger.vsync_event_phase_offset_ns is set but expecting "
254 "duration");
255
256 LOG_ALWAYS_FATAL_IF(sysprop::vsync_sf_event_phase_offset_ns(-1) != -1,
257 "ro.surface_flinger.vsync_sf_event_phase_offset_ns is set but expecting "
258 "duration");
259
260 const auto validateProperty = [](const char* prop) {
261 LOG_ALWAYS_FATAL_IF(getProperty(prop).has_value(),
262 "%s is set to %" PRId64 " but expecting duration", prop,
263 getProperty(prop).value_or(-1));
264 };
265
266 validateProperty("debug.sf.early_phase_offset_ns");
267 validateProperty("debug.sf.early_gl_phase_offset_ns");
268 validateProperty("debug.sf.early_app_phase_offset_ns");
269 validateProperty("debug.sf.early_gl_app_phase_offset_ns");
270 validateProperty("debug.sf.high_fps_late_app_phase_offset_ns");
271 validateProperty("debug.sf.high_fps_late_sf_phase_offset_ns");
272 validateProperty("debug.sf.high_fps_early_phase_offset_ns");
273 validateProperty("debug.sf.high_fps_early_gl_phase_offset_ns");
274 validateProperty("debug.sf.high_fps_early_app_phase_offset_ns");
275 validateProperty("debug.sf.high_fps_early_gl_app_phase_offset_ns");
276 }
277
278 namespace {
sfDurationToOffset(std::chrono::nanoseconds sfDuration,nsecs_t vsyncDuration)279 nsecs_t sfDurationToOffset(std::chrono::nanoseconds sfDuration, nsecs_t vsyncDuration) {
280 return vsyncDuration - sfDuration.count() % vsyncDuration;
281 }
282
appDurationToOffset(std::chrono::nanoseconds appDuration,std::chrono::nanoseconds sfDuration,nsecs_t vsyncDuration)283 nsecs_t appDurationToOffset(std::chrono::nanoseconds appDuration,
284 std::chrono::nanoseconds sfDuration, nsecs_t vsyncDuration) {
285 return vsyncDuration - (appDuration + sfDuration).count() % vsyncDuration;
286 }
287 } // namespace
288
constructOffsets(nsecs_t vsyncDuration) const289 VsyncConfigSet WorkDuration::constructOffsets(nsecs_t vsyncDuration) const {
290 const auto sfDurationFixup = [vsyncDuration](nsecs_t duration) {
291 return duration == -1 ? std::chrono::nanoseconds(vsyncDuration) - 1ms
292 : std::chrono::nanoseconds(duration);
293 };
294
295 const auto appDurationFixup = [vsyncDuration](nsecs_t duration) {
296 return duration == -1 ? std::chrono::nanoseconds(vsyncDuration)
297 : std::chrono::nanoseconds(duration);
298 };
299
300 const auto sfEarlyDuration = sfDurationFixup(mSfEarlyDuration);
301 const auto appEarlyDuration = appDurationFixup(mAppEarlyDuration);
302 const auto sfEarlyGpuDuration = sfDurationFixup(mSfEarlyGpuDuration);
303 const auto appEarlyGpuDuration = appDurationFixup(mAppEarlyGpuDuration);
304 const auto sfDuration = sfDurationFixup(mSfDuration);
305 const auto appDuration = appDurationFixup(mAppDuration);
306
307 return {
308 .early =
309 {
310
311 .sfOffset = sfEarlyDuration.count() < vsyncDuration
312 ? sfDurationToOffset(sfEarlyDuration, vsyncDuration)
313 : sfDurationToOffset(sfEarlyDuration, vsyncDuration) -
314 vsyncDuration,
315
316 .appOffset = appDurationToOffset(appEarlyDuration, sfEarlyDuration,
317 vsyncDuration),
318
319 .sfWorkDuration = sfEarlyDuration,
320 .appWorkDuration = appEarlyDuration,
321 },
322 .earlyGpu =
323 {
324
325 .sfOffset = sfEarlyGpuDuration.count() < vsyncDuration
326
327 ? sfDurationToOffset(sfEarlyGpuDuration, vsyncDuration)
328 : sfDurationToOffset(sfEarlyGpuDuration, vsyncDuration) -
329 vsyncDuration,
330
331 .appOffset = appDurationToOffset(appEarlyGpuDuration,
332 sfEarlyGpuDuration, vsyncDuration),
333 .sfWorkDuration = sfEarlyGpuDuration,
334 .appWorkDuration = appEarlyGpuDuration,
335 },
336 .late =
337 {
338
339 .sfOffset = sfDuration.count() < vsyncDuration
340
341 ? sfDurationToOffset(sfDuration, vsyncDuration)
342 : sfDurationToOffset(sfDuration, vsyncDuration) - vsyncDuration,
343
344 .appOffset =
345 appDurationToOffset(appDuration, sfDuration, vsyncDuration),
346
347 .sfWorkDuration = sfDuration,
348 .appWorkDuration = appDuration,
349 },
350 .hwcMinWorkDuration = std::chrono::nanoseconds(mHwcMinWorkDuration),
351 };
352 }
353
WorkDuration(Fps currentRefreshRate)354 WorkDuration::WorkDuration(Fps currentRefreshRate)
355 : WorkDuration(currentRefreshRate, getProperty("debug.sf.late.sf.duration").value_or(-1),
356 getProperty("debug.sf.late.app.duration").value_or(-1),
357 getProperty("debug.sf.early.sf.duration").value_or(mSfDuration),
358 getProperty("debug.sf.early.app.duration").value_or(mAppDuration),
359 getProperty("debug.sf.earlyGl.sf.duration").value_or(mSfDuration),
360 getProperty("debug.sf.earlyGl.app.duration").value_or(mAppDuration),
361 getProperty("debug.sf.hwc.min.duration").value_or(0)) {
362 validateSysprops();
363 }
364
WorkDuration(Fps currentRefreshRate,nsecs_t sfDuration,nsecs_t appDuration,nsecs_t sfEarlyDuration,nsecs_t appEarlyDuration,nsecs_t sfEarlyGpuDuration,nsecs_t appEarlyGpuDuration,nsecs_t hwcMinWorkDuration)365 WorkDuration::WorkDuration(Fps currentRefreshRate, nsecs_t sfDuration, nsecs_t appDuration,
366 nsecs_t sfEarlyDuration, nsecs_t appEarlyDuration,
367 nsecs_t sfEarlyGpuDuration, nsecs_t appEarlyGpuDuration,
368 nsecs_t hwcMinWorkDuration)
369 : VsyncConfiguration(currentRefreshRate),
370 mSfDuration(sfDuration),
371 mAppDuration(appDuration),
372 mSfEarlyDuration(sfEarlyDuration),
373 mAppEarlyDuration(appEarlyDuration),
374 mSfEarlyGpuDuration(sfEarlyGpuDuration),
375 mAppEarlyGpuDuration(appEarlyGpuDuration),
376 mHwcMinWorkDuration(hwcMinWorkDuration) {}
377
378 } // namespace android::scheduler::impl
379