[PATCH v2] drm/msm: UAPI error reporting
Akhil P Oommen
quic_akhilpo at quicinc.com
Thu Jan 2 20:07:21 UTC 2025
On 12/30/2024 9:30 PM, Rob Clark wrote:
> From: Rob Clark <robdclark at chromium.org>
>
> Debugging incorrect UAPI usage tends to be a bit painful, so add a
> helper macro to make it easier to add debug logging which can be enabled
> at runtime via drm.debug.
>
> Signed-off-by: Rob Clark <robdclark at chromium.org>
> ---
> drivers/gpu/drm/msm/adreno/adreno_gpu.c | 21 ++++----
> drivers/gpu/drm/msm/msm_drv.c | 2 +-
> drivers/gpu/drm/msm/msm_drv.h | 7 +++
> drivers/gpu/drm/msm/msm_gem_submit.c | 64 +++++++++++--------------
> drivers/gpu/drm/msm/msm_submitqueue.c | 2 +-
> 5 files changed, 47 insertions(+), 49 deletions(-)
>
> diff --git a/drivers/gpu/drm/msm/adreno/adreno_gpu.c b/drivers/gpu/drm/msm/adreno/adreno_gpu.c
> index 076be0473eb5..9649c0bd0a38 100644
> --- a/drivers/gpu/drm/msm/adreno/adreno_gpu.c
> +++ b/drivers/gpu/drm/msm/adreno/adreno_gpu.c
> @@ -310,10 +310,11 @@ int adreno_get_param(struct msm_gpu *gpu, struct msm_file_private *ctx,
> uint32_t param, uint64_t *value, uint32_t *len)
> {
> struct adreno_gpu *adreno_gpu = to_adreno_gpu(gpu);
> + struct drm_device *drm = gpu->dev;
>
> /* No pointer params yet */
> if (*len != 0)
> - return -EINVAL;
> + return UERR(EINVAL, drm, "invalid len");
>
> switch (param) {
> case MSM_PARAM_GPU_ID:
> @@ -365,12 +366,12 @@ int adreno_get_param(struct msm_gpu *gpu, struct msm_file_private *ctx,
> return 0;
> case MSM_PARAM_VA_START:
> if (ctx->aspace == gpu->aspace)
> - return -EINVAL;
> + return UERR(EINVAL, drm, "requires per-process pgtables");
> *value = ctx->aspace->va_start;
> return 0;
> case MSM_PARAM_VA_SIZE:
> if (ctx->aspace == gpu->aspace)
> - return -EINVAL;
> + return UERR(EINVAL, drm, "requires per-process pgtables");
> *value = ctx->aspace->va_size;
> return 0;
> case MSM_PARAM_HIGHEST_BANK_BIT:
> @@ -386,14 +387,15 @@ int adreno_get_param(struct msm_gpu *gpu, struct msm_file_private *ctx,
> *value = adreno_gpu->ubwc_config.macrotile_mode;
> return 0;
> default:
> - DBG("%s: invalid param: %u", gpu->name, param);
> - return -EINVAL;
> + return UERR(EINVAL, drm, "%s: invalid param: %u", gpu->name, param);
> }
> }
>
> int adreno_set_param(struct msm_gpu *gpu, struct msm_file_private *ctx,
> uint32_t param, uint64_t value, uint32_t len)
> {
> + struct drm_device *drm = gpu->dev;
> +
> switch (param) {
> case MSM_PARAM_COMM:
> case MSM_PARAM_CMDLINE:
> @@ -401,11 +403,11 @@ int adreno_set_param(struct msm_gpu *gpu, struct msm_file_private *ctx,
> * that should be a reasonable upper bound
> */
> if (len > PAGE_SIZE)
> - return -EINVAL;
> + return UERR(EINVAL, drm, "invalid len");
> break;
> default:
> if (len != 0)
> - return -EINVAL;
> + return UERR(EINVAL, drm, "invalid len");
> }
>
> switch (param) {
> @@ -434,11 +436,10 @@ int adreno_set_param(struct msm_gpu *gpu, struct msm_file_private *ctx,
> }
> case MSM_PARAM_SYSPROF:
> if (!capable(CAP_SYS_ADMIN))
> - return -EPERM;
> + return UERR(EPERM, drm, "invalid permissions");
> return msm_file_private_set_sysprof(ctx, gpu, value);
> default:
> - DBG("%s: invalid param: %u", gpu->name, param);
> - return -EINVAL;
> + return UERR(EINVAL, drm, "%s: invalid param: %u", gpu->name, param);
> }
> }
>
> diff --git a/drivers/gpu/drm/msm/msm_drv.c b/drivers/gpu/drm/msm/msm_drv.c
> index c2dd8ef6d6dc..2aefb8becda0 100644
> --- a/drivers/gpu/drm/msm/msm_drv.c
> +++ b/drivers/gpu/drm/msm/msm_drv.c
> @@ -538,7 +538,7 @@ static int msm_ioctl_gem_info_set_iova(struct drm_device *dev,
>
> /* Only supported if per-process address space is supported: */
> if (priv->gpu->aspace == ctx->aspace)
> - return -EOPNOTSUPP;
> + return UERR(EOPNOTSUPP, dev, "requires per-process pgtables");
>
> if (should_fail(&fail_gem_iova, obj->size))
> return -ENOMEM;
> diff --git a/drivers/gpu/drm/msm/msm_drv.h b/drivers/gpu/drm/msm/msm_drv.h
> index d8c9a1b19263..fee31680a6d5 100644
> --- a/drivers/gpu/drm/msm/msm_drv.h
> +++ b/drivers/gpu/drm/msm/msm_drv.h
> @@ -28,6 +28,7 @@
>
> #include <drm/drm_atomic.h>
> #include <drm/drm_atomic_helper.h>
> +#include <drm/drm_print.h>
> #include <drm/drm_probe_helper.h>
> #include <drm/display/drm_dsc.h>
> #include <drm/msm_drm.h>
> @@ -506,6 +507,12 @@ void msm_hrtimer_work_init(struct msm_hrtimer_work *work,
> clockid_t clock_id,
> enum hrtimer_mode mode);
>
> +/* Helper for returning a UABI error with optional logging which can make
> + * it easier for userspace to understand what it is doing wrong.
> + */
> +#define UERR(err, drm, fmt, ...) \
> + ({ DRM_DEV_DEBUG_DRIVER((drm)->dev, fmt, ##__VA_ARGS__); -(err); })
> +
> #define DBG(fmt, ...) DRM_DEBUG_DRIVER(fmt"\n", ##__VA_ARGS__)
> #define VERB(fmt, ...) if (0) DRM_DEBUG_DRIVER(fmt"\n", ##__VA_ARGS__)
>
> diff --git a/drivers/gpu/drm/msm/msm_gem_submit.c b/drivers/gpu/drm/msm/msm_gem_submit.c
> index fba78193127d..be6e793f34bd 100644
> --- a/drivers/gpu/drm/msm/msm_gem_submit.c
> +++ b/drivers/gpu/drm/msm/msm_gem_submit.c
> @@ -20,8 +20,8 @@
> /* For userspace errors, use DRM_UT_DRIVER.. so that userspace can enable
> * error msgs for debugging, but we don't spam dmesg by default
> */
> -#define SUBMIT_ERROR(submit, fmt, ...) \
> - DRM_DEV_DEBUG_DRIVER((submit)->dev->dev, fmt, ##__VA_ARGS__)
> +#define SUBMIT_ERROR(err, submit, fmt, ...) \
> + UERR(err, (submit)->dev, fmt, ##__VA_ARGS__)
>
> /*
> * Cmdstream submission:
> @@ -142,8 +142,7 @@ static int submit_lookup_objects(struct msm_gem_submit *submit,
>
> if ((submit_bo.flags & ~MSM_SUBMIT_BO_FLAGS) ||
> !(submit_bo.flags & MANDATORY_FLAGS)) {
> - SUBMIT_ERROR(submit, "invalid flags: %x\n", submit_bo.flags);
> - ret = -EINVAL;
> + ret = SUBMIT_ERROR(EINVAL, submit, "invalid flags: %x\n", submit_bo.flags);
> i = 0;
> goto out;
> }
> @@ -162,8 +161,7 @@ static int submit_lookup_objects(struct msm_gem_submit *submit,
> */
> obj = idr_find(&file->object_idr, submit->bos[i].handle);
> if (!obj) {
> - SUBMIT_ERROR(submit, "invalid handle %u at index %u\n", submit->bos[i].handle, i);
> - ret = -EINVAL;
> + ret = SUBMIT_ERROR(EINVAL, submit, "invalid handle %u at index %u\n", submit->bos[i].handle, i);
> goto out_unlock;
> }
>
> @@ -206,14 +204,12 @@ static int submit_lookup_cmds(struct msm_gem_submit *submit,
> case MSM_SUBMIT_CMD_CTX_RESTORE_BUF:
> break;
> default:
> - SUBMIT_ERROR(submit, "invalid type: %08x\n", submit_cmd.type);
> - return -EINVAL;
> + return SUBMIT_ERROR(EINVAL, submit, "invalid type: %08x\n", submit_cmd.type);
> }
>
> if (submit_cmd.size % 4) {
> - SUBMIT_ERROR(submit, "non-aligned cmdstream buffer size: %u\n",
> - submit_cmd.size);
> - ret = -EINVAL;
> + ret = SUBMIT_ERROR(EINVAL, submit, "non-aligned cmdstream buffer size: %u\n",
> + submit_cmd.size);
> goto out;
> }
>
> @@ -371,9 +367,8 @@ static int submit_bo(struct msm_gem_submit *submit, uint32_t idx,
> struct drm_gem_object **obj, uint64_t *iova)
> {
> if (idx >= submit->nr_bos) {
> - SUBMIT_ERROR(submit, "invalid buffer index: %u (out of %u)\n",
> - idx, submit->nr_bos);
> - return -EINVAL;
> + return SUBMIT_ERROR(EINVAL, submit, "invalid buffer index: %u (out of %u)\n",
> + idx, submit->nr_bos);
> }
>
> if (obj)
> @@ -392,10 +387,8 @@ static int submit_reloc(struct msm_gem_submit *submit, struct drm_gem_object *ob
> uint32_t *ptr;
> int ret = 0;
>
> - if (offset % 4) {
> - SUBMIT_ERROR(submit, "non-aligned cmdstream buffer: %u\n", offset);
> - return -EINVAL;
> - }
> + if (offset % 4)
> + return SUBMIT_ERROR(EINVAL, submit, "non-aligned cmdstream buffer: %u\n", offset);
>
> /* For now, just map the entire thing. Eventually we probably
> * to do it page-by-page, w/ kmap() if not vmap()d..
> @@ -414,9 +407,8 @@ static int submit_reloc(struct msm_gem_submit *submit, struct drm_gem_object *ob
> uint64_t iova;
>
> if (submit_reloc.submit_offset % 4) {
> - SUBMIT_ERROR(submit, "non-aligned reloc offset: %u\n",
> - submit_reloc.submit_offset);
> - ret = -EINVAL;
> + ret = SUBMIT_ERROR(EINVAL, submit, "non-aligned reloc offset: %u\n",
> + submit_reloc.submit_offset);
> goto out;
> }
>
> @@ -425,8 +417,7 @@ static int submit_reloc(struct msm_gem_submit *submit, struct drm_gem_object *ob
>
> if ((off >= (obj->size / 4)) ||
> (off < last_offset)) {
> - SUBMIT_ERROR(submit, "invalid offset %u at reloc %u\n", off, i);
> - ret = -EINVAL;
> + ret = SUBMIT_ERROR(EINVAL, submit, "invalid offset %u at reloc %u\n", off, i);
> goto out;
> }
>
> @@ -513,12 +504,12 @@ static struct drm_syncobj **msm_parse_deps(struct msm_gem_submit *submit,
>
> if (syncobj_desc.point &&
> !drm_core_check_feature(submit->dev, DRIVER_SYNCOBJ_TIMELINE)) {
> - ret = -EOPNOTSUPP;
> + ret = SUBMIT_ERROR(EOPNOTSUPP, submit, "syncobj timeline unsupported");
> break;
> }
>
> if (syncobj_desc.flags & ~MSM_SUBMIT_SYNCOBJ_FLAGS) {
> - ret = -EINVAL;
> + ret = -SUBMIT_ERROR(EINVAL, submit, "invalid syncobj flags");
> break;
> }
>
> @@ -531,7 +522,7 @@ static struct drm_syncobj **msm_parse_deps(struct msm_gem_submit *submit,
> syncobjs[i] =
> drm_syncobj_find(file, syncobj_desc.handle);
> if (!syncobjs[i]) {
> - ret = -EINVAL;
> + ret = SUBMIT_ERROR(EINVAL, submit, "invalid syncobj handle");
Just to be more useful, probably we can print the index or the handle
here. Anyway
Reviewed-by: Akhil P Oommen <quic_akhilpo at quicinc.com>
-Akhil.
More information about the dri-devel
mailing list