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.
189 lines
4.1 KiB
189 lines
4.1 KiB
3 years ago
|
/*
|
||
|
* Copyright (c) 2019 Samsung Electronics Co., Ltd.
|
||
|
* http://www.samsung.com
|
||
|
*
|
||
|
* Samsung TN debugging code
|
||
|
*
|
||
|
* 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.
|
||
|
*/
|
||
|
|
||
|
#define pr_fmt(fmt) KBUILD_MODNAME ":%s() " fmt, __func__
|
||
|
|
||
|
#include <linux/kernel.h>
|
||
|
#include <linux/module.h>
|
||
|
#include <linux/fs.h>
|
||
|
#include <linux/file.h>
|
||
|
#include <linux/slab.h>
|
||
|
#include <linux/uaccess.h>
|
||
|
#include <linux/device.h>
|
||
|
|
||
|
#include <linux/sec_param.h>
|
||
|
#include <linux/sec_class.h>
|
||
|
|
||
|
#define SEC_EDTBO_FILENAME "/spu/edtbo/edtbo.img"
|
||
|
#define EDTBO_FIEMAP_MAGIC 0x7763
|
||
|
#define EDTBO_FIEMAP_COUNT 128
|
||
|
|
||
|
static unsigned long edtbo_ver = 0;
|
||
|
|
||
|
static int __init init_sysup_edtbo(char *str)
|
||
|
{
|
||
|
edtbo_ver = simple_strtoul(str, NULL, 10);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
early_param("androidboot.edtbo", init_sysup_edtbo);
|
||
|
|
||
|
static int fiemap_chk_ranges(struct super_block *sb,
|
||
|
u64 start, u64 len, u64 *new_len)
|
||
|
{
|
||
|
u64 maxbytes = (u64) sb->s_maxbytes;
|
||
|
|
||
|
*new_len = len;
|
||
|
|
||
|
if (len == 0)
|
||
|
return -EINVAL;
|
||
|
|
||
|
if (start > maxbytes)
|
||
|
return -EFBIG;
|
||
|
|
||
|
/*
|
||
|
* Shrink request scope to what the fs can actually handle.
|
||
|
*/
|
||
|
if (len > maxbytes || (maxbytes - len) < start)
|
||
|
*new_len = maxbytes - start;
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static ssize_t sec_edtbo_update_store(struct device *dev,
|
||
|
struct device_attribute *attr,
|
||
|
const char *buf, size_t size)
|
||
|
{
|
||
|
struct fiemap *pfiemap;
|
||
|
struct fiemap_extent_info fieinfo = { 0, };
|
||
|
struct file *fp;
|
||
|
mm_segment_t old_fs;
|
||
|
struct inode *inode;
|
||
|
u64 len;
|
||
|
int update, error;
|
||
|
u32 allocsize = 0;
|
||
|
u32 fiemap_magic = EDTBO_FIEMAP_MAGIC;
|
||
|
|
||
|
error = sscanf(buf, "%d", &update);
|
||
|
if (error < 0 || update != 1)
|
||
|
return -EINVAL;
|
||
|
|
||
|
old_fs = get_fs();
|
||
|
set_fs(KERNEL_DS);
|
||
|
|
||
|
fp = filp_open(SEC_EDTBO_FILENAME, O_RDONLY, S_IRUSR);
|
||
|
if (IS_ERR(fp)) {
|
||
|
pr_err("file open error\n");
|
||
|
error = -ENOENT;
|
||
|
goto fp_err;
|
||
|
}
|
||
|
|
||
|
inode = file_inode(fp);
|
||
|
if (!inode->i_op->fiemap) {
|
||
|
error = -EOPNOTSUPP;
|
||
|
goto open_err;
|
||
|
}
|
||
|
|
||
|
allocsize = EDTBO_FIEMAP_COUNT * sizeof(struct fiemap_extent) + offsetof(struct fiemap, fm_extents);
|
||
|
pfiemap = kmalloc(allocsize, GFP_KERNEL | __GFP_ZERO);
|
||
|
if (!pfiemap) {
|
||
|
error = -ENOMEM;
|
||
|
goto open_err;
|
||
|
}
|
||
|
|
||
|
pfiemap->fm_length = ULONG_MAX;
|
||
|
pfiemap->fm_extent_count = EDTBO_FIEMAP_COUNT;
|
||
|
|
||
|
error = fiemap_chk_ranges(inode->i_sb, pfiemap->fm_start, pfiemap->fm_length, &len);
|
||
|
if (error)
|
||
|
goto alloc_err;
|
||
|
|
||
|
fieinfo.fi_flags = pfiemap->fm_flags;
|
||
|
fieinfo.fi_extents_max = pfiemap->fm_extent_count;
|
||
|
fieinfo.fi_extents_start = pfiemap->fm_extents;
|
||
|
|
||
|
error = inode->i_op->fiemap(inode, &fieinfo, pfiemap->fm_start, len);
|
||
|
if (error)
|
||
|
goto alloc_err;
|
||
|
|
||
|
pfiemap->fm_flags = fieinfo.fi_flags;
|
||
|
pfiemap->fm_mapped_extents = fieinfo.fi_extents_mapped;
|
||
|
|
||
|
if (!pfiemap->fm_mapped_extents) {
|
||
|
error = -EFAULT;
|
||
|
goto alloc_err;
|
||
|
}
|
||
|
|
||
|
sec_set_param(param_index_fiemap_result, pfiemap);
|
||
|
sec_set_param(param_index_fiemap_update, &fiemap_magic);
|
||
|
|
||
|
error = size;
|
||
|
|
||
|
alloc_err:
|
||
|
if (pfiemap)
|
||
|
kfree(pfiemap);
|
||
|
open_err:
|
||
|
filp_close(fp, NULL);
|
||
|
fp_err:
|
||
|
set_fs(old_fs);
|
||
|
|
||
|
return error;
|
||
|
}
|
||
|
|
||
|
static ssize_t sec_edtbo_version_show(struct device *dev,
|
||
|
struct device_attribute *attr, char *buf)
|
||
|
{
|
||
|
return sprintf(buf, "%lu\n", edtbo_ver);
|
||
|
}
|
||
|
|
||
|
static struct device_attribute sec_sysup_edtbo_update_attr =
|
||
|
__ATTR(edtbo_update, 0220, NULL, sec_edtbo_update_store);
|
||
|
|
||
|
static struct device_attribute sec_sysup_edtbo_version_attr =
|
||
|
__ATTR(edtbo_version, 0440, sec_edtbo_version_show, NULL);
|
||
|
|
||
|
static struct attribute *sec_sysup_attributes[] = {
|
||
|
&sec_sysup_edtbo_update_attr.attr,
|
||
|
&sec_sysup_edtbo_version_attr.attr,
|
||
|
NULL,
|
||
|
};
|
||
|
|
||
|
static struct attribute_group sec_sysup_attr_group = {
|
||
|
.attrs = sec_sysup_attributes,
|
||
|
};
|
||
|
|
||
|
static int __init sec_sysup_init(void)
|
||
|
{
|
||
|
int ret = 0;
|
||
|
struct device *dev;
|
||
|
|
||
|
pr_info("start\n");
|
||
|
|
||
|
dev = ___sec_device_create(NULL, "sec_sysup");
|
||
|
if(!dev)
|
||
|
pr_err("sec device create failed!\n");
|
||
|
|
||
|
ret = sysfs_create_group(&dev->kobj, &sec_sysup_attr_group);
|
||
|
if (ret)
|
||
|
pr_err("could not create sysfs node\n");
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static void __exit sec_sysup_exit(void)
|
||
|
{
|
||
|
pr_info("exit\n");
|
||
|
}
|
||
|
|
||
|
module_init(sec_sysup_init);
|
||
|
module_exit(sec_sysup_exit);
|
||
|
|