* 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 "hdi_layer.h"
#include <cerrno>
#include <fstream>
#include <libsync.h>
#include <securec.h>
#include <sstream>
#include <string>
#include <sys/time.h>
#include <unistd.h>
#include "display_buffer_vdi_impl.h"
#include "v1_0/display_composer_type.h"
namespace OHOS {
namespace HDI {
namespace DISPLAY {
uint32_t HdiLayer::mIdleId = 0;
std::unordered_set<uint32_t> HdiLayer::mIdSets;
std::shared_ptr<IDisplayBufferVdi> g_buffer;
constexpr int TIME_BUFFER_MAX_LEN = 15;
const std::string PATH_PREFIX = "/data/local/traces/";
HdiLayerBuffer::HdiLayerBuffer(const BufferHandle &hdl)
: mPhyAddr(hdl.phyAddr), mHeight(hdl.height), mWidth(hdl.width), mStride(hdl.stride), mFormat(hdl.format)
{
DISPLAY_LOGD();
mFd = dup(hdl.fd);
mHandle = hdl;
if (mFd < 0) {
DISPLAY_LOGE("the fd : %{public}d dup failed errno %{public}d", hdl.fd, errno);
}
}
HdiLayerBuffer::~HdiLayerBuffer()
{
DISPLAY_LOGD();
if (mFd >= 0) {
close(mFd);
}
}
HdiLayerBuffer &HdiLayerBuffer::operator = (const BufferHandle &right)
{
DISPLAY_LOGD();
if (mFd >= 0) {
close(mFd);
}
mFd = dup(right.fd);
mPhyAddr = right.phyAddr;
mWidth = right.width;
mHeight = right.height;
mStride = right.stride;
mFormat = right.format;
return *this;
}
uint32_t HdiLayer::GetIdleId()
{
const uint32_t oldIdleId = mIdleId;
uint32_t id = INVALIDE_LAYER_ID;
mIdleId = mIdleId % INVALIDE_LAYER_ID;
do {
auto iter = mIdSets.find(mIdleId);
if (iter == mIdSets.end()) {
id = mIdleId;
break;
}
mIdleId = (mIdleId + 1) % INVALIDE_LAYER_ID;
} while (oldIdleId != mIdleId);
mIdSets.emplace(id);
mIdleId++;
DISPLAY_LOGD("id %{public}d mIdleId %{public}d", id, mIdleId);
return id;
}
int32_t HdiLayer::Init()
{
DISPLAY_LOGD();
uint32_t id = GetIdleId();
DISPLAY_CHK_RETURN((id == INVALIDE_LAYER_ID), DISPLAY_FAILURE, DISPLAY_LOGE("have no id to used"));
mId = id;
return DISPLAY_SUCCESS;
}
int32_t HdiLayer::SetLayerRegion(IRect *rect)
{
DISPLAY_CHK_RETURN((rect == nullptr), DISPLAY_NULL_PTR, DISPLAY_LOGE("in rect is nullptr"));
DISPLAY_LOGD(" displayRect x: %{public}d y : %{public}d w : %{public}d h : %{public}d", rect->x, rect->y,
rect->w, rect->h);
mDisplayRect = *rect;
return DISPLAY_SUCCESS;
}
int32_t HdiLayer::SetLayerCrop(IRect *rect)
{
DISPLAY_CHK_RETURN((rect == nullptr), DISPLAY_NULL_PTR, DISPLAY_LOGE("in rect is nullptr"));
DISPLAY_LOGD("id : %{public}d crop x: %{public}d y : %{public}d w : %{public}d h : %{public}d", mId,
rect->x, rect->y, rect->w, rect->h);
mCrop = *rect;
return DISPLAY_SUCCESS;
}
void HdiLayer::SetLayerZorder(uint32_t zorder)
{
DISPLAY_LOGD("id : %{public}d zorder : %{public}d ", mId, zorder);
mZorder = zorder;
}
int32_t HdiLayer::SetLayerPreMulti(bool preMul)
{
DISPLAY_LOGD();
mPreMul = preMul;
return DISPLAY_SUCCESS;
}
int32_t HdiLayer::SetLayerAlpha(LayerAlpha *alpha)
{
DISPLAY_CHK_RETURN((alpha == nullptr), DISPLAY_NULL_PTR, DISPLAY_LOGE("in alpha is nullptr"));
DISPLAY_LOGD("enable alpha %{public}d galpha 0x%{public}x", alpha->enGlobalAlpha, alpha->gAlpha);
mAlpha = *alpha;
return DISPLAY_SUCCESS;
}
int32_t HdiLayer::SetLayerTransformMode(TransformType type)
{
DISPLAY_LOGD("TransformType %{public}d", type);
mTransformType = type;
return DISPLAY_SUCCESS;
}
int32_t HdiLayer::SetLayerDirtyRegion(IRect *region)
{
DISPLAY_CHK_RETURN((region == nullptr), DISPLAY_FAILURE, DISPLAY_LOGE("the in rect is null"));
DISPLAY_LOGD("id : %{public}d DirtyRegion x: %{public}d y : %{public}d w : %{public}d h : %{public}d", mId,
region->x, region->y, region->w, region->h);
return DISPLAY_SUCCESS;
}
int32_t HdiLayer::SetLayerVisibleRegion(uint32_t num, IRect *rect)
{
DISPLAY_LOGD("id : %{public}d DirtyRegion x: %{public}d y : %{public}d w : %{public}d h : %{public}d", mId,
rect->x, rect->y, rect->w, rect->h);
return DISPLAY_SUCCESS;
}
static std::string GetFileName(const BufferHandle *buffer)
{
struct timeval tv;
char nowStr[TIME_BUFFER_MAX_LEN] = {0};
gettimeofday(&tv, nullptr);
if (strftime(nowStr, sizeof(nowStr), "%m-%d-%H-%M-%S", localtime(&tv.tv_sec)) == 0) {
DISPLAY_LOGE("strftime failed");
return "";
};
std::ostringstream strStream;
strStream << "hdi_layer_" << nowStr << "_" << tv.tv_usec << "_" << buffer->width << "_" << buffer->height << ".img";
return strStream.str();
}
static int32_t DumpLayerBuffer(BufferHandle *buffer)
{
CHECK_NULLPOINTER_RETURN_VALUE(buffer, DISPLAY_NULL_PTR);
int32_t ret = 0;
if (g_buffer== nullptr) {
IDisplayBufferVdi* dispBuf = new DisplayBufferVdiImpl();
DISPLAY_CHK_RETURN((dispBuf == nullptr), DISPLAY_FAILURE, DISPLAY_LOGE("dispBuf init failed"));
g_buffer.reset(dispBuf);
}
std::string fileName = GetFileName(buffer);
DISPLAY_CHK_RETURN((fileName == ""), DISPLAY_FAILURE, DISPLAY_LOGE("GetFileName failed"));
DISPLAY_LOGI("fileName = %{public}s", fileName.c_str());
std::stringstream filePath;
filePath << PATH_PREFIX << fileName;
std::ofstream rawDataFile(filePath.str(), std::ofstream::binary);
DISPLAY_CHK_RETURN((!rawDataFile.good()), DISPLAY_FAILURE, DISPLAY_LOGE("open file failed, %{public}s",
std::strerror(errno)));
void *buffAddr = g_buffer->Mmap(*buffer);
DISPLAY_CHK_RETURN((buffAddr == nullptr), DISPLAY_FAILURE, DISPLAY_LOGE("Mmap buffer failed"));
rawDataFile.write(static_cast<const char *>(buffAddr), buffer->size);
rawDataFile.close();
ret = g_buffer->Unmap(*buffer);
DISPLAY_CHK_RETURN((ret != DISPLAY_SUCCESS), DISPLAY_FAILURE, DISPLAY_LOGE("Unmap buffer failed"));
return DISPLAY_SUCCESS;
}
int32_t HdiLayer::SetLayerBuffer(const BufferHandle *buffer, int32_t fence)
{
DISPLAY_LOGD();
DISPLAY_CHK_RETURN((buffer == nullptr), DISPLAY_NULL_PTR, DISPLAY_LOGE("buffer is nullptr"));
std::unique_ptr<HdiLayerBuffer> layerbuffer = std::make_unique<HdiLayerBuffer>(*buffer);
mHdiBuffer = std::move(layerbuffer);
mAcquireFence = dup(fence);
if (access("/data/hdi_dump_layer", F_OK) != -1) {
if (DumpLayerBuffer(const_cast<BufferHandle *>(buffer)) != DISPLAY_SUCCESS) {
DISPLAY_LOGE("dump layer buffer failed");
}
}
return DISPLAY_SUCCESS;
}
int32_t HdiLayer::SetLayerCompositionType(CompositionType type)
{
DISPLAY_LOGD("CompositionType type %{public}d", type);
mCompositionType = type;
return DISPLAY_SUCCESS;
}
int32_t HdiLayer::SetLayerBlendType(BlendType type)
{
DISPLAY_LOGD("BlendType type %{public}d", type);
mBlendType = type;
return DISPLAY_SUCCESS;
}
void HdiLayer::SetPixel(const BufferHandle &handle, int x, int y, uint32_t color)
{
const int32_t pixelBytes = 4;
DISPLAY_CHK_RETURN_NOT_VALUE((handle.format <= 0),
DISPLAY_LOGE("CheckPixel do not support format %{public}d", handle.format));
DISPLAY_CHK_RETURN_NOT_VALUE((handle.virAddr == nullptr), DISPLAY_LOGE("CheckPixel viraddr is null must map it"));
DISPLAY_CHK_RETURN_NOT_VALUE((x < 0 || x >= handle.width),
DISPLAY_LOGE("CheckPixel invalid parameter x:%{public}d width:%{public}d", x, handle.width));
DISPLAY_CHK_RETURN_NOT_VALUE((y < 0 || y >= handle.height),
DISPLAY_LOGE("CheckPixel invalid parameter y:%{public}d height:%{public}d", y, handle.height));
int32_t position = y * handle.width + x;
if ((position * pixelBytes) > handle.size) {
DISPLAY_LOGE("the pixel postion outside\n");
}
uint32_t *pixel = reinterpret_cast<uint32_t *>(handle.virAddr) + position;
*pixel = color;
}
void HdiLayer::ClearColor(uint32_t color)
{
DISPLAY_LOGD();
BufferHandle &handle = mHdiBuffer->mHandle;
for (int32_t x = 0; x < handle.width; x++) {
for (int32_t y = 0; y < handle.height; y++) {
SetPixel(handle, x, y, color);
}
}
}
void HdiLayer::WaitAcquireFence()
{
int fd = GetAcquireFenceFd();
if (fd < 0) {
DISPLAY_LOGE("fd is invalid");
return;
}
sync_wait(fd, mFenceTimeOut);
}
}
}
}