• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2012 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 //#define LOG_NDEBUG 0
17 
18 #define ATRACE_TAG (ATRACE_TAG_GRAPHICS | ATRACE_TAG_HAL)
19 
20 #include "ExynosPrimaryDisplay.h"
21 
22 #include <android-base/properties.h>
23 
24 #include <linux/fb.h>
25 #include <poll.h>
26 
27 #include <chrono>
28 #include <fstream>
29 
30 #include "../libvrr/FileNode.h"
31 #include "../libvrr/VariableRefreshRateVersion.h"
32 #include "../libvrr/interface/Panel_def.h"
33 #include "BrightnessController.h"
34 #include "DisplayTe2Manager.h"
35 #include "ExynosDevice.h"
36 #include "ExynosDisplayDrmInterface.h"
37 #include "ExynosDisplayDrmInterfaceModule.h"
38 #include "ExynosExternalDisplay.h"
39 #include "ExynosHWCDebug.h"
40 #include "ExynosHWCHelper.h"
41 #include "ExynosLayer.h"
42 #include "HistogramController.h"
43 
44 extern struct exynos_hwc_control exynosHWCControl;
45 
46 using namespace SOC_VERSION;
47 
48 namespace {
49 
50 using android::hardware::graphics::composer::kPanelRefreshCtrlFrameInsertionAutoMode;
51 using android::hardware::graphics::composer::kPanelRefreshCtrlFrameInsertionAutoModeOffset;
52 using android::hardware::graphics::composer::kPanelRefreshCtrlMrrV1OverV2;
53 using android::hardware::graphics::composer::kPanelRefreshCtrlMrrV1OverV2Offset;
54 using android::hardware::graphics::composer::kRefreshControlNodeEnabled;
55 using android::hardware::graphics::composer::kRefreshControlNodeName;
56 
57 constexpr auto nsecsPerSec = std::chrono::nanoseconds(1s).count();
58 
59 inline constexpr int kDefaultNotifyExpectedPresentConfigHeadsUpNs =
60         std::chrono::nanoseconds(30ms).count();
61 inline constexpr int kDefaultNotifyExpectedPresentConfigTimeoutNs =
62         std::chrono::nanoseconds(30ms).count();
63 
64 static constexpr int kMaximumPropertyIdentifierLength = 128;
65 
66 static const std::map<const DisplayType, const std::string> panelSysfsPath =
67         {{DisplayType::DISPLAY_PRIMARY, "/sys/devices/platform/exynos-drm/primary-panel/"},
68 #ifdef USES_IDISPLAY_INTF_SEC
69          {DisplayType::DISPLAY_SECONDARY, "/sys/devices/platform/exynos-drm/secondary-panel/"}
70 #endif
71 
72 };
73 } // namespace
74 
getPropertyBootModeStr(const int32_t dispId)75 static String8 getPropertyBootModeStr(const int32_t dispId) {
76     String8 str;
77     if (dispId == 0) {
78         str.appendFormat("persist.vendor.display.primary.boot_config");
79     } else {
80         str.appendFormat("persist.vendor.display.%d.primary.boot_config", dispId);
81     }
82     return str;
83 }
84 
loadPanelGammaCalibration(const std::string & file)85 static std::string loadPanelGammaCalibration(const std::string &file) {
86     std::ifstream ifs(file);
87 
88     if (!ifs.is_open()) {
89         ALOGW("Unable to open gamma calibration '%s', error = %s", file.c_str(), strerror(errno));
90         return {};
91     }
92 
93     std::string raw_data, gamma;
94     char ch;
95     while (std::getline(ifs, raw_data, '\r')) {
96         gamma.append(raw_data);
97         gamma.append(1, ' ');
98         ifs.get(ch);
99         if (ch != '\n') {
100             gamma.append(1, ch);
101         }
102     }
103     ifs.close();
104 
105     /* eliminate space character in the last byte */
106     if (!gamma.empty()) {
107         gamma.pop_back();
108     }
109 
110     return gamma;
111 }
112 
ExynosPrimaryDisplay(uint32_t index,ExynosDevice * device,const std::string & displayName)113 ExynosPrimaryDisplay::ExynosPrimaryDisplay(uint32_t index, ExynosDevice* device,
114                                            const std::string& displayName)
115       : ExynosDisplay(HWC_DISPLAY_PRIMARY, index, device, displayName),
116         mMinIdleRefreshRate(0),
117         mRrThrottleFps{0},
118         mRrThrottleNanos{0},
119         mRefreshRateDelayNanos(0),
120         mLastRefreshRateAppliedNanos(0),
121         mAppliedActiveConfig(0),
122         mDisplayIdleTimerEnabled(false),
123         mDisplayIdleTimerNanos{0},
124         mDisplayIdleDelayNanos(-1),
125         mDisplayNeedHandleIdleExit(false) {
126     // TODO : Hard coded here
127     mNumMaxPriorityAllowed = 5;
128 
129     /* Initialization */
130     mFramesToReachLhbmPeakBrightness =
131             property_get_int32("vendor.primarydisplay.lhbm.frames_to_reach_peak_brightness", 3);
132 
133     DisplayType displayType = getDcDisplayType();
134     std::string displayTypeIdentifier;
135     if (displayType == DisplayType::DISPLAY_PRIMARY) {
136         displayTypeIdentifier = "primarydisplay";
137     } else if (displayType == DisplayType::DISPLAY_EXTERNAL) {
138         displayTypeIdentifier = "externaldisplay";
139     }
140 #ifdef USES_IDISPLAY_INTF_SEC
141     else if (displayType == DisplayType::DISPLAY_SECONDARY) {
142         displayTypeIdentifier = "secondarydisplay";
143     }
144 #endif
145     if (!displayTypeIdentifier.empty()) {
146         // Blocking zone
147         auto propertyName =
148                 "ro.vendor." + displayTypeIdentifier + ".blocking_zone.min_refresh_rate_by_nits";
149         auto minRefreshRateByBrightnessString = android::base::GetProperty(propertyName, "");
150         ALOGD("%s brightness blocking zone propterty = %s", __func__,
151               minRefreshRateByBrightnessString.c_str());
152         if (!minRefreshRateByBrightnessString.empty()) {
153             std::vector<std::string> patterns;
154             size_t pos = 0;
155             while ((pos = minRefreshRateByBrightnessString.find(',')) != std::string::npos) {
156                 patterns.emplace_back(minRefreshRateByBrightnessString.substr(0, pos));
157                 minRefreshRateByBrightnessString.erase(0, pos + 1);
158             }
159             patterns.emplace_back(minRefreshRateByBrightnessString);
160             std::string brightnessString, fpsString;
161             for (auto& pattern : patterns) {
162                 int brightness, fps;
163                 char* endPos;
164                 pos = pattern.find(':');
165                 if (pos == std::string::npos) {
166                     ALOGE("%s(): cannot find the delimiter ':' of the pattern {brightness}:{fps} "
167                           "in "
168                           "pattern = %s",
169                           __func__, pattern.c_str());
170                     break;
171                 } else {
172                     brightnessString = pattern.substr(0, pos);
173                     pattern.erase(0, pos + 1);
174                     if (pattern.empty()) {
175                         ALOGE("%s(): miss the {fps} of the pattern = %s", __func__,
176                               pattern.c_str());
177                         break;
178                     } else {
179                         fpsString = pattern;
180                     }
181                     brightness = brightnessString.empty()
182                             ? INT_MAX
183                             : std::strtol(brightnessString.c_str(), &endPos, 10);
184                     fps = std::strtol(fpsString.c_str(), &endPos, 10);
185                 }
186                 mBrightnessBlockingZonesLookupTable[brightness] = fps;
187             }
188             ALOGI("Brightness blocking zone lookup table:");
189             int upperBound;
190             int lowerBound = INT_MIN;
191             for (const auto& brightnessBlockingZone : mBrightnessBlockingZonesLookupTable) {
192                 upperBound = brightnessBlockingZone.first;
193                 ALOGI("Brightness blocking zone: range [%s %s) fps = %d",
194                       (lowerBound == INT_MIN ? "Min" : std::to_string(lowerBound).c_str()),
195                       (upperBound == INT_MAX ? "Max" : std::to_string(upperBound).c_str()),
196                       brightnessBlockingZone.second);
197                 lowerBound = upperBound;
198             }
199         }
200         // XRR version
201         auto xrrVersion =
202                 android::hardware::graphics::composer::getDisplayXrrVersion(displayTypeIdentifier);
203         mXrrSettings.versionInfo.majorVersion = xrrVersion.first;
204         mXrrSettings.versionInfo.minorVersion = xrrVersion.second;
205         ALOGI("%s() xRR version = %d.%d", __func__, mXrrSettings.versionInfo.majorVersion,
206               mXrrSettings.versionInfo.minorVersion);
207 
208         uint32_t refreshControlCommand = 0;
209         if (mXrrSettings.versionInfo.needVrrParameters()) {
210             char pathBuffer[PROP_VALUE_MAX] = {0};
211             sprintf(pathBuffer, "ro.vendor.%s.vrr.expected_present.headsup_ns",
212                     displayTypeIdentifier.c_str());
213             mXrrSettings.notifyExpectedPresentConfig.HeadsUpNs =
214                     property_get_int32(pathBuffer, kDefaultNotifyExpectedPresentConfigHeadsUpNs);
215             sprintf(pathBuffer, "ro.vendor.%s.vrr.expected_present.timeout_ns",
216                     displayTypeIdentifier.c_str());
217             mXrrSettings.notifyExpectedPresentConfig.TimeoutNs =
218                     property_get_int32(pathBuffer, kDefaultNotifyExpectedPresentConfigTimeoutNs);
219             mXrrSettings.configChangeCallback =
220                     std::bind(&ExynosPrimaryDisplay::onConfigChange, this, std::placeholders::_1);
221         } else {
222             setBit(refreshControlCommand, kPanelRefreshCtrlMrrV1OverV2Offset);
223             setBit(refreshControlCommand, kPanelRefreshCtrlFrameInsertionAutoModeOffset);
224         }
225 
226         std::string displayFileNodePath = getPanelSysfsPath();
227         if (displayFileNodePath.empty()) {
228             ALOGE("%s(): cannot find file node %s of display %s", __func__,
229                   displayFileNodePath.c_str(), mDisplayName.c_str());
230         } else {
231             auto& fileNodeManager =
232                     android::hardware::graphics::composer::FileNodeManager::getInstance();
233             auto fileNode = fileNodeManager.getFileNode(displayFileNodePath);
234             auto content = fileNode->readString(kRefreshControlNodeName);
235             if (content.has_value() &&
236                 !(content.value().compare(0, kRefreshControlNodeEnabled.length(),
237                                           kRefreshControlNodeEnabled))) {
238                 bool ret = fileNode->WriteUint32(kRefreshControlNodeName, refreshControlCommand);
239                 if (!ret) {
240                     ALOGE("%s(): write command to file node %s%s failed.", __func__,
241                           displayFileNodePath.c_str(), kRefreshControlNodeName.c_str());
242                 }
243             } else {
244                 ALOGI("%s(): refresh control is not supported", __func__);
245             }
246         }
247     }
248 
249     // Allow to enable dynamic recomposition after every power on
250     // since it will always be disabled for every power off
251     // TODO(b/268474771): to enable DR by default if video mode panel is detected
252     if (property_get_int32("vendor.display.dynamic_recomposition", 0) & (1 << index)) {
253         mDRDefault = true;
254         mDREnable = true;
255     }
256 
257     // Prepare multi resolution
258     // Will be exynosHWCControl.multiResoultion
259     mResolutionInfo.nNum = 1;
260     mResolutionInfo.nResolution[0].w = 1440;
261     mResolutionInfo.nResolution[0].h = 2960;
262     mResolutionInfo.nDSCYSliceSize[0] = 40;
263     mResolutionInfo.nDSCXSliceSize[0] = 1440 / 2;
264     mResolutionInfo.nPanelType[0] = PANEL_DSC;
265     mResolutionInfo.nResolution[1].w = 1080;
266     mResolutionInfo.nResolution[1].h = 2220;
267     mResolutionInfo.nDSCYSliceSize[1] = 30;
268     mResolutionInfo.nDSCXSliceSize[1] = 1080 / 2;
269     mResolutionInfo.nPanelType[1] = PANEL_DSC;
270     mResolutionInfo.nResolution[2].w = 720;
271     mResolutionInfo.nResolution[2].h = 1480;
272     mResolutionInfo.nDSCYSliceSize[2] = 74;
273     mResolutionInfo.nDSCXSliceSize[2] = 720;
274     mResolutionInfo.nPanelType[2] = PANEL_LEGACY;
275 
276     char value[PROPERTY_VALUE_MAX];
277     const char *earlyWakeupNodeBase = early_wakeup_node_0_base;
278     if (getDcDisplayType() == DisplayType::DISPLAY_SECONDARY &&
279         property_get("vendor.display.secondary_early_wakeup_node", value, "") > 0) {
280         earlyWakeupNodeBase = value;
281     }
282     mEarlyWakeupDispFd = fopen(earlyWakeupNodeBase, "w");
283     if (mEarlyWakeupDispFd == nullptr)
284         ALOGE("open %s failed! %s", earlyWakeupNodeBase, strerror(errno));
285     mBrightnessController = std::make_unique<BrightnessController>(
__anonfd9a683f0202() 286             mIndex, [this]() { mDevice->onRefresh(mDisplayId); },
__anonfd9a683f0302() 287             [this]() { updatePresentColorConversionInfo(); });
288     mHistogramController = std::make_unique<HistogramController>(this);
289 
290     mDisplayControl.multiThreadedPresent = true;
291 
292     int fixedTe2DefaultRateHz =
293             property_get_int32("vendor.primarydisplay.fixed_te2.default_rate_hz", 0);
294     if (fixedTe2DefaultRateHz) {
295         mDisplayTe2Manager =
296                 std::make_unique<DisplayTe2Manager>(this, mIndex, fixedTe2DefaultRateHz);
297     }
298 }
299 
~ExynosPrimaryDisplay()300 ExynosPrimaryDisplay::~ExynosPrimaryDisplay()
301 {
302     if (mEarlyWakeupDispFd) {
303         fclose(mEarlyWakeupDispFd);
304         mEarlyWakeupDispFd = nullptr;
305     }
306 
307     if (mDisplayNeedHandleIdleExitOfs.is_open()) {
308         mDisplayNeedHandleIdleExitOfs.close();
309     }
310 }
311 
setDDIScalerEnable(int width,int height)312 void ExynosPrimaryDisplay::setDDIScalerEnable(int width, int height) {
313 
314     if (exynosHWCControl.setDDIScaler == false) return;
315 
316     ALOGI("DDISCALER Info : setDDIScalerEnable(w=%d,h=%d)", width, height);
317     mNewScaledWidth = width;
318     mNewScaledHeight = height;
319     mXres = width;
320     mYres = height;
321 }
322 
getDDIScalerMode(int width,int height)323 int ExynosPrimaryDisplay::getDDIScalerMode(int width, int height) {
324 
325     if (exynosHWCControl.setDDIScaler == false) return 1;
326 
327     // Check if panel support support resolution or not.
328     for (uint32_t i=0; i < mResolutionInfo.nNum; i++) {
329         if (mResolutionInfo.nResolution[i].w * mResolutionInfo.nResolution[i].h ==
330                 static_cast<uint32_t>(width * height))
331             return i + 1;
332     }
333 
334     return 1; // WQHD
335 }
336 
doDisplayConfigInternal(hwc2_config_t config)337 int32_t ExynosPrimaryDisplay::doDisplayConfigInternal(hwc2_config_t config) {
338     if (!mPowerModeState.has_value() || (*mPowerModeState != HWC2_POWER_MODE_ON) ||
339         !isConfigSettingEnabled()) {
340         mPendingConfig = config;
341         mConfigRequestState = hwc_request_state_t::SET_CONFIG_STATE_DONE;
342         DISPLAY_LOGI("%s:: Pending desired Config: %d", __func__, config);
343         return NO_ERROR;
344     }
345     return ExynosDisplay::doDisplayConfigInternal(config);
346 }
347 
getActiveConfigInternal(hwc2_config_t * outConfig)348 int32_t ExynosPrimaryDisplay::getActiveConfigInternal(hwc2_config_t *outConfig) {
349     if (outConfig && mPendingConfig != UINT_MAX) {
350         *outConfig = mPendingConfig;
351         return HWC2_ERROR_NONE;
352     }
353     return ExynosDisplay::getActiveConfigInternal(outConfig);
354 }
355 
setActiveConfigInternal(hwc2_config_t config,bool force)356 int32_t ExynosPrimaryDisplay::setActiveConfigInternal(hwc2_config_t config, bool force) {
357     hwc2_config_t cur_config;
358 
359     getActiveConfigInternal(&cur_config);
360     if (cur_config == config) {
361         ALOGI("%s:: Same display config is set", __func__);
362         return HWC2_ERROR_NONE;
363     }
364     if (!mPowerModeState.has_value() || (*mPowerModeState != HWC2_POWER_MODE_ON) ||
365         !isConfigSettingEnabled()) {
366         mPendingConfig = config;
367         return HWC2_ERROR_NONE;
368     }
369     return ExynosDisplay::setActiveConfigInternal(config, force);
370 }
371 
372 // If a display is Off, applyPendingConfig() calls to setActiveConfig() that also
373 // power on the display by a blocking commit.
applyPendingConfig()374 int32_t ExynosPrimaryDisplay::applyPendingConfig() {
375     if (!isConfigSettingEnabled()) {
376         ALOGI("%s:: config setting is disabled", __func__);
377         if (mPowerModeState.has_value() && (*mPowerModeState == HWC2_POWER_MODE_ON)) {
378             ALOGI("%s:: skip apply pending config", __func__);
379             return HWC2_ERROR_NONE;
380         }
381     }
382 
383     hwc2_config_t config;
384     if (mPendingConfig != UINT_MAX) {
385         ALOGI("%s:: mPendingConfig: %d", __func__, mPendingConfig);
386         config = mPendingConfig;
387         mPendingConfig = UINT_MAX;
388     } else {
389         getActiveConfigInternal(&config);
390     }
391 
392     return ExynosDisplay::setActiveConfigInternal(config, true);
393 }
394 
setBootDisplayConfig(int32_t config)395 int32_t ExynosPrimaryDisplay::setBootDisplayConfig(int32_t config) {
396     auto hwcConfig = static_cast<hwc2_config_t>(config);
397 
398     const auto &it = mDisplayConfigs.find(hwcConfig);
399     if (it == mDisplayConfigs.end()) {
400         DISPLAY_LOGE("%s: invalid config %d", __func__, config);
401         return HWC2_ERROR_BAD_CONFIG;
402     }
403 
404     const auto &mode = it->second;
405     if (mode.vsyncPeriod == 0)
406         return HWC2_ERROR_BAD_CONFIG;
407 
408     char modeStr[PROPERTY_VALUE_MAX];
409     int ret = snprintf(modeStr, sizeof(modeStr), "%dx%d@%d:%d", mode.width, mode.height,
410                        mode.refreshRate, nanoSec2Hz(mode.vsyncPeriod));
411     if (ret <= 0)
412         return HWC2_ERROR_BAD_CONFIG;
413 
414     ALOGD("%s: mode=%s (%d)", __func__, modeStr, config);
415     ret = property_set(getPropertyBootModeStr(mDisplayId).c_str(), modeStr);
416 
417     return !ret ? HWC2_ERROR_NONE : HWC2_ERROR_BAD_CONFIG;
418 }
419 
clearBootDisplayConfig()420 int32_t ExynosPrimaryDisplay::clearBootDisplayConfig() {
421     auto ret = property_set(getPropertyBootModeStr(mDisplayId).c_str(), nullptr);
422 
423     ALOGD("%s: clearing boot mode", __func__);
424     return !ret ? HWC2_ERROR_NONE : HWC2_ERROR_BAD_CONFIG;
425 }
426 
getPreferredDisplayConfigInternal(int32_t * outConfig)427 int32_t ExynosPrimaryDisplay::getPreferredDisplayConfigInternal(int32_t *outConfig) {
428     char modeStr[PROPERTY_VALUE_MAX];
429     auto ret = property_get(getPropertyBootModeStr(mDisplayId).c_str(), modeStr, "");
430 
431     if (ret <= 0) {
432         return mDisplayInterface->getDefaultModeId(outConfig);
433     }
434 
435     int width, height;
436     int fps = 0, vsyncRate = 0;
437 
438     ret = sscanf(modeStr, "%dx%d@%d:%d", &width, &height, &fps, &vsyncRate);
439     if (ret < 4) {
440         ret = sscanf(modeStr, "%dx%d@%d", &width, &height, &fps);
441         if ((ret < 3) || !fps) {
442             ALOGW("%s: unable to find boot config for mode: %s", __func__, modeStr);
443             return HWC2_ERROR_BAD_CONFIG;
444         }
445         if (lookupDisplayConfigs(width, height, fps, fps, outConfig) != HWC2_ERROR_NONE) {
446             ALOGE("%s: kernel doesn't support mode: %s", __func__, modeStr);
447             return HWC2_ERROR_BAD_CONFIG;
448         }
449         ret = setBootDisplayConfig(*outConfig);
450         if (ret == HWC2_ERROR_NONE)
451             ALOGI("%s: succeeded to replace %s with new format", __func__, modeStr);
452         else
453             ALOGE("%s: failed to replace %s with new format", __func__, modeStr);
454         return ret;
455     }
456     if (!fps || !vsyncRate || (fps > vsyncRate)) {
457         ALOGE("%s: bad boot config: %s", __func__, modeStr);
458         return HWC2_ERROR_BAD_CONFIG;
459     }
460     return lookupDisplayConfigs(width, height, fps, vsyncRate, outConfig);
461 }
462 
setPowerOn()463 int32_t ExynosPrimaryDisplay::setPowerOn() {
464     ATRACE_CALL();
465     updateAppliedActiveConfig(0, 0);
466     int ret = NO_ERROR;
467     if (mDisplayId != 0 || !mFirstPowerOn) {
468         if (mDevice->hasOtherDisplayOn(this)) {
469             mResourceManager->prepareResources(mDisplayId);
470             // TODO: This is useful for cmd mode, and b/282094671 tries to handles video mode
471             mDisplayInterface->triggerClearDisplayPlanes();
472         }
473         ret = applyPendingConfig();
474     }
475 
476     if (!mPowerModeState.has_value() || (*mPowerModeState == HWC2_POWER_MODE_OFF)) {
477         // check the dynamic recomposition thread by following display
478         mDevice->checkDynamicRecompositionThread();
479         if (ret) {
480             mDisplayInterface->setPowerMode(HWC2_POWER_MODE_ON);
481         }
482         setGeometryChanged(GEOMETRY_DISPLAY_POWER_ON);
483     }
484 
485     {
486         std::lock_guard<std::mutex> lock(mPowerModeMutex);
487         mPowerModeState = HWC2_POWER_MODE_ON;
488         if (mNotifyPowerOn) {
489             mPowerOnCondition.notify_one();
490             mNotifyPowerOn = false;
491         }
492     }
493 
494     if (mDisplayTe2Manager) mDisplayTe2Manager->restoreTe2FromDozeMode();
495 
496     if (mFirstPowerOn) {
497         firstPowerOn();
498     }
499 
500     return HWC2_ERROR_NONE;
501 }
502 
setPowerOff()503 int32_t ExynosPrimaryDisplay::setPowerOff() {
504     ATRACE_CALL();
505 
506     clearDisplay(true);
507 
508     // check the dynamic recomposition thread by following display
509     mDevice->checkDynamicRecompositionThread();
510 
511     mDisplayInterface->setPowerMode(HWC2_POWER_MODE_OFF);
512 
513     {
514         std::lock_guard<std::mutex> lock(mPowerModeMutex);
515         mPowerModeState = HWC2_POWER_MODE_OFF;
516     }
517 
518     /* It should be called from validate() when the screen is on */
519     mSkipFrame = true;
520     setGeometryChanged(GEOMETRY_DISPLAY_POWER_OFF);
521     if ((mRenderingState >= RENDERING_STATE_VALIDATED) &&
522         (mRenderingState < RENDERING_STATE_PRESENTED))
523         closeFencesForSkipFrame(RENDERING_STATE_VALIDATED);
524     mRenderingState = RENDERING_STATE_NONE;
525 
526     // in the case user turns off screen when LHBM is on
527     // TODO: b/236433238 considering a lock for mLhbmOn state
528     mLhbmOn = false;
529     return HWC2_ERROR_NONE;
530 }
531 
setPowerDoze(hwc2_power_mode_t mode)532 int32_t ExynosPrimaryDisplay::setPowerDoze(hwc2_power_mode_t mode) {
533     ATRACE_CALL();
534 
535     if (!mDisplayInterface->isDozeModeAvailable()) {
536         return HWC2_ERROR_UNSUPPORTED;
537     }
538 
539     if (mPowerModeState.has_value() &&
540         ((*mPowerModeState == HWC2_POWER_MODE_OFF) || (*mPowerModeState == HWC2_POWER_MODE_ON))) {
541         if (mDisplayInterface->setLowPowerMode()) {
542             ALOGI("Not support LP mode.");
543             return HWC2_ERROR_UNSUPPORTED;
544         }
545     }
546 
547     {
548         std::lock_guard<std::mutex> lock(mPowerModeMutex);
549         mPowerModeState = mode;
550     }
551 
552     // LHBM will be disabled in the kernel while entering AOD mode if it's
553     // already enabled. Reset the state to avoid the sync problem.
554     mBrightnessController->resetLhbmState();
555     mLhbmOn = false;
556 
557     ExynosDisplay::updateRefreshRateHint();
558 
559     if (mDisplayTe2Manager) mDisplayTe2Manager->updateTe2ForDozeMode();
560 
561     return HWC2_ERROR_NONE;
562 }
563 
setPowerMode(int32_t mode)564 int32_t ExynosPrimaryDisplay::setPowerMode(int32_t mode) {
565     Mutex::Autolock lock(mDisplayMutex);
566 
567     if (mode == static_cast<int32_t>(ext_hwc2_power_mode_t::PAUSE)) {
568         mode = HWC2_POWER_MODE_OFF;
569         mPauseDisplay = true;
570     } else if (mode == static_cast<int32_t>(ext_hwc2_power_mode_t::RESUME)) {
571         mode = HWC2_POWER_MODE_ON;
572         mPauseDisplay = false;
573     } else if (mPauseDisplay) {
574         ALOGI("Skip power mode transition due to pause display.");
575         return HWC2_ERROR_NONE;
576     }
577 
578     if (mPowerModeState.has_value() && (mode == static_cast<int32_t>(mPowerModeState.value()))) {
579         ALOGI("Skip power mode transition due to the same power state.");
580         return HWC2_ERROR_NONE;
581     }
582 
583     if (mode == HWC2_POWER_MODE_ON && mDevice->mNumPrimaryDisplays >= 2) {
584         ExynosDisplay* external_display =
585                 mDevice->getDisplay(getDisplayId(HWC_DISPLAY_EXTERNAL, 0));
586         ExynosDisplayDrmInterface* external_display_intf = external_display
587                 ? static_cast<ExynosDisplayDrmInterface*>(external_display->mDisplayInterface.get())
588                 : nullptr;
589         if (external_display_intf && external_display_intf->borrowedCrtcFrom() == this) {
590             ALOGI("Display %s is powering on, adjusting decon assignments",
591                   mDisplayTraceName.c_str());
592             hwc2_config_t activeConfig = 0;
593             external_display->getActiveConfig(&activeConfig);
594             external_display->clearDisplay(true);
595             external_display->setPowerMode(HWC2_POWER_MODE_OFF);
596             // Restore the original decon assigned to external display, this will ensure
597             // primary displays remain on the same initially assigned decons.
598             external_display_intf->swapCrtcs(this);
599             // This display is about to be powered on, but its mPowerModeState is not updated yet,
600             // so we need to exclude it from consideration explicitly here.
601             ExynosDisplay* poweredOffPrimaryDisplay = mDevice->findPoweredOffPrimaryDisplay(this);
602             if (poweredOffPrimaryDisplay) {
603                 ALOGI("Found powered off primary display %s, will use its decon for external "
604                       "display",
605                       poweredOffPrimaryDisplay->mDisplayTraceName.c_str());
606                 external_display_intf->swapCrtcs(poweredOffPrimaryDisplay);
607             } else {
608                 ALOGE("Could not find a powered off primary display!");
609             }
610             external_display->mActiveConfig = 0;
611             external_display->setActiveConfig(activeConfig);
612             external_display->setPowerMode(HWC2_POWER_MODE_ON);
613         }
614     }
615 
616     int fb_blank = (mode != HWC2_POWER_MODE_OFF) ? FB_BLANK_UNBLANK : FB_BLANK_POWERDOWN;
617     ALOGD("%s:: FBIOBLANK mode(%d), blank(%d)", __func__, mode, fb_blank);
618 
619     if (fb_blank == FB_BLANK_POWERDOWN)
620         mDREnable = false;
621     else
622         mDREnable = mDRDefault;
623 
624     if (mOperationRateManager) {
625         mOperationRateManager->onPowerMode(mode);
626         mBrightnessController->processOperationRate(
627                 mOperationRateManager->getTargetOperationRate());
628     }
629 
630     std::lock_guard<std::mutex> minIdleRefreshRateLock(mMinIdleRefreshRateMutex);
631     if (mVariableRefreshRateController) {
632         if ((mode == HWC2_POWER_MODE_DOZE) || (mode == HWC2_POWER_MODE_DOZE_SUSPEND)) {
633             mVariableRefreshRateController->setFixedRefreshRateRange(kMinIdleRefreshRateForDozeMode,
634                                                                      0);
635         }
636         mVariableRefreshRateController->preSetPowerMode(mode);
637     }
638 
639     int32_t res = HWC2_ERROR_BAD_PARAMETER;
640     switch (mode) {
641         case HWC2_POWER_MODE_DOZE:
642         case HWC2_POWER_MODE_DOZE_SUSPEND: {
643             if (mode == HWC2_POWER_MODE_DOZE && mDisplayInterface->needRefreshOnLP()) {
644                 ALOGI("Refresh before setting power doze.");
645                 mDevice->onRefresh(mDisplayId);
646             }
647             res = setPowerDoze(static_cast<hwc2_power_mode_t>(mode));
648             break;
649         }
650         case HWC2_POWER_MODE_OFF:
651             res = setPowerOff();
652             break;
653         case HWC2_POWER_MODE_ON:
654             res = setPowerOn();
655             break;
656         default:
657             return res;
658     }
659     if (res != HWC2_ERROR_NONE) {
660         return res;
661     }
662 
663     ExynosDisplay::updateRefreshRateHint();
664     if (mVariableRefreshRateController) {
665         mVariableRefreshRateController->postSetPowerMode(mode);
666         if (mode == HWC2_POWER_MODE_ON) {
667             mVariableRefreshRateController->setFixedRefreshRateRange(mMinIdleRefreshRate,
668                                                                      mRefreshRateDelayNanos);
669         }
670     }
671     return res;
672 }
673 
firstPowerOn()674 void ExynosPrimaryDisplay::firstPowerOn() {
675     SetCurrentPanelGammaSource(DisplayType::DISPLAY_PRIMARY, PanelGammaSource::GAMMA_CALIBRATION);
676     mFirstPowerOn = false;
677     getDisplayIdleTimerEnabled(mDisplayIdleTimerEnabled);
678     initDisplayHandleIdleExit();
679 }
680 
getHDRException(ExynosLayer * __unused layer)681 bool ExynosPrimaryDisplay::getHDRException(ExynosLayer* __unused layer)
682 {
683     return false;
684 }
685 
initDisplayInterface(uint32_t interfaceType)686 void ExynosPrimaryDisplay::initDisplayInterface(uint32_t interfaceType)
687 {
688     if (interfaceType == INTERFACE_TYPE_DRM)
689         mDisplayInterface = std::make_unique<ExynosPrimaryDisplayDrmInterfaceModule>((ExynosDisplay *)this);
690     else
691         LOG_ALWAYS_FATAL("%s::Unknown interface type(%d)",
692                 __func__, interfaceType);
693     mDisplayInterface->init(this);
694 
695     if (mXrrSettings.versionInfo.needVrrParameters()) {
696         mDisplayInterface->setXrrSettings(mXrrSettings);
697     }
698 
699     mDpuData.init(mMaxWindowNum, mDevice->getSpecialPlaneNum(mDisplayId));
700     mLastDpuData.init(mMaxWindowNum, mDevice->getSpecialPlaneNum(mDisplayId));
701     ALOGI("window configs size(%zu) rcd configs zie(%zu)", mDpuData.configs.size(),
702           mDpuData.rcdConfigs.size());
703 }
704 
getPanelSysfsPath(const DisplayType & type) const705 std::string ExynosPrimaryDisplay::getPanelSysfsPath(const DisplayType& type) const {
706     if ((type < DisplayType::DISPLAY_PRIMARY) || (type >= DisplayType::DISPLAY_MAX)) {
707         ALOGE("Invalid display panel type %d", type);
708         return {};
709     }
710 
711     const auto& iter = panelSysfsPath.find(type);
712     if (iter == panelSysfsPath.end()) {
713         return {};
714     }
715 
716     return iter->second;
717 }
718 
SetCurrentPanelGammaSource(const DisplayType type,const PanelGammaSource & source)719 int32_t ExynosPrimaryDisplay::SetCurrentPanelGammaSource(const DisplayType type,
720                                                          const PanelGammaSource &source) {
721     std::string &&panel_sysfs_path = getPanelSysfsPath(type);
722     if (panel_sysfs_path.empty()) {
723         return HWC2_ERROR_UNSUPPORTED;
724     }
725 
726     std::ifstream ifs;
727     std::string &&path = panel_sysfs_path + "panel_name";
728     ifs.open(path, std::ifstream::in);
729     if (!ifs.is_open()) {
730         ALOGW("Unable to access panel name path '%s' (%s)", path.c_str(), strerror(errno));
731         return HWC2_ERROR_UNSUPPORTED;
732     }
733     std::string panel_name;
734     std::getline(ifs, panel_name);
735     ifs.close();
736 
737     path = panel_sysfs_path + "serial_number";
738     ifs.open(path, std::ifstream::in);
739     if (!ifs.is_open()) {
740         ALOGW("Unable to access panel id path '%s' (%s)", path.c_str(), strerror(errno));
741         return HWC2_ERROR_UNSUPPORTED;
742     }
743     std::string panel_id;
744     std::getline(ifs, panel_id);
745     ifs.close();
746 
747     std::string gamma_node = panel_sysfs_path + "gamma";
748     if (access(gamma_node.c_str(), W_OK)) {
749         ALOGW("Unable to access panel gamma calibration node '%s' (%s)", gamma_node.c_str(),
750               strerror(errno));
751         return HWC2_ERROR_UNSUPPORTED;
752     }
753 
754     std::string &&gamma_data = "default";
755     if (source == PanelGammaSource::GAMMA_CALIBRATION) {
756         std::string gamma_cal_file(kDisplayCalFilePath);
757         gamma_cal_file.append(kPanelGammaCalFilePrefix)
758                 .append(1, '_')
759                 .append(panel_name)
760                 .append(1, '_')
761                 .append(panel_id)
762                 .append(".cal");
763         if (access(gamma_cal_file.c_str(), R_OK)) {
764             ALOGI("Fail to access `%s` (%s), try golden gamma calibration", gamma_cal_file.c_str(),
765                   strerror(errno));
766             gamma_cal_file = kDisplayCalFilePath;
767             gamma_cal_file.append(kPanelGammaCalFilePrefix)
768                     .append(1, '_')
769                     .append(panel_name)
770                     .append(".cal");
771         }
772         gamma_data = loadPanelGammaCalibration(gamma_cal_file);
773     }
774 
775     if (gamma_data.empty()) {
776         return HWC2_ERROR_UNSUPPORTED;
777     }
778 
779     std::ofstream ofs(gamma_node);
780     if (!ofs.is_open()) {
781         ALOGW("Unable to open gamma node '%s', error = %s", gamma_node.c_str(), strerror(errno));
782         return HWC2_ERROR_UNSUPPORTED;
783     }
784     ofs.write(gamma_data.c_str(), gamma_data.size());
785     ofs.close();
786 
787     currentPanelGammaSource = source;
788     return HWC2_ERROR_NONE;
789 }
790 
isLhbmSupported()791 bool ExynosPrimaryDisplay::isLhbmSupported() {
792     return mBrightnessController->isLhbmSupported();
793 }
794 
isConfigSettingEnabled()795 bool ExynosPrimaryDisplay::isConfigSettingEnabled() {
796     int64_t msSinceDisabled =
797             (systemTime(SYSTEM_TIME_MONOTONIC) - mConfigSettingDisabledTimestamp) / 1000000;
798     return !mConfigSettingDisabled || msSinceDisabled > kConfigDisablingMaxDurationMs;
799 }
800 
enableConfigSetting(bool en)801 void ExynosPrimaryDisplay::enableConfigSetting(bool en) {
802     DISPLAY_ATRACE_INT("ConfigSettingDisabled", !en);
803     ALOGI("%s:: mConfigSettingDisabled: %d", __func__, !en);
804     if (!en) {
805         mConfigSettingDisabled = true;
806         mConfigSettingDisabledTimestamp = systemTime(SYSTEM_TIME_MONOTONIC);
807         return;
808     }
809 
810     mConfigSettingDisabled = false;
811 }
812 
getDisplayConfigs(uint32_t * outNumConfigs,hwc2_config_t * outConfigs)813 int32_t ExynosPrimaryDisplay::getDisplayConfigs(uint32_t* outNumConfigs,
814                                                 hwc2_config_t* outConfigs) {
815     int32_t ret = ExynosDisplay::getDisplayConfigs(outNumConfigs, outConfigs);
816     if (ret == HWC2_ERROR_NONE) {
817         if (mXrrSettings.versionInfo.hasVrrController() && mDisplayConfigs.size()) {
818             if (!mVariableRefreshRateController) {
819                 mVariableRefreshRateController =
820                         VariableRefreshRateController::CreateInstance(this, getPanelName());
821                 std::unordered_map<hwc2_config_t, VrrConfig_t> vrrConfigs;
822                 for (const auto& it : mDisplayConfigs) {
823                     if (!it.second.vrrConfig.has_value()) {
824                         ALOGE("Both pseudo and full VRR modes should include VRR configurations.");
825                         return HWC2_ERROR_BAD_CONFIG;
826                     }
827                     vrrConfigs[it.first] = it.second.vrrConfig.value();
828                 }
829                 mVariableRefreshRateController->setVrrConfigurations(std::move(vrrConfigs));
830                 hwc2_config_t activeConfig;
831                 if (ExynosDisplay::getActiveConfig(&activeConfig) == HWC2_ERROR_NONE) {
832                     mVariableRefreshRateController->setActiveVrrConfiguration(activeConfig);
833                     mVariableRefreshRateController->setEnable(true);
834                 }
835             }
836         }
837     }
838     return ret;
839 }
840 
presentDisplay(int32_t * outRetireFence)841 int32_t ExynosPrimaryDisplay::presentDisplay(int32_t* outRetireFence) {
842     auto res = ExynosDisplay::presentDisplay(outRetireFence);
843     // Forward presentDisplay if there is a listener.
844     const auto presentListener = getPresentListener();
845     if (res == HWC2_ERROR_NONE && presentListener) {
846         presentListener->onPresent(*outRetireFence);
847     }
848     return res;
849 }
850 
onVsync(int64_t timestamp)851 void ExynosPrimaryDisplay::onVsync(int64_t timestamp) {
852     const auto vsyncListener = getVsyncListener();
853     if (vsyncListener) {
854         vsyncListener->onVsync(timestamp, 0);
855     }
856 }
857 
notifyExpectedPresent(int64_t timestamp,int32_t frameIntervalNs)858 int32_t ExynosPrimaryDisplay::notifyExpectedPresent(int64_t timestamp, int32_t frameIntervalNs) {
859     if (mVariableRefreshRateController) {
860         mVariableRefreshRateController->notifyExpectedPresent(timestamp, frameIntervalNs);
861     }
862     return NO_ERROR;
863 }
864 
setPresentTimeoutParameters(int timeoutNs,const std::vector<std::pair<uint32_t,uint32_t>> & settings)865 int32_t ExynosPrimaryDisplay::setPresentTimeoutParameters(
866         int timeoutNs, const std::vector<std::pair<uint32_t, uint32_t>>& settings) {
867     if (mVariableRefreshRateController) {
868         mVariableRefreshRateController->setPresentTimeoutParameters(timeoutNs, settings);
869         return NO_ERROR;
870     }
871     return HWC2_ERROR_UNSUPPORTED;
872 }
873 
setPresentTimeoutController(uint32_t controllerType)874 int32_t ExynosPrimaryDisplay::setPresentTimeoutController(uint32_t controllerType) {
875     if (mVariableRefreshRateController) {
876         mVariableRefreshRateController->setPresentTimeoutController(controllerType);
877         return NO_ERROR;
878     }
879     return HWC2_ERROR_UNSUPPORTED;
880 }
881 
setRefreshRateChangedCallbackDebugEnabled(bool enabled)882 int32_t ExynosPrimaryDisplay::setRefreshRateChangedCallbackDebugEnabled(bool enabled) {
883     if (mVariableRefreshRateController) {
884         mVariableRefreshRateController->enableRefreshRateCalculator(enabled);
885     } else {
886         ExynosDisplay::setRefreshRateChangedCallbackDebugEnabled(enabled);
887     }
888     return NO_ERROR;
889 }
890 
setLhbmDisplayConfigLocked(uint32_t peakRate)891 int32_t ExynosPrimaryDisplay::setLhbmDisplayConfigLocked(uint32_t peakRate) {
892     auto hwConfig = mDisplayInterface->getActiveModeId();
893     auto config = getConfigId(peakRate, mDisplayConfigs[hwConfig].width,
894                               mDisplayConfigs[hwConfig].height);
895     if (config == UINT_MAX) {
896         DISPLAY_LOGE("%s: failed to get config for rate=%d", __func__, peakRate);
897         return -EINVAL;
898     }
899 
900     if (mPendingConfig == UINT_MAX && mActiveConfig != config) mPendingConfig = mActiveConfig;
901     if (config != hwConfig) {
902         if (ExynosDisplay::setActiveConfigInternal(config, true) == HWC2_ERROR_NONE) {
903             DISPLAY_LOGI("%s: succeeded to set config=%d rate=%d", __func__, config, peakRate);
904         } else {
905             DISPLAY_LOGW("%s: failed to set config=%d rate=%d", __func__, config, peakRate);
906         }
907     } else {
908         DISPLAY_LOGI("%s: keep config=%d rate=%d", __func__, config, peakRate);
909     }
910     enableConfigSetting(false);
911     return OK;
912 }
913 
restoreLhbmDisplayConfigLocked()914 void ExynosPrimaryDisplay::restoreLhbmDisplayConfigLocked() {
915     enableConfigSetting(true);
916     hwc2_config_t pendingConfig = mPendingConfig;
917     auto hwConfig = mDisplayInterface->getActiveModeId();
918     if (pendingConfig != UINT_MAX && pendingConfig != hwConfig) {
919         if (applyPendingConfig() == HWC2_ERROR_NONE) {
920             DISPLAY_LOGI("%s: succeeded to set config=%d rate=%d", __func__, pendingConfig,
921                          getRefreshRate(pendingConfig));
922         } else {
923             DISPLAY_LOGE("%s: failed to set config=%d rate=%d", __func__, pendingConfig,
924                          getRefreshRate(pendingConfig));
925         }
926     } else {
927         mPendingConfig = UINT_MAX;
928         DISPLAY_LOGI("%s: keep config=%d rate=%d", __func__, hwConfig, getRefreshRate(hwConfig));
929     }
930 }
931 
932 // This function should be called by other threads (e.g. sensor HAL).
933 // HWCService can call this function but it should be for test purpose only.
setLhbmState(bool enabled)934 int32_t ExynosPrimaryDisplay::setLhbmState(bool enabled) {
935     int ret = OK;
936     // NOTE: mLhbmOn could be set to false at any time by setPowerOff in another
937     // thread. Make sure no side effect if that happens. Or add lock if we have
938     // to when new code is added.
939     DISPLAY_ATRACE_CALL();
940     DISPLAY_LOGI("%s: enabled=%d", __func__, enabled);
941     {
942         ATRACE_NAME("wait_for_power_on");
943         std::unique_lock<std::mutex> lock(mPowerModeMutex);
944         if (!mPowerModeState.has_value() || (*mPowerModeState != HWC2_POWER_MODE_ON)) {
945             mNotifyPowerOn = true;
946             if (!mPowerOnCondition.wait_for(lock, std::chrono::milliseconds(2000), [this]() {
947                     return (mPowerModeState.has_value() &&
948                             (*mPowerModeState == HWC2_POWER_MODE_ON));
949                 })) {
950                 DISPLAY_LOGW("%s: wait for power mode on timeout !", __func__);
951                 return TIMED_OUT;
952             }
953         }
954     }
955 
956     auto lhbmSysfs = mBrightnessController->GetPanelSysfileByIndex(
957             BrightnessController::kLocalHbmModeFileNode);
958     ret = mBrightnessController->checkSysfsStatus(lhbmSysfs,
959                                          {std::to_string(static_cast<int>(
960                                                  BrightnessController::LhbmMode::DISABLED))},
961                                          0);
962     bool wasDisabled = ret == OK;
963     if (!enabled && wasDisabled) {
964         DISPLAY_LOGW("%s: lhbm is at DISABLED state, skip disabling", __func__);
965         return NO_ERROR;
966     } else if (enabled && !wasDisabled) {
967         requestLhbm(true);
968         DISPLAY_LOGI("%s: lhbm is at ENABLING or ENABLED state, re-enable to reset timeout timer",
969                      __func__);
970         return NO_ERROR;
971     }
972 
973     std::vector<std::string> checkingValue;
974     if (!enabled) {
975         ATRACE_NAME("disable_lhbm");
976         requestLhbm(false);
977         {
978             ATRACE_NAME("wait_for_lhbm_off_cmd");
979             checkingValue = {
980                     std::to_string(static_cast<int>(BrightnessController::LhbmMode::DISABLED))};
981             ret = mBrightnessController->checkSysfsStatus(lhbmSysfs, checkingValue,
982                                                           ms2ns(kSysfsCheckTimeoutMs));
983             if (ret != OK) {
984                 DISPLAY_LOGW("%s: failed to send lhbm-off cmd", __func__);
985             }
986         }
987         {
988             Mutex::Autolock lock(mDisplayMutex);
989             restoreLhbmDisplayConfigLocked();
990         }
991         setLHBMRefreshRateThrottle(0);
992         mLhbmOn = false;
993         return NO_ERROR;
994     }
995 
996     ATRACE_NAME("enable_lhbm");
997     int64_t lhbmWaitForRrNanos, lhbmEnablingNanos, lhbmEnablingDoneNanos;
998     bool enablingStateSupported = !mFramesToReachLhbmPeakBrightness;
999     uint32_t peakRate = 0;
1000     auto rrSysfs = mBrightnessController->GetPanelRefreshRateSysfile();
1001     lhbmWaitForRrNanos = systemTime(SYSTEM_TIME_MONOTONIC);
1002     {
1003         Mutex::Autolock lock(mDisplayMutex);
1004         peakRate = getPeakRefreshRate();
1005         if (peakRate < 60) {
1006             DISPLAY_LOGE("%s: invalid peak rate=%d", __func__, peakRate);
1007             return -EINVAL;
1008         }
1009         ret = setLhbmDisplayConfigLocked(peakRate);
1010         if (ret != OK) return ret;
1011     }
1012 
1013     if (mBrightnessController->fileExists(rrSysfs)) {
1014         ATRACE_NAME("wait_for_peak_rate_cmd");
1015         ret = mBrightnessController->checkSysfsStatus(rrSysfs, {std::to_string(peakRate)},
1016                                                       ms2ns(kLhbmWaitForPeakRefreshRateMs));
1017         if (ret != OK) {
1018             DISPLAY_LOGW("%s: failed to poll peak refresh rate=%d, ret=%d", __func__, peakRate,
1019                          ret);
1020         }
1021     } else {
1022         ATRACE_NAME("wait_for_peak_rate_blindly");
1023         DISPLAY_LOGW("%s: missing refresh rate path: %s", __func__, rrSysfs.c_str());
1024         // blindly wait for (3 full frames + 1 frame uncertainty) to ensure DM finishes
1025         // switching refresh rate
1026         for (int32_t i = 0; i < 4; i++) {
1027             if (mDisplayInterface->waitVBlank()) {
1028                 DISPLAY_LOGE("%s: failed to blindly wait for peak refresh rate=%d, i=%d", __func__,
1029                              peakRate, i);
1030                 ret = -ENODEV;
1031                 goto enable_err;
1032             }
1033         }
1034     }
1035 
1036     setLHBMRefreshRateThrottle(kLhbmRefreshRateThrottleMs);
1037     checkingValue = {std::to_string(static_cast<int>(BrightnessController::LhbmMode::ENABLING)),
1038                      std::to_string(static_cast<int>(BrightnessController::LhbmMode::ENABLED))};
1039     lhbmEnablingNanos = systemTime(SYSTEM_TIME_MONOTONIC);
1040     requestLhbm(true);
1041     {
1042         ATRACE_NAME("wait_for_lhbm_on_cmd");
1043         ret = mBrightnessController->checkSysfsStatus(lhbmSysfs, checkingValue,
1044                                                       ms2ns(kSysfsCheckTimeoutMs));
1045         if (ret != OK) {
1046             DISPLAY_LOGE("%s: failed to enable lhbm", __func__);
1047             setLHBMRefreshRateThrottle(0);
1048             goto enable_err;
1049         }
1050     }
1051 
1052     lhbmEnablingDoneNanos = systemTime(SYSTEM_TIME_MONOTONIC);
1053     {
1054         ATRACE_NAME("wait_for_peak_brightness");
1055         if (enablingStateSupported) {
1056             ret = mBrightnessController->checkSysfsStatus(lhbmSysfs,
1057                                             {std::to_string(static_cast<int>(
1058                                                     BrightnessController::LhbmMode::ENABLED))},
1059                                             ms2ns(kSysfsCheckTimeoutMs));
1060             if (ret != OK) {
1061                 DISPLAY_LOGE("%s: failed to wait for lhbm becoming effective", __func__);
1062                 goto enable_err;
1063             }
1064         } else {
1065             // lhbm takes effect at next vblank
1066             for (int32_t i = mFramesToReachLhbmPeakBrightness + 1; i > 0; i--) {
1067                 ret = mDisplayInterface->waitVBlank();
1068                 if (ret) {
1069                     DISPLAY_LOGE("%s: failed to wait vblank for peak brightness, %d", __func__, i);
1070                     goto enable_err;
1071                 }
1072             }
1073         }
1074     }
1075     DISPLAY_LOGI("%s: latency: %04d = %03d|rr@%03d + %03d|en + %03d|boost@%s", __func__,
1076                  getTimestampDeltaMs(0, lhbmWaitForRrNanos),
1077                  getTimestampDeltaMs(lhbmEnablingNanos, lhbmWaitForRrNanos), peakRate,
1078                  getTimestampDeltaMs(lhbmEnablingDoneNanos, lhbmEnablingNanos),
1079                  getTimestampDeltaMs(0, lhbmEnablingDoneNanos),
1080                  enablingStateSupported ? "polling" : "fixed");
1081 
1082     mLhbmOn = true;
1083     if (!mPowerModeState.has_value() || (*mPowerModeState == HWC2_POWER_MODE_OFF && mLhbmOn)) {
1084         mLhbmOn = false;
1085         DISPLAY_LOGE("%s: power off during request lhbm on", __func__);
1086         return -EINVAL;
1087     }
1088     return NO_ERROR;
1089 enable_err:
1090     {
1091         // We may receive LHBM request during the power off sequence due to the
1092         // race condition between display and sensor. If the failure happens
1093         // after requestLhbm(), we will get a wrong LHBM state in the 1st commit
1094         // after power on. We should reset the state in this case.
1095         std::unique_lock<std::mutex> lock(mPowerModeMutex);
1096         if (!mPowerModeState.has_value() || (*mPowerModeState == HWC2_POWER_MODE_OFF)) {
1097             DISPLAY_LOGW("%s: request lhbm during power off sequence, reset the state", __func__);
1098             mBrightnessController->resetLhbmState();
1099         }
1100     }
1101 
1102     Mutex::Autolock lock(mDisplayMutex);
1103     restoreLhbmDisplayConfigLocked();
1104     return ret;
1105 }
1106 
getLhbmState()1107 bool ExynosPrimaryDisplay::getLhbmState() {
1108     return mLhbmOn;
1109 }
1110 
setLHBMRefreshRateThrottle(const uint32_t delayMs)1111 void ExynosPrimaryDisplay::setLHBMRefreshRateThrottle(const uint32_t delayMs) {
1112     ATRACE_CALL();
1113 
1114     if (delayMs) {
1115         // make new throttle take effect
1116         mLastRefreshRateAppliedNanos = systemTime(SYSTEM_TIME_MONOTONIC);
1117         DISPLAY_ATRACE_INT64("LastRefreshRateAppliedMs", ns2ms(mLastRefreshRateAppliedNanos));
1118     }
1119 
1120     setRefreshRateThrottleNanos(std::chrono::duration_cast<std::chrono::nanoseconds>(
1121                                         std::chrono::milliseconds(delayMs))
1122                                         .count(),
1123                                 RrThrottleRequester::LHBM);
1124 }
1125 
setEarlyWakeupDisplay()1126 void ExynosPrimaryDisplay::setEarlyWakeupDisplay() {
1127     if (mEarlyWakeupDispFd) {
1128         writeFileNode(mEarlyWakeupDispFd, 1);
1129     }
1130 }
1131 
setExpectedPresentTime(uint64_t timestamp,int frameIntervalNs)1132 void ExynosPrimaryDisplay::setExpectedPresentTime(uint64_t timestamp, int frameIntervalNs) {
1133     mExpectedPresentTimeAndInterval.store(std::make_tuple(timestamp, frameIntervalNs));
1134     // Forward presentDisplay if there is a listener.
1135     const auto presentListener = getPresentListener();
1136     if (presentListener) {
1137         presentListener->setExpectedPresentTime(timestamp, frameIntervalNs);
1138     }
1139 }
1140 
getPendingExpectedPresentTime()1141 uint64_t ExynosPrimaryDisplay::getPendingExpectedPresentTime() {
1142     if (mExpectedPresentTimeAndInterval.is_dirty()) {
1143         return std::get<0>(mExpectedPresentTimeAndInterval.get());
1144     }
1145 
1146     return 0;
1147 }
1148 
getPendingFrameInterval()1149 int ExynosPrimaryDisplay::getPendingFrameInterval() {
1150     if (mExpectedPresentTimeAndInterval.is_dirty()) {
1151         return std::get<1>(mExpectedPresentTimeAndInterval.get());
1152     }
1153 
1154     return 0;
1155 }
1156 
applyExpectedPresentTime()1157 void ExynosPrimaryDisplay::applyExpectedPresentTime() {
1158     mExpectedPresentTimeAndInterval.clear_dirty();
1159 }
1160 
setDisplayIdleTimer(const int32_t timeoutMs)1161 int32_t ExynosPrimaryDisplay::setDisplayIdleTimer(const int32_t timeoutMs) {
1162     bool support = false;
1163     if (getDisplayIdleTimerSupport(support) || support == false) {
1164         return HWC2_ERROR_UNSUPPORTED;
1165     }
1166 
1167     if (timeoutMs < 0) {
1168         return HWC2_ERROR_BAD_PARAMETER;
1169     }
1170 
1171     if (timeoutMs > 0) {
1172         setDisplayIdleDelayNanos(std::chrono::duration_cast<std::chrono::nanoseconds>(
1173                                          std::chrono::milliseconds(timeoutMs))
1174                                          .count(),
1175                                  DispIdleTimerRequester::SF);
1176     }
1177 
1178     bool enabled = (timeoutMs > 0);
1179     if (enabled != mDisplayIdleTimerEnabled) {
1180         if (setDisplayIdleTimerEnabled(enabled) == NO_ERROR) {
1181             mDisplayIdleTimerEnabled = enabled;
1182         }
1183     }
1184 
1185     return HWC2_ERROR_NONE;
1186 }
1187 
getDisplayIdleTimerEnabled(bool & enabled)1188 int32_t ExynosPrimaryDisplay::getDisplayIdleTimerEnabled(bool &enabled) {
1189     bool support = false;
1190     if (getDisplayIdleTimerSupport(support) || support == false) {
1191         return HWC2_ERROR_UNSUPPORTED;
1192     }
1193 
1194     const std::string path = getPanelSysfsPath() + "panel_idle";
1195     std::ifstream ifs(path);
1196     if (!ifs.is_open()) {
1197         ALOGW("%s() unable to open node '%s', error = %s", __func__, path.c_str(), strerror(errno));
1198         return errno;
1199     } else {
1200         std::string panel_idle;
1201         std::getline(ifs, panel_idle);
1202         ifs.close();
1203         enabled = (panel_idle == "1");
1204         ALOGI("%s() get panel_idle(%d) from the sysfs node", __func__, enabled);
1205     }
1206     return NO_ERROR;
1207 }
1208 
setDisplayIdleTimerEnabled(const bool enabled)1209 int32_t ExynosPrimaryDisplay::setDisplayIdleTimerEnabled(const bool enabled) {
1210     const std::string path = getPanelSysfsPath() + "panel_idle";
1211     std::ofstream ofs(path);
1212     if (!ofs.is_open()) {
1213         ALOGW("%s() unable to open node '%s', error = %s", __func__, path.c_str(), strerror(errno));
1214         return errno;
1215     } else {
1216         ofs << enabled;
1217         ofs.close();
1218         ALOGI("%s() writes panel_idle(%d) to the sysfs node", __func__, enabled);
1219     }
1220     return NO_ERROR;
1221 }
1222 
setDisplayIdleDelayNanos(const int32_t delayNanos,const DispIdleTimerRequester requester)1223 int32_t ExynosPrimaryDisplay::setDisplayIdleDelayNanos(const int32_t delayNanos,
1224                                                        const DispIdleTimerRequester requester) {
1225     std::lock_guard<std::mutex> lock(mDisplayIdleDelayMutex);
1226 
1227     int64_t maxDelayNanos = 0;
1228     mDisplayIdleTimerNanos[toUnderlying(requester)] = delayNanos;
1229     for (uint32_t i = 0; i < toUnderlying(DispIdleTimerRequester::MAX); i++) {
1230         if (mDisplayIdleTimerNanos[i] > maxDelayNanos) {
1231             maxDelayNanos = mDisplayIdleTimerNanos[i];
1232         }
1233     }
1234 
1235     if (mDisplayIdleDelayNanos == maxDelayNanos) {
1236         return NO_ERROR;
1237     }
1238 
1239     mDisplayIdleDelayNanos = maxDelayNanos;
1240 
1241     const int32_t displayIdleDelayMs = std::chrono::duration_cast<std::chrono::milliseconds>(
1242                                                std::chrono::nanoseconds(mDisplayIdleDelayNanos))
1243                                                .count();
1244     const std::string path = getPanelSysfsPath() + "idle_delay_ms";
1245     std::ofstream ofs(path);
1246     if (!ofs.is_open()) {
1247         ALOGW("%s() unable to open node '%s', error = %s", __func__, path.c_str(), strerror(errno));
1248         return errno;
1249     } else {
1250         ofs << displayIdleDelayMs;
1251         ALOGI("%s() writes idle_delay_ms(%d) to the sysfs node (0x%x)", __func__,
1252               displayIdleDelayMs, ofs.rdstate());
1253         ofs.close();
1254     }
1255     return NO_ERROR;
1256 }
1257 
initDisplayHandleIdleExit()1258 void ExynosPrimaryDisplay::initDisplayHandleIdleExit() {
1259     if (bool support; getDisplayIdleTimerSupport(support) || support == false) {
1260         return;
1261     }
1262 
1263     const std::string path = getPanelSysfsPath() + "panel_need_handle_idle_exit";
1264     mDisplayNeedHandleIdleExitOfs.open(path, std::ofstream::out);
1265     if (!mDisplayNeedHandleIdleExitOfs.is_open()) {
1266         ALOGI("%s() '%s' doesn't exist(%s)", __func__, path.c_str(), strerror(errno));
1267     }
1268 
1269     setDisplayNeedHandleIdleExit(false, true);
1270 }
1271 
setDisplayNeedHandleIdleExit(const bool needed,const bool force)1272 void ExynosPrimaryDisplay::setDisplayNeedHandleIdleExit(const bool needed, const bool force) {
1273     if (!mDisplayNeedHandleIdleExitOfs.is_open()) {
1274         return;
1275     }
1276 
1277     if (needed == mDisplayNeedHandleIdleExit && !force) {
1278         return;
1279     }
1280 
1281     mDisplayNeedHandleIdleExitOfs << needed;
1282     if (mDisplayNeedHandleIdleExitOfs.fail()) {
1283         ALOGW("%s() failed to write panel_need_handle_idle_exit(%d) to sysfs node %s", __func__,
1284               needed, strerror(errno));
1285         return;
1286     }
1287 
1288     mDisplayNeedHandleIdleExitOfs.flush();
1289     if (mDisplayNeedHandleIdleExitOfs.fail()) {
1290         ALOGW("%s() failed to flush panel_need_handle_idle_exit(%d) to sysfs node %s", __func__,
1291               needed, strerror(errno));
1292         return;
1293     }
1294 
1295     ALOGI("%s() writes panel_need_handle_idle_exit(%d) to sysfs node", __func__, needed);
1296     mDisplayNeedHandleIdleExit = needed;
1297 }
1298 
handleDisplayIdleEnter(const uint32_t idleTeRefreshRate)1299 void ExynosPrimaryDisplay::handleDisplayIdleEnter(const uint32_t idleTeRefreshRate) {
1300     bool needed = false;
1301     {
1302         Mutex::Autolock lock1(mDisplayMutex);
1303         uint32_t btsRefreshRate = getBtsRefreshRate();
1304         if (idleTeRefreshRate <= btsRefreshRate) {
1305             return;
1306         }
1307         Mutex::Autolock lock2(mDRMutex);
1308         for (size_t i = 0; i < mLayers.size(); i++) {
1309             if (mLayers[i]->mOtfMPP && mLayers[i]->mM2mMPP == nullptr &&
1310                 !mLayers[i]->checkBtsCap(idleTeRefreshRate)) {
1311                 needed = true;
1312                 break;
1313             }
1314         }
1315     }
1316 
1317     setDisplayNeedHandleIdleExit(needed, false);
1318 }
1319 
setFixedTe2Rate(const int targetTe2RateHz)1320 int32_t ExynosPrimaryDisplay::setFixedTe2Rate(const int targetTe2RateHz) {
1321     if (mDisplayTe2Manager) {
1322         return mDisplayTe2Manager->setFixedTe2Rate(targetTe2RateHz);
1323     } else {
1324         return HWC2_ERROR_UNSUPPORTED;
1325     }
1326 }
1327 
setMinIdleRefreshRate(const int targetFps,const RrThrottleRequester requester)1328 int32_t ExynosPrimaryDisplay::setMinIdleRefreshRate(const int targetFps,
1329                                                     const RrThrottleRequester requester) {
1330     if (targetFps < 0) {
1331         return HWC2_ERROR_BAD_PARAMETER;
1332     }
1333     int fps = targetFps;
1334     if ((requester == RrThrottleRequester::BRIGHTNESS) &&
1335         (!mBrightnessBlockingZonesLookupTable.empty())) {
1336         std::unique_lock<std::mutex> lock(mPowerModeMutex);
1337         // Only check the BRIGHTNESS vote when the power is on.
1338         if (mPowerModeState.has_value() && (*mPowerModeState == HWC2_POWER_MODE_ON)) {
1339             auto res = mBrightnessController->getBrightnessNitsAndMode();
1340             if (res != std::nullopt) {
1341                 const auto it =
1342                         mBrightnessBlockingZonesLookupTable.upper_bound(std::get<0>(res.value()));
1343                 ALOGD("%s requester = BRIGHTNESS, brightness = %f nits, vote minimum refresh rate "
1344                       "to %d",
1345                       __func__, std::get<0>(res.value()), it->second);
1346                 fps = std::max(fps, it->second);
1347                 if (mDisplayConfigs.count(mActiveConfig)) {
1348                     if (fps > mDisplayConfigs[mActiveConfig].refreshRate) {
1349                         ALOGI("%s() The brightness blocking zone votes for the FPS = %d, which is "
1350                               "higher than the maximum refresh rate of the current configuration = "
1351                               "%d",
1352                               __func__, fps, mDisplayConfigs[mActiveConfig].refreshRate);
1353                         fps = mDisplayConfigs[mActiveConfig].refreshRate;
1354                     }
1355                 }
1356             }
1357         }
1358     }
1359 
1360     std::lock_guard<std::mutex> lock(mMinIdleRefreshRateMutex);
1361     if (fps == mRrThrottleFps[toUnderlying(requester)]) return NO_ERROR;
1362 
1363     ALOGD("%s requester %u, fps %d", __func__, toUnderlying(requester), fps);
1364     mRrThrottleFps[toUnderlying(requester)] = fps;
1365     int maxMinIdleFps = 0;
1366     for (uint32_t i = 0; i < toUnderlying(RrThrottleRequester::MAX); i++) {
1367         if (mRrThrottleFps[i] > maxMinIdleFps) {
1368             maxMinIdleFps = mRrThrottleFps[i];
1369         }
1370     }
1371 
1372     {
1373         std::lock_guard<std::mutex> lock(mPowerModeMutex);
1374         bool dozeMode = (mPowerModeState.has_value() &&
1375                          (*mPowerModeState == HWC2_POWER_MODE_DOZE ||
1376                           *mPowerModeState == HWC2_POWER_MODE_DOZE_SUSPEND));
1377         // Currently only proximity sensor will request the min refresh rate via this API with
1378         // PIXEL_DISP (or TEST for the debugging command). It will request a non-zero value,
1379         // e.g. 30Hz, if it's active, and request zero if it's inactive. So we can know its state
1380         // and update the TE2 option accordingly.
1381         if (mDisplayTe2Manager &&
1382             (requester == RrThrottleRequester::PIXEL_DISP ||
1383              requester == RrThrottleRequester::TEST)) {
1384             bool proximityActive = !!targetFps;
1385             ALOGD("%s: proximity state %s, min %dhz, doze mode %d", __func__,
1386                   proximityActive ? "active" : "inactive", targetFps, dozeMode);
1387             mDisplayTe2Manager->updateTe2OptionForProximity(proximityActive, targetFps, dozeMode);
1388         }
1389 
1390         if (maxMinIdleFps == mMinIdleRefreshRate) return NO_ERROR;
1391 
1392         if (mVariableRefreshRateController) {
1393             if (dozeMode && maxMinIdleFps != kMinIdleRefreshRateForDozeMode) {
1394                 ALOGW("%s: setting %dhz in doze mode (expect %dhz)", __func__, maxMinIdleFps,
1395                       kMinIdleRefreshRateForDozeMode);
1396                 mMinIdleRefreshRate = maxMinIdleFps;
1397                 return NO_ERROR;
1398             }
1399 
1400             int ret = mVariableRefreshRateController
1401                               ->setFixedRefreshRateRange(maxMinIdleFps, mRefreshRateDelayNanos);
1402             if (ret >= 0) {
1403                 mMinIdleRefreshRate = maxMinIdleFps;
1404                 return NO_ERROR;
1405             }
1406             return ret;
1407         }
1408     }
1409 
1410     const std::string path = getPanelSysfsPath() + "min_vrefresh";
1411     std::ofstream ofs(path);
1412     if (!ofs.is_open()) {
1413         ALOGW("%s Unable to open node '%s', error = %s", __func__, path.c_str(), strerror(errno));
1414         return errno;
1415     } else {
1416         ofs << maxMinIdleFps;
1417         ofs.close();
1418         ALOGI("ExynosPrimaryDisplay::%s() writes min_vrefresh(%d) to the sysfs node", __func__,
1419               maxMinIdleFps);
1420     }
1421     mMinIdleRefreshRate = maxMinIdleFps;
1422     return NO_ERROR;
1423 }
1424 
setRefreshRateThrottleNanos(const int64_t delayNanos,const RrThrottleRequester requester)1425 int32_t ExynosPrimaryDisplay::setRefreshRateThrottleNanos(const int64_t delayNanos,
1426                                                           const RrThrottleRequester requester) {
1427     ATRACE_CALL();
1428     if (delayNanos < 0) {
1429         ALOGW("%s() set invalid delay(%" PRId64 ")", __func__, delayNanos);
1430         return BAD_VALUE;
1431     }
1432 
1433     int32_t ret = NO_ERROR;
1434     int64_t maxDelayNanos = 0;
1435     {
1436         std::lock_guard<std::mutex> lock(mIdleRefreshRateThrottleMutex);
1437         if (delayNanos == mRrThrottleNanos[toUnderlying(requester)]) return NO_ERROR;
1438 
1439         ALOGI("%s() requester(%u) set delay to %" PRId64 "ns", __func__, toUnderlying(requester),
1440               delayNanos);
1441         mRrThrottleNanos[toUnderlying(requester)] = delayNanos;
1442         uint32_t maxDelayRequester = toUnderlying(RrThrottleRequester::MAX);
1443         for (uint32_t i = 0; i < toUnderlying(RrThrottleRequester::MAX); i++) {
1444             if (mRrThrottleNanos[i] > maxDelayNanos) {
1445                 maxDelayRequester = i;
1446                 maxDelayNanos = mRrThrottleNanos[i];
1447             }
1448         }
1449 
1450         DISPLAY_ATRACE_INT("RefreshRateDelayRequester", maxDelayRequester);
1451         DISPLAY_ATRACE_INT64("RefreshRateDelay", ns2ms(maxDelayNanos));
1452         if (mRefreshRateDelayNanos == maxDelayNanos) {
1453             return NO_ERROR;
1454         }
1455 
1456         if (mVariableRefreshRateController) {
1457             int ret = mVariableRefreshRateController->setFixedRefreshRateRange(mMinIdleRefreshRate,
1458                                                                                maxDelayNanos);
1459             if (ret >= 0) {
1460                 mRefreshRateDelayNanos = maxDelayNanos;
1461                 return NO_ERROR;
1462             }
1463             return ret;
1464         }
1465 
1466         ret = setDisplayIdleDelayNanos(maxDelayNanos, DispIdleTimerRequester::RR_THROTTLE);
1467         if (ret == NO_ERROR) {
1468             mRefreshRateDelayNanos = maxDelayNanos;
1469         }
1470     }
1471 
1472     Mutex::Autolock lock(mDisplayMutex);
1473     if (ret == NO_ERROR) {
1474         recalculateTimelineLocked(maxDelayNanos);
1475     }
1476 
1477     return ret;
1478 }
1479 
dump(String8 & result)1480 void ExynosPrimaryDisplay::dump(String8 &result) {
1481     ExynosDisplay::dump(result);
1482     result.appendFormat("Display idle timer: %s\n",
1483                         (mDisplayIdleTimerEnabled) ? "enabled" : "disabled");
1484     for (uint32_t i = 0; i < toUnderlying(DispIdleTimerRequester::MAX); i++) {
1485         result.appendFormat("\t[%u] vote to %" PRId64 " ns\n", i, mDisplayIdleTimerNanos[i]);
1486     }
1487 
1488     if (!mBrightnessBlockingZonesLookupTable.empty()) {
1489         int upperBound;
1490         int lowerBound = INT_MIN;
1491         result.appendFormat("Brightness blocking zone lookup table:\n");
1492         for (const auto& brightnessBlockingZone : mBrightnessBlockingZonesLookupTable) {
1493             upperBound = brightnessBlockingZone.first;
1494             result.appendFormat("\tBrightness blocking zone: range [%s %s) fps = %d\n",
1495                                 (lowerBound == INT_MIN ? "Min"
1496                                                        : std::to_string(lowerBound).c_str()),
1497                                 (upperBound == INT_MAX ? "Max"
1498                                                        : std::to_string(upperBound).c_str()),
1499                                 brightnessBlockingZone.second);
1500             lowerBound = upperBound;
1501         }
1502     } else {
1503         result.appendFormat("\n");
1504     }
1505 
1506     result.appendFormat("Min idle refresh rate: %d\n", mMinIdleRefreshRate);
1507     for (uint32_t i = 0; i < toUnderlying(RrThrottleRequester::MAX); i++) {
1508         result.appendFormat("\t[%u] vote to %d hz\n", i, mRrThrottleFps[i]);
1509     }
1510 
1511     result.appendFormat("Refresh rate delay: %" PRId64 " ns\n", mRefreshRateDelayNanos);
1512     for (uint32_t i = 0; i < toUnderlying(RrThrottleRequester::MAX); i++) {
1513         result.appendFormat("\t[%u] vote to %" PRId64 " ns\n", i, mRrThrottleNanos[i]);
1514     }
1515     result.appendFormat("\n");
1516 }
1517 
calculateTimelineLocked(hwc2_config_t config,hwc_vsync_period_change_constraints_t * vsyncPeriodChangeConstraints,hwc_vsync_period_change_timeline_t * outTimeline)1518 void ExynosPrimaryDisplay::calculateTimelineLocked(
1519         hwc2_config_t config, hwc_vsync_period_change_constraints_t* vsyncPeriodChangeConstraints,
1520         hwc_vsync_period_change_timeline_t* outTimeline) {
1521     ATRACE_CALL();
1522     int64_t desiredUpdateTimeNanos = vsyncPeriodChangeConstraints->desiredTimeNanos;
1523     const int64_t origDesiredUpdateTimeNanos = desiredUpdateTimeNanos;
1524     int64_t threshold = 0;
1525     int64_t lastUpdateDelta = 0;
1526     int64_t actualChangeTime = 0;
1527     bool isDelayed = false;
1528 
1529     /* actualChangeTime includes transient duration */
1530     mDisplayInterface->getVsyncAppliedTime(config, &actualChangeTime);
1531 
1532     outTimeline->refreshRequired = true;
1533 
1534     /* When the refresh rate changes from high to low, check if RR throttle is needed */
1535     {
1536         std::lock_guard<std::mutex> lock(mIdleRefreshRateThrottleMutex);
1537         threshold = mRefreshRateDelayNanos;
1538         mRrUseDelayNanos = 0;
1539         mIsRrNeedCheckDelay =
1540                 mDisplayConfigs[mActiveConfig].vsyncPeriod < mDisplayConfigs[config].vsyncPeriod;
1541         if (threshold != 0 && mLastRefreshRateAppliedNanos != 0 && mIsRrNeedCheckDelay) {
1542             lastUpdateDelta = desiredUpdateTimeNanos - mLastRefreshRateAppliedNanos;
1543             if (lastUpdateDelta < threshold) {
1544                 /* in this case, the active config change needs to be delayed */
1545                 isDelayed = true;
1546                 desiredUpdateTimeNanos += threshold - lastUpdateDelta;
1547                 mRrUseDelayNanos = threshold;
1548             }
1549         }
1550     }
1551 
1552     mVsyncPeriodChangeConstraints.desiredTimeNanos = desiredUpdateTimeNanos;
1553 
1554     getConfigAppliedTime(mVsyncPeriodChangeConstraints.desiredTimeNanos, actualChangeTime,
1555                          outTimeline->newVsyncAppliedTimeNanos, outTimeline->refreshTimeNanos);
1556 
1557     const nsecs_t now = systemTime(SYSTEM_TIME_MONOTONIC);
1558     DISPLAY_LOGD_AND_ATRACE_NAME(eDebugDisplayConfig,
1559                                  "requested config : %d(%d)->%d(%d), isDelay:%d,"
1560                                  " delta %" PRId64 ", delay %" PRId64 ", threshold %" PRId64 ", "
1561                                  "now:%" PRId64 ", desired %" PRId64 "->%" PRId64
1562                                  ", newVsyncAppliedTimeNanos : %" PRId64
1563                                  ", refreshTimeNanos:%" PRId64
1564                                  ", mLastRefreshRateAppliedNanos:%" PRId64,
1565                                  mActiveConfig, mDisplayConfigs[mActiveConfig].vsyncPeriod, config,
1566                                  mDisplayConfigs[config].vsyncPeriod, isDelayed,
1567                                  ns2ms(lastUpdateDelta), ns2ms(threshold - lastUpdateDelta),
1568                                  ns2ms(threshold), ns2ms(now), ns2ms(origDesiredUpdateTimeNanos),
1569                                  ns2ms(mVsyncPeriodChangeConstraints.desiredTimeNanos),
1570                                  ns2ms(outTimeline->newVsyncAppliedTimeNanos),
1571                                  ns2ms(outTimeline->refreshTimeNanos),
1572                                  ns2ms(mLastRefreshRateAppliedNanos));
1573 
1574     DISPLAY_ATRACE_INT64("TimeToApplyConfig",
1575                          ns2ms(mVsyncPeriodChangeConstraints.desiredTimeNanos - now));
1576 }
1577 
recalculateTimelineLocked(int64_t refreshRateDelayNanos)1578 void ExynosPrimaryDisplay::recalculateTimelineLocked(int64_t refreshRateDelayNanos) {
1579     ATRACE_CALL();
1580 
1581     if (mConfigRequestState != hwc_request_state_t::SET_CONFIG_STATE_PENDING) {
1582         return;
1583     }
1584 
1585     std::lock_guard<std::mutex> lock(mIdleRefreshRateThrottleMutex);
1586     if (!mIsRrNeedCheckDelay) {
1587         return;
1588     }
1589 
1590     int64_t desiredUpdateTimeNanos = mVsyncPeriodChangeConstraints.desiredTimeNanos;
1591     if (mRrUseDelayNanos) {
1592         desiredUpdateTimeNanos += (refreshRateDelayNanos - mRrUseDelayNanos);
1593     } else {
1594         int64_t lastUpdateDelta =
1595                 mVsyncPeriodChangeConstraints.desiredTimeNanos - mLastRefreshRateAppliedNanos;
1596         if (lastUpdateDelta < refreshRateDelayNanos) {
1597             desiredUpdateTimeNanos += (refreshRateDelayNanos - lastUpdateDelta);
1598         }
1599     }
1600     mRrUseDelayNanos = refreshRateDelayNanos;
1601 
1602     const nsecs_t now = systemTime(SYSTEM_TIME_MONOTONIC);
1603     if (desiredUpdateTimeNanos > now) {
1604         mVsyncPeriodChangeConstraints.desiredTimeNanos = desiredUpdateTimeNanos;
1605     } else {
1606         mVsyncPeriodChangeConstraints.desiredTimeNanos = now;
1607     }
1608 
1609     DISPLAY_ATRACE_INT64("TimeToApplyConfig",
1610                          ns2ms(mVsyncPeriodChangeConstraints.desiredTimeNanos - now));
1611 }
1612 
updateAppliedActiveConfig(const hwc2_config_t newConfig,const int64_t ts)1613 void ExynosPrimaryDisplay::updateAppliedActiveConfig(const hwc2_config_t newConfig,
1614                                                      const int64_t ts) {
1615     ATRACE_CALL();
1616     if (mAppliedActiveConfig == 0 ||
1617         getDisplayVsyncPeriodFromConfig(mAppliedActiveConfig) !=
1618                 getDisplayVsyncPeriodFromConfig(newConfig)) {
1619         DISPLAY_LOGD(eDebugDisplayConfig,
1620                      "%s mAppliedActiveConfig(%d->%d), mLastRefreshRateAppliedNanos(%" PRIu64
1621                      " -> %" PRIu64 ")",
1622                      __func__, mAppliedActiveConfig, newConfig, mLastRefreshRateAppliedNanos, ts);
1623         mLastRefreshRateAppliedNanos = ts;
1624         DISPLAY_ATRACE_INT64("LastRefreshRateAppliedMs", ns2ms(mLastRefreshRateAppliedNanos));
1625     }
1626 
1627     mAppliedActiveConfig = newConfig;
1628 }
1629 
checkBtsReassignResource(const int32_t vsyncPeriod,const int32_t btsVsyncPeriod)1630 void ExynosPrimaryDisplay::checkBtsReassignResource(const int32_t vsyncPeriod,
1631                                                     const int32_t btsVsyncPeriod) {
1632     ATRACE_CALL();
1633     uint32_t refreshRate = static_cast<uint32_t>(round(nsecsPerSec / vsyncPeriod * 0.1f) * 10);
1634 
1635     Mutex::Autolock lock(mDRMutex);
1636     if (vsyncPeriod < btsVsyncPeriod) {
1637         for (size_t i = 0; i < mLayers.size(); i++) {
1638             if (mLayers[i]->mOtfMPP && mLayers[i]->mM2mMPP == nullptr &&
1639                 !mLayers[i]->checkBtsCap(refreshRate)) {
1640                 mLayers[i]->setGeometryChanged(GEOMETRY_DEVICE_CONFIG_CHANGED);
1641                 break;
1642             }
1643         }
1644     } else if (vsyncPeriod > btsVsyncPeriod) {
1645         for (size_t i = 0; i < mLayers.size(); i++) {
1646             if (mLayers[i]->mOtfMPP && mLayers[i]->mM2mMPP) {
1647                 float srcWidth = mLayers[i]->mSourceCrop.right - mLayers[i]->mSourceCrop.left;
1648                 float srcHeight = mLayers[i]->mSourceCrop.bottom - mLayers[i]->mSourceCrop.top;
1649                 float resolution = srcWidth * srcHeight * refreshRate / 1000;
1650                 float ratioVertical = static_cast<float>(mLayers[i]->mDisplayFrame.bottom -
1651                                                          mLayers[i]->mDisplayFrame.top) /
1652                         mYres;
1653 
1654                 if (mLayers[i]->mOtfMPP->checkDownscaleCap(resolution, ratioVertical)) {
1655                     mLayers[i]->setGeometryChanged(GEOMETRY_DEVICE_CONFIG_CHANGED);
1656                     break;
1657                 }
1658             }
1659         }
1660     }
1661 }
1662 
isDbmSupported()1663 bool ExynosPrimaryDisplay::isDbmSupported() {
1664     return mBrightnessController->isDbmSupported();
1665 }
1666 
setDbmState(bool enabled)1667 int32_t ExynosPrimaryDisplay::setDbmState(bool enabled) {
1668     mBrightnessController->processDimBrightness(enabled);
1669     return NO_ERROR;
1670 }
1671 
getPresentListener()1672 PresentListener* ExynosPrimaryDisplay::getPresentListener() {
1673     if (mVariableRefreshRateController) {
1674         return mVariableRefreshRateController.get();
1675     }
1676     return nullptr;
1677 }
1678 
getVsyncListener()1679 VsyncListener* ExynosPrimaryDisplay::getVsyncListener() {
1680     if (mVariableRefreshRateController) {
1681         return mVariableRefreshRateController.get();
1682     }
1683     return nullptr;
1684 }
1685 
onConfigChange(int configId)1686 void ExynosPrimaryDisplay::onConfigChange(int configId) {
1687     if (mVariableRefreshRateController) {
1688         return mVariableRefreshRateController->setActiveVrrConfiguration(configId);
1689     }
1690 }
1691 
getPanelName()1692 const std::string& ExynosPrimaryDisplay::getPanelName() {
1693     if (!mPanelName.empty()) {
1694         return mPanelName;
1695     }
1696 
1697     const std::string& sysfs = getPanelSysfsPath();
1698     if (!sysfs.empty()) {
1699         std::string sysfs_rel("panel_name");
1700         if (readLineFromFile(sysfs + "/" + sysfs_rel, mPanelName, '\n') != OK) {
1701             ALOGE("failed reading %s/%s", sysfs.c_str(), sysfs_rel.c_str());
1702         }
1703     }
1704     return mPanelName;
1705 }
1706 
registerRefreshRateChangeListener(std::shared_ptr<RefreshRateChangeListener> listener)1707 int32_t ExynosPrimaryDisplay::registerRefreshRateChangeListener(
1708         std::shared_ptr<RefreshRateChangeListener> listener) {
1709     if (!mXrrSettings.versionInfo.hasVrrController()) return HWC2_ERROR_UNSUPPORTED;
1710 
1711     if (mVariableRefreshRateController) {
1712         mVariableRefreshRateController->registerRefreshRateChangeListener(listener);
1713         return NO_ERROR;
1714     } else {
1715         return -EINVAL;
1716     }
1717 }
1718