/* * Copyright (C) 2012-2018 Rob Clark * * Permission is hereby granted, free of charge, to any person obtaining a * copy of this software and associated documentation files (the "Software"), * to deal in the Software without restriction, including without limitation * the rights to use, copy, modify, merge, publish, distribute, sublicense, * and/or sell copies of the Software, and to permit persons to whom the * Software is furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice (including the next * paragraph) shall be included in all copies or substantial portions of the * Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * SOFTWARE. * * Authors: * Rob Clark */ #include "msm_priv.h" static int bo_allocate(struct msm_bo *msm_bo) { struct fd_bo *bo = &msm_bo->base; if (!msm_bo->offset) { struct drm_msm_gem_info req = { .handle = bo->handle, .info = MSM_INFO_GET_OFFSET, }; int ret; /* if the buffer is already backed by pages then this * doesn't actually do anything (other than giving us * the offset) */ ret = drmCommandWriteRead(bo->dev->fd, DRM_MSM_GEM_INFO, &req, sizeof(req)); if (ret) { ERROR_MSG("alloc failed: %s", strerror(errno)); return ret; } msm_bo->offset = req.value; } return 0; } static int msm_bo_offset(struct fd_bo *bo, uint64_t *offset) { struct msm_bo *msm_bo = to_msm_bo(bo); int ret = bo_allocate(msm_bo); if (ret) return ret; *offset = msm_bo->offset; return 0; } static int msm_bo_cpu_prep(struct fd_bo *bo, struct fd_pipe *pipe, uint32_t op) { struct drm_msm_gem_cpu_prep req = { .handle = bo->handle, .op = op, }; get_abs_timeout(&req.timeout, PIPE_TIMEOUT_INFINITE); return drmCommandWrite(bo->dev->fd, DRM_MSM_GEM_CPU_PREP, &req, sizeof(req)); } static void msm_bo_cpu_fini(struct fd_bo *bo) { struct drm_msm_gem_cpu_fini req = { .handle = bo->handle, }; drmCommandWrite(bo->dev->fd, DRM_MSM_GEM_CPU_FINI, &req, sizeof(req)); } static int msm_bo_madvise(struct fd_bo *bo, int willneed) { struct drm_msm_gem_madvise req = { .handle = bo->handle, .madv = willneed ? MSM_MADV_WILLNEED : MSM_MADV_DONTNEED, }; int ret; /* older kernels do not support this: */ if (bo->dev->version < FD_VERSION_MADVISE) return willneed; ret = drmCommandWriteRead(bo->dev->fd, DRM_MSM_GEM_MADVISE, &req, sizeof(req)); if (ret) return ret; return req.retained; } static uint64_t msm_bo_iova(struct fd_bo *bo) { struct drm_msm_gem_info req = { .handle = bo->handle, .info = MSM_INFO_GET_IOVA, }; int ret; ret = drmCommandWriteRead(bo->dev->fd, DRM_MSM_GEM_INFO, &req, sizeof(req)); if (ret) return 0; return req.value; } static void msm_bo_set_name(struct fd_bo *bo, const char *fmt, va_list ap) { struct drm_msm_gem_info req = { .handle = bo->handle, .info = MSM_INFO_SET_NAME, }; char buf[32]; int sz; if (bo->dev->version < FD_VERSION_SOFTPIN) return; sz = vsnprintf(buf, sizeof(buf), fmt, ap); req.value = VOID2U64(buf); req.len = MIN2(sz, sizeof(buf)); drmCommandWrite(bo->dev->fd, DRM_MSM_GEM_INFO, &req, sizeof(req)); } static void msm_bo_destroy(struct fd_bo *bo) { struct msm_bo *msm_bo = to_msm_bo(bo); free(msm_bo); } static const struct fd_bo_funcs funcs = { .offset = msm_bo_offset, .cpu_prep = msm_bo_cpu_prep, .cpu_fini = msm_bo_cpu_fini, .madvise = msm_bo_madvise, .iova = msm_bo_iova, .set_name = msm_bo_set_name, .destroy = msm_bo_destroy, }; /* allocate a buffer handle: */ static int new_handle(struct fd_device *dev, uint32_t size, uint32_t flags, uint32_t *handle) { struct drm_msm_gem_new req = { .size = size, }; int ret; if (flags & FD_BO_SCANOUT) req.flags |= MSM_BO_SCANOUT; if (flags & FD_BO_GPUREADONLY) req.flags |= MSM_BO_GPU_READONLY; if (flags & FD_BO_CACHED_COHERENT) req.flags |= MSM_BO_CACHED_COHERENT; else req.flags |= MSM_BO_WC; ret = drmCommandWriteRead(dev->fd, DRM_MSM_GEM_NEW, &req, sizeof(req)); if (ret) return ret; *handle = req.handle; return 0; } /* allocate a new buffer object */ struct fd_bo * msm_bo_new(struct fd_device *dev, uint32_t size, uint32_t flags) { uint32_t handle; int ret; ret = new_handle(dev, size, flags, &handle); if (ret) return NULL; return msm_bo_from_handle(dev, size, handle); } /* allocate a new buffer object from existing handle (import) */ struct fd_bo * msm_bo_from_handle(struct fd_device *dev, uint32_t size, uint32_t handle) { struct msm_bo *msm_bo; struct fd_bo *bo; msm_bo = calloc(1, sizeof(*msm_bo)); if (!msm_bo) return NULL; bo = &msm_bo->base; bo->size = size; bo->handle = handle; bo->funcs = &funcs; fd_bo_init_common(bo, dev); return bo; }