blob: 0401a87e4f3124bf4f443f173a291aa75b8ca100 [file] [log] [blame]
/*
* Copyright 2023 Google LLC
* SPDX-License-Identifier: MIT
*/
#include "GrallocMinigbm.h"
#include <cros_gralloc/cros_gralloc_handle.h>
#include <stdlib.h>
#include <sys/user.h>
#include <unistd.h>
#include <vndk/hardware_buffer.h>
#include <cinttypes>
#include <cstring>
#include "VirtGpu.h"
#include "util/log.h"
namespace gfxstream {
MinigbmGralloc::MinigbmGralloc(int32_t descriptor) {
mDevice.reset(createPlatformVirtGpuDevice(kCapsetNone, descriptor));
}
GrallocType MinigbmGralloc::getGrallocType() { return GRALLOC_TYPE_MINIGBM; }
uint32_t MinigbmGralloc::createColorBuffer(int width, int height, uint32_t glformat) {
// Only supported format for pbuffers in gfxstream should be RGBA8
const uint32_t kVirglFormatRGBA = 67; // VIRGL_FORMAT_R8G8B8A8_UNORM;
uint32_t virtgpu_format = 0;
uint32_t bpp = 0;
switch (glformat) {
case kGlRGB:
mesa_logi("Note: egl wanted GL_RGB, still using RGBA");
virtgpu_format = kVirglFormatRGBA;
bpp = 4;
break;
case kGlRGBA:
virtgpu_format = kVirglFormatRGBA;
bpp = 4;
break;
default:
mesa_logi("Note: egl wanted 0x%x, still using RGBA", glformat);
virtgpu_format = kVirglFormatRGBA;
bpp = 4;
break;
}
uint32_t stride = bpp * width;
auto resource = mDevice->createResource(width, height, stride, stride * height, virtgpu_format,
PIPE_TEXTURE_2D, VIRGL_BIND_RENDER_TARGET);
uint32_t handle = resource->getResourceHandle();
resource->intoRaw();
return handle;
}
int MinigbmGralloc::allocate(uint32_t width, uint32_t height, uint32_t format, uint64_t usage,
AHardwareBuffer** outputAhb) {
struct AHardwareBuffer_Desc desc = {
.width = width,
.height = height,
.layers = 1,
.format = format,
.usage = usage,
};
return AHardwareBuffer_allocate(&desc, outputAhb);
}
void MinigbmGralloc::acquire(AHardwareBuffer* ahb) { AHardwareBuffer_acquire(ahb); }
void MinigbmGralloc::release(AHardwareBuffer* ahb) { AHardwareBuffer_release(ahb); }
int MinigbmGralloc::lock(AHardwareBuffer* ahb, uint8_t** ptr) {
return AHardwareBuffer_lock(ahb, AHARDWAREBUFFER_USAGE_CPU_READ_RARELY, -1, nullptr,
reinterpret_cast<void**>(ptr));
}
int MinigbmGralloc::lockPlanes(AHardwareBuffer* ahb, std::vector<LockedPlane>* ahbPlanes) {
mesa_loge("%s: unimplemented", __func__);
return -1;
}
int MinigbmGralloc::unlock(AHardwareBuffer* ahb) { return AHardwareBuffer_unlock(ahb, nullptr); }
uint32_t MinigbmGralloc::getHostHandle(const native_handle_t* handle) {
cros_gralloc_handle const* cros_handle = reinterpret_cast<cros_gralloc_handle const*>(handle);
struct VirtGpuExternalHandle hnd = {
.osHandle = dup(cros_handle->fds[0]),
.type = kMemHandleDmabuf,
};
auto resource = mDevice->importBlob(hnd);
if (!resource) {
return 0;
}
if (resource->wait()) {
return 0;
}
return resource->getResourceHandle();
}
uint32_t MinigbmGralloc::getHostHandle(const AHardwareBuffer* ahb) {
const native_handle_t* handle = AHardwareBuffer_getNativeHandle(ahb);
return getHostHandle(handle);
}
const native_handle_t* MinigbmGralloc::getNativeHandle(const AHardwareBuffer* ahb) {
return AHardwareBuffer_getNativeHandle(ahb);
}
int MinigbmGralloc::getFormat(const native_handle_t* handle) {
return ((cros_gralloc_handle*)handle)->droid_format;
}
int MinigbmGralloc::getFormat(const AHardwareBuffer* ahb) {
const native_handle_t* handle = AHardwareBuffer_getNativeHandle(ahb);
return ((cros_gralloc_handle*)handle)->droid_format;
}
uint32_t MinigbmGralloc::getFormatDrmFourcc(const native_handle_t* handle) {
return ((cros_gralloc_handle*)handle)->format;
}
uint32_t MinigbmGralloc::getFormatDrmFourcc(const AHardwareBuffer* ahb) {
const native_handle_t* handle = AHardwareBuffer_getNativeHandle(ahb);
return getFormatDrmFourcc(handle);
}
uint32_t MinigbmGralloc::getWidth(const AHardwareBuffer* ahb) {
AHardwareBuffer_Desc desc = {};
AHardwareBuffer_describe(ahb, &desc);
return desc.width;
}
uint32_t MinigbmGralloc::getHeight(const AHardwareBuffer* ahb) {
AHardwareBuffer_Desc desc = {};
AHardwareBuffer_describe(ahb, &desc);
return desc.height;
}
size_t MinigbmGralloc::getAllocatedSize(const native_handle_t* handle) {
cros_gralloc_handle const* cros_handle = reinterpret_cast<cros_gralloc_handle const*>(handle);
struct VirtGpuExternalHandle hnd = {
.osHandle = dup(cros_handle->fds[0]),
.type = kMemHandleDmabuf,
};
auto resource = mDevice->importBlob(hnd);
if (!resource) {
return 0;
}
if (resource->wait()) {
return 0;
}
return resource->getSize();
}
size_t MinigbmGralloc::getAllocatedSize(const AHardwareBuffer* ahb) {
const native_handle_t* handle = AHardwareBuffer_getNativeHandle(ahb);
return getAllocatedSize(handle);
}
int MinigbmGralloc::getId(const AHardwareBuffer* ahb, uint64_t* id) {
#if ANDROID_API_LEVEL >= 31
return AHardwareBuffer_getId(ahb, id);
#else
(void)ahb;
*id = 0;
return 0;
#endif
}
int32_t MinigbmGralloc::getDataspace(const AHardwareBuffer* ahb) {
#if ANDROID_API_LEVEL >= 34
return AHardwareBuffer_getDataSpace(ahb);
#else
(void)ahb;
return GFXSTREAM_AHB_DATASPACE_UNKNOWN;
#endif
}
} // namespace gfxstream