/* * Copyright (c) 2021-2023 Huawei Device Co., Ltd. * 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 "drm_vsync_worker.h" #include #include "display_log.h" #include "drm_device.h" namespace OHOS { namespace HDI { namespace DISPLAY { DrmVsyncWorker::DrmVsyncWorker() {} int32_t DrmVsyncWorker::Init(int fd) { DISPLAY_CHK_RETURN((fd < 0), DISPLAY_FAILURE, DISPLAY_LOGE("the fd is invalid")); mDrmFd = fd; DISPLAY_LOGD("the drm fd is %{public}d", fd); mThread = std::make_unique([this]() { WorkThread(); }); DISPLAY_CHK_RETURN((mThread == nullptr), DISPLAY_FAILURE, DISPLAY_LOGE("can not create thread")); mRunning = true; return DISPLAY_SUCCESS; } DrmVsyncWorker &DrmVsyncWorker::GetInstance() { static DrmVsyncWorker instance; static std::once_flag once; std::call_once(once, [&]() { int ret = instance.Init(DrmDevice::GetDrmFd()); if (ret != DISPLAY_SUCCESS) { DISPLAY_LOGE("Vsync Worker Init failed"); } }); return instance; } DrmVsyncWorker::~DrmVsyncWorker() { DISPLAY_LOGD(); { std::lock_guard lg(mMutex); mRunning = false; } DISPLAY_LOGD(); mCondition.notify_one(); if (mThread != nullptr) { mThread->join(); } DISPLAY_LOGD(); } bool DrmVsyncWorker::WaitSignalAndCheckRuning() { std::unique_lock ul(mMutex); mCondition.wait(ul, [this]() { return (mEnable || !mRunning); }); return mRunning; } uint64_t DrmVsyncWorker::WaitNextVBlank(unsigned int &sq) { constexpr uint64_t SEC_TO_NSEC = 1000 * 1000 * 1000; constexpr uint64_t USEC_TO_NSEC = 1000; drmVBlank vblank = { .request = drmVBlankReq { .type = DRM_VBLANK_RELATIVE, .sequence = 1, .signal = 0, } }; /* The drmWaitVBlank need set the crtc pipe when there are multi crtcs in the system. */ if (mCallBack->GetPipe() == 1) vblank.request.type = drmVBlankSeqType((int)(vblank.request.type) | (int)DRM_VBLANK_SECONDARY); int ret = drmWaitVBlank(mDrmFd, &vblank); DISPLAY_CHK_RETURN((ret < 0), 0, DISPLAY_LOGE("wait vblank failed ret : %{public}d errno %{public}d", ret, errno)); sq = vblank.reply.sequence; return (uint64_t)(vblank.reply.tval_sec * SEC_TO_NSEC + vblank.reply.tval_usec * USEC_TO_NSEC); } void DrmVsyncWorker::EnableVsync(bool enable) { DISPLAY_LOGD(); { std::lock_guard lg(mMutex); mEnable = enable; } mCondition.notify_one(); } void DrmVsyncWorker::WorkThread() { DISPLAY_LOGD(); unsigned int seq = 0; while (WaitSignalAndCheckRuning()) { // wait the vblank uint64_t time = WaitNextVBlank(seq); if (mCallBack != nullptr) { mCallBack->Vsync(seq, time); } else { DISPLAY_LOGE("the callbac is nullptr"); } } } void DrmVsyncWorker::ReqesterVBlankCb(std::shared_ptr &cb) { DISPLAY_LOGD(); DISPLAY_CHK_RETURN_NOT_VALUE((cb == nullptr), DISPLAY_LOGE("the VBlankCallback is nullptr ")); mCallBack = cb; } } // namespace OHOS } // namespace HDI } // namespace DISPLAY