* 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 <chrono>
#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<std::thread>([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<std::mutex> lg(mMutex);
mRunning = false;
}
DISPLAY_LOGD();
mCondition.notify_one();
if (mThread != nullptr) {
mThread->join();
}
DISPLAY_LOGD();
}
bool DrmVsyncWorker::WaitSignalAndCheckRuning()
{
std::unique_lock<std::mutex> 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,
}
};
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<std::mutex> lg(mMutex);
mEnable = enable;
}
mCondition.notify_one();
}
void DrmVsyncWorker::WorkThread()
{
DISPLAY_LOGD();
unsigned int seq = 0;
while (WaitSignalAndCheckRuning()) {
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<VsyncCallBack> &cb)
{
DISPLAY_LOGD();
DISPLAY_CHK_RETURN_NOT_VALUE((cb == nullptr), DISPLAY_LOGE("the VBlankCallback is nullptr "));
mCallBack = cb;
}
}
}
}