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.
857 lines
21 KiB
857 lines
21 KiB
/* Copyright (c) 2014-2019, 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/sysfs.h>
|
|
#include <linux/device.h>
|
|
|
|
#include "kgsl_device.h"
|
|
#include "kgsl_gmu.h"
|
|
#include "kgsl_hfi.h"
|
|
#include "adreno.h"
|
|
|
|
struct adreno_sysfs_attribute {
|
|
struct device_attribute attr;
|
|
unsigned int (*show)(struct adreno_device *adreno_dev);
|
|
int (*store)(struct adreno_device *adreno_dev, unsigned int val);
|
|
};
|
|
|
|
#define _ADRENO_SYSFS_ATTR(_name, __show, __store) \
|
|
struct adreno_sysfs_attribute adreno_attr_##_name = { \
|
|
.attr = __ATTR(_name, 0644, __show, __store), \
|
|
.show = _ ## _name ## _show, \
|
|
.store = _ ## _name ## _store, \
|
|
}
|
|
|
|
#define _ADRENO_SYSFS_ATTR_RO(_name, __show) \
|
|
struct adreno_sysfs_attribute adreno_attr_##_name = { \
|
|
.attr = __ATTR(_name, 0444, __show, NULL), \
|
|
.show = _ ## _name ## _show, \
|
|
.store = NULL, \
|
|
}
|
|
|
|
#define ADRENO_SYSFS_ATTR(_a) \
|
|
container_of((_a), struct adreno_sysfs_attribute, attr)
|
|
|
|
static struct adreno_device *_get_adreno_dev(struct device *dev)
|
|
{
|
|
struct kgsl_device *device = kgsl_device_from_dev(dev);
|
|
|
|
return device ? ADRENO_DEVICE(device) : NULL;
|
|
}
|
|
|
|
static int _ft_policy_store(struct adreno_device *adreno_dev,
|
|
unsigned int val)
|
|
{
|
|
adreno_dev->ft_policy = val & KGSL_FT_POLICY_MASK;
|
|
return 0;
|
|
}
|
|
|
|
static unsigned int _ft_policy_show(struct adreno_device *adreno_dev)
|
|
{
|
|
return adreno_dev->ft_policy;
|
|
}
|
|
|
|
static int _preempt_level_store(struct adreno_device *adreno_dev,
|
|
unsigned int val)
|
|
{
|
|
struct adreno_preemption *preempt = &adreno_dev->preempt;
|
|
|
|
if (val <= 2)
|
|
preempt->preempt_level = val;
|
|
return 0;
|
|
}
|
|
|
|
static unsigned int _preempt_level_show(struct adreno_device *adreno_dev)
|
|
{
|
|
struct adreno_preemption *preempt = &adreno_dev->preempt;
|
|
|
|
return preempt->preempt_level;
|
|
}
|
|
|
|
static int _usesgmem_store(struct adreno_device *adreno_dev,
|
|
unsigned int val)
|
|
{
|
|
struct adreno_preemption *preempt = &adreno_dev->preempt;
|
|
|
|
preempt->usesgmem = val ? 1 : 0;
|
|
return 0;
|
|
}
|
|
|
|
static unsigned int _usesgmem_show(struct adreno_device *adreno_dev)
|
|
{
|
|
struct adreno_preemption *preempt = &adreno_dev->preempt;
|
|
|
|
return preempt->usesgmem;
|
|
}
|
|
|
|
static int _skipsaverestore_store(struct adreno_device *adreno_dev,
|
|
unsigned int val)
|
|
{
|
|
struct adreno_preemption *preempt = &adreno_dev->preempt;
|
|
|
|
preempt->skipsaverestore = val ? 1 : 0;
|
|
return 0;
|
|
}
|
|
|
|
static unsigned int _skipsaverestore_show(struct adreno_device *adreno_dev)
|
|
{
|
|
struct adreno_preemption *preempt = &adreno_dev->preempt;
|
|
|
|
return preempt->skipsaverestore;
|
|
}
|
|
|
|
static int _ft_pagefault_policy_store(struct adreno_device *adreno_dev,
|
|
unsigned int val)
|
|
{
|
|
struct kgsl_device *device = KGSL_DEVICE(adreno_dev);
|
|
int ret = 0;
|
|
|
|
mutex_lock(&device->mutex);
|
|
val &= KGSL_FT_PAGEFAULT_MASK;
|
|
|
|
if (test_bit(ADRENO_DEVICE_STARTED, &adreno_dev->priv))
|
|
ret = kgsl_mmu_set_pagefault_policy(&device->mmu,
|
|
(unsigned long) val);
|
|
|
|
if (ret == 0)
|
|
adreno_dev->ft_pf_policy = val;
|
|
|
|
mutex_unlock(&device->mutex);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static unsigned int _ft_pagefault_policy_show(struct adreno_device *adreno_dev)
|
|
{
|
|
return adreno_dev->ft_pf_policy;
|
|
}
|
|
|
|
static int _gpu_llc_slice_enable_store(struct adreno_device *adreno_dev,
|
|
unsigned int val)
|
|
{
|
|
adreno_dev->gpu_llc_slice_enable = val ? true : false;
|
|
return 0;
|
|
}
|
|
|
|
static unsigned int _gpu_llc_slice_enable_show(struct adreno_device *adreno_dev)
|
|
{
|
|
return adreno_dev->gpu_llc_slice_enable;
|
|
}
|
|
|
|
static int _gpuhtw_llc_slice_enable_store(struct adreno_device *adreno_dev,
|
|
unsigned int val)
|
|
{
|
|
adreno_dev->gpuhtw_llc_slice_enable = val ? true : false;
|
|
return 0;
|
|
}
|
|
|
|
static unsigned int
|
|
_gpuhtw_llc_slice_enable_show(struct adreno_device *adreno_dev)
|
|
{
|
|
return adreno_dev->gpuhtw_llc_slice_enable;
|
|
}
|
|
|
|
static int _ft_long_ib_detect_store(struct adreno_device *adreno_dev,
|
|
unsigned int val)
|
|
{
|
|
adreno_dev->long_ib_detect = val;
|
|
return 0;
|
|
}
|
|
|
|
static unsigned int _ft_long_ib_detect_show(struct adreno_device *adreno_dev)
|
|
{
|
|
return adreno_dev->long_ib_detect;
|
|
}
|
|
|
|
static int _ft_hang_intr_status_store(struct adreno_device *adreno_dev,
|
|
unsigned int val)
|
|
{
|
|
struct kgsl_device *device = KGSL_DEVICE(adreno_dev);
|
|
int ret = 0;
|
|
|
|
if (val == test_bit(ADRENO_DEVICE_HANG_INTR, &adreno_dev->priv))
|
|
return 0;
|
|
|
|
mutex_lock(&device->mutex);
|
|
change_bit(ADRENO_DEVICE_HANG_INTR, &adreno_dev->priv);
|
|
|
|
if (test_bit(ADRENO_DEVICE_STARTED, &adreno_dev->priv)) {
|
|
kgsl_pwrctrl_change_state(device, KGSL_STATE_ACTIVE);
|
|
} else if (device->state == KGSL_STATE_INIT) {
|
|
ret = -EACCES;
|
|
change_bit(ADRENO_DEVICE_HANG_INTR, &adreno_dev->priv);
|
|
}
|
|
|
|
mutex_unlock(&device->mutex);
|
|
return ret;
|
|
}
|
|
|
|
static unsigned int _ft_hang_intr_status_show(struct adreno_device *adreno_dev)
|
|
{
|
|
return test_bit(ADRENO_DEVICE_HANG_INTR, &adreno_dev->priv);
|
|
}
|
|
|
|
static int _pwrctrl_store(struct adreno_device *adreno_dev,
|
|
unsigned int val, unsigned int flag)
|
|
{
|
|
struct kgsl_device *device = KGSL_DEVICE(adreno_dev);
|
|
|
|
if (val == test_bit(flag, &adreno_dev->pwrctrl_flag))
|
|
return 0;
|
|
|
|
return kgsl_change_flag(device, flag, &adreno_dev->pwrctrl_flag);
|
|
}
|
|
|
|
static int _preemption_store(struct adreno_device *adreno_dev,
|
|
unsigned int val)
|
|
{
|
|
struct kgsl_device *device = KGSL_DEVICE(adreno_dev);
|
|
struct kgsl_context *context;
|
|
struct adreno_context *drawctxt;
|
|
int id, ret;
|
|
|
|
mutex_lock(&device->mutex);
|
|
|
|
if (!(ADRENO_FEATURE(adreno_dev, ADRENO_PREEMPTION)) ||
|
|
(test_bit(ADRENO_DEVICE_PREEMPTION,
|
|
&adreno_dev->priv) == val)) {
|
|
mutex_unlock(&device->mutex);
|
|
return 0;
|
|
}
|
|
|
|
ret = kgsl_pwrctrl_change_state(device, KGSL_STATE_SUSPEND);
|
|
if (ret) {
|
|
mutex_unlock(&device->mutex);
|
|
return ret;
|
|
}
|
|
change_bit(ADRENO_DEVICE_PREEMPTION, &adreno_dev->priv);
|
|
adreno_dev->cur_rb = &(adreno_dev->ringbuffers[0]);
|
|
|
|
/* Update the ringbuffer for each draw context */
|
|
write_lock(&device->context_lock);
|
|
idr_for_each_entry(&device->context_idr, context, id) {
|
|
drawctxt = ADRENO_CONTEXT(context);
|
|
drawctxt->rb = adreno_ctx_get_rb(adreno_dev, drawctxt);
|
|
}
|
|
write_unlock(&device->context_lock);
|
|
|
|
kgsl_pwrctrl_change_state(device, KGSL_STATE_SLUMBER);
|
|
|
|
mutex_unlock(&device->mutex);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static unsigned int _preemption_show(struct adreno_device *adreno_dev)
|
|
{
|
|
return adreno_is_preemption_enabled(adreno_dev);
|
|
}
|
|
|
|
static int _hwcg_store(struct adreno_device *adreno_dev,
|
|
unsigned int val)
|
|
{
|
|
return _pwrctrl_store(adreno_dev, val, ADRENO_HWCG_CTRL);
|
|
}
|
|
|
|
static unsigned int _hwcg_show(struct adreno_device *adreno_dev)
|
|
{
|
|
return test_bit(ADRENO_HWCG_CTRL, &adreno_dev->pwrctrl_flag);
|
|
}
|
|
|
|
static int _throttling_store(struct adreno_device *adreno_dev,
|
|
unsigned int val)
|
|
{
|
|
return _pwrctrl_store(adreno_dev, val, ADRENO_THROTTLING_CTRL);
|
|
}
|
|
|
|
static unsigned int _throttling_show(struct adreno_device *adreno_dev)
|
|
{
|
|
return test_bit(ADRENO_THROTTLING_CTRL, &adreno_dev->pwrctrl_flag);
|
|
}
|
|
|
|
static int _sptp_pc_store(struct adreno_device *adreno_dev,
|
|
unsigned int val)
|
|
{
|
|
return _pwrctrl_store(adreno_dev, val, ADRENO_SPTP_PC_CTRL);
|
|
}
|
|
|
|
static unsigned int _sptp_pc_show(struct adreno_device *adreno_dev)
|
|
{
|
|
return test_bit(ADRENO_SPTP_PC_CTRL, &adreno_dev->pwrctrl_flag);
|
|
}
|
|
|
|
static int _lm_store(struct adreno_device *adreno_dev, unsigned int val)
|
|
{
|
|
return _pwrctrl_store(adreno_dev, val, ADRENO_LM_CTRL);
|
|
}
|
|
|
|
static unsigned int _lm_show(struct adreno_device *adreno_dev)
|
|
{
|
|
return test_bit(ADRENO_LM_CTRL, &adreno_dev->pwrctrl_flag);
|
|
}
|
|
|
|
static int _ifpc_store(struct adreno_device *adreno_dev, unsigned int val)
|
|
{
|
|
return adreno_gmu_ifpc_store(adreno_dev, val);
|
|
}
|
|
|
|
static unsigned int _ifpc_show(struct adreno_device *adreno_dev)
|
|
{
|
|
return adreno_gmu_ifpc_show(adreno_dev);
|
|
}
|
|
|
|
static unsigned int _ifpc_count_show(struct adreno_device *adreno_dev)
|
|
{
|
|
return adreno_dev->ifpc_count;
|
|
}
|
|
|
|
static unsigned int _preempt_count_show(struct adreno_device *adreno_dev)
|
|
{
|
|
struct adreno_preemption *preempt = &adreno_dev->preempt;
|
|
|
|
return preempt->count;
|
|
}
|
|
|
|
static unsigned int acd_data_index;
|
|
static DEFINE_SPINLOCK(acd_data_index_lock);
|
|
|
|
static unsigned int _acd_data_index_show(struct adreno_device *adreno_dev)
|
|
{
|
|
unsigned int val;
|
|
|
|
spin_lock(&acd_data_index_lock);
|
|
val = acd_data_index;
|
|
spin_unlock(&acd_data_index_lock);
|
|
return val;
|
|
}
|
|
|
|
static int _acd_data_index_store(struct adreno_device *adreno_dev,
|
|
unsigned int val)
|
|
{
|
|
struct kgsl_device *device = KGSL_DEVICE(adreno_dev);
|
|
struct gmu_device *gmu = KGSL_GMU_DEVICE(device);
|
|
struct hfi_acd_table_cmd *cmd;
|
|
|
|
if (!ADRENO_FEATURE(adreno_dev, ADRENO_ACD))
|
|
return -EINVAL;
|
|
|
|
cmd = &gmu->hfi.acd_tbl_cmd;
|
|
|
|
if (val >= (cmd->stride * cmd->num_levels))
|
|
return -EINVAL;
|
|
|
|
spin_lock(&acd_data_index_lock);
|
|
acd_data_index = val;
|
|
spin_unlock(&acd_data_index_lock);
|
|
return 0;
|
|
}
|
|
|
|
static unsigned int _acd_show(struct adreno_device *adreno_dev)
|
|
{
|
|
return test_bit(ADRENO_ACD_CTRL, &adreno_dev->pwrctrl_flag);
|
|
}
|
|
|
|
static int _acd_store(struct adreno_device *adreno_dev, unsigned int val)
|
|
{
|
|
struct kgsl_device *device = KGSL_DEVICE(adreno_dev);
|
|
|
|
if (test_bit(ADRENO_ACD_CTRL, &adreno_dev->pwrctrl_flag) == val)
|
|
return 0;
|
|
|
|
return gmu_core_acd_set(device, val);
|
|
}
|
|
|
|
static unsigned int _acd_version_show(struct adreno_device *adreno_dev)
|
|
{
|
|
struct kgsl_device *device = KGSL_DEVICE(adreno_dev);
|
|
struct gmu_device *gmu = KGSL_GMU_DEVICE(device);
|
|
struct hfi_acd_table_cmd *cmd;
|
|
|
|
if (!ADRENO_FEATURE(adreno_dev, ADRENO_ACD))
|
|
return 0;
|
|
|
|
cmd = &gmu->hfi.acd_tbl_cmd;
|
|
return cmd->version;
|
|
}
|
|
|
|
static int _acd_version_store(struct adreno_device *adreno_dev,
|
|
unsigned int val)
|
|
{
|
|
struct kgsl_device *device = KGSL_DEVICE(adreno_dev);
|
|
struct gmu_device *gmu = KGSL_GMU_DEVICE(device);
|
|
struct hfi_acd_table_cmd *cmd;
|
|
|
|
if (!ADRENO_FEATURE(adreno_dev, ADRENO_ACD))
|
|
return -EINVAL;
|
|
|
|
cmd = &gmu->hfi.acd_tbl_cmd;
|
|
cmd->version = val;
|
|
return 0;
|
|
}
|
|
|
|
static unsigned int _acd_stride_show(struct adreno_device *adreno_dev)
|
|
{
|
|
struct kgsl_device *device = KGSL_DEVICE(adreno_dev);
|
|
struct gmu_device *gmu = KGSL_GMU_DEVICE(device);
|
|
struct hfi_acd_table_cmd *cmd;
|
|
|
|
if (!ADRENO_FEATURE(adreno_dev, ADRENO_ACD))
|
|
return 0;
|
|
|
|
cmd = &gmu->hfi.acd_tbl_cmd;
|
|
return cmd->stride;
|
|
}
|
|
|
|
static int _acd_stride_store(struct adreno_device *adreno_dev, unsigned int val)
|
|
{
|
|
struct kgsl_device *device = KGSL_DEVICE(adreno_dev);
|
|
struct gmu_device *gmu = KGSL_GMU_DEVICE(device);
|
|
struct hfi_acd_table_cmd *cmd;
|
|
|
|
if (!ADRENO_FEATURE(adreno_dev, ADRENO_ACD))
|
|
return -EINVAL;
|
|
|
|
if (!val || val > MAX_ACD_STRIDE)
|
|
return -EINVAL;
|
|
|
|
cmd = &gmu->hfi.acd_tbl_cmd;
|
|
cmd->stride = val;
|
|
return 0;
|
|
}
|
|
|
|
static unsigned int _acd_num_levels_show(struct adreno_device *adreno_dev)
|
|
{
|
|
struct kgsl_device *device = KGSL_DEVICE(adreno_dev);
|
|
struct gmu_device *gmu = KGSL_GMU_DEVICE(device);
|
|
struct hfi_acd_table_cmd *cmd;
|
|
|
|
if (!ADRENO_FEATURE(adreno_dev, ADRENO_ACD))
|
|
return 0;
|
|
|
|
cmd = &gmu->hfi.acd_tbl_cmd;
|
|
return cmd->num_levels;
|
|
}
|
|
|
|
static int _acd_num_levels_store(struct adreno_device *adreno_dev,
|
|
unsigned int val)
|
|
{
|
|
struct kgsl_device *device = KGSL_DEVICE(adreno_dev);
|
|
struct gmu_device *gmu = KGSL_GMU_DEVICE(device);
|
|
struct hfi_acd_table_cmd *cmd;
|
|
|
|
if (!ADRENO_FEATURE(adreno_dev, ADRENO_ACD))
|
|
return -EINVAL;
|
|
|
|
if (!val || val > MAX_ACD_NUM_LEVELS)
|
|
return -EINVAL;
|
|
|
|
cmd = &gmu->hfi.acd_tbl_cmd;
|
|
cmd->num_levels = val;
|
|
return 0;
|
|
}
|
|
|
|
static unsigned int _acd_enable_by_level_show(struct adreno_device *adreno_dev)
|
|
{
|
|
struct kgsl_device *device = KGSL_DEVICE(adreno_dev);
|
|
struct gmu_device *gmu = KGSL_GMU_DEVICE(device);
|
|
struct hfi_acd_table_cmd *cmd;
|
|
|
|
if (!ADRENO_FEATURE(adreno_dev, ADRENO_ACD))
|
|
return 0;
|
|
|
|
cmd = &gmu->hfi.acd_tbl_cmd;
|
|
return cmd->enable_by_level;
|
|
}
|
|
|
|
static int _acd_enable_by_level_store(struct adreno_device *adreno_dev,
|
|
unsigned int val)
|
|
{
|
|
struct kgsl_device *device = KGSL_DEVICE(adreno_dev);
|
|
struct gmu_device *gmu = KGSL_GMU_DEVICE(device);
|
|
struct hfi_acd_table_cmd *cmd;
|
|
|
|
if (!ADRENO_FEATURE(adreno_dev, ADRENO_ACD))
|
|
return -EINVAL;
|
|
|
|
cmd = &gmu->hfi.acd_tbl_cmd;
|
|
|
|
if (hweight32(val) != cmd->num_levels)
|
|
return -EINVAL;
|
|
|
|
cmd->enable_by_level = val;
|
|
return 0;
|
|
}
|
|
|
|
static unsigned int _acd_data_show(struct adreno_device *adreno_dev)
|
|
{
|
|
struct kgsl_device *device = KGSL_DEVICE(adreno_dev);
|
|
struct gmu_device *gmu = KGSL_GMU_DEVICE(device);
|
|
struct hfi_acd_table_cmd *cmd;
|
|
unsigned int index;
|
|
|
|
if (!ADRENO_FEATURE(adreno_dev, ADRENO_ACD))
|
|
return 0;
|
|
|
|
spin_lock(&acd_data_index_lock);
|
|
index = acd_data_index;
|
|
spin_unlock(&acd_data_index_lock);
|
|
|
|
cmd = &gmu->hfi.acd_tbl_cmd;
|
|
return cmd->data[index];
|
|
}
|
|
|
|
static int _acd_data_store(struct adreno_device *adreno_dev, unsigned int val)
|
|
{
|
|
struct kgsl_device *device = KGSL_DEVICE(adreno_dev);
|
|
struct gmu_device *gmu = KGSL_GMU_DEVICE(device);
|
|
struct hfi_acd_table_cmd *cmd;
|
|
unsigned int index;
|
|
|
|
if (!ADRENO_FEATURE(adreno_dev, ADRENO_ACD))
|
|
return -EINVAL;
|
|
|
|
spin_lock(&acd_data_index_lock);
|
|
index = acd_data_index;
|
|
spin_unlock(&acd_data_index_lock);
|
|
|
|
cmd = &gmu->hfi.acd_tbl_cmd;
|
|
cmd->data[index] = val;
|
|
return 0;
|
|
}
|
|
|
|
static unsigned int _perfcounter_show(struct adreno_device *adreno_dev)
|
|
{
|
|
return adreno_dev->perfcounter;
|
|
}
|
|
|
|
static int _perfcounter_store(struct adreno_device *adreno_dev,
|
|
unsigned int val)
|
|
{
|
|
struct kgsl_device *device = KGSL_DEVICE(adreno_dev);
|
|
|
|
if (adreno_dev->perfcounter == val)
|
|
return 0;
|
|
|
|
mutex_lock(&device->mutex);
|
|
|
|
/* Power down the GPU before changing the state */
|
|
kgsl_pwrctrl_change_state(device, KGSL_STATE_SUSPEND);
|
|
adreno_dev->perfcounter = val;
|
|
kgsl_pwrctrl_change_state(device, KGSL_STATE_SLUMBER);
|
|
|
|
mutex_unlock(&device->mutex);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static ssize_t _sysfs_store_u32(struct device *dev,
|
|
struct device_attribute *attr,
|
|
const char *buf, size_t count)
|
|
{
|
|
struct adreno_device *adreno_dev = _get_adreno_dev(dev);
|
|
struct adreno_sysfs_attribute *_attr = ADRENO_SYSFS_ATTR(attr);
|
|
unsigned int val = 0;
|
|
int ret;
|
|
|
|
if (adreno_dev == NULL)
|
|
return 0;
|
|
|
|
ret = kgsl_sysfs_store(buf, &val);
|
|
|
|
if (!ret && _attr->store)
|
|
ret = _attr->store(adreno_dev, val);
|
|
|
|
return (ssize_t) ret < 0 ? ret : count;
|
|
}
|
|
|
|
static ssize_t _sysfs_show_u32(struct device *dev,
|
|
struct device_attribute *attr,
|
|
char *buf)
|
|
{
|
|
struct adreno_device *adreno_dev = _get_adreno_dev(dev);
|
|
struct adreno_sysfs_attribute *_attr = ADRENO_SYSFS_ATTR(attr);
|
|
unsigned int val = 0;
|
|
|
|
if (adreno_dev == NULL)
|
|
return 0;
|
|
|
|
if (_attr->show)
|
|
val = _attr->show(adreno_dev);
|
|
|
|
return snprintf(buf, PAGE_SIZE, "0x%X\n", val);
|
|
}
|
|
|
|
static ssize_t _sysfs_store_bool(struct device *dev,
|
|
struct device_attribute *attr,
|
|
const char *buf, size_t count)
|
|
{
|
|
struct adreno_device *adreno_dev = _get_adreno_dev(dev);
|
|
struct adreno_sysfs_attribute *_attr = ADRENO_SYSFS_ATTR(attr);
|
|
unsigned int val = 0;
|
|
int ret;
|
|
|
|
if (adreno_dev == NULL)
|
|
return 0;
|
|
|
|
ret = kgsl_sysfs_store(buf, &val);
|
|
|
|
if (!ret && _attr->store)
|
|
ret = _attr->store(adreno_dev, val ? 1 : 0);
|
|
|
|
return (ssize_t) ret < 0 ? ret : count;
|
|
}
|
|
|
|
static ssize_t _sysfs_show_bool(struct device *dev,
|
|
struct device_attribute *attr,
|
|
char *buf)
|
|
{
|
|
struct adreno_device *adreno_dev = _get_adreno_dev(dev);
|
|
struct adreno_sysfs_attribute *_attr = ADRENO_SYSFS_ATTR(attr);
|
|
unsigned int val = 0;
|
|
|
|
if (adreno_dev == NULL)
|
|
return 0;
|
|
|
|
if (_attr->show)
|
|
val = _attr->show(adreno_dev);
|
|
|
|
return snprintf(buf, PAGE_SIZE, "%d\n", val);
|
|
}
|
|
|
|
#define ADRENO_SYSFS_BOOL(_name) \
|
|
_ADRENO_SYSFS_ATTR(_name, _sysfs_show_bool, _sysfs_store_bool)
|
|
|
|
#define ADRENO_SYSFS_U32(_name) \
|
|
_ADRENO_SYSFS_ATTR(_name, _sysfs_show_u32, _sysfs_store_u32)
|
|
|
|
#define ADRENO_SYSFS_RO_U32(_name) \
|
|
_ADRENO_SYSFS_ATTR_RO(_name, _sysfs_show_u32)
|
|
|
|
static ADRENO_SYSFS_U32(ft_policy);
|
|
static ADRENO_SYSFS_U32(ft_pagefault_policy);
|
|
static ADRENO_SYSFS_U32(preempt_level);
|
|
static ADRENO_SYSFS_RO_U32(preempt_count);
|
|
static ADRENO_SYSFS_BOOL(usesgmem);
|
|
static ADRENO_SYSFS_BOOL(skipsaverestore);
|
|
static ADRENO_SYSFS_BOOL(ft_long_ib_detect);
|
|
static ADRENO_SYSFS_BOOL(ft_hang_intr_status);
|
|
static ADRENO_SYSFS_BOOL(gpu_llc_slice_enable);
|
|
static ADRENO_SYSFS_BOOL(gpuhtw_llc_slice_enable);
|
|
|
|
static DEVICE_INT_ATTR(wake_nice, 0644, adreno_wake_nice);
|
|
|
|
static ADRENO_SYSFS_BOOL(sptp_pc);
|
|
static ADRENO_SYSFS_BOOL(lm);
|
|
static ADRENO_SYSFS_BOOL(preemption);
|
|
static ADRENO_SYSFS_BOOL(hwcg);
|
|
static ADRENO_SYSFS_BOOL(throttling);
|
|
static ADRENO_SYSFS_BOOL(ifpc);
|
|
static ADRENO_SYSFS_RO_U32(ifpc_count);
|
|
static ADRENO_SYSFS_BOOL(acd);
|
|
static ADRENO_SYSFS_BOOL(perfcounter);
|
|
|
|
static ADRENO_SYSFS_U32(acd_data_index);
|
|
static ADRENO_SYSFS_U32(acd_version);
|
|
static ADRENO_SYSFS_U32(acd_stride);
|
|
static ADRENO_SYSFS_U32(acd_num_levels);
|
|
static ADRENO_SYSFS_U32(acd_enable_by_level);
|
|
static ADRENO_SYSFS_U32(acd_data);
|
|
|
|
static const struct device_attribute *_attr_list[] = {
|
|
&adreno_attr_ft_policy.attr,
|
|
&adreno_attr_ft_pagefault_policy.attr,
|
|
&adreno_attr_ft_long_ib_detect.attr,
|
|
&adreno_attr_ft_hang_intr_status.attr,
|
|
&dev_attr_wake_nice.attr,
|
|
&adreno_attr_sptp_pc.attr,
|
|
&adreno_attr_lm.attr,
|
|
&adreno_attr_preemption.attr,
|
|
&adreno_attr_hwcg.attr,
|
|
&adreno_attr_throttling.attr,
|
|
&adreno_attr_gpu_llc_slice_enable.attr,
|
|
&adreno_attr_gpuhtw_llc_slice_enable.attr,
|
|
&adreno_attr_preempt_level.attr,
|
|
&adreno_attr_usesgmem.attr,
|
|
&adreno_attr_skipsaverestore.attr,
|
|
&adreno_attr_ifpc.attr,
|
|
&adreno_attr_ifpc_count.attr,
|
|
&adreno_attr_preempt_count.attr,
|
|
&adreno_attr_acd.attr,
|
|
&adreno_attr_perfcounter.attr,
|
|
&adreno_attr_acd_data_index.attr,
|
|
&adreno_attr_acd_version.attr,
|
|
&adreno_attr_acd_stride.attr,
|
|
&adreno_attr_acd_num_levels.attr,
|
|
&adreno_attr_acd_enable_by_level.attr,
|
|
&adreno_attr_acd_data.attr,
|
|
NULL,
|
|
};
|
|
|
|
/* Add a ppd directory for controlling different knobs from sysfs */
|
|
struct adreno_ppd_attribute {
|
|
struct attribute attr;
|
|
ssize_t (*show)(struct kgsl_device *device, char *buf);
|
|
ssize_t (*store)(struct kgsl_device *device, const char *buf,
|
|
size_t count);
|
|
};
|
|
|
|
#define PPD_ATTR(_name, _mode, _show, _store) \
|
|
struct adreno_ppd_attribute attr_##_name = { \
|
|
.attr = { .name = __stringify(_name), .mode = _mode }, \
|
|
.show = _show, \
|
|
.store = _store, \
|
|
}
|
|
|
|
#define to_ppd_attr(a) \
|
|
container_of((a), struct adreno_ppd_attribute, attr)
|
|
|
|
#define kobj_to_device(a) \
|
|
container_of((a), struct kgsl_device, ppd_kobj)
|
|
|
|
static ssize_t ppd_enable_store(struct kgsl_device *device,
|
|
const char *buf, size_t count)
|
|
{
|
|
struct adreno_device *adreno_dev = ADRENO_DEVICE(device);
|
|
unsigned int ppd_on = 0;
|
|
int ret;
|
|
|
|
if (!adreno_is_a430v2(adreno_dev) ||
|
|
!ADRENO_FEATURE(adreno_dev, ADRENO_PPD))
|
|
return count;
|
|
|
|
ret = kgsl_sysfs_store(buf, &ppd_on);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
ppd_on = (ppd_on) ? 1 : 0;
|
|
|
|
if (ppd_on == test_bit(ADRENO_PPD_CTRL, &adreno_dev->pwrctrl_flag))
|
|
return count;
|
|
|
|
ret = kgsl_change_flag(device, ADRENO_PPD_CTRL,
|
|
&adreno_dev->pwrctrl_flag);
|
|
return ret ? ret : count;
|
|
}
|
|
|
|
static ssize_t ppd_enable_show(struct kgsl_device *device,
|
|
char *buf)
|
|
{
|
|
struct adreno_device *adreno_dev = ADRENO_DEVICE(device);
|
|
|
|
return snprintf(buf, PAGE_SIZE, "%u\n",
|
|
test_bit(ADRENO_PPD_CTRL, &adreno_dev->pwrctrl_flag));
|
|
}
|
|
/* Add individual ppd attributes here */
|
|
static PPD_ATTR(enable, 0644, ppd_enable_show, ppd_enable_store);
|
|
|
|
static ssize_t ppd_sysfs_show(struct kobject *kobj,
|
|
struct attribute *attr, char *buf)
|
|
{
|
|
struct adreno_ppd_attribute *pattr = to_ppd_attr(attr);
|
|
struct kgsl_device *device = kobj_to_device(kobj);
|
|
ssize_t ret = -EIO;
|
|
|
|
if (device != NULL && pattr->show != NULL)
|
|
ret = pattr->show(device, buf);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static ssize_t ppd_sysfs_store(struct kobject *kobj,
|
|
struct attribute *attr, const char *buf, size_t count)
|
|
{
|
|
struct adreno_ppd_attribute *pattr = to_ppd_attr(attr);
|
|
struct kgsl_device *device = kobj_to_device(kobj);
|
|
ssize_t ret = -EIO;
|
|
|
|
if (device != NULL && pattr->store != NULL)
|
|
ret = pattr->store(device, buf, count);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static const struct sysfs_ops ppd_sysfs_ops = {
|
|
.show = ppd_sysfs_show,
|
|
.store = ppd_sysfs_store,
|
|
};
|
|
|
|
static struct kobj_type ktype_ppd = {
|
|
.sysfs_ops = &ppd_sysfs_ops,
|
|
};
|
|
|
|
static void ppd_sysfs_close(struct adreno_device *adreno_dev)
|
|
{
|
|
struct kgsl_device *device = KGSL_DEVICE(adreno_dev);
|
|
|
|
if (!ADRENO_FEATURE(adreno_dev, ADRENO_PPD))
|
|
return;
|
|
|
|
sysfs_remove_file(&device->ppd_kobj, &attr_enable.attr);
|
|
kobject_put(&device->ppd_kobj);
|
|
}
|
|
|
|
static int ppd_sysfs_init(struct adreno_device *adreno_dev)
|
|
{
|
|
struct kgsl_device *device = KGSL_DEVICE(adreno_dev);
|
|
int ret;
|
|
|
|
if (!ADRENO_FEATURE(adreno_dev, ADRENO_PPD))
|
|
return -ENODEV;
|
|
|
|
ret = kobject_init_and_add(&device->ppd_kobj, &ktype_ppd,
|
|
&device->dev->kobj, "ppd");
|
|
|
|
if (ret == 0)
|
|
ret = sysfs_create_file(&device->ppd_kobj, &attr_enable.attr);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* adreno_sysfs_close() - Take down the adreno sysfs files
|
|
* @adreno_dev: Pointer to the adreno device
|
|
*
|
|
* Take down the sysfs files on when the device goes away
|
|
*/
|
|
void adreno_sysfs_close(struct adreno_device *adreno_dev)
|
|
{
|
|
struct kgsl_device *device = KGSL_DEVICE(adreno_dev);
|
|
|
|
ppd_sysfs_close(adreno_dev);
|
|
kgsl_remove_device_sysfs_files(device->dev, _attr_list);
|
|
}
|
|
|
|
/**
|
|
* adreno_sysfs_init() - Initialize adreno sysfs files
|
|
* @adreno_dev: Pointer to the adreno device
|
|
*
|
|
* Initialize many of the adreno specific sysfs files especially for fault
|
|
* tolerance and power control
|
|
*/
|
|
int adreno_sysfs_init(struct adreno_device *adreno_dev)
|
|
{
|
|
struct kgsl_device *device = KGSL_DEVICE(adreno_dev);
|
|
int ret;
|
|
|
|
ret = kgsl_create_device_sysfs_files(device->dev, _attr_list);
|
|
|
|
/* Add the PPD directory and files */
|
|
if (ret == 0) {
|
|
/* Notify userspace */
|
|
kobject_uevent(&device->dev->kobj, KOBJ_ADD);
|
|
ppd_sysfs_init(adreno_dev);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|