2016-07-08 19:13:13 +03:00
/*
* ACPI configfs support
*
* Copyright ( c ) 2016 Intel Corporation
*
* 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 .
*/
2016-07-08 19:13:14 +03:00
# define pr_fmt(fmt) "ACPI configfs: " fmt
2016-07-08 19:13:13 +03:00
# include <linux/init.h>
# include <linux/module.h>
# include <linux/configfs.h>
# include <linux/acpi.h>
2017-06-09 21:36:31 +03:00
# include "acpica/accommon.h"
# include "acpica/actables.h"
2016-07-08 19:13:14 +03:00
static struct config_group * acpi_table_group ;
struct acpi_table {
struct config_item cfg ;
struct acpi_table_header * header ;
2017-06-09 21:36:31 +03:00
u32 index ;
2016-07-08 19:13:14 +03:00
} ;
static ssize_t acpi_table_aml_write ( struct config_item * cfg ,
const void * data , size_t size )
{
const struct acpi_table_header * header = data ;
struct acpi_table * table ;
int ret ;
table = container_of ( cfg , struct acpi_table , cfg ) ;
if ( table - > header ) {
pr_err ( " table already loaded \n " ) ;
return - EBUSY ;
}
if ( header - > length ! = size ) {
pr_err ( " invalid table length \n " ) ;
return - EINVAL ;
}
if ( memcmp ( header - > signature , ACPI_SIG_SSDT , 4 ) ) {
pr_err ( " invalid table signature \n " ) ;
return - EINVAL ;
}
table = container_of ( cfg , struct acpi_table , cfg ) ;
table - > header = kmemdup ( header , header - > length , GFP_KERNEL ) ;
if ( ! table - > header )
return - ENOMEM ;
2017-06-09 21:36:31 +03:00
ACPI_INFO ( ( " Host-directed Dynamic ACPI Table Load: " ) ) ;
ret = acpi_tb_install_and_load_table (
ACPI_PTR_TO_PHYSADDR ( table - > header ) ,
ACPI_TABLE_ORIGIN_EXTERNAL_VIRTUAL , FALSE ,
& table - > index ) ;
2016-07-08 19:13:14 +03:00
if ( ret ) {
kfree ( table - > header ) ;
table - > header = NULL ;
}
return ret ;
}
static inline struct acpi_table_header * get_header ( struct config_item * cfg )
{
struct acpi_table * table = container_of ( cfg , struct acpi_table , cfg ) ;
if ( ! table - > header )
pr_err ( " table not loaded \n " ) ;
return table - > header ;
}
static ssize_t acpi_table_aml_read ( struct config_item * cfg ,
void * data , size_t size )
{
struct acpi_table_header * h = get_header ( cfg ) ;
if ( ! h )
return - EINVAL ;
if ( data )
memcpy ( data , h , h - > length ) ;
return h - > length ;
}
# define MAX_ACPI_TABLE_SIZE (128 * 1024)
CONFIGFS_BIN_ATTR ( acpi_table_ , aml , NULL , MAX_ACPI_TABLE_SIZE ) ;
struct configfs_bin_attribute * acpi_table_bin_attrs [ ] = {
& acpi_table_attr_aml ,
NULL ,
} ;
ssize_t acpi_table_signature_show ( struct config_item * cfg , char * str )
{
struct acpi_table_header * h = get_header ( cfg ) ;
if ( ! h )
return - EINVAL ;
return sprintf ( str , " %.*s \n " , ACPI_NAME_SIZE , h - > signature ) ;
}
ssize_t acpi_table_length_show ( struct config_item * cfg , char * str )
{
struct acpi_table_header * h = get_header ( cfg ) ;
if ( ! h )
return - EINVAL ;
return sprintf ( str , " %d \n " , h - > length ) ;
}
ssize_t acpi_table_revision_show ( struct config_item * cfg , char * str )
{
struct acpi_table_header * h = get_header ( cfg ) ;
if ( ! h )
return - EINVAL ;
return sprintf ( str , " %d \n " , h - > revision ) ;
}
ssize_t acpi_table_oem_id_show ( struct config_item * cfg , char * str )
{
struct acpi_table_header * h = get_header ( cfg ) ;
if ( ! h )
return - EINVAL ;
return sprintf ( str , " %.*s \n " , ACPI_OEM_ID_SIZE , h - > oem_id ) ;
}
ssize_t acpi_table_oem_table_id_show ( struct config_item * cfg , char * str )
{
struct acpi_table_header * h = get_header ( cfg ) ;
if ( ! h )
return - EINVAL ;
return sprintf ( str , " %.*s \n " , ACPI_OEM_TABLE_ID_SIZE , h - > oem_table_id ) ;
}
ssize_t acpi_table_oem_revision_show ( struct config_item * cfg , char * str )
{
struct acpi_table_header * h = get_header ( cfg ) ;
if ( ! h )
return - EINVAL ;
return sprintf ( str , " %d \n " , h - > oem_revision ) ;
}
ssize_t acpi_table_asl_compiler_id_show ( struct config_item * cfg , char * str )
{
struct acpi_table_header * h = get_header ( cfg ) ;
if ( ! h )
return - EINVAL ;
return sprintf ( str , " %.*s \n " , ACPI_NAME_SIZE , h - > asl_compiler_id ) ;
}
ssize_t acpi_table_asl_compiler_revision_show ( struct config_item * cfg ,
char * str )
{
struct acpi_table_header * h = get_header ( cfg ) ;
if ( ! h )
return - EINVAL ;
return sprintf ( str , " %d \n " , h - > asl_compiler_revision ) ;
}
CONFIGFS_ATTR_RO ( acpi_table_ , signature ) ;
CONFIGFS_ATTR_RO ( acpi_table_ , length ) ;
CONFIGFS_ATTR_RO ( acpi_table_ , revision ) ;
CONFIGFS_ATTR_RO ( acpi_table_ , oem_id ) ;
CONFIGFS_ATTR_RO ( acpi_table_ , oem_table_id ) ;
CONFIGFS_ATTR_RO ( acpi_table_ , oem_revision ) ;
CONFIGFS_ATTR_RO ( acpi_table_ , asl_compiler_id ) ;
CONFIGFS_ATTR_RO ( acpi_table_ , asl_compiler_revision ) ;
struct configfs_attribute * acpi_table_attrs [ ] = {
& acpi_table_attr_signature ,
& acpi_table_attr_length ,
& acpi_table_attr_revision ,
& acpi_table_attr_oem_id ,
& acpi_table_attr_oem_table_id ,
& acpi_table_attr_oem_revision ,
& acpi_table_attr_asl_compiler_id ,
& acpi_table_attr_asl_compiler_revision ,
NULL ,
} ;
static struct config_item_type acpi_table_type = {
. ct_owner = THIS_MODULE ,
. ct_bin_attrs = acpi_table_bin_attrs ,
. ct_attrs = acpi_table_attrs ,
} ;
static struct config_item * acpi_table_make_item ( struct config_group * group ,
const char * name )
{
struct acpi_table * table ;
table = kzalloc ( sizeof ( * table ) , GFP_KERNEL ) ;
if ( ! table )
return ERR_PTR ( - ENOMEM ) ;
config_item_init_type_name ( & table - > cfg , name , & acpi_table_type ) ;
return & table - > cfg ;
}
2017-06-09 21:36:31 +03:00
static void acpi_table_drop_item ( struct config_group * group ,
struct config_item * cfg )
{
struct acpi_table * table = container_of ( cfg , struct acpi_table , cfg ) ;
ACPI_INFO ( ( " Host-directed Dynamic ACPI Table Unload " ) ) ;
acpi_tb_unload_table ( table - > index ) ;
}
2016-07-08 19:13:14 +03:00
struct configfs_group_operations acpi_table_group_ops = {
. make_item = acpi_table_make_item ,
2017-06-09 21:36:31 +03:00
. drop_item = acpi_table_drop_item ,
2016-07-08 19:13:14 +03:00
} ;
static struct config_item_type acpi_tables_type = {
. ct_owner = THIS_MODULE ,
. ct_group_ops = & acpi_table_group_ops ,
} ;
2016-07-08 19:13:13 +03:00
static struct config_item_type acpi_root_group_type = {
. ct_owner = THIS_MODULE ,
} ;
static struct configfs_subsystem acpi_configfs = {
. su_group = {
. cg_item = {
. ci_namebuf = " acpi " ,
. ci_type = & acpi_root_group_type ,
} ,
} ,
. su_mutex = __MUTEX_INITIALIZER ( acpi_configfs . su_mutex ) ,
} ;
static int __init acpi_configfs_init ( void )
{
int ret ;
struct config_group * root = & acpi_configfs . su_group ;
config_group_init ( root ) ;
ret = configfs_register_subsystem ( & acpi_configfs ) ;
if ( ret )
return ret ;
2016-07-08 19:13:14 +03:00
acpi_table_group = configfs_register_default_group ( root , " table " ,
& acpi_tables_type ) ;
return PTR_ERR_OR_ZERO ( acpi_table_group ) ;
2016-07-08 19:13:13 +03:00
}
module_init ( acpi_configfs_init ) ;
static void __exit acpi_configfs_exit ( void )
{
2016-07-08 19:13:14 +03:00
configfs_unregister_default_group ( acpi_table_group ) ;
2016-07-08 19:13:13 +03:00
configfs_unregister_subsystem ( & acpi_configfs ) ;
}
module_exit ( acpi_configfs_exit ) ;
MODULE_AUTHOR ( " Octavian Purdila <octavian.purdila@intel.com> " ) ;
MODULE_DESCRIPTION ( " ACPI configfs support " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;