/* * Copyright (C) 2012 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "ExynosExternalDisplay.h" #include #include #include #include "ExynosDevice.h" #include "ExynosDisplayDrmInterface.h" #include "ExynosDisplayDrmInterfaceModule.h" #include "ExynosHWCDebug.h" #include "ExynosHWCHelper.h" #include "ExynosLayer.h" #include "pixelstats-display.h" #define SKIP_FRAME_COUNT 3 extern struct exynos_hwc_control exynosHWCControl; using namespace SOC_VERSION; ExynosExternalDisplay::ExynosExternalDisplay(uint32_t index, ExynosDevice* device, const std::string& displayName) : ExynosDisplay(HWC_DISPLAY_EXTERNAL, index, device, displayName) { DISPLAY_LOGD(eDebugExternalDisplay, ""); mEnabled = false; mBlanked = false; mXres = 0; mYres = 0; mXdpi = 0; mYdpi = 0; mVsyncPeriod = 0; mSkipStartFrame = 0; mSkipFrameCount = -1; mIsSkipFrame = false; mVirtualDisplayState = 0; mDRDefault = true; mDREnable = false; //TODO : Hard coded currently mNumMaxPriorityAllowed = 1; mPowerModeState = (hwc2_power_mode_t)HWC_POWER_MODE_OFF; } ExynosExternalDisplay::~ExynosExternalDisplay() { } void ExynosExternalDisplay::init() { } void ExynosExternalDisplay::deInit() { } int ExynosExternalDisplay::openExternalDisplay() { DISPLAY_LOGD(eDebugExternalDisplay, ""); int ret = 0; mSkipFrameCount = SKIP_FRAME_COUNT; mSkipStartFrame = 0; mPlugState = true; if (mLayers.size() != 0) { mLayers.clear(); } DISPLAY_LOGD(eDebugExternalDisplay, "open fd for External Display(%d)", ret); return ret; } void ExynosExternalDisplay::closeExternalDisplay() { DISPLAY_LOGD(eDebugExternalDisplay, ""); setVsyncEnabledInternal(HWC2_VSYNC_DISABLE); if (mPowerModeState.has_value() && (*mPowerModeState != (hwc2_power_mode_t)HWC_POWER_MODE_OFF)) { if (mDisplayInterface->setPowerMode(HWC_POWER_MODE_OFF) < 0) { DISPLAY_LOGE("%s: set powermode ioctl failed errno : %d", __func__, errno); return; } } mPowerModeState = (hwc2_power_mode_t)HWC_POWER_MODE_OFF; DISPLAY_LOGD(eDebugExternalDisplay, "Close fd for External Display"); mPlugState = false; mEnabled = false; mBlanked = false; mSkipFrameCount = SKIP_FRAME_COUNT; for (size_t i = 0; i < mLayers.size(); i++) { ExynosLayer *layer = mLayers[i]; layer->mAcquireFence = fence_close(layer->mAcquireFence, this, FENCE_TYPE_SRC_ACQUIRE, FENCE_IP_LAYER); layer->mReleaseFence = -1; layer->mLayerBuffer = NULL; } mClientCompositionInfo.initializeInfosComplete(this); mExynosCompositionInfo.initializeInfosComplete(this); } int ExynosExternalDisplay::getDisplayConfigs(uint32_t* outNumConfigs, hwc2_config_t* outConfigs) { DISPLAY_LOGD(eDebugExternalDisplay, ""); int32_t ret = mDisplayInterface->getDisplayConfigs(outNumConfigs, outConfigs); if (ret) DISPLAY_LOGE("%s: failed to getDisplayConfigs, ret(%d)", __func__, ret); if (outConfigs) { char modeStr[PROPERTY_VALUE_MAX] = "\0"; int32_t width, height, fps, config; int32_t err = HWC2_ERROR_BAD_CONFIG; if (property_get("vendor.display.external.preferred_mode", modeStr, "") > 0) { if (sscanf(modeStr, "%dx%d@%d", &width, &height, &fps) == 3) { err = lookupDisplayConfigs(width, height, fps, fps, &config); if (err != HWC2_ERROR_NONE) { DISPLAY_LOGW("%s: display does not support preferred mode %dx%d@%d", __func__, width, height, fps); } } else { DISPLAY_LOGW("%s: vendor.display.external.preferred_mode: bad format", __func__); } } if (err == HWC2_ERROR_NONE) { mActiveConfig = config; } else { err = lookupDisplayConfigsRelaxed(1920, 1080, 60, &config); if (err == HWC2_ERROR_NONE) { mActiveConfig = config; } else { mActiveConfig = outConfigs[0]; } } displayConfigs_t displayConfig = mDisplayConfigs[mActiveConfig]; mXres = displayConfig.width; mYres = displayConfig.height; mVsyncPeriod = displayConfig.vsyncPeriod; mRefreshRate = displayConfig.refreshRate; if (mDisplayInterface->mType == INTERFACE_TYPE_DRM) { ret = mDisplayInterface->setActiveConfig(mActiveConfig); if (ret) { DISPLAY_LOGE("%s: failed to setActiveConfigs, ret(%d)", __func__, ret); return ret; } } } return ret; } int32_t ExynosExternalDisplay::getActiveConfig(hwc2_config_t* outConfig) { DISPLAY_LOGD(eDebugExternalDisplay, ""); if (!mHpdStatus) return HWC2_ERROR_BAD_DISPLAY; if (mActiveConfig == UINT_MAX) return HWC2_ERROR_BAD_CONFIG; *outConfig = mActiveConfig; return HWC2_ERROR_NONE; } bool ExynosExternalDisplay::handleRotate() { // FIXME: HWC2_COMPOSITION_SCREENSHOT is not dfeind in AOSP // HWC guys should fix this. if (mSkipStartFrame < SKIP_EXTERNAL_FRAME) { #if 0 for (size_t i = 0; i < mLayers.size(); i++) { ExynosLayer *layer = mLayers[i]; if (layer->mCompositionType == HWC2_COMPOSITION_SCREENSHOT) layer->mCompositionType = HWC2_COMPOSITION_DEVICE; } #endif mIsSkipFrame = false; return false; } #if 0 for (size_t i = 0; i < mLayers.size(); i++) { ExynosLayer *layer = mLayers[i]; if (layer->mCompositionType == HWC2_COMPOSITION_SCREENSHOT) { DISPLAY_LOGD(eDebugExternalDisplay, "include rotation animation layer"); layer->mOverlayInfo = eSkipRotateAnim; for (size_t j = 0; j < mLayers.size(); j++) { ExynosLayer *skipLayer = mLayers[j]; skipLayer->mValidateCompositionType = HWC2_COMPOSITION_DEVICE; } mIsSkipFrame = true; return true; } } #endif mIsSkipFrame = false; return false; } bool ExynosExternalDisplay::checkRotate() { // FIXME: HWC2_COMPOSITION_SCREENSHOT is not dfeind in AOSP // HWC guys should fix this. #if 0 for (size_t i = 0; i < mLayers.size(); i++) { ExynosLayer *layer = mLayers[i]; if (layer->mCompositionType == HWC2_COMPOSITION_SCREENSHOT) { return true; } } #endif return false; } int32_t ExynosExternalDisplay::validateDisplay( uint32_t* outNumTypes, uint32_t* outNumRequests) { Mutex::Autolock lock(mExternalMutex); DISPLAY_LOGD(eDebugExternalDisplay, ""); int32_t ret; mSkipFrame = false; if (mSkipStartFrame < SKIP_EXTERNAL_FRAME) { ALOGI("[ExternalDisplay] %s : setGeometryChanged [%d/%d]", __func__, mSkipStartFrame, SKIP_EXTERNAL_FRAME); initDisplay(); /* * geometry should be set before ExynosDisplay::validateDisplay is called * not to skip resource assignment */ if (mPlugState) setGeometryChanged(GEOMETRY_DEVICE_DISPLAY_ADDED); else setGeometryChanged(GEOMETRY_DEVICE_DISPLAY_REMOVED); } if (handleRotate()) { if ((ret = mResourceManager->initResourcesState(this)) != NO_ERROR) DISPLAY_LOGE("[ExternalDisplay] %s : initResourcesState fail, ret(%d)", __func__, ret); mDevice->setGeometryChanged(GEOMETRY_LAYER_UNKNOWN_CHANGED); mClientCompositionInfo.initializeInfos(this); mExynosCompositionInfo.initializeInfos(this); mRenderingState = RENDERING_STATE_VALIDATED; return HWC2_ERROR_NONE; } if (mSkipStartFrame < SKIP_EXTERNAL_FRAME) { /* * Set mIsSkipFrame before calling ExynosDisplay::validateDisplay() * startPostProcessing() that is called by ExynosDisplay::validateDisplay() * checks mIsSkipFrame. */ mIsSkipFrame = true; } ret = ExynosDisplay::validateDisplay(outNumTypes, outNumRequests); if (mSkipStartFrame < SKIP_EXTERNAL_FRAME) { initDisplay(); mRenderingState = RENDERING_STATE_VALIDATED; uint32_t changed_count = 0; for (size_t i = 0; i < mLayers.size(); i++) { ExynosLayer *layer = mLayers[i]; if (layer && (layer->mValidateCompositionType == HWC2_COMPOSITION_DEVICE || layer->mValidateCompositionType == HWC2_COMPOSITION_EXYNOS)) { layer->mValidateCompositionType = HWC2_COMPOSITION_CLIENT; layer->mReleaseFence = layer->mAcquireFence; changed_count++; } } mSkipStartFrame++; *outNumTypes += changed_count; ALOGI("[ExternalDisplay] %s : Skip start frame [%d/%d]", __func__, mSkipStartFrame, SKIP_EXTERNAL_FRAME); } return ret; } int32_t ExynosExternalDisplay::canSkipValidate() { /* * SurfaceFlinger may call vadlidate, present for a few frame * even though external display is disconnected. * Cammands for primary display can be discarded if validate is skipped * in this case. HWC should return error not to skip validate. */ if ((mHpdStatus == false) || (mBlanked == true)) return SKIP_ERR_DISP_NOT_CONNECTED; if ((mSkipStartFrame > (SKIP_EXTERNAL_FRAME - 1)) && (mEnabled == false) && (mPowerModeState.has_value() && (*mPowerModeState == (hwc2_power_mode_t)HWC_POWER_MODE_NORMAL))) return SKIP_ERR_DISP_NOT_POWER_ON; if (checkRotate() || (mIsSkipFrame) || (mSkipStartFrame < SKIP_EXTERNAL_FRAME)) return SKIP_ERR_FORCE_VALIDATE; return ExynosDisplay::canSkipValidate(); } int32_t ExynosExternalDisplay::presentDisplay( int32_t* outRetireFence) { Mutex::Autolock lock(mExternalMutex); DISPLAY_LOGD(eDebugExternalDisplay, ""); int32_t ret; if (mSkipFrame) { ALOGI("[%d] presentDisplay is skipped by mSkipFrame", mDisplayId); closeFencesForSkipFrame(RENDERING_STATE_PRESENTED); setGeometryChanged(GEOMETRY_DISPLAY_FORCE_VALIDATE); *outRetireFence = -1; for (size_t i=0; i < mLayers.size(); i++) { mLayers[i]->mReleaseFence = -1; } if (mRenderingState == RENDERING_STATE_NONE) { ALOGD("\tThis is the first frame after power on"); ret = HWC2_ERROR_NONE; } else { ALOGD("\tThis is the second frame after power on"); ret = HWC2_ERROR_NOT_VALIDATED; } mRenderingState = RENDERING_STATE_PRESENTED; mDevice->onRefresh(mDisplayId); return ret; } if ((mIsSkipFrame) || (mHpdStatus == false) || (mBlanked == true)) { if ((exynosHWCControl.skipValidate == true) && ((mRenderingState == RENDERING_STATE_PRESENTED) || (mRenderingState == RENDERING_STATE_NONE))) { if (mDevice->canSkipValidate() == false) { mRenderingState = RENDERING_STATE_NONE; return HWC2_ERROR_NOT_VALIDATED; } else { DISPLAY_LOGD(eDebugSkipValidate, "validate is skipped"); } } *outRetireFence = -1; for (size_t i = 0; i < mLayers.size(); i++) { ExynosLayer *layer = mLayers[i]; layer->mAcquireFence = fence_close(layer->mAcquireFence, this, FENCE_TYPE_SRC_ACQUIRE, FENCE_IP_LAYER); layer->mReleaseFence = -1; } mClientCompositionInfo.mAcquireFence = fence_close(mClientCompositionInfo.mAcquireFence, this, FENCE_TYPE_SRC_ACQUIRE, FENCE_IP_FB); mClientCompositionInfo.mReleaseFence = -1; /* this frame is not presented, but mRenderingState is updated to RENDERING_STATE_PRESENTED */ initDisplay(); /* * Resource assignment information was initialized during skipping frames * So resource assignment for the first displayed frame after skpping frames * should not be skipped */ setGeometryChanged(GEOMETRY_DISPLAY_FORCE_VALIDATE); mDevice->onRefresh(mDisplayId); return HWC2_ERROR_NONE; } ret = ExynosDisplay::presentDisplay(outRetireFence); return ret; } int32_t ExynosExternalDisplay::setClientTarget( buffer_handle_t target, int32_t acquireFence, int32_t /*android_dataspace_t*/ dataspace) { buffer_handle_t handle = NULL; if (target != NULL) handle = target; if ((mClientCompositionInfo.mHasCompositionLayer == true) && (handle == NULL) && (mClientCompositionInfo.mSkipFlag == false)) { /* * openExternalDisplay() can be called between validateDisplay and getChangedCompositionTypes. * Then getChangedCompositionTypes() returns no layer because openExternalDisplay() clears mLayers. * SurfaceFlinger might not change compositionType to HWC2_COMPOSITION_CLIENT. * Handle can be NULL in this case. It is not error case. */ if (mSkipStartFrame == 0) { if (acquireFence >= 0) fence_close(acquireFence, this, FENCE_TYPE_SRC_ACQUIRE, FENCE_IP_FB); acquireFence = -1; mClientCompositionInfo.setTargetBuffer(this, handle, acquireFence, (android_dataspace)dataspace); return NO_ERROR; } } return ExynosDisplay::setClientTarget(target, acquireFence, dataspace); } int ExynosExternalDisplay::enable() { ALOGI("[ExternalDisplay] %s +", __func__); if (mEnabled) return HWC2_ERROR_NONE; if (mHpdStatus == false) { ALOGI("HPD is not connected"); return HWC2_ERROR_NONE; } if (openExternalDisplay() < 0) return HWC2_ERROR_UNSUPPORTED; if (mDisplayInterface->setPowerMode(HWC_POWER_MODE_NORMAL) < 0) { DISPLAY_LOGE("set powermode ioctl failed errno : %d", errno); return HWC2_ERROR_UNSUPPORTED; } mEnabled = true; mPowerModeState = (hwc2_power_mode_t)HWC_POWER_MODE_NORMAL; reportUsage(true); ALOGI("[ExternalDisplay] %s -", __func__); return HWC2_ERROR_NONE; } int ExynosExternalDisplay::disable() { ALOGI("[ExternalDisplay] %s +", __func__); if (mHpdStatus) { /* * DP cable is connected and link is up * * Currently, we don't power down here for two reasons: * - power up would require DP link re-training (slow) * - DP audio can continue playing while display is blank */ if (mEnabled) clearDisplay(false); return HWC2_ERROR_NONE; } if (mSkipStartFrame > (SKIP_EXTERNAL_FRAME - 1)) { clearDisplay(true); } else { ALOGI("Skip clearDisplay to avoid resource conflict"); } if (mDisplayInterface->setPowerMode(HWC_POWER_MODE_OFF) < 0) { DISPLAY_LOGE("set powermode ioctl failed errno : %d", errno); return HWC2_ERROR_UNSUPPORTED; } if (mEnabled) reportUsage(false); mEnabled = false; mPowerModeState = (hwc2_power_mode_t)HWC_POWER_MODE_OFF; ALOGI("[ExternalDisplay] %s -", __func__); return HWC2_ERROR_NONE; } int32_t ExynosExternalDisplay::setPowerMode( int32_t /*hwc2_power_mode_t*/ mode) { Mutex::Autolock lock(mExternalMutex); { Mutex::Autolock lock(mDisplayMutex); /* TODO state check routine should be added */ int fb_blank = 0; int err = 0; if (mode == HWC_POWER_MODE_OFF) { fb_blank = FB_BLANK_POWERDOWN; err = disable(); } else if (mode == HWC_POWER_MODE_NORMAL) { fb_blank = FB_BLANK_UNBLANK; err = enable(); } else { DISPLAY_LOGE("unsupported powermode: %d", mode); return HWC2_ERROR_UNSUPPORTED; } if (err != 0) { DISPLAY_LOGE("set powermode ioctl failed errno : %d", errno); return HWC2_ERROR_UNSUPPORTED; } if (fb_blank == FB_BLANK_POWERDOWN) mDREnable = false; else if (fb_blank == FB_BLANK_UNBLANK) mDREnable = mDRDefault; // check the dynamic recomposition thread by following display power status mDevice->checkDynamicRecompositionThread(); DISPLAY_LOGD(eDebugExternalDisplay, "%s:: mode(%d), blank(%d)", __func__, mode, fb_blank); if (mode == HWC_POWER_MODE_OFF) { /* It should be called from validate() when the screen is on */ mSkipFrame = true; setGeometryChanged(GEOMETRY_DISPLAY_POWER_OFF); if ((mRenderingState >= RENDERING_STATE_VALIDATED) && (mRenderingState < RENDERING_STATE_PRESENTED)) closeFencesForSkipFrame(RENDERING_STATE_VALIDATED); mRenderingState = RENDERING_STATE_NONE; } else { setGeometryChanged(GEOMETRY_DISPLAY_POWER_ON); } } return HWC2_ERROR_NONE; } int32_t ExynosExternalDisplay::startPostProcessing() { if ((mHpdStatus == false) || (mBlanked == true) || mIsSkipFrame) { ALOGI("%s:: skip startPostProcessing display(%d) mHpdStatus(%d)", __func__, mDisplayId, mHpdStatus); return NO_ERROR; } return ExynosDisplay::startPostProcessing(); } bool ExynosExternalDisplay::getHDRException(ExynosLayer* __unused layer) { bool ret = false; if (mExternalHdrSupported) { ret = true; } return ret; } void ExynosExternalDisplay::handleHotplugEvent(bool hpdStatus) { mHpdStatus = hpdStatus; if (mHpdStatus) { if (openExternalDisplay() < 0) { ALOGE("Failed to openExternalDisplay"); mHpdStatus = false; return; } mDREnable = mDRDefault; } else { disable(); closeExternalDisplay(); mDREnable = false; } mDevice->checkDynamicRecompositionThread(); ALOGI("HPD status changed to %s, mDisplayId %d, mDisplayFd %d", mHpdStatus ? "enabled" : "disabled", mDisplayId, mDisplayInterface->getDisplayFd()); } void ExynosExternalDisplay::initDisplayInterface(uint32_t interfaceType) { if (interfaceType == INTERFACE_TYPE_DRM) mDisplayInterface = std::make_unique((ExynosDisplay *)this); else LOG_ALWAYS_FATAL("%s::Unknown interface type(%d)", __func__, interfaceType); mDisplayInterface->init(this); } void ExynosExternalDisplay::reportUsage(bool enabled) { auto refreshRate = nanoSec2Hz(mVsyncPeriod); auto manufacturerInfo = mDisplayInterface->getManufacturerInfo(); auto productId = mDisplayInterface->getProductId(); ATRACE_NAME("report ext usage"); reportDisplayPortUsage(mXres, mYres, refreshRate, manufacturerInfo, productId, enabled); }