/* Copyright (c) 2013-2018, The Linux Foundation. All rights reserved. * Copyright (c) 2022 Qualcomm Innovation Center, Inc. All rights reserved. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 and * only version 2 as published by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * */ #include #include #include "kgsl.h" #include "kgsl_compat.h" #include "adreno.h" #include "adreno_compat.h" int adreno_getproperty_compat(struct kgsl_device *device, unsigned int type, void __user *value, size_t sizebytes) { int status = -EINVAL; struct adreno_device *adreno_dev = ADRENO_DEVICE(device); switch (type) { case KGSL_PROP_DEVICE_INFO: { struct kgsl_devinfo_compat devinfo; if (sizebytes != sizeof(devinfo)) { status = -EINVAL; break; } memset(&devinfo, 0, sizeof(devinfo)); devinfo.device_id = device->id + 1; devinfo.chip_id = adreno_dev->chipid; devinfo.mmu_enabled = MMU_FEATURE(&device->mmu, KGSL_MMU_PAGED); devinfo.gmem_gpubaseaddr = adreno_dev->gmem_base; devinfo.gmem_sizebytes = adreno_dev->gmem_size; if (copy_to_user(value, &devinfo, sizeof(devinfo)) != 0) { status = -EFAULT; break; } status = 0; } break; case KGSL_PROP_DEVICE_SHADOW: { struct kgsl_shadowprop_compat shadowprop; if (sizebytes != sizeof(shadowprop)) { status = -EINVAL; break; } memset(&shadowprop, 0, sizeof(shadowprop)); if (device->memstore.hostptr) { /* * NOTE: with mmu enabled, gpuaddr doesn't mean * anything to mmap(). * NOTE: shadowprop.gpuaddr is uint32 * (because legacy) and the memstore gpuaddr is * 64 bit. Cast the memstore gpuaddr to uint32. */ shadowprop.gpuaddr = (unsigned int) device->memstore.gpuaddr; shadowprop.size = (unsigned int) device->memstore.size; /* * GSL needs this to be set, even if it * appears to be meaningless */ shadowprop.flags = KGSL_FLAGS_INITIALIZED | KGSL_FLAGS_PER_CONTEXT_TIMESTAMPS; } if (copy_to_user(value, &shadowprop, sizeof(shadowprop))) { status = -EFAULT; break; } status = 0; } break; case KGSL_PROP_DEVICE_QDSS_STM: { struct kgsl_qdss_stm_prop qdssprop = {0}; struct kgsl_memdesc *qdss_desc = kgsl_mmu_get_qdss_global_entry(device); if (sizebytes != sizeof(qdssprop)) { status = -EINVAL; break; } if (qdss_desc) { qdssprop.gpuaddr = qdss_desc->gpuaddr; qdssprop.size = qdss_desc->size; } if (copy_to_user(value, &qdssprop, sizeof(qdssprop))) { status = -EFAULT; break; } status = 0; } break; case KGSL_PROP_DEVICE_QTIMER: { struct kgsl_qtimer_prop qtimerprop = {0}; struct kgsl_memdesc *qtimer_desc = kgsl_mmu_get_qtimer_global_entry(device); if (sizebytes != sizeof(qtimerprop)) { status = -EINVAL; break; } if (qtimer_desc) { qtimerprop.gpuaddr = qtimer_desc->gpuaddr; qtimerprop.size = qtimer_desc->size; } if (copy_to_user(value, &qtimerprop, sizeof(qtimerprop))) { status = -EFAULT; break; } status = 0; } break; default: /* * Call the adreno_getproperty to check if the property type * was KGSL_PROP_MMU_ENABLE or KGSL_PROP_INTERRUPT_WAITS */ status = device->ftbl->getproperty(device, type, value, sizebytes); } return status; } int adreno_setproperty_compat(struct kgsl_device_private *dev_priv, unsigned int type, void __user *value, unsigned int sizebytes) { int status = -EINVAL; struct kgsl_device *device = dev_priv->device; switch (type) { case KGSL_PROP_PWR_CONSTRAINT: case KGSL_PROP_L3_PWR_CONSTRAINT: { struct kgsl_device_constraint_compat constraint32; struct kgsl_device_constraint constraint; struct kgsl_context *context; if (sizebytes != sizeof(constraint32)) break; if (copy_from_user(&constraint32, value, sizeof(constraint32))) { status = -EFAULT; break; } /* Populate the real constraint type from the compat */ constraint.type = constraint32.type; constraint.context_id = constraint32.context_id; constraint.data = compat_ptr(constraint32.data); constraint.size = (size_t)constraint32.size; context = kgsl_context_get_owner(dev_priv, constraint.context_id); if (context == NULL) break; status = adreno_set_constraint(device, context, &constraint); kgsl_context_put(context); } break; default: /* * Call adreno_setproperty in case the property type was * KGSL_PROP_PWRCTRL */ status = device->ftbl->setproperty(dev_priv, type, value, sizebytes); } return status; } static long adreno_ioctl_perfcounter_query_compat( struct kgsl_device_private *dev_priv, unsigned int cmd, void *data) { struct adreno_device *adreno_dev = ADRENO_DEVICE(dev_priv->device); struct kgsl_perfcounter_query_compat *query32 = data; struct kgsl_perfcounter_query query; long result; query.groupid = query32->groupid; query.countables = to_user_ptr(query32->countables); query.count = query32->count; query.max_counters = query32->max_counters; result = adreno_perfcounter_query_group(adreno_dev, query.groupid, query.countables, query.count, &query.max_counters); query32->max_counters = query.max_counters; return result; } static long adreno_ioctl_perfcounter_read_compat( struct kgsl_device_private *dev_priv, unsigned int cmd, void *data) { struct adreno_device *adreno_dev = ADRENO_DEVICE(dev_priv->device); struct kgsl_perfcounter_read_compat *read32 = data; struct kgsl_perfcounter_read read; /* * When performance counter zapping is enabled, the counters are cleared * across context switches. Reading the counters when they are zapped is * not permitted. */ if (!adreno_dev->perfcounter) return -EPERM; read.reads = (struct kgsl_perfcounter_read_group __user *) (uintptr_t)read32->reads; read.count = read32->count; return adreno_perfcounter_read_group(adreno_dev, read.reads, read.count); } static struct kgsl_ioctl adreno_compat_ioctl_funcs[] = { { IOCTL_KGSL_PERFCOUNTER_GET, adreno_ioctl_perfcounter_get }, { IOCTL_KGSL_PERFCOUNTER_PUT, adreno_ioctl_perfcounter_put }, { IOCTL_KGSL_PERFCOUNTER_QUERY_COMPAT, adreno_ioctl_perfcounter_query_compat }, { IOCTL_KGSL_PERFCOUNTER_READ_COMPAT, adreno_ioctl_perfcounter_read_compat }, }; long adreno_compat_ioctl(struct kgsl_device_private *dev_priv, unsigned int cmd, unsigned long arg) { return adreno_ioctl_helper(dev_priv, cmd, arg, adreno_compat_ioctl_funcs, ARRAY_SIZE(adreno_compat_ioctl_funcs)); }