Skip to content

Commit

Permalink
weird thing to miss
Browse files Browse the repository at this point in the history
  • Loading branch information
SolDev69 committed Nov 29, 2023
1 parent e6ea463 commit 8089471
Showing 1 changed file with 208 additions and 0 deletions.
208 changes: 208 additions & 0 deletions src/freedreno/drm/kgsl_bo.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,208 @@
#include "kgsl_priv.h"
#include "util/os_mman.h"

static uint64_t
kgsl_bo_iova(struct fd_bo *bo)
{
struct kgsl_bo *kgsl_bo = to_kgsl_bo(bo);
return kgsl_bo->iova;
}

static void
kgsl_bo_set_name(struct fd_bo *bo, const char *fmt, va_list ap)
{
/* This function is a no op for KGSL */
return;
}

static int
kgsl_bo_offset(struct fd_bo *bo, uint64_t *offset)
{
/* from tu_kgsl.c - offset for mmap needs to be */
/* the returned alloc id shifted over 12 */
*offset = bo->handle << 12;
return 0;
}

static int
kgsl_bo_madvise(struct fd_bo *bo, int willneed)
{
/* KGSL does not support this, so simply return willneed */
return willneed;
}

static int
kgsl_bo_cpu_prep(struct fd_bo *bo, struct fd_pipe *pipe, uint32_t op)
{
/* only need to handle implicit sync here which is a NOP for KGSL */
return 0;
}

void
kgsl_bo_close_handle(struct fd_device *dev, uint32_t handle)
{
struct kgsl_gpumem_free_id req = {
.id = handle
};

kgsl_pipe_safe_ioctl(dev->fd, IOCTL_KGSL_GPUMEM_FREE_ID, &req);
}

static void
kgsl_bo_destroy(struct fd_bo *bo)
{
/* KGSL will immediately delete the BO when we close
* the handle, so wait on all fences to ensure
* the GPU is done using it before we destory it
*/
for (uint32_t i = 0; i < bo->nr_fences; i++) {
struct fd_pipe *pipe = bo->fences[i]->pipe;
pipe->funcs->wait(pipe, bo->fences[i], ~0);
}

fd_bo_fini_common(bo);
}

static void *
kgsl_bo_map(struct fd_bo *bo)
{
struct kgsl_bo *kgsl_bo = to_kgsl_bo(bo);
if (!bo->map) {
if (kgsl_bo->bo_type == KGSL_BO_IMPORT) {
/* in `fd_bo_map` if it tries to mmap this BO. mmap logic is copied from
* https://android.googlesource.com/platform/hardware/libhardware/+/master/modules/gralloc/mapper.cpp#44
*/
bo->map = os_mmap(0, bo->size, PROT_READ | PROT_WRITE, MAP_SHARED, kgsl_bo->import_fd, 0);
} else {
uint64_t offset;
int ret;

ret = bo->funcs->offset(bo, &offset);
if (ret) {
return NULL;
}

bo->map = os_mmap(0, bo->size, PROT_READ | PROT_WRITE, MAP_SHARED,
bo->dev->fd, offset);
if (bo->map == MAP_FAILED) {
ERROR_MSG("mmap failed: %s", strerror(errno));
bo->map = NULL;
}
}

if (bo->map == MAP_FAILED) {
ERROR_MSG("mmap failed: %s", strerror(errno));
bo->map = NULL;
}
}
return bo->map;
}

static const struct fd_bo_funcs bo_funcs = {
.iova = kgsl_bo_iova,
.set_name = kgsl_bo_set_name,
.offset = kgsl_bo_offset,
.map = kgsl_bo_map,
.madvise = kgsl_bo_madvise,
.cpu_prep = kgsl_bo_cpu_prep,
.destroy = kgsl_bo_destroy,
};

/* Size is not used by KGSL */
struct fd_bo *
kgsl_bo_from_handle(struct fd_device *dev, uint32_t size, uint32_t handle)
{
struct fd_bo *bo;
int ret;
struct kgsl_gpuobj_info req = {
.id = handle,
};

ret = kgsl_pipe_safe_ioctl(dev->fd,
IOCTL_KGSL_GPUOBJ_INFO, &req);

if (ret) {
ERROR_MSG("Failed to get handle info (%s)", strerror(errno));
return NULL;
}

struct kgsl_bo *kgsl_bo = calloc(1, sizeof(*kgsl_bo));
if (!kgsl_bo)
return NULL;

bo = &kgsl_bo->base;
bo->dev = dev;
bo->size = req.size;
bo->handle = req.id;
bo->funcs = &bo_funcs;

kgsl_bo->iova = req.gpuaddr;

fd_bo_init_common(bo, dev);

return bo;
}

struct fd_bo *
kgsl_bo_from_dmabuf(struct fd_device *dev, int fd)
{
struct fd_bo *bo;
struct kgsl_gpuobj_import_dma_buf import_dmabuf = {
.fd = fd,
};
struct kgsl_gpuobj_import req = {
.priv = (uintptr_t)&import_dmabuf,
.priv_len = sizeof(import_dmabuf),
.flags = 0,
.type = KGSL_USER_MEM_TYPE_DMABUF,
};
int ret;

ret = kgsl_pipe_safe_ioctl(dev->fd,
IOCTL_KGSL_GPUOBJ_IMPORT, &req);

if (ret) {
ERROR_MSG("Failed to import dma-buf (%s)", strerror(errno));
return NULL;
}

bo = fd_bo_from_handle(dev, req.id, 0);

struct kgsl_bo *kgsl_bo = to_kgsl_bo(bo);
kgsl_bo->bo_type = KGSL_BO_IMPORT;
kgsl_bo->import_fd = fd;

return bo;
}

struct fd_bo *
kgsl_bo_new(struct fd_device *dev, uint32_t size, uint32_t flags)
{
int ret;
struct fd_bo *bo;
struct kgsl_gpumem_alloc_id req = {
.size = size,
};

if (flags & FD_BO_GPUREADONLY)
req.flags |= KGSL_MEMFLAGS_GPUREADONLY;

ret = kgsl_pipe_safe_ioctl(dev->fd, IOCTL_KGSL_GPUMEM_ALLOC_ID, &req);

if (ret) {
ERROR_MSG("GPUMEM_ALLOC_ID failed (%s)", strerror(errno));
return NULL;
}

bo = kgsl_bo_from_handle(dev, size, req.id);

struct kgsl_bo *kgsl_bo = to_kgsl_bo(bo);
kgsl_bo->bo_type = KGSL_BO_NATIVE;

if (!bo) {
ERROR_MSG("Failed to allocate buffer object");
return NULL;
}

return bo;
}

0 comments on commit 8089471

Please sign in to comment.