2019-05-19 13:08:20 +01:00
// SPDX-License-Identifier: GPL-2.0-only
2011-05-26 12:26:24 +02:00
/*
2013-01-17 14:24:05 +01:00
* custom_method . c - debugfs interface for customizing ACPI control method
2011-05-26 12:26:24 +02:00
*/
# include <linux/init.h>
# include <linux/module.h>
# include <linux/kernel.h>
# include <linux/uaccess.h>
# include <linux/debugfs.h>
2013-12-03 08:49:16 +08:00
# include <linux/acpi.h>
2019-08-19 17:17:50 -07:00
# include <linux/security.h>
2011-05-26 12:26:24 +02:00
# include "internal.h"
MODULE_LICENSE ( " GPL " ) ;
static struct dentry * cm_dentry ;
/* /sys/kernel/debug/acpi/custom_method */
2021-03-27 20:08:21 +08:00
static ssize_t cm_write ( struct file * file , const char __user * user_buf ,
2011-05-26 12:26:24 +02:00
size_t count , loff_t * ppos )
{
static char * buf ;
static u32 max_size ;
static u32 uncopied_bytes ;
struct acpi_table_header table ;
acpi_status status ;
2019-08-19 17:17:50 -07:00
int ret ;
ret = security_locked_down ( LOCKDOWN_ACPI_TABLES ) ;
if ( ret )
return ret ;
2011-05-26 12:26:24 +02:00
if ( ! ( * ppos ) ) {
/* parse the table header to get the table length */
if ( count < = sizeof ( struct acpi_table_header ) )
return - EINVAL ;
if ( copy_from_user ( & table , user_buf ,
sizeof ( struct acpi_table_header ) ) )
return - EFAULT ;
uncopied_bytes = max_size = table . length ;
2021-04-27 13:54:33 -05:00
/* make sure the buf is not allocated */
kfree ( buf ) ;
2011-05-26 12:26:24 +02:00
buf = kzalloc ( max_size , GFP_KERNEL ) ;
if ( ! buf )
return - ENOMEM ;
}
if ( buf = = NULL )
return - EINVAL ;
if ( ( * ppos > max_size ) | |
( * ppos + count > max_size ) | |
( * ppos + count < count ) | |
2019-08-16 00:08:27 -05:00
( count > uncopied_bytes ) ) {
kfree ( buf ) ;
2021-04-23 10:28:17 -05:00
buf = NULL ;
2011-05-26 12:26:24 +02:00
return - EINVAL ;
2019-08-16 00:08:27 -05:00
}
2011-05-26 12:26:24 +02:00
if ( copy_from_user ( buf + ( * ppos ) , user_buf , count ) ) {
kfree ( buf ) ;
buf = NULL ;
return - EFAULT ;
}
uncopied_bytes - = count ;
* ppos + = count ;
if ( ! uncopied_bytes ) {
status = acpi_install_method ( buf ) ;
kfree ( buf ) ;
buf = NULL ;
if ( ACPI_FAILURE ( status ) )
return - EINVAL ;
2013-01-21 17:17:39 +10:30
add_taint ( TAINT_OVERRIDDEN_ACPI_TABLE , LOCKDEP_NOW_UNRELIABLE ) ;
2011-05-26 12:26:24 +02:00
}
return count ;
}
static const struct file_operations cm_fops = {
. write = cm_write ,
. llseek = default_llseek ,
} ;
static int __init acpi_custom_method_init ( void )
{
cm_dentry = debugfs_create_file ( " custom_method " , S_IWUSR ,
acpi_debugfs_dir , NULL , & cm_fops ) ;
return 0 ;
}
static void __exit acpi_custom_method_exit ( void )
{
2018-08-18 18:49:20 +08:00
debugfs_remove ( cm_dentry ) ;
2018-03-21 15:09:32 -07:00
}
2011-05-26 12:26:24 +02:00
module_init ( acpi_custom_method_init ) ;
module_exit ( acpi_custom_method_exit ) ;