kernel_samsung_a53x/sound/soc/samsung/abox/abox_proc.c

260 lines
6.1 KiB
C
Raw Normal View History

2024-06-15 16:02:09 -03:00
// SPDX-License-Identifier: GPL-2.0-or-later
/*
* ALSA SoC - Samsung Abox Proc FS driver
*
* Copyright (c) 2020 Samsung Electronics Co. Ltd.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation.
*/
#include <linux/debugfs.h>
#include <linux/proc_fs.h>
#include <linux/device.h>
#include <linux/slab.h>
#include "abox.h"
#include "abox_proc.h"
#define ROOT_DIR_NAME "abox"
static struct proc_dir_entry *root;
/* for backward compatibility */
static struct dentry *debugfs_link;
void *abox_proc_data(const struct file *file)
{
return PDE_DATA(file_inode(file));
}
struct proc_dir_entry *abox_proc_mkdir(const char *name,
struct proc_dir_entry *parent)
{
if (!parent)
parent = root;
return proc_mkdir(name, parent);
}
void abox_proc_remove_file(struct proc_dir_entry *pde)
{
proc_remove(pde);
}
struct proc_dir_entry *abox_proc_create_file(const char *name, umode_t mode,
struct proc_dir_entry *parent,
const struct proc_ops *fops, void *data, size_t size)
{
struct proc_dir_entry *pde;
if (!parent)
parent = root;
pde = proc_create_data(name, mode, parent, fops, data);
if (!IS_ERR(pde))
proc_set_size(pde, size);
return pde;
}
static ssize_t abox_proc_bin_read(struct file *file, char __user *buf,
size_t count, loff_t *pos)
{
struct abox_proc_bin *bin = abox_proc_data(file);
return simple_read_from_buffer(buf, count, pos, bin->data, bin->size);
}
static const struct proc_ops abox_proc_bin_fops = {
.proc_read = abox_proc_bin_read,
.proc_open = simple_open,
.proc_lseek = default_llseek,
};
struct proc_dir_entry *abox_proc_create_bin(const char *name, umode_t mode,
struct proc_dir_entry *parent, struct abox_proc_bin *bin)
{
return abox_proc_create_file(name, mode, parent, &abox_proc_bin_fops,
bin, bin->size);
}
static ssize_t abox_proc_u64_read(struct file *file, char __user *buf,
size_t count, loff_t *pos)
{
unsigned long long *val = abox_proc_data(file);
char buf_val[24]; /* enough to store a u64 and "\n\0" */
size_t size;
size = scnprintf(buf_val, sizeof(buf_val), "%llu\n", *val);
return simple_read_from_buffer(buf, count, pos, buf_val, size);
}
static ssize_t abox_proc_u64_write(struct file *file, const char __user *buf,
size_t count, loff_t *pos)
{
unsigned long long *val = abox_proc_data(file);
int ret;
ret = kstrtoull_from_user(buf, count, 0, val);
if (ret < 0)
return ret;
return count;
}
static const struct proc_ops abox_proc_u64_fops = {
.proc_read = abox_proc_u64_read,
.proc_write = abox_proc_u64_write,
.proc_open = simple_open,
.proc_lseek = no_llseek,
};
struct proc_dir_entry *abox_proc_create_u64(const char *name, umode_t mode,
struct proc_dir_entry *parent, u64 *data)
{
return abox_proc_create_file(name, mode, parent, &abox_proc_u64_fops,
data, 0);
}
static ssize_t abox_proc_u32_read(struct file *file, char __user *buf,
size_t count, loff_t *pos)
{
unsigned int *val = abox_proc_data(file);
char buf_val[24]; /* enough to store a u32 and "\n\0" */
size_t size;
size = scnprintf(buf_val, sizeof(buf_val), "%u\n", *val);
return simple_read_from_buffer(buf, count, pos, buf_val, size);
}
static ssize_t abox_proc_u32_write(struct file *file, const char __user *buf,
size_t count, loff_t *pos)
{
unsigned int *val = abox_proc_data(file);
int ret;
ret = kstrtouint_from_user(buf, count, 0, val);
if (ret < 0)
return ret;
return count;
}
static const struct proc_ops abox_proc_u32_fops = {
.proc_read = abox_proc_u32_read,
.proc_write = abox_proc_u32_write,
.proc_open = simple_open,
.proc_lseek = no_llseek,
};
struct proc_dir_entry *abox_proc_create_u32(const char *name, umode_t mode,
struct proc_dir_entry *parent, u32 *data)
{
return abox_proc_create_file(name, mode, parent, &abox_proc_u32_fops,
data, 0);
}
static ssize_t abox_proc_bool_read(struct file *file, char __user *buf,
size_t count, loff_t *pos)
{
bool *val = abox_proc_data(file);
char buf_val[4]; /* enough to store a bool and "\n\0" */
if (*val)
buf_val[0] = 'Y';
else
buf_val[0] = 'N';
buf_val[1] = '\n';
buf_val[2] = 0x00;
return simple_read_from_buffer(buf, count, pos, buf_val, 2);
}
static ssize_t abox_proc_bool_write(struct file *file, const char __user *buf,
size_t count, loff_t *pos)
{
bool *val = abox_proc_data(file);
int ret;
ret = kstrtobool_from_user(buf, count, val);
if (ret < 0)
return ret;
return count;
}
static const struct proc_ops abox_proc_bool_fops = {
.proc_read = abox_proc_bool_read,
.proc_write = abox_proc_bool_write,
.proc_open = simple_open,
.proc_lseek = no_llseek,
};
struct proc_dir_entry *abox_proc_create_bool(const char *name, umode_t mode,
struct proc_dir_entry *parent, bool *data)
{
return abox_proc_create_file(name, mode, parent, &abox_proc_bool_fops,
data, 0);
}
int abox_proc_symlink_attr(struct device *dev, const char *name,
struct proc_dir_entry *parent)
{
char *attr_path;
struct proc_dir_entry *pde;
if (!parent)
parent = root;
if (is_abox(dev))
attr_path = kasprintf(GFP_KERNEL,
"/sys/devices/platform/%s/%s",
dev_name(dev), name);
else
attr_path = kasprintf(GFP_KERNEL,
"/sys/devices/platform/%s/%s/%s",
dev_name(dev->parent), dev_name(dev), name);
pde = proc_symlink(name, parent, attr_path);
kfree(attr_path);
return PTR_ERR_OR_ZERO(pde);
}
int abox_proc_symlink_dev(struct device *dev, const char *name)
{
char *sys_path;
struct proc_dir_entry *pde;
if (is_abox(dev))
sys_path = kasprintf(GFP_KERNEL, "/sys/devices/platform/%s",
dev_name(dev));
else
sys_path = kasprintf(GFP_KERNEL, "/sys/devices/platform/%s/%s",
dev_name(dev->parent), dev_name(dev));
pde = proc_symlink(name, root, sys_path);
kfree(sys_path);
return PTR_ERR_OR_ZERO(pde);
}
int abox_proc_probe(void)
{
root = proc_mkdir(ROOT_DIR_NAME, NULL);
/* for backward compatibility */
debugfs_link = debugfs_create_symlink(ROOT_DIR_NAME, NULL,
"/proc/"ROOT_DIR_NAME);
return PTR_ERR_OR_ZERO(root);
}
void abox_proc_remove(void)
{
/* for backward compatibility */
debugfs_remove(debugfs_link);
proc_remove(root);
}