2014-12-12 11:46:38 -08:00
/*
2015-05-30 08:09:04 +03:00
* Copyright ( C ) 2014 , 2015 Intel Corporation
2014-12-12 11:46:38 -08:00
*
* Authors :
* Jarkko Sakkinen < jarkko . sakkinen @ linux . intel . com >
*
* Maintained by : < tpmdd - devel @ lists . sourceforge . net >
*
* This file contains TPM2 protocol implementations of the commands
* used by the kernel internally .
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation ; version 2
* of the License .
*/
# include "tpm.h"
2015-11-05 21:43:06 +02:00
# include <crypto/hash_info.h>
2015-05-30 08:09:04 +03:00
# include <keys/trusted-type.h>
enum tpm2_object_attributes {
2016-02-13 11:51:23 +02:00
TPM2_OA_USER_WITH_AUTH = BIT ( 6 ) ,
} ;
enum tpm2_session_attributes {
TPM2_SA_CONTINUE_SESSION = BIT ( 0 ) ,
2015-05-30 08:09:04 +03:00
} ;
2014-12-12 11:46:38 -08:00
struct tpm2_startup_in {
__be16 startup_type ;
} __packed ;
struct tpm2_self_test_in {
u8 full_test ;
} __packed ;
struct tpm2_pcr_read_in {
__be32 pcr_selects_cnt ;
__be16 hash_alg ;
u8 pcr_select_size ;
u8 pcr_select [ TPM2_PCR_SELECT_MIN ] ;
} __packed ;
struct tpm2_pcr_read_out {
__be32 update_cnt ;
__be32 pcr_selects_cnt ;
__be16 hash_alg ;
u8 pcr_select_size ;
u8 pcr_select [ TPM2_PCR_SELECT_MIN ] ;
__be32 digests_cnt ;
__be16 digest_size ;
u8 digest [ TPM_DIGEST_SIZE ] ;
} __packed ;
struct tpm2_null_auth_area {
__be32 handle ;
__be16 nonce_size ;
u8 attributes ;
__be16 auth_size ;
} __packed ;
struct tpm2_pcr_extend_in {
__be32 pcr_idx ;
__be32 auth_area_size ;
struct tpm2_null_auth_area auth_area ;
__be32 digest_cnt ;
__be16 hash_alg ;
u8 digest [ TPM_DIGEST_SIZE ] ;
} __packed ;
struct tpm2_get_tpm_pt_in {
__be32 cap_id ;
__be32 property_id ;
__be32 property_cnt ;
} __packed ;
struct tpm2_get_tpm_pt_out {
u8 more_data ;
__be32 subcap_id ;
__be32 property_cnt ;
__be32 property_id ;
__be32 value ;
} __packed ;
struct tpm2_get_random_in {
__be16 size ;
} __packed ;
struct tpm2_get_random_out {
__be16 size ;
u8 buffer [ TPM_MAX_RNG_DATA ] ;
} __packed ;
union tpm2_cmd_params {
struct tpm2_startup_in startup_in ;
struct tpm2_self_test_in selftest_in ;
struct tpm2_pcr_read_in pcrread_in ;
struct tpm2_pcr_read_out pcrread_out ;
struct tpm2_pcr_extend_in pcrextend_in ;
struct tpm2_get_tpm_pt_in get_tpm_pt_in ;
struct tpm2_get_tpm_pt_out get_tpm_pt_out ;
struct tpm2_get_random_in getrandom_in ;
struct tpm2_get_random_out getrandom_out ;
} ;
struct tpm2_cmd {
tpm_cmd_header header ;
union tpm2_cmd_params params ;
} __packed ;
2015-11-05 21:43:06 +02:00
struct tpm2_hash {
unsigned int crypto_id ;
unsigned int tpm_id ;
} ;
static struct tpm2_hash tpm2_hash_map [ ] = {
{ HASH_ALGO_SHA1 , TPM2_ALG_SHA1 } ,
{ HASH_ALGO_SHA256 , TPM2_ALG_SHA256 } ,
{ HASH_ALGO_SHA384 , TPM2_ALG_SHA384 } ,
{ HASH_ALGO_SHA512 , TPM2_ALG_SHA512 } ,
{ HASH_ALGO_SM3_256 , TPM2_ALG_SM3_256 } ,
} ;
2014-12-12 11:46:38 -08:00
/*
* Array with one entry per ordinal defining the maximum amount
* of time the chip could take to return the result . The values
* of the SHORT , MEDIUM , and LONG durations are taken from the
* PC Client Profile ( PTP ) specification .
*/
static const u8 tpm2_ordinal_duration [ TPM2_CC_LAST - TPM2_CC_FIRST + 1 ] = {
TPM_UNDEFINED , /* 11F */
TPM_UNDEFINED , /* 120 */
TPM_LONG , /* 121 */
TPM_UNDEFINED , /* 122 */
TPM_UNDEFINED , /* 123 */
TPM_UNDEFINED , /* 124 */
TPM_UNDEFINED , /* 125 */
TPM_UNDEFINED , /* 126 */
TPM_UNDEFINED , /* 127 */
TPM_UNDEFINED , /* 128 */
TPM_LONG , /* 129 */
TPM_UNDEFINED , /* 12a */
TPM_UNDEFINED , /* 12b */
TPM_UNDEFINED , /* 12c */
TPM_UNDEFINED , /* 12d */
TPM_UNDEFINED , /* 12e */
TPM_UNDEFINED , /* 12f */
TPM_UNDEFINED , /* 130 */
TPM_UNDEFINED , /* 131 */
TPM_UNDEFINED , /* 132 */
TPM_UNDEFINED , /* 133 */
TPM_UNDEFINED , /* 134 */
TPM_UNDEFINED , /* 135 */
TPM_UNDEFINED , /* 136 */
TPM_UNDEFINED , /* 137 */
TPM_UNDEFINED , /* 138 */
TPM_UNDEFINED , /* 139 */
TPM_UNDEFINED , /* 13a */
TPM_UNDEFINED , /* 13b */
TPM_UNDEFINED , /* 13c */
TPM_UNDEFINED , /* 13d */
TPM_MEDIUM , /* 13e */
TPM_UNDEFINED , /* 13f */
TPM_UNDEFINED , /* 140 */
TPM_UNDEFINED , /* 141 */
TPM_UNDEFINED , /* 142 */
TPM_LONG , /* 143 */
TPM_MEDIUM , /* 144 */
TPM_UNDEFINED , /* 145 */
TPM_UNDEFINED , /* 146 */
TPM_UNDEFINED , /* 147 */
TPM_UNDEFINED , /* 148 */
TPM_UNDEFINED , /* 149 */
TPM_UNDEFINED , /* 14a */
TPM_UNDEFINED , /* 14b */
TPM_UNDEFINED , /* 14c */
TPM_UNDEFINED , /* 14d */
TPM_LONG , /* 14e */
TPM_UNDEFINED , /* 14f */
TPM_UNDEFINED , /* 150 */
TPM_UNDEFINED , /* 151 */
TPM_UNDEFINED , /* 152 */
TPM_UNDEFINED , /* 153 */
TPM_UNDEFINED , /* 154 */
TPM_UNDEFINED , /* 155 */
TPM_UNDEFINED , /* 156 */
TPM_UNDEFINED , /* 157 */
TPM_UNDEFINED , /* 158 */
TPM_UNDEFINED , /* 159 */
TPM_UNDEFINED , /* 15a */
TPM_UNDEFINED , /* 15b */
TPM_MEDIUM , /* 15c */
TPM_UNDEFINED , /* 15d */
TPM_UNDEFINED , /* 15e */
TPM_UNDEFINED , /* 15f */
TPM_UNDEFINED , /* 160 */
TPM_UNDEFINED , /* 161 */
TPM_UNDEFINED , /* 162 */
TPM_UNDEFINED , /* 163 */
TPM_UNDEFINED , /* 164 */
TPM_UNDEFINED , /* 165 */
TPM_UNDEFINED , /* 166 */
TPM_UNDEFINED , /* 167 */
TPM_UNDEFINED , /* 168 */
TPM_UNDEFINED , /* 169 */
TPM_UNDEFINED , /* 16a */
TPM_UNDEFINED , /* 16b */
TPM_UNDEFINED , /* 16c */
TPM_UNDEFINED , /* 16d */
TPM_UNDEFINED , /* 16e */
TPM_UNDEFINED , /* 16f */
TPM_UNDEFINED , /* 170 */
TPM_UNDEFINED , /* 171 */
TPM_UNDEFINED , /* 172 */
TPM_UNDEFINED , /* 173 */
TPM_UNDEFINED , /* 174 */
TPM_UNDEFINED , /* 175 */
TPM_UNDEFINED , /* 176 */
TPM_LONG , /* 177 */
TPM_UNDEFINED , /* 178 */
TPM_UNDEFINED , /* 179 */
TPM_MEDIUM , /* 17a */
TPM_LONG , /* 17b */
TPM_UNDEFINED , /* 17c */
TPM_UNDEFINED , /* 17d */
TPM_UNDEFINED , /* 17e */
TPM_UNDEFINED , /* 17f */
TPM_UNDEFINED , /* 180 */
TPM_UNDEFINED , /* 181 */
TPM_MEDIUM , /* 182 */
TPM_UNDEFINED , /* 183 */
TPM_UNDEFINED , /* 184 */
TPM_MEDIUM , /* 185 */
TPM_MEDIUM , /* 186 */
TPM_UNDEFINED , /* 187 */
TPM_UNDEFINED , /* 188 */
TPM_UNDEFINED , /* 189 */
TPM_UNDEFINED , /* 18a */
TPM_UNDEFINED , /* 18b */
TPM_UNDEFINED , /* 18c */
TPM_UNDEFINED , /* 18d */
TPM_UNDEFINED , /* 18e */
TPM_UNDEFINED /* 18f */
} ;
# define TPM2_PCR_READ_IN_SIZE \
( sizeof ( struct tpm_input_header ) + \
sizeof ( struct tpm2_pcr_read_in ) )
static const struct tpm_input_header tpm2_pcrread_header = {
. tag = cpu_to_be16 ( TPM2_ST_NO_SESSIONS ) ,
. length = cpu_to_be32 ( TPM2_PCR_READ_IN_SIZE ) ,
. ordinal = cpu_to_be32 ( TPM2_CC_PCR_READ )
} ;
/**
* tpm2_pcr_read ( ) - read a PCR value
* @ chip : TPM chip to use .
* @ pcr_idx : index of the PCR to read .
* @ ref_buf : buffer to store the resulting hash ,
*
* 0 is returned when the operation is successful . If a negative number is
* returned it remarks a POSIX error code . If a positive number is returned
* it remarks a TPM error .
*/
int tpm2_pcr_read ( struct tpm_chip * chip , int pcr_idx , u8 * res_buf )
{
int rc ;
struct tpm2_cmd cmd ;
u8 * buf ;
if ( pcr_idx > = TPM2_PLATFORM_PCR )
return - EINVAL ;
cmd . header . in = tpm2_pcrread_header ;
cmd . params . pcrread_in . pcr_selects_cnt = cpu_to_be32 ( 1 ) ;
cmd . params . pcrread_in . hash_alg = cpu_to_be16 ( TPM2_ALG_SHA1 ) ;
cmd . params . pcrread_in . pcr_select_size = TPM2_PCR_SELECT_MIN ;
memset ( cmd . params . pcrread_in . pcr_select , 0 ,
sizeof ( cmd . params . pcrread_in . pcr_select ) ) ;
cmd . params . pcrread_in . pcr_select [ pcr_idx > > 3 ] = 1 < < ( pcr_idx & 0x7 ) ;
rc = tpm_transmit_cmd ( chip , & cmd , sizeof ( cmd ) ,
" attempting to read a pcr value " ) ;
if ( rc = = 0 ) {
buf = cmd . params . pcrread_out . digest ;
memcpy ( res_buf , buf , TPM_DIGEST_SIZE ) ;
}
return rc ;
}
# define TPM2_GET_PCREXTEND_IN_SIZE \
( sizeof ( struct tpm_input_header ) + \
sizeof ( struct tpm2_pcr_extend_in ) )
static const struct tpm_input_header tpm2_pcrextend_header = {
. tag = cpu_to_be16 ( TPM2_ST_SESSIONS ) ,
. length = cpu_to_be32 ( TPM2_GET_PCREXTEND_IN_SIZE ) ,
. ordinal = cpu_to_be32 ( TPM2_CC_PCR_EXTEND )
} ;
/**
* tpm2_pcr_extend ( ) - extend a PCR value
* @ chip : TPM chip to use .
* @ pcr_idx : index of the PCR .
* @ hash : hash value to use for the extend operation .
*
* 0 is returned when the operation is successful . If a negative number is
* returned it remarks a POSIX error code . If a positive number is returned
* it remarks a TPM error .
*/
int tpm2_pcr_extend ( struct tpm_chip * chip , int pcr_idx , const u8 * hash )
{
struct tpm2_cmd cmd ;
int rc ;
cmd . header . in = tpm2_pcrextend_header ;
cmd . params . pcrextend_in . pcr_idx = cpu_to_be32 ( pcr_idx ) ;
cmd . params . pcrextend_in . auth_area_size =
cpu_to_be32 ( sizeof ( struct tpm2_null_auth_area ) ) ;
cmd . params . pcrextend_in . auth_area . handle =
cpu_to_be32 ( TPM2_RS_PW ) ;
cmd . params . pcrextend_in . auth_area . nonce_size = 0 ;
cmd . params . pcrextend_in . auth_area . attributes = 0 ;
cmd . params . pcrextend_in . auth_area . auth_size = 0 ;
cmd . params . pcrextend_in . digest_cnt = cpu_to_be32 ( 1 ) ;
cmd . params . pcrextend_in . hash_alg = cpu_to_be16 ( TPM2_ALG_SHA1 ) ;
memcpy ( cmd . params . pcrextend_in . digest , hash , TPM_DIGEST_SIZE ) ;
rc = tpm_transmit_cmd ( chip , & cmd , sizeof ( cmd ) ,
" attempting extend a PCR value " ) ;
return rc ;
}
# define TPM2_GETRANDOM_IN_SIZE \
( sizeof ( struct tpm_input_header ) + \
sizeof ( struct tpm2_get_random_in ) )
static const struct tpm_input_header tpm2_getrandom_header = {
. tag = cpu_to_be16 ( TPM2_ST_NO_SESSIONS ) ,
. length = cpu_to_be32 ( TPM2_GETRANDOM_IN_SIZE ) ,
. ordinal = cpu_to_be32 ( TPM2_CC_GET_RANDOM )
} ;
/**
* tpm2_get_random ( ) - get random bytes from the TPM RNG
* @ chip : TPM chip to use
* @ out : destination buffer for the random bytes
* @ max : the max number of bytes to write to @ out
*
* 0 is returned when the operation is successful . If a negative number is
* returned it remarks a POSIX error code . If a positive number is returned
* it remarks a TPM error .
*/
int tpm2_get_random ( struct tpm_chip * chip , u8 * out , size_t max )
{
struct tpm2_cmd cmd ;
u32 recd ;
u32 num_bytes ;
int err ;
int total = 0 ;
int retries = 5 ;
u8 * dest = out ;
num_bytes = min_t ( u32 , max , sizeof ( cmd . params . getrandom_out . buffer ) ) ;
if ( ! out | | ! num_bytes | |
max > sizeof ( cmd . params . getrandom_out . buffer ) )
return - EINVAL ;
do {
cmd . header . in = tpm2_getrandom_header ;
cmd . params . getrandom_in . size = cpu_to_be16 ( num_bytes ) ;
err = tpm_transmit_cmd ( chip , & cmd , sizeof ( cmd ) ,
" attempting get random " ) ;
if ( err )
break ;
recd = min_t ( u32 , be16_to_cpu ( cmd . params . getrandom_out . size ) ,
num_bytes ) ;
memcpy ( dest , cmd . params . getrandom_out . buffer , recd ) ;
dest + = recd ;
total + = recd ;
num_bytes - = recd ;
} while ( retries - - & & total < max ) ;
return total ? total : - EIO ;
}
# define TPM2_GET_TPM_PT_IN_SIZE \
( sizeof ( struct tpm_input_header ) + \
sizeof ( struct tpm2_get_tpm_pt_in ) )
static const struct tpm_input_header tpm2_get_tpm_pt_header = {
. tag = cpu_to_be16 ( TPM2_ST_NO_SESSIONS ) ,
. length = cpu_to_be32 ( TPM2_GET_TPM_PT_IN_SIZE ) ,
. ordinal = cpu_to_be32 ( TPM2_CC_GET_CAPABILITY )
} ;
2015-05-30 08:09:04 +03:00
/**
* Append TPMS_AUTH_COMMAND to the buffer . The buffer must be allocated with
* tpm_buf_alloc ( ) .
*
* @ param buf : an allocated tpm_buf instance
* @ param nonce : the session nonce , may be NULL if not used
* @ param nonce_len : the session nonce length , may be 0 if not used
* @ param attributes : the session attributes
* @ param hmac : the session HMAC or password , may be NULL if not used
* @ param hmac_len : the session HMAC or password length , maybe 0 if not used
*/
static void tpm2_buf_append_auth ( struct tpm_buf * buf , u32 session_handle ,
const u8 * nonce , u16 nonce_len ,
u8 attributes ,
const u8 * hmac , u16 hmac_len )
{
tpm_buf_append_u32 ( buf , 9 + nonce_len + hmac_len ) ;
tpm_buf_append_u32 ( buf , session_handle ) ;
tpm_buf_append_u16 ( buf , nonce_len ) ;
if ( nonce & & nonce_len )
tpm_buf_append ( buf , nonce , nonce_len ) ;
tpm_buf_append_u8 ( buf , attributes ) ;
tpm_buf_append_u16 ( buf , hmac_len ) ;
if ( hmac & & hmac_len )
tpm_buf_append ( buf , hmac , hmac_len ) ;
}
/**
* tpm2_seal_trusted ( ) - seal a trusted key
* @ chip_num : A specific chip number for the request or TPM_ANY_NUM
* @ options : authentication values and other options
* @ payload : the key data in clear and encrypted form
*
* Returns < 0 on error and 0 on success .
*/
int tpm2_seal_trusted ( struct tpm_chip * chip ,
struct trusted_key_payload * payload ,
struct trusted_key_options * options )
{
unsigned int blob_len ;
struct tpm_buf buf ;
2015-11-05 21:43:06 +02:00
u32 hash ;
int i ;
2015-05-30 08:09:04 +03:00
int rc ;
2015-11-05 21:43:06 +02:00
for ( i = 0 ; i < ARRAY_SIZE ( tpm2_hash_map ) ; i + + ) {
if ( options - > hash = = tpm2_hash_map [ i ] . crypto_id ) {
hash = tpm2_hash_map [ i ] . tpm_id ;
break ;
}
}
if ( i = = ARRAY_SIZE ( tpm2_hash_map ) )
return - EINVAL ;
2015-05-30 08:09:04 +03:00
rc = tpm_buf_init ( & buf , TPM2_ST_SESSIONS , TPM2_CC_CREATE ) ;
if ( rc )
return rc ;
tpm_buf_append_u32 ( & buf , options - > keyhandle ) ;
tpm2_buf_append_auth ( & buf , TPM2_RS_PW ,
NULL /* nonce */ , 0 ,
0 /* session_attributes */ ,
options - > keyauth /* hmac */ ,
TPM_DIGEST_SIZE ) ;
/* sensitive */
2015-10-30 14:57:02 +02:00
tpm_buf_append_u16 ( & buf , 4 + TPM_DIGEST_SIZE + payload - > key_len + 1 ) ;
2015-05-30 08:09:04 +03:00
tpm_buf_append_u16 ( & buf , TPM_DIGEST_SIZE ) ;
tpm_buf_append ( & buf , options - > blobauth , TPM_DIGEST_SIZE ) ;
2015-10-30 14:57:02 +02:00
tpm_buf_append_u16 ( & buf , payload - > key_len + 1 ) ;
2015-05-30 08:09:04 +03:00
tpm_buf_append ( & buf , payload - > key , payload - > key_len ) ;
2015-10-30 14:57:02 +02:00
tpm_buf_append_u8 ( & buf , payload - > migratable ) ;
2015-05-30 08:09:04 +03:00
/* public */
2016-01-06 16:43:30 +02:00
tpm_buf_append_u16 ( & buf , 14 + options - > policydigest_len ) ;
2015-05-30 08:09:04 +03:00
tpm_buf_append_u16 ( & buf , TPM2_ALG_KEYEDHASH ) ;
2015-11-05 21:43:06 +02:00
tpm_buf_append_u16 ( & buf , hash ) ;
2015-10-31 17:53:44 +02:00
/* policy */
2016-01-06 16:43:30 +02:00
if ( options - > policydigest_len ) {
2015-10-31 17:53:44 +02:00
tpm_buf_append_u32 ( & buf , 0 ) ;
2016-01-06 16:43:30 +02:00
tpm_buf_append_u16 ( & buf , options - > policydigest_len ) ;
2015-10-31 17:53:44 +02:00
tpm_buf_append ( & buf , options - > policydigest ,
2016-01-06 16:43:30 +02:00
options - > policydigest_len ) ;
2015-10-31 17:53:44 +02:00
} else {
2016-02-13 11:51:23 +02:00
tpm_buf_append_u32 ( & buf , TPM2_OA_USER_WITH_AUTH ) ;
2015-10-31 17:53:44 +02:00
tpm_buf_append_u16 ( & buf , 0 ) ;
}
/* public parameters */
2015-05-30 08:09:04 +03:00
tpm_buf_append_u16 ( & buf , TPM2_ALG_NULL ) ;
tpm_buf_append_u16 ( & buf , 0 ) ;
/* outside info */
tpm_buf_append_u16 ( & buf , 0 ) ;
/* creation PCR */
tpm_buf_append_u32 ( & buf , 0 ) ;
if ( buf . flags & TPM_BUF_OVERFLOW ) {
rc = - E2BIG ;
goto out ;
}
rc = tpm_transmit_cmd ( chip , buf . data , PAGE_SIZE , " sealing data " ) ;
if ( rc )
goto out ;
blob_len = be32_to_cpup ( ( __be32 * ) & buf . data [ TPM_HEADER_SIZE ] ) ;
if ( blob_len > MAX_BLOB_SIZE ) {
rc = - E2BIG ;
goto out ;
}
memcpy ( payload - > blob , & buf . data [ TPM_HEADER_SIZE + 4 ] , blob_len ) ;
payload - > blob_len = blob_len ;
out :
tpm_buf_destroy ( & buf ) ;
2015-11-05 21:43:06 +02:00
if ( rc > 0 ) {
if ( ( rc & TPM2_RC_HASH ) = = TPM2_RC_HASH )
rc = - EINVAL ;
else
rc = - EPERM ;
}
2015-05-30 08:09:04 +03:00
return rc ;
}
static int tpm2_load ( struct tpm_chip * chip ,
struct trusted_key_payload * payload ,
struct trusted_key_options * options ,
u32 * blob_handle )
{
struct tpm_buf buf ;
unsigned int private_len ;
unsigned int public_len ;
unsigned int blob_len ;
int rc ;
private_len = be16_to_cpup ( ( __be16 * ) & payload - > blob [ 0 ] ) ;
if ( private_len > ( payload - > blob_len - 2 ) )
return - E2BIG ;
public_len = be16_to_cpup ( ( __be16 * ) & payload - > blob [ 2 + private_len ] ) ;
blob_len = private_len + public_len + 4 ;
if ( blob_len > payload - > blob_len )
return - E2BIG ;
rc = tpm_buf_init ( & buf , TPM2_ST_SESSIONS , TPM2_CC_LOAD ) ;
if ( rc )
return rc ;
tpm_buf_append_u32 ( & buf , options - > keyhandle ) ;
tpm2_buf_append_auth ( & buf , TPM2_RS_PW ,
NULL /* nonce */ , 0 ,
0 /* session_attributes */ ,
options - > keyauth /* hmac */ ,
TPM_DIGEST_SIZE ) ;
tpm_buf_append ( & buf , payload - > blob , blob_len ) ;
if ( buf . flags & TPM_BUF_OVERFLOW ) {
rc = - E2BIG ;
goto out ;
}
rc = tpm_transmit_cmd ( chip , buf . data , PAGE_SIZE , " loading blob " ) ;
if ( ! rc )
* blob_handle = be32_to_cpup (
( __be32 * ) & buf . data [ TPM_HEADER_SIZE ] ) ;
out :
tpm_buf_destroy ( & buf ) ;
if ( rc > 0 )
rc = - EPERM ;
return rc ;
}
static void tpm2_flush_context ( struct tpm_chip * chip , u32 handle )
{
struct tpm_buf buf ;
int rc ;
rc = tpm_buf_init ( & buf , TPM2_ST_NO_SESSIONS , TPM2_CC_FLUSH_CONTEXT ) ;
if ( rc ) {
2016-02-29 12:29:47 -05:00
dev_warn ( & chip - > dev , " 0x%08x was not flushed, out of memory \n " ,
2015-05-30 08:09:04 +03:00
handle ) ;
return ;
}
tpm_buf_append_u32 ( & buf , handle ) ;
rc = tpm_transmit_cmd ( chip , buf . data , PAGE_SIZE , " flushing context " ) ;
if ( rc )
2016-02-29 12:29:47 -05:00
dev_warn ( & chip - > dev , " 0x%08x was not flushed, rc=%d \n " , handle ,
2015-05-30 08:09:04 +03:00
rc ) ;
tpm_buf_destroy ( & buf ) ;
}
static int tpm2_unseal ( struct tpm_chip * chip ,
struct trusted_key_payload * payload ,
struct trusted_key_options * options ,
u32 blob_handle )
{
struct tpm_buf buf ;
2015-10-30 14:57:02 +02:00
u16 data_len ;
u8 * data ;
2015-05-30 08:09:04 +03:00
int rc ;
rc = tpm_buf_init ( & buf , TPM2_ST_SESSIONS , TPM2_CC_UNSEAL ) ;
if ( rc )
return rc ;
tpm_buf_append_u32 ( & buf , blob_handle ) ;
2015-10-31 17:53:44 +02:00
tpm2_buf_append_auth ( & buf ,
options - > policyhandle ?
options - > policyhandle : TPM2_RS_PW ,
2015-05-30 08:09:04 +03:00
NULL /* nonce */ , 0 ,
2016-02-13 11:51:23 +02:00
TPM2_SA_CONTINUE_SESSION ,
2015-05-30 08:09:04 +03:00
options - > blobauth /* hmac */ ,
TPM_DIGEST_SIZE ) ;
rc = tpm_transmit_cmd ( chip , buf . data , PAGE_SIZE , " unsealing " ) ;
if ( rc > 0 )
rc = - EPERM ;
if ( ! rc ) {
2015-10-30 14:57:02 +02:00
data_len = be16_to_cpup (
2015-05-30 08:09:04 +03:00
( __be16 * ) & buf . data [ TPM_HEADER_SIZE + 4 ] ) ;
2015-10-30 14:57:02 +02:00
data = & buf . data [ TPM_HEADER_SIZE + 6 ] ;
2015-05-30 08:09:04 +03:00
2015-10-30 14:57:02 +02:00
memcpy ( payload - > key , data , data_len - 1 ) ;
payload - > key_len = data_len - 1 ;
payload - > migratable = data [ data_len - 1 ] ;
2015-05-30 08:09:04 +03:00
}
tpm_buf_destroy ( & buf ) ;
return rc ;
}
/**
* tpm_unseal_trusted ( ) - unseal a trusted key
* @ chip_num : A specific chip number for the request or TPM_ANY_NUM
* @ options : authentication values and other options
* @ payload : the key data in clear and encrypted form
*
* Returns < 0 on error and 0 on success .
*/
int tpm2_unseal_trusted ( struct tpm_chip * chip ,
struct trusted_key_payload * payload ,
struct trusted_key_options * options )
{
u32 blob_handle ;
int rc ;
rc = tpm2_load ( chip , payload , options , & blob_handle ) ;
if ( rc )
return rc ;
rc = tpm2_unseal ( chip , payload , options , blob_handle ) ;
tpm2_flush_context ( chip , blob_handle ) ;
return rc ;
}
2014-12-12 11:46:38 -08:00
/**
* tpm2_get_tpm_pt ( ) - get value of a TPM_CAP_TPM_PROPERTIES type property
* @ chip : TPM chip to use .
* @ property_id : property ID .
* @ value : output variable .
* @ desc : passed to tpm_transmit_cmd ( )
*
* 0 is returned when the operation is successful . If a negative number is
* returned it remarks a POSIX error code . If a positive number is returned
* it remarks a TPM error .
*/
ssize_t tpm2_get_tpm_pt ( struct tpm_chip * chip , u32 property_id , u32 * value ,
const char * desc )
{
struct tpm2_cmd cmd ;
int rc ;
cmd . header . in = tpm2_get_tpm_pt_header ;
cmd . params . get_tpm_pt_in . cap_id = cpu_to_be32 ( TPM2_CAP_TPM_PROPERTIES ) ;
cmd . params . get_tpm_pt_in . property_id = cpu_to_be32 ( property_id ) ;
cmd . params . get_tpm_pt_in . property_cnt = cpu_to_be32 ( 1 ) ;
rc = tpm_transmit_cmd ( chip , & cmd , sizeof ( cmd ) , desc ) ;
if ( ! rc )
2016-07-14 18:07:18 -07:00
* value = be32_to_cpu ( cmd . params . get_tpm_pt_out . value ) ;
2014-12-12 11:46:38 -08:00
return rc ;
}
# define TPM2_STARTUP_IN_SIZE \
( sizeof ( struct tpm_input_header ) + \
sizeof ( struct tpm2_startup_in ) )
static const struct tpm_input_header tpm2_startup_header = {
. tag = cpu_to_be16 ( TPM2_ST_NO_SESSIONS ) ,
. length = cpu_to_be32 ( TPM2_STARTUP_IN_SIZE ) ,
. ordinal = cpu_to_be32 ( TPM2_CC_STARTUP )
} ;
/**
* tpm2_startup ( ) - send startup command to the TPM chip
* @ chip : TPM chip to use .
* @ startup_type startup type . The value is either
* TPM_SU_CLEAR or TPM_SU_STATE .
*
* 0 is returned when the operation is successful . If a negative number is
* returned it remarks a POSIX error code . If a positive number is returned
* it remarks a TPM error .
*/
2016-07-12 11:41:49 -06:00
static int tpm2_startup ( struct tpm_chip * chip , u16 startup_type )
2014-12-12 11:46:38 -08:00
{
struct tpm2_cmd cmd ;
cmd . header . in = tpm2_startup_header ;
cmd . params . startup_in . startup_type = cpu_to_be16 ( startup_type ) ;
return tpm_transmit_cmd ( chip , & cmd , sizeof ( cmd ) ,
" attempting to start the TPM " ) ;
}
# define TPM2_SHUTDOWN_IN_SIZE \
( sizeof ( struct tpm_input_header ) + \
sizeof ( struct tpm2_startup_in ) )
static const struct tpm_input_header tpm2_shutdown_header = {
. tag = cpu_to_be16 ( TPM2_ST_NO_SESSIONS ) ,
. length = cpu_to_be32 ( TPM2_SHUTDOWN_IN_SIZE ) ,
. ordinal = cpu_to_be32 ( TPM2_CC_SHUTDOWN )
} ;
/**
* tpm2_shutdown ( ) - send shutdown command to the TPM chip
* @ chip : TPM chip to use .
* @ shutdown_type shutdown type . The value is either
* TPM_SU_CLEAR or TPM_SU_STATE .
*/
2015-01-29 07:43:47 +02:00
void tpm2_shutdown ( struct tpm_chip * chip , u16 shutdown_type )
2014-12-12 11:46:38 -08:00
{
struct tpm2_cmd cmd ;
2015-01-29 07:43:47 +02:00
int rc ;
2014-12-12 11:46:38 -08:00
cmd . header . in = tpm2_shutdown_header ;
cmd . params . startup_in . startup_type = cpu_to_be16 ( shutdown_type ) ;
2015-01-29 07:43:47 +02:00
rc = tpm_transmit_cmd ( chip , & cmd , sizeof ( cmd ) , " stopping the TPM " ) ;
/* In places where shutdown command is sent there's no much we can do
* except print the error code on a system failure .
*/
if ( rc < 0 )
2016-02-29 12:29:47 -05:00
dev_warn ( & chip - > dev , " transmit returned %d while stopping the TPM " ,
2015-01-29 07:43:47 +02:00
rc ) ;
2014-12-12 11:46:38 -08:00
}
/*
* tpm2_calc_ordinal_duration ( ) - maximum duration for a command
* @ chip : TPM chip to use .
* @ ordinal : command code number .
*
* 0 is returned when the operation is successful . If a negative number is
* returned it remarks a POSIX error code . If a positive number is returned
* it remarks a TPM error .
*/
unsigned long tpm2_calc_ordinal_duration ( struct tpm_chip * chip , u32 ordinal )
{
int index = TPM_UNDEFINED ;
int duration = 0 ;
if ( ordinal > = TPM2_CC_FIRST & & ordinal < = TPM2_CC_LAST )
index = tpm2_ordinal_duration [ ordinal - TPM2_CC_FIRST ] ;
if ( index ! = TPM_UNDEFINED )
2016-03-31 22:56:59 +02:00
duration = chip - > duration [ index ] ;
2014-12-12 11:46:38 -08:00
if ( duration < = 0 )
duration = 2 * 60 * HZ ;
return duration ;
}
EXPORT_SYMBOL_GPL ( tpm2_calc_ordinal_duration ) ;
# define TPM2_SELF_TEST_IN_SIZE \
( sizeof ( struct tpm_input_header ) + \
sizeof ( struct tpm2_self_test_in ) )
static const struct tpm_input_header tpm2_selftest_header = {
. tag = cpu_to_be16 ( TPM2_ST_NO_SESSIONS ) ,
. length = cpu_to_be32 ( TPM2_SELF_TEST_IN_SIZE ) ,
. ordinal = cpu_to_be32 ( TPM2_CC_SELF_TEST )
} ;
/**
* tpm2_continue_selftest ( ) - start a self test
* @ chip : TPM chip to use
* @ full : test all commands instead of testing only those that were not
* previously tested .
*
* 0 is returned when the operation is successful . If a negative number is
* returned it remarks a POSIX error code . If a positive number is returned
* it remarks a TPM error .
*/
static int tpm2_start_selftest ( struct tpm_chip * chip , bool full )
{
int rc ;
struct tpm2_cmd cmd ;
cmd . header . in = tpm2_selftest_header ;
cmd . params . selftest_in . full_test = full ;
rc = tpm_transmit_cmd ( chip , & cmd , TPM2_SELF_TEST_IN_SIZE ,
" continue selftest " ) ;
/* At least some prototype chips seem to give RC_TESTING error
* immediately . This is a workaround for that .
*/
if ( rc = = TPM2_RC_TESTING ) {
2016-02-29 12:29:47 -05:00
dev_warn ( & chip - > dev , " Got RC_TESTING, ignoring \n " ) ;
2014-12-12 11:46:38 -08:00
rc = 0 ;
}
return rc ;
}
/**
* tpm2_do_selftest ( ) - run a full self test
* @ chip : TPM chip to use
*
* During the self test TPM2 commands return with the error code RC_TESTING .
* Waiting is done by issuing PCR read until it executes successfully .
*
* 0 is returned when the operation is successful . If a negative number is
* returned it remarks a POSIX error code . If a positive number is returned
* it remarks a TPM error .
*/
2016-07-12 11:41:49 -06:00
static int tpm2_do_selftest ( struct tpm_chip * chip )
2014-12-12 11:46:38 -08:00
{
int rc ;
unsigned int loops ;
unsigned int delay_msec = 100 ;
unsigned long duration ;
struct tpm2_cmd cmd ;
int i ;
duration = tpm2_calc_ordinal_duration ( chip , TPM2_CC_SELF_TEST ) ;
loops = jiffies_to_msecs ( duration ) / delay_msec ;
rc = tpm2_start_selftest ( chip , true ) ;
if ( rc )
return rc ;
for ( i = 0 ; i < loops ; i + + ) {
/* Attempt to read a PCR value */
cmd . header . in = tpm2_pcrread_header ;
cmd . params . pcrread_in . pcr_selects_cnt = cpu_to_be32 ( 1 ) ;
cmd . params . pcrread_in . hash_alg = cpu_to_be16 ( TPM2_ALG_SHA1 ) ;
cmd . params . pcrread_in . pcr_select_size = TPM2_PCR_SELECT_MIN ;
cmd . params . pcrread_in . pcr_select [ 0 ] = 0x01 ;
cmd . params . pcrread_in . pcr_select [ 1 ] = 0x00 ;
cmd . params . pcrread_in . pcr_select [ 2 ] = 0x00 ;
rc = tpm_transmit_cmd ( chip , ( u8 * ) & cmd , sizeof ( cmd ) , NULL ) ;
if ( rc < 0 )
break ;
rc = be32_to_cpu ( cmd . header . out . return_code ) ;
if ( rc ! = TPM2_RC_TESTING )
break ;
msleep ( delay_msec ) ;
}
return rc ;
}
/**
* tpm2_gen_interrupt ( ) - generate an interrupt
* @ chip : TPM chip to use
*
* 0 is returned when the operation is successful . If a negative number is
* returned it remarks a POSIX error code . If a positive number is returned
* it remarks a TPM error .
*/
2015-02-04 16:21:09 +02:00
int tpm2_gen_interrupt ( struct tpm_chip * chip )
2014-12-12 11:46:38 -08:00
{
u32 dummy ;
2015-02-04 16:21:09 +02:00
return tpm2_get_tpm_pt ( chip , 0x100 , & dummy ,
" attempting to generate an interrupt " ) ;
2014-12-12 11:46:38 -08:00
}
EXPORT_SYMBOL_GPL ( tpm2_gen_interrupt ) ;
2015-02-04 16:21:09 +02:00
/**
* tpm2_probe ( ) - probe TPM 2.0
* @ chip : TPM chip to use
*
* Send idempotent TPM 2.0 command and see whether TPM 2.0 chip replied based on
* the reply tag .
*/
int tpm2_probe ( struct tpm_chip * chip )
{
struct tpm2_cmd cmd ;
int rc ;
cmd . header . in = tpm2_get_tpm_pt_header ;
cmd . params . get_tpm_pt_in . cap_id = cpu_to_be32 ( TPM2_CAP_TPM_PROPERTIES ) ;
cmd . params . get_tpm_pt_in . property_id = cpu_to_be32 ( 0x100 ) ;
cmd . params . get_tpm_pt_in . property_cnt = cpu_to_be32 ( 1 ) ;
rc = tpm_transmit ( chip , ( const char * ) & cmd , sizeof ( cmd ) ) ;
if ( rc < 0 )
return rc ;
else if ( rc < TPM_HEADER_SIZE )
return - EFAULT ;
if ( be16_to_cpu ( cmd . header . out . tag ) = = TPM2_ST_NO_SESSIONS )
chip - > flags | = TPM_CHIP_FLAG_TPM2 ;
return 0 ;
}
EXPORT_SYMBOL_GPL ( tpm2_probe ) ;
2016-07-12 11:41:49 -06:00
/**
* tpm2_auto_startup - Perform the standard automatic TPM initialization
* sequence
* @ chip : TPM chip to use
*
* Returns 0 on success , < 0 in case of fatal error .
*/
int tpm2_auto_startup ( struct tpm_chip * chip )
{
int rc ;
rc = tpm_get_timeouts ( chip ) ;
if ( rc )
goto out ;
rc = tpm2_do_selftest ( chip ) ;
if ( rc ! = TPM2_RC_INITIALIZE ) {
dev_err ( & chip - > dev , " TPM self test failed \n " ) ;
goto out ;
}
if ( rc = = TPM2_RC_INITIALIZE ) {
rc = tpm2_startup ( chip , TPM2_SU_CLEAR ) ;
if ( rc )
goto out ;
rc = tpm2_do_selftest ( chip ) ;
if ( rc ) {
dev_err ( & chip - > dev , " TPM self test failed \n " ) ;
goto out ;
}
}
return rc ;
out :
if ( rc > 0 )
rc = - ENODEV ;
return rc ;
}