2012-08-22 18:47:22 +08:00
# include <linux/acpi.h>
# include "tpm.h"
# define TPM_PPI_REVISION_ID 1
# define TPM_PPI_FN_VERSION 1
# define TPM_PPI_FN_SUBREQ 2
# define TPM_PPI_FN_GETREQ 3
# define TPM_PPI_FN_GETACT 4
# define TPM_PPI_FN_GETRSP 5
# define TPM_PPI_FN_SUBREQ2 7
# define TPM_PPI_FN_GETOPR 8
# define PPI_TPM_REQ_MAX 22
# define PPI_VS_REQ_START 128
# define PPI_VS_REQ_END 255
# define PPI_VERSION_LEN 3
2013-12-19 20:38:17 +08:00
static const u8 tpm_ppi_uuid [ ] = {
0xA6 , 0xFA , 0xDD , 0x3D ,
0x1B , 0x36 ,
0xB4 , 0x4E ,
0xA4 , 0x24 ,
0x8D , 0x10 , 0x08 , 0x9D , 0x16 , 0x53
} ;
static char tpm_ppi_version [ PPI_VERSION_LEN + 1 ] ;
static acpi_handle tpm_ppi_handle ;
2012-08-22 18:47:22 +08:00
static acpi_status ppi_callback ( acpi_handle handle , u32 level , void * context ,
void * * return_value )
{
2013-12-19 20:38:18 +08:00
union acpi_object * obj ;
2013-12-19 20:38:15 +08:00
2013-12-19 20:38:18 +08:00
if ( ! acpi_check_dsm ( handle , tpm_ppi_uuid , TPM_PPI_REVISION_ID ,
1 < < TPM_PPI_FN_VERSION ) )
2013-12-19 20:38:17 +08:00
return AE_OK ;
2013-12-19 20:38:18 +08:00
/* Cache version string */
obj = acpi_evaluate_dsm_typed ( handle , tpm_ppi_uuid ,
TPM_PPI_REVISION_ID , TPM_PPI_FN_VERSION ,
NULL , ACPI_TYPE_STRING ) ;
if ( obj ) {
strlcpy ( tpm_ppi_version , obj - > string . pointer ,
PPI_VERSION_LEN + 1 ) ;
ACPI_FREE ( obj ) ;
2012-08-22 18:47:22 +08:00
}
2013-12-19 20:38:15 +08:00
2013-12-19 20:38:18 +08:00
* return_value = handle ;
return AE_CTRL_TERMINATE ;
2012-08-22 18:47:22 +08:00
}
2013-12-19 20:38:17 +08:00
static inline union acpi_object *
tpm_eval_dsm ( int func , acpi_object_type type , union acpi_object * argv4 )
2012-08-22 18:47:22 +08:00
{
2013-12-19 20:38:17 +08:00
BUG_ON ( ! tpm_ppi_handle ) ;
return acpi_evaluate_dsm_typed ( tpm_ppi_handle , tpm_ppi_uuid ,
TPM_PPI_REVISION_ID , func , argv4 , type ) ;
2012-08-22 18:47:22 +08:00
}
2013-08-29 20:39:11 +08:00
static ssize_t tpm_show_ppi_version ( struct device * dev ,
struct device_attribute * attr , char * buf )
2012-08-22 18:47:22 +08:00
{
2013-12-19 20:38:17 +08:00
return scnprintf ( buf , PAGE_SIZE , " %s \n " , tpm_ppi_version ) ;
2012-08-22 18:47:22 +08:00
}
2013-08-29 20:39:11 +08:00
static ssize_t tpm_show_ppi_request ( struct device * dev ,
struct device_attribute * attr , char * buf )
2012-08-22 18:47:22 +08:00
{
2013-12-19 20:38:17 +08:00
ssize_t size = - EINVAL ;
union acpi_object * obj ;
obj = tpm_eval_dsm ( TPM_PPI_FN_GETREQ , ACPI_TYPE_PACKAGE , NULL ) ;
if ( ! obj )
2012-08-22 18:47:22 +08:00
return - ENXIO ;
/*
* output . pointer should be of package type , including two integers .
* The first is function return code , 0 means success and 1 means
* error . The second is pending TPM operation requested by the OS , 0
* means none and > 0 means operation value .
*/
2013-12-19 20:38:17 +08:00
if ( obj - > package . count = = 2 & &
obj - > package . elements [ 0 ] . type = = ACPI_TYPE_INTEGER & &
obj - > package . elements [ 1 ] . type = = ACPI_TYPE_INTEGER ) {
if ( obj - > package . elements [ 0 ] . integer . value )
size = - EFAULT ;
2012-08-22 18:47:22 +08:00
else
2013-12-19 20:38:17 +08:00
size = scnprintf ( buf , PAGE_SIZE , " %llu \n " ,
obj - > package . elements [ 1 ] . integer . value ) ;
2012-08-22 18:47:22 +08:00
}
2013-12-19 20:38:17 +08:00
ACPI_FREE ( obj ) ;
return size ;
2012-08-22 18:47:22 +08:00
}
2013-08-29 20:39:11 +08:00
static ssize_t tpm_store_ppi_request ( struct device * dev ,
struct device_attribute * attr ,
const char * buf , size_t count )
2012-08-22 18:47:22 +08:00
{
u32 req ;
u64 ret ;
2013-12-19 20:38:17 +08:00
int func = TPM_PPI_FN_SUBREQ ;
union acpi_object * obj , tmp ;
union acpi_object argv4 = ACPI_INIT_DSM_ARGV4 ( 1 , & tmp ) ;
2012-08-22 18:47:22 +08:00
/*
* the function to submit TPM operation request to pre - os environment
* is updated with function index from SUBREQ to SUBREQ2 since PPI
* version 1.1
*/
2013-12-19 20:38:18 +08:00
if ( acpi_check_dsm ( tpm_ppi_handle , tpm_ppi_uuid , TPM_PPI_REVISION_ID ,
1 < < TPM_PPI_FN_SUBREQ2 ) )
2013-12-19 20:38:17 +08:00
func = TPM_PPI_FN_SUBREQ2 ;
2012-08-22 18:47:22 +08:00
/*
* PPI spec defines params [ 3 ] . type as ACPI_TYPE_PACKAGE . Some BIOS
* accept buffer / string / integer type , but some BIOS accept buffer /
* string / package type . For PPI version 1.0 and 1.1 , use buffer type
* for compatibility , and use package type since 1.2 according to spec .
*/
2013-12-19 20:38:17 +08:00
if ( strcmp ( tpm_ppi_version , " 1.2 " ) < 0 ) {
if ( sscanf ( buf , " %d " , & req ) ! = 1 )
return - EINVAL ;
argv4 . type = ACPI_TYPE_BUFFER ;
argv4 . buffer . length = sizeof ( req ) ;
argv4 . buffer . pointer = ( u8 * ) & req ;
2012-08-22 18:47:22 +08:00
} else {
2013-12-19 20:38:17 +08:00
tmp . type = ACPI_TYPE_INTEGER ;
if ( sscanf ( buf , " %llu " , & tmp . integer . value ) ! = 1 )
return - EINVAL ;
}
obj = tpm_eval_dsm ( func , ACPI_TYPE_INTEGER , & argv4 ) ;
if ( ! obj ) {
return - ENXIO ;
2012-08-22 18:47:22 +08:00
} else {
2013-12-19 20:38:17 +08:00
ret = obj - > integer . value ;
ACPI_FREE ( obj ) ;
2012-08-22 18:47:22 +08:00
}
if ( ret = = 0 )
2013-12-19 20:38:17 +08:00
return ( acpi_status ) count ;
return ( ret = = 1 ) ? - EPERM : - EFAULT ;
2012-08-22 18:47:22 +08:00
}
2013-08-29 20:39:11 +08:00
static ssize_t tpm_show_ppi_transition_action ( struct device * dev ,
struct device_attribute * attr ,
char * buf )
2012-08-22 18:47:22 +08:00
{
u32 ret ;
2013-12-19 20:38:17 +08:00
acpi_status status ;
union acpi_object * obj = NULL ;
union acpi_object tmp = {
. buffer . type = ACPI_TYPE_BUFFER ,
. buffer . length = 0 ,
. buffer . pointer = NULL
} ;
static char * info [ ] = {
2012-08-22 18:47:22 +08:00
" None " ,
" Shutdown " ,
" Reboot " ,
" OS Vendor-specific " ,
" Error " ,
} ;
/*
* PPI spec defines params [ 3 ] . type as empty package , but some platforms
* ( e . g . Capella with PPI 1.0 ) need integer / string / buffer type , so for
* compatibility , define params [ 3 ] . type as buffer , if PPI version < 1.2
*/
2013-12-19 20:38:17 +08:00
if ( strcmp ( tpm_ppi_version , " 1.2 " ) < 0 )
obj = & tmp ;
obj = tpm_eval_dsm ( TPM_PPI_FN_GETACT , ACPI_TYPE_INTEGER , obj ) ;
if ( ! obj ) {
return - ENXIO ;
} else {
ret = obj - > integer . value ;
ACPI_FREE ( obj ) ;
2012-08-22 18:47:22 +08:00
}
2013-12-19 20:38:17 +08:00
2012-08-22 18:47:22 +08:00
if ( ret < ARRAY_SIZE ( info ) - 1 )
status = scnprintf ( buf , PAGE_SIZE , " %d: %s \n " , ret , info [ ret ] ) ;
else
status = scnprintf ( buf , PAGE_SIZE , " %d: %s \n " , ret ,
info [ ARRAY_SIZE ( info ) - 1 ] ) ;
return status ;
}
2013-08-29 20:39:11 +08:00
static ssize_t tpm_show_ppi_response ( struct device * dev ,
struct device_attribute * attr ,
char * buf )
2012-08-22 18:47:22 +08:00
{
2013-12-19 20:38:17 +08:00
acpi_status status = - EINVAL ;
union acpi_object * obj , * ret_obj ;
u64 req , res ;
obj = tpm_eval_dsm ( TPM_PPI_FN_GETRSP , ACPI_TYPE_PACKAGE , NULL ) ;
if ( ! obj )
2012-08-22 18:47:22 +08:00
return - ENXIO ;
/*
* parameter output . pointer should be of package type , including
* 3 integers . The first means function return code , the second means
* most recent TPM operation request , and the last means response to
* the most recent TPM operation request . Only if the first is 0 , and
* the second integer is not 0 , the response makes sense .
*/
2013-12-19 20:38:17 +08:00
ret_obj = obj - > package . elements ;
if ( obj - > package . count < 3 | |
ret_obj [ 0 ] . type ! = ACPI_TYPE_INTEGER | |
ret_obj [ 1 ] . type ! = ACPI_TYPE_INTEGER | |
ret_obj [ 2 ] . type ! = ACPI_TYPE_INTEGER )
2012-08-22 18:47:22 +08:00
goto cleanup ;
2013-12-19 20:38:17 +08:00
if ( ret_obj [ 0 ] . integer . value ) {
2012-08-22 18:47:22 +08:00
status = - EFAULT ;
goto cleanup ;
}
2013-12-19 20:38:17 +08:00
req = ret_obj [ 1 ] . integer . value ;
res = ret_obj [ 2 ] . integer . value ;
if ( req ) {
if ( res = = 0 )
2012-08-22 18:47:22 +08:00
status = scnprintf ( buf , PAGE_SIZE , " %llu %s \n " , req ,
" 0: Success " ) ;
2013-12-19 20:38:17 +08:00
else if ( res = = 0xFFFFFFF0 )
2012-08-22 18:47:22 +08:00
status = scnprintf ( buf , PAGE_SIZE , " %llu %s \n " , req ,
" 0xFFFFFFF0: User Abort " ) ;
2013-12-19 20:38:17 +08:00
else if ( res = = 0xFFFFFFF1 )
2012-08-22 18:47:22 +08:00
status = scnprintf ( buf , PAGE_SIZE , " %llu %s \n " , req ,
" 0xFFFFFFF1: BIOS Failure " ) ;
2013-12-19 20:38:17 +08:00
else if ( res > = 1 & & res < = 0x00000FFF )
2012-08-22 18:47:22 +08:00
status = scnprintf ( buf , PAGE_SIZE , " %llu %llu: %s \n " ,
2013-12-19 20:38:17 +08:00
req , res , " Corresponding TPM error " ) ;
2012-08-22 18:47:22 +08:00
else
status = scnprintf ( buf , PAGE_SIZE , " %llu %llu: %s \n " ,
2013-12-19 20:38:17 +08:00
req , res , " Error " ) ;
2012-08-22 18:47:22 +08:00
} else {
status = scnprintf ( buf , PAGE_SIZE , " %llu: %s \n " ,
2013-12-19 20:38:17 +08:00
req , " No Recent Request " ) ;
2012-08-22 18:47:22 +08:00
}
2013-12-19 20:38:17 +08:00
2012-08-22 18:47:22 +08:00
cleanup :
2013-12-19 20:38:17 +08:00
ACPI_FREE ( obj ) ;
2012-08-22 18:47:22 +08:00
return status ;
}
static ssize_t show_ppi_operations ( char * buf , u32 start , u32 end )
{
int i ;
u32 ret ;
2013-12-19 20:38:17 +08:00
char * str = buf ;
union acpi_object * obj , tmp ;
union acpi_object argv = ACPI_INIT_DSM_ARGV4 ( 1 , & tmp ) ;
static char * info [ ] = {
2012-08-22 18:47:22 +08:00
" Not implemented " ,
" BIOS only " ,
" Blocked for OS by BIOS " ,
" User required " ,
" User not required " ,
} ;
2013-12-19 20:38:18 +08:00
if ( ! acpi_check_dsm ( tpm_ppi_handle , tpm_ppi_uuid , TPM_PPI_REVISION_ID ,
1 < < TPM_PPI_FN_GETOPR ) )
2012-08-22 18:47:22 +08:00
return - EPERM ;
2013-12-19 20:38:17 +08:00
tmp . integer . type = ACPI_TYPE_INTEGER ;
2012-08-22 18:47:22 +08:00
for ( i = start ; i < = end ; i + + ) {
2013-12-19 20:38:17 +08:00
tmp . integer . value = i ;
obj = tpm_eval_dsm ( TPM_PPI_FN_GETOPR , ACPI_TYPE_INTEGER , & argv ) ;
if ( ! obj ) {
2012-08-22 18:47:22 +08:00
return - ENOMEM ;
2013-12-19 20:38:17 +08:00
} else {
ret = obj - > integer . value ;
ACPI_FREE ( obj ) ;
}
2012-08-22 18:47:22 +08:00
if ( ret > 0 & & ret < ARRAY_SIZE ( info ) )
str + = scnprintf ( str , PAGE_SIZE , " %d %d: %s \n " ,
i , ret , info [ ret ] ) ;
}
2013-12-19 20:38:17 +08:00
2012-08-22 18:47:22 +08:00
return str - buf ;
}
2013-08-29 20:39:11 +08:00
static ssize_t tpm_show_ppi_tcg_operations ( struct device * dev ,
struct device_attribute * attr ,
char * buf )
2012-08-22 18:47:22 +08:00
{
return show_ppi_operations ( buf , 0 , PPI_TPM_REQ_MAX ) ;
}
2013-08-29 20:39:11 +08:00
static ssize_t tpm_show_ppi_vs_operations ( struct device * dev ,
struct device_attribute * attr ,
char * buf )
2012-08-22 18:47:22 +08:00
{
return show_ppi_operations ( buf , PPI_VS_REQ_START , PPI_VS_REQ_END ) ;
}
static DEVICE_ATTR ( version , S_IRUGO , tpm_show_ppi_version , NULL ) ;
static DEVICE_ATTR ( request , S_IRUGO | S_IWUSR | S_IWGRP ,
tpm_show_ppi_request , tpm_store_ppi_request ) ;
static DEVICE_ATTR ( transition_action , S_IRUGO ,
tpm_show_ppi_transition_action , NULL ) ;
static DEVICE_ATTR ( response , S_IRUGO , tpm_show_ppi_response , NULL ) ;
static DEVICE_ATTR ( tcg_operations , S_IRUGO , tpm_show_ppi_tcg_operations , NULL ) ;
static DEVICE_ATTR ( vs_operations , S_IRUGO , tpm_show_ppi_vs_operations , NULL ) ;
static struct attribute * ppi_attrs [ ] = {
& dev_attr_version . attr ,
& dev_attr_request . attr ,
& dev_attr_transition_action . attr ,
& dev_attr_response . attr ,
& dev_attr_tcg_operations . attr ,
& dev_attr_vs_operations . attr , NULL ,
} ;
static struct attribute_group ppi_attr_grp = {
2012-10-09 17:35:22 +08:00
. name = " ppi " ,
2012-08-22 18:47:22 +08:00
. attrs = ppi_attrs
} ;
2012-10-09 17:35:22 +08:00
int tpm_add_ppi ( struct kobject * parent )
2012-08-22 18:47:22 +08:00
{
2013-12-19 20:38:17 +08:00
/* Cache TPM ACPI handle and version string */
acpi_walk_namespace ( ACPI_TYPE_DEVICE , ACPI_ROOT_OBJECT , ACPI_UINT32_MAX ,
2013-12-19 20:38:18 +08:00
ppi_callback , NULL , NULL , & tpm_ppi_handle ) ;
2014-05-12 01:13:28 +02:00
return tpm_ppi_handle ? sysfs_create_group ( parent , & ppi_attr_grp ) : 0 ;
2012-10-09 17:35:22 +08:00
}
void tpm_remove_ppi ( struct kobject * parent )
{
2014-05-12 01:13:28 +02:00
if ( tpm_ppi_handle )
sysfs_remove_group ( parent , & ppi_attr_grp ) ;
2012-08-22 18:47:22 +08:00
}