2019-06-01 10:08:55 +02:00
/* SPDX-License-Identifier: GPL-2.0-only */
2005-11-13 16:07:41 -08:00
/*
* Copyright ( C ) 2005 IBM Corporation
*
* Authors :
* Kylene Hall < kjhall @ us . ibm . com >
*
2007-08-22 14:01:04 -07:00
* Maintained by : < tpmdd - devel @ lists . sourceforge . net >
2005-11-13 16:07:41 -08:00
*
* Device driver for TCG / TCPA TPM ( trusted platform module ) .
* Specifications at www . trustedcomputinggroup . org
*
* These difference are required on power because the device must be
* discovered through the device tree and iomap must be used to get
* around the need for holes in the io_page_mask . This does not happen
* automatically because the tpm is not a normal pci device and lives
* under the root node .
*/
2016-03-23 07:10:22 +02:00
struct tpm_atmel_priv {
int region_size ;
int have_region ;
2016-03-23 08:16:09 +02:00
unsigned long base ;
2016-03-31 22:56:55 +02:00
void __iomem * iobase ;
2016-03-23 07:10:22 +02:00
} ;
2005-11-13 16:07:41 -08:00
# ifdef CONFIG_PPC64
2007-05-12 10:36:56 -07:00
# include <asm/prom.h>
2016-03-31 22:56:55 +02:00
# define atmel_getb(priv, offset) readb(priv->iobase + offset)
# define atmel_putb(val, priv, offset) writeb(val, priv->iobase + offset)
2005-11-13 16:07:41 -08:00
# define atmel_request_region request_mem_region
# define atmel_release_region release_mem_region
2005-11-18 01:10:58 -08:00
2006-04-22 02:37:26 -07:00
static inline void atmel_put_base_addr ( void __iomem * iobase )
2005-11-13 16:07:41 -08:00
{
2006-04-22 02:37:26 -07:00
iounmap ( iobase ) ;
2005-11-13 16:07:41 -08:00
}
2006-04-22 02:37:26 -07:00
static void __iomem * atmel_get_base_addr ( unsigned long * base , int * region_size )
2005-11-13 16:07:41 -08:00
{
struct device_node * dn ;
unsigned long address , size ;
2006-07-12 15:41:30 +10:00
const unsigned int * reg ;
2005-11-13 16:07:41 -08:00
int reglen ;
int naddrc ;
int nsizec ;
dn = of_find_node_by_name ( NULL , " tpm " ) ;
if ( ! dn )
2005-11-18 01:10:58 -08:00
return NULL ;
2005-11-13 16:07:41 -08:00
2007-05-03 17:26:52 +10:00
if ( ! of_device_is_compatible ( dn , " AT97SC3201 " ) ) {
2005-11-13 16:07:41 -08:00
of_node_put ( dn ) ;
2005-11-18 01:10:58 -08:00
return NULL ;
2005-11-13 16:07:41 -08:00
}
2007-05-01 13:54:02 +10:00
reg = of_get_property ( dn , " reg " , & reglen ) ;
2007-04-03 10:56:50 +10:00
naddrc = of_n_addr_cells ( dn ) ;
2007-04-03 10:57:48 +10:00
nsizec = of_n_size_cells ( dn ) ;
2005-11-13 16:07:41 -08:00
of_node_put ( dn ) ;
if ( naddrc = = 2 )
address = ( ( unsigned long ) reg [ 0 ] < < 32 ) | reg [ 1 ] ;
else
address = reg [ 0 ] ;
if ( nsizec = = 2 )
size =
( ( unsigned long ) reg [ naddrc ] < < 32 ) | reg [ naddrc + 1 ] ;
else
size = reg [ naddrc ] ;
2006-04-22 02:37:26 -07:00
* base = address ;
* region_size = size ;
return ioremap ( * base , * region_size ) ;
2005-11-13 16:07:41 -08:00
}
# else
2016-03-23 08:16:09 +02:00
# define atmel_getb(chip, offset) inb(atmel_get_priv(chip)->base + offset)
# define atmel_putb(val, chip, offset) \
outb ( val , atmel_get_priv ( chip ) - > base + offset )
2005-11-13 16:07:41 -08:00
# define atmel_request_region request_region
# define atmel_release_region release_region
/* Atmel definitions */
enum tpm_atmel_addr {
TPM_ATMEL_BASE_ADDR_LO = 0x08 ,
TPM_ATMEL_BASE_ADDR_HI = 0x09
} ;
2017-01-25 16:47:39 +02:00
static inline int tpm_read_index ( int base , int index )
{
outb ( index , base ) ;
return inb ( base + 1 ) & 0xFF ;
}
2005-11-13 16:07:41 -08:00
/* Verify this is a 1.1 Atmel TPM */
static int atmel_verify_tpm11 ( void )
{
/* verify that it is an Atmel part */
if ( tpm_read_index ( TPM_ADDR , 4 ) ! = ' A ' | |
tpm_read_index ( TPM_ADDR , 5 ) ! = ' T ' | |
tpm_read_index ( TPM_ADDR , 6 ) ! = ' M ' | |
tpm_read_index ( TPM_ADDR , 7 ) ! = ' L ' )
return 1 ;
/* query chip for its version number */
if ( tpm_read_index ( TPM_ADDR , 0x00 ) ! = 1 | |
tpm_read_index ( TPM_ADDR , 0x01 ) ! = 1 )
return 1 ;
/* This is an atmel supported part */
return 0 ;
}
2006-04-22 02:37:26 -07:00
static inline void atmel_put_base_addr ( void __iomem * iobase )
2005-11-13 16:07:41 -08:00
{
}
/* Determine where to talk to device */
2006-04-22 02:37:26 -07:00
static void __iomem * atmel_get_base_addr ( unsigned long * base , int * region_size )
2005-11-13 16:07:41 -08:00
{
int lo , hi ;
if ( atmel_verify_tpm11 ( ) ! = 0 )
2005-11-18 01:10:58 -08:00
return NULL ;
2005-11-13 16:07:41 -08:00
lo = tpm_read_index ( TPM_ADDR , TPM_ATMEL_BASE_ADDR_LO ) ;
hi = tpm_read_index ( TPM_ADDR , TPM_ATMEL_BASE_ADDR_HI ) ;
2006-04-22 02:37:26 -07:00
* base = ( hi < < 8 ) | lo ;
* region_size = 2 ;
2005-11-13 16:07:41 -08:00
2006-04-22 02:37:26 -07:00
return ioport_map ( * base , * region_size ) ;
2005-11-13 16:07:41 -08:00
}
# endif