You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
265 lines
6.8 KiB
265 lines
6.8 KiB
/* 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 <linux/uaccess.h>
|
|
#include <linux/ioctl.h>
|
|
|
|
#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));
|
|
}
|
|
|