cachepc-linux

Fork of AMDESE/linux with modifications for CachePC side-channel attack
git clone https://git.sinitax.com/sinitax/cachepc-linux
Log | Files | Refs | README | LICENSE | sfeed.txt

hidma_mgmt_sys.c (6803B)


      1// SPDX-License-Identifier: GPL-2.0-only
      2/*
      3 * Qualcomm Technologies HIDMA Management SYS interface
      4 *
      5 * Copyright (c) 2015, The Linux Foundation. All rights reserved.
      6 */
      7
      8#include <linux/sysfs.h>
      9#include <linux/platform_device.h>
     10
     11#include "hidma_mgmt.h"
     12
     13struct hidma_chan_attr {
     14	struct hidma_mgmt_dev *mdev;
     15	int index;
     16	struct kobj_attribute attr;
     17};
     18
     19struct hidma_mgmt_fileinfo {
     20	char *name;
     21	int mode;
     22	int (*get)(struct hidma_mgmt_dev *mdev);
     23	int (*set)(struct hidma_mgmt_dev *mdev, u64 val);
     24};
     25
     26#define IMPLEMENT_GETSET(name)					\
     27static int get_##name(struct hidma_mgmt_dev *mdev)		\
     28{								\
     29	return mdev->name;					\
     30}								\
     31static int set_##name(struct hidma_mgmt_dev *mdev, u64 val)	\
     32{								\
     33	u64 tmp;						\
     34	int rc;							\
     35								\
     36	tmp = mdev->name;					\
     37	mdev->name = val;					\
     38	rc = hidma_mgmt_setup(mdev);				\
     39	if (rc)							\
     40		mdev->name = tmp;				\
     41	return rc;						\
     42}
     43
     44#define DECLARE_ATTRIBUTE(name, mode)				\
     45	{#name, mode, get_##name, set_##name}
     46
     47IMPLEMENT_GETSET(hw_version_major)
     48IMPLEMENT_GETSET(hw_version_minor)
     49IMPLEMENT_GETSET(max_wr_xactions)
     50IMPLEMENT_GETSET(max_rd_xactions)
     51IMPLEMENT_GETSET(max_write_request)
     52IMPLEMENT_GETSET(max_read_request)
     53IMPLEMENT_GETSET(dma_channels)
     54IMPLEMENT_GETSET(chreset_timeout_cycles)
     55
     56static int set_priority(struct hidma_mgmt_dev *mdev, unsigned int i, u64 val)
     57{
     58	u64 tmp;
     59	int rc;
     60
     61	if (i >= mdev->dma_channels)
     62		return -EINVAL;
     63
     64	tmp = mdev->priority[i];
     65	mdev->priority[i] = val;
     66	rc = hidma_mgmt_setup(mdev);
     67	if (rc)
     68		mdev->priority[i] = tmp;
     69	return rc;
     70}
     71
     72static int set_weight(struct hidma_mgmt_dev *mdev, unsigned int i, u64 val)
     73{
     74	u64 tmp;
     75	int rc;
     76
     77	if (i >= mdev->dma_channels)
     78		return -EINVAL;
     79
     80	tmp = mdev->weight[i];
     81	mdev->weight[i] = val;
     82	rc = hidma_mgmt_setup(mdev);
     83	if (rc)
     84		mdev->weight[i] = tmp;
     85	return rc;
     86}
     87
     88static struct hidma_mgmt_fileinfo hidma_mgmt_files[] = {
     89	DECLARE_ATTRIBUTE(hw_version_major, S_IRUGO),
     90	DECLARE_ATTRIBUTE(hw_version_minor, S_IRUGO),
     91	DECLARE_ATTRIBUTE(dma_channels, S_IRUGO),
     92	DECLARE_ATTRIBUTE(chreset_timeout_cycles, S_IRUGO),
     93	DECLARE_ATTRIBUTE(max_wr_xactions, S_IRUGO),
     94	DECLARE_ATTRIBUTE(max_rd_xactions, S_IRUGO),
     95	DECLARE_ATTRIBUTE(max_write_request, S_IRUGO),
     96	DECLARE_ATTRIBUTE(max_read_request, S_IRUGO),
     97};
     98
     99static ssize_t show_values(struct device *dev, struct device_attribute *attr,
    100			   char *buf)
    101{
    102	struct hidma_mgmt_dev *mdev = dev_get_drvdata(dev);
    103	unsigned int i;
    104
    105	buf[0] = 0;
    106
    107	for (i = 0; i < ARRAY_SIZE(hidma_mgmt_files); i++) {
    108		if (strcmp(attr->attr.name, hidma_mgmt_files[i].name) == 0) {
    109			sprintf(buf, "%d\n", hidma_mgmt_files[i].get(mdev));
    110			break;
    111		}
    112	}
    113	return strlen(buf);
    114}
    115
    116static ssize_t set_values(struct device *dev, struct device_attribute *attr,
    117			  const char *buf, size_t count)
    118{
    119	struct hidma_mgmt_dev *mdev = dev_get_drvdata(dev);
    120	unsigned long tmp;
    121	unsigned int i;
    122	int rc;
    123
    124	rc = kstrtoul(buf, 0, &tmp);
    125	if (rc)
    126		return rc;
    127
    128	for (i = 0; i < ARRAY_SIZE(hidma_mgmt_files); i++) {
    129		if (strcmp(attr->attr.name, hidma_mgmt_files[i].name) == 0) {
    130			rc = hidma_mgmt_files[i].set(mdev, tmp);
    131			if (rc)
    132				return rc;
    133
    134			break;
    135		}
    136	}
    137	return count;
    138}
    139
    140static ssize_t show_values_channel(struct kobject *kobj,
    141				   struct kobj_attribute *attr, char *buf)
    142{
    143	struct hidma_chan_attr *chattr;
    144	struct hidma_mgmt_dev *mdev;
    145
    146	buf[0] = 0;
    147	chattr = container_of(attr, struct hidma_chan_attr, attr);
    148	mdev = chattr->mdev;
    149	if (strcmp(attr->attr.name, "priority") == 0)
    150		sprintf(buf, "%d\n", mdev->priority[chattr->index]);
    151	else if (strcmp(attr->attr.name, "weight") == 0)
    152		sprintf(buf, "%d\n", mdev->weight[chattr->index]);
    153
    154	return strlen(buf);
    155}
    156
    157static ssize_t set_values_channel(struct kobject *kobj,
    158				  struct kobj_attribute *attr, const char *buf,
    159				  size_t count)
    160{
    161	struct hidma_chan_attr *chattr;
    162	struct hidma_mgmt_dev *mdev;
    163	unsigned long tmp;
    164	int rc;
    165
    166	chattr = container_of(attr, struct hidma_chan_attr, attr);
    167	mdev = chattr->mdev;
    168
    169	rc = kstrtoul(buf, 0, &tmp);
    170	if (rc)
    171		return rc;
    172
    173	if (strcmp(attr->attr.name, "priority") == 0) {
    174		rc = set_priority(mdev, chattr->index, tmp);
    175		if (rc)
    176			return rc;
    177	} else if (strcmp(attr->attr.name, "weight") == 0) {
    178		rc = set_weight(mdev, chattr->index, tmp);
    179		if (rc)
    180			return rc;
    181	}
    182	return count;
    183}
    184
    185static int create_sysfs_entry(struct hidma_mgmt_dev *dev, char *name, int mode)
    186{
    187	struct device_attribute *attrs;
    188	char *name_copy;
    189
    190	attrs = devm_kmalloc(&dev->pdev->dev,
    191			     sizeof(struct device_attribute), GFP_KERNEL);
    192	if (!attrs)
    193		return -ENOMEM;
    194
    195	name_copy = devm_kstrdup(&dev->pdev->dev, name, GFP_KERNEL);
    196	if (!name_copy)
    197		return -ENOMEM;
    198
    199	attrs->attr.name = name_copy;
    200	attrs->attr.mode = mode;
    201	attrs->show = show_values;
    202	attrs->store = set_values;
    203	sysfs_attr_init(&attrs->attr);
    204
    205	return device_create_file(&dev->pdev->dev, attrs);
    206}
    207
    208static int create_sysfs_entry_channel(struct hidma_mgmt_dev *mdev, char *name,
    209				      int mode, int index,
    210				      struct kobject *parent)
    211{
    212	struct hidma_chan_attr *chattr;
    213	char *name_copy;
    214
    215	chattr = devm_kmalloc(&mdev->pdev->dev, sizeof(*chattr), GFP_KERNEL);
    216	if (!chattr)
    217		return -ENOMEM;
    218
    219	name_copy = devm_kstrdup(&mdev->pdev->dev, name, GFP_KERNEL);
    220	if (!name_copy)
    221		return -ENOMEM;
    222
    223	chattr->mdev = mdev;
    224	chattr->index = index;
    225	chattr->attr.attr.name = name_copy;
    226	chattr->attr.attr.mode = mode;
    227	chattr->attr.show = show_values_channel;
    228	chattr->attr.store = set_values_channel;
    229	sysfs_attr_init(&chattr->attr.attr);
    230
    231	return sysfs_create_file(parent, &chattr->attr.attr);
    232}
    233
    234int hidma_mgmt_init_sys(struct hidma_mgmt_dev *mdev)
    235{
    236	unsigned int i;
    237	int rc;
    238	int required;
    239	struct kobject *chanops;
    240
    241	required = sizeof(*mdev->chroots) * mdev->dma_channels;
    242	mdev->chroots = devm_kmalloc(&mdev->pdev->dev, required, GFP_KERNEL);
    243	if (!mdev->chroots)
    244		return -ENOMEM;
    245
    246	chanops = kobject_create_and_add("chanops", &mdev->pdev->dev.kobj);
    247	if (!chanops)
    248		return -ENOMEM;
    249
    250	/* create each channel directory here */
    251	for (i = 0; i < mdev->dma_channels; i++) {
    252		char name[20];
    253
    254		snprintf(name, sizeof(name), "chan%d", i);
    255		mdev->chroots[i] = kobject_create_and_add(name, chanops);
    256		if (!mdev->chroots[i])
    257			return -ENOMEM;
    258	}
    259
    260	/* populate common parameters */
    261	for (i = 0; i < ARRAY_SIZE(hidma_mgmt_files); i++) {
    262		rc = create_sysfs_entry(mdev, hidma_mgmt_files[i].name,
    263					hidma_mgmt_files[i].mode);
    264		if (rc)
    265			return rc;
    266	}
    267
    268	/* populate parameters that are per channel */
    269	for (i = 0; i < mdev->dma_channels; i++) {
    270		rc = create_sysfs_entry_channel(mdev, "priority",
    271						(S_IRUGO | S_IWUGO), i,
    272						mdev->chroots[i]);
    273		if (rc)
    274			return rc;
    275
    276		rc = create_sysfs_entry_channel(mdev, "weight",
    277						(S_IRUGO | S_IWUGO), i,
    278						mdev->chroots[i]);
    279		if (rc)
    280			return rc;
    281	}
    282
    283	return 0;
    284}
    285EXPORT_SYMBOL_GPL(hidma_mgmt_init_sys);