2013-04-05 22:10:34 +03:00
/*
*
* Intel Management Engine Interface ( Intel MEI ) Linux driver
* Copyright ( c ) 2012 - 2013 , Intel Corporation .
*
* This program is free software ; you can redistribute it and / or modify it
* under the terms and conditions of the GNU General Public License ,
* version 2 , as published by the Free Software Foundation .
*
* This program is distributed in the hope it will be useful , but WITHOUT
* ANY WARRANTY ; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE . See the GNU General Public License for
* more details .
*
*/
# include <linux/slab.h>
# include <linux/kernel.h>
# include <linux/device.h>
# include <linux/debugfs.h>
# include <linux/mei.h>
# include "mei_dev.h"
2015-01-11 00:07:16 +02:00
# include "client.h"
2013-04-05 22:10:34 +03:00
# include "hw.h"
static ssize_t mei_dbgfs_read_meclients ( struct file * fp , char __user * ubuf ,
size_t cnt , loff_t * ppos )
{
struct mei_device * dev = fp - > private_data ;
2015-02-10 10:39:31 +02:00
struct mei_me_client * me_cl ;
2014-09-29 16:31:34 +03:00
size_t bufsz = 1 ;
char * buf ;
2014-08-21 14:29:13 +03:00
int i = 0 ;
2013-04-05 22:10:34 +03:00
int pos = 0 ;
int ret ;
2015-01-11 00:07:16 +02:00
# define HDR \
" |id|fix| UUID |con|msg len|sb|refc| \n "
2013-04-05 22:10:34 +03:00
2015-02-10 10:39:31 +02:00
down_read ( & dev - > me_clients_rwsem ) ;
2014-09-29 16:31:34 +03:00
list_for_each_entry ( me_cl , & dev - > me_clients , list )
bufsz + + ;
bufsz * = sizeof ( HDR ) + 1 ;
buf = kzalloc ( bufsz , GFP_KERNEL ) ;
if ( ! buf ) {
2015-02-10 10:39:31 +02:00
up_read ( & dev - > me_clients_rwsem ) ;
2014-09-29 16:31:34 +03:00
return - ENOMEM ;
}
pos + = scnprintf ( buf + pos , bufsz - pos , HDR ) ;
2016-02-07 23:35:17 +02:00
# undef HDR
2014-09-29 16:31:34 +03:00
2014-01-08 22:31:46 +02:00
/* if the driver is not enabled the list won't be consistent */
2013-04-05 22:10:34 +03:00
if ( dev - > dev_state ! = MEI_DEV_ENABLED )
goto out ;
2015-02-10 10:39:31 +02:00
list_for_each_entry ( me_cl , & dev - > me_clients , list ) {
2013-04-05 22:10:34 +03:00
2015-02-10 10:39:31 +02:00
if ( mei_me_cl_get ( me_cl ) ) {
2015-01-11 00:07:16 +02:00
pos + = scnprintf ( buf + pos , bufsz - pos ,
" %2d|%2d|%3d|%pUl|%3d|%7d|%2d|%4d| \n " ,
i + + , me_cl - > client_id ,
me_cl - > props . fixed_address ,
& me_cl - > props . protocol_name ,
me_cl - > props . max_number_of_connections ,
me_cl - > props . max_msg_length ,
me_cl - > props . single_recv_buf ,
atomic_read ( & me_cl - > refcnt . refcount ) ) ;
2015-02-10 10:39:31 +02:00
mei_me_cl_put ( me_cl ) ;
}
2013-04-05 22:10:34 +03:00
}
2015-02-10 10:39:31 +02:00
2013-04-05 22:10:34 +03:00
out :
2015-02-10 10:39:31 +02:00
up_read ( & dev - > me_clients_rwsem ) ;
2013-04-05 22:10:34 +03:00
ret = simple_read_from_buffer ( ubuf , cnt , ppos , buf , pos ) ;
kfree ( buf ) ;
return ret ;
}
static const struct file_operations mei_dbgfs_fops_meclients = {
2013-04-09 14:30:19 +08:00
. open = simple_open ,
2013-04-05 22:10:34 +03:00
. read = mei_dbgfs_read_meclients ,
. llseek = generic_file_llseek ,
} ;
2014-01-14 23:21:41 +02:00
static ssize_t mei_dbgfs_read_active ( struct file * fp , char __user * ubuf ,
size_t cnt , loff_t * ppos )
{
struct mei_device * dev = fp - > private_data ;
struct mei_cl * cl ;
2016-02-07 23:35:17 +02:00
size_t bufsz = 1 ;
2014-01-14 23:21:41 +02:00
char * buf ;
int i = 0 ;
int pos = 0 ;
int ret ;
2016-02-07 23:35:17 +02:00
# define HDR " |me|host|state|rd|wr|\n"
2014-01-14 23:21:41 +02:00
if ( ! dev )
return - ENODEV ;
2016-02-07 23:35:17 +02:00
mutex_lock ( & dev - > device_lock ) ;
/*
* if the driver is not enabled the list won ' t be consistent ,
* we output empty table
*/
if ( dev - > dev_state = = MEI_DEV_ENABLED )
list_for_each_entry ( cl , & dev - > file_list , link )
bufsz + + ;
bufsz * = sizeof ( HDR ) + 1 ;
2014-01-14 23:21:41 +02:00
buf = kzalloc ( bufsz , GFP_KERNEL ) ;
2016-02-07 23:35:17 +02:00
if ( ! buf ) {
mutex_unlock ( & dev - > device_lock ) ;
2014-01-14 23:21:41 +02:00
return - ENOMEM ;
2016-02-07 23:35:17 +02:00
}
2014-01-14 23:21:41 +02:00
2016-02-07 23:35:17 +02:00
pos + = scnprintf ( buf + pos , bufsz - pos , HDR ) ;
# undef HDR
2014-01-14 23:21:41 +02:00
2014-09-29 16:31:47 +03:00
/* if the driver is not enabled the list won't be consistent */
2014-01-14 23:21:41 +02:00
if ( dev - > dev_state ! = MEI_DEV_ENABLED )
goto out ;
list_for_each_entry ( cl , & dev - > file_list , link ) {
pos + = scnprintf ( buf + pos , bufsz - pos ,
2016-02-07 23:35:17 +02:00
" %3d|%2d|%4d|%5d|%2d|%2d| \n " ,
2015-05-04 09:43:54 +03:00
i , mei_cl_me_id ( cl ) , cl - > host_client_id , cl - > state ,
2015-02-10 10:39:46 +02:00
! list_empty ( & cl - > rd_completed ) , cl - > writing_state ) ;
2014-01-14 23:21:41 +02:00
i + + ;
}
out :
mutex_unlock ( & dev - > device_lock ) ;
ret = simple_read_from_buffer ( ubuf , cnt , ppos , buf , pos ) ;
kfree ( buf ) ;
return ret ;
}
static const struct file_operations mei_dbgfs_fops_active = {
. open = simple_open ,
. read = mei_dbgfs_read_active ,
. llseek = generic_file_llseek ,
} ;
2013-04-05 22:10:34 +03:00
static ssize_t mei_dbgfs_read_devstate ( struct file * fp , char __user * ubuf ,
size_t cnt , loff_t * ppos )
{
struct mei_device * dev = fp - > private_data ;
const size_t bufsz = 1024 ;
char * buf = kzalloc ( bufsz , GFP_KERNEL ) ;
int pos = 0 ;
int ret ;
if ( ! buf )
return - ENOMEM ;
2014-09-29 16:31:33 +03:00
pos + = scnprintf ( buf + pos , bufsz - pos , " dev: %s \n " ,
2013-04-05 22:10:34 +03:00
mei_dev_state_str ( dev - > dev_state ) ) ;
2014-09-29 16:31:33 +03:00
pos + = scnprintf ( buf + pos , bufsz - pos , " hbm: %s \n " ,
mei_hbm_state_str ( dev - > hbm_state ) ) ;
2015-05-21 13:35:48 +03:00
2016-02-07 23:35:18 +02:00
if ( dev - > hbm_state > = MEI_HBM_ENUM_CLIENTS & &
dev - > hbm_state < = MEI_HBM_STARTED ) {
2015-05-21 13:35:48 +03:00
pos + = scnprintf ( buf + pos , bufsz - pos , " hbm features: \n " ) ;
pos + = scnprintf ( buf + pos , bufsz - pos , " \t PG: %01d \n " ,
dev - > hbm_f_pg_supported ) ;
2015-07-23 21:37:12 +03:00
pos + = scnprintf ( buf + pos , bufsz - pos , " \t DC: %01d \n " ,
dev - > hbm_f_dc_supported ) ;
2016-02-07 23:35:42 +02:00
pos + = scnprintf ( buf + pos , bufsz - pos , " \t IE: %01d \n " ,
dev - > hbm_f_ie_supported ) ;
2015-07-23 21:37:13 +03:00
pos + = scnprintf ( buf + pos , bufsz - pos , " \t DOT: %01d \n " ,
dev - > hbm_f_dot_supported ) ;
2015-07-26 09:54:17 +03:00
pos + = scnprintf ( buf + pos , bufsz - pos , " \t EV: %01d \n " ,
dev - > hbm_f_ev_supported ) ;
2016-02-07 23:35:38 +02:00
pos + = scnprintf ( buf + pos , bufsz - pos , " \t FA: %01d \n " ,
dev - > hbm_f_fa_supported ) ;
2015-05-21 13:35:48 +03:00
}
2014-09-29 16:31:33 +03:00
pos + = scnprintf ( buf + pos , bufsz - pos , " pg: %s, %s \n " ,
mei_pg_is_enabled ( dev ) ? " ENABLED " : " DISABLED " ,
mei_pg_state_str ( mei_pg_state ( dev ) ) ) ;
2013-04-05 22:10:34 +03:00
ret = simple_read_from_buffer ( ubuf , cnt , ppos , buf , pos ) ;
kfree ( buf ) ;
return ret ;
}
static const struct file_operations mei_dbgfs_fops_devstate = {
2013-04-09 14:30:19 +08:00
. open = simple_open ,
2013-04-05 22:10:34 +03:00
. read = mei_dbgfs_read_devstate ,
. llseek = generic_file_llseek ,
} ;
2016-02-07 23:35:38 +02:00
static ssize_t mei_dbgfs_write_allow_fa ( struct file * file ,
const char __user * user_buf ,
size_t count , loff_t * ppos )
{
struct mei_device * dev ;
int ret ;
dev = container_of ( file - > private_data ,
struct mei_device , allow_fixed_address ) ;
ret = debugfs_write_file_bool ( file , user_buf , count , ppos ) ;
if ( ret < 0 )
return ret ;
dev - > override_fixed_address = true ;
return ret ;
}
static const struct file_operations mei_dbgfs_fops_allow_fa = {
. open = simple_open ,
. read = debugfs_read_file_bool ,
. write = mei_dbgfs_write_allow_fa ,
. llseek = generic_file_llseek ,
} ;
2013-04-05 22:10:34 +03:00
/**
* mei_dbgfs_deregister - Remove the debugfs files and directories
2014-09-29 16:31:49 +03:00
*
* @ dev : the mei device structure
2013-04-05 22:10:34 +03:00
*/
void mei_dbgfs_deregister ( struct mei_device * dev )
{
if ( ! dev - > dbgfs_dir )
return ;
debugfs_remove_recursive ( dev - > dbgfs_dir ) ;
dev - > dbgfs_dir = NULL ;
}
/**
2014-09-29 16:31:49 +03:00
* mei_dbgfs_register - Add the debugfs files
2013-04-05 22:10:34 +03:00
*
2014-09-29 16:31:49 +03:00
* @ dev : the mei device structure
* @ name : the mei device name
2014-09-29 16:31:50 +03:00
*
* Return : 0 on success , < 0 on failure .
2013-04-05 22:10:34 +03:00
*/
int mei_dbgfs_register ( struct mei_device * dev , const char * name )
{
struct dentry * dir , * f ;
2014-09-29 16:31:37 +03:00
2013-04-05 22:10:34 +03:00
dir = debugfs_create_dir ( name , NULL ) ;
if ( ! dir )
return - ENOMEM ;
2015-08-24 15:27:37 +03:00
dev - > dbgfs_dir = dir ;
2013-04-05 22:10:34 +03:00
f = debugfs_create_file ( " meclients " , S_IRUSR , dir ,
dev , & mei_dbgfs_fops_meclients ) ;
if ( ! f ) {
2014-09-29 16:31:42 +03:00
dev_err ( dev - > dev , " meclients: registration failed \n " ) ;
2013-04-05 22:10:34 +03:00
goto err ;
}
2014-01-14 23:21:41 +02:00
f = debugfs_create_file ( " active " , S_IRUSR , dir ,
dev , & mei_dbgfs_fops_active ) ;
if ( ! f ) {
2015-10-26 01:13:49 +02:00
dev_err ( dev - > dev , " active: registration failed \n " ) ;
2014-01-14 23:21:41 +02:00
goto err ;
}
2013-04-05 22:10:34 +03:00
f = debugfs_create_file ( " devstate " , S_IRUSR , dir ,
dev , & mei_dbgfs_fops_devstate ) ;
if ( ! f ) {
2014-09-29 16:31:42 +03:00
dev_err ( dev - > dev , " devstate: registration failed \n " ) ;
2013-04-05 22:10:34 +03:00
goto err ;
}
2016-02-07 23:35:38 +02:00
f = debugfs_create_file ( " allow_fixed_address " , S_IRUSR | S_IWUSR , dir ,
& dev - > allow_fixed_address ,
& mei_dbgfs_fops_allow_fa ) ;
2015-05-04 09:43:57 +03:00
if ( ! f ) {
dev_err ( dev - > dev , " allow_fixed_address: registration failed \n " ) ;
goto err ;
}
2013-04-05 22:10:34 +03:00
return 0 ;
err :
mei_dbgfs_deregister ( dev ) ;
return - ENODEV ;
}