2005-04-16 15:20:36 -07:00
/*
* Copyright ( C ) 2004 IBM Corporation
*
* Authors :
* Leendert van Doorn < leendert @ watson . ibm . com >
* Dave Safford < safford @ watson . ibm . com >
* Reiner Sailer < sailer @ watson . ibm . com >
* Kylene Hall < kjhall @ us . ibm . com >
*
2007-08-22 14:01:04 -07:00
* Maintained by : < tpmdd - devel @ lists . sourceforge . net >
2005-04-16 15:20:36 -07:00
*
* Device driver for TCG / TCPA TPM ( trusted platform module ) .
* Specifications at www . trustedcomputinggroup . org
*
* 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 <linux/module.h>
# include <linux/delay.h>
# include <linux/fs.h>
2007-05-08 00:32:02 -07:00
# include <linux/mutex.h>
2006-10-18 13:55:46 -04:00
# include <linux/sched.h>
2005-04-16 15:20:36 -07:00
# include <linux/miscdevice.h>
2005-11-01 15:14:05 +00:00
# include <linux/platform_device.h>
2006-03-25 03:07:35 -08:00
# include <linux/io.h>
2005-04-16 15:20:36 -07:00
2005-06-23 22:01:48 -07:00
enum tpm_timeout {
TPM_TIMEOUT = 5 , /* msecs */
} ;
2005-04-16 15:20:36 -07:00
/* TPM addresses */
2005-06-23 22:01:48 -07:00
enum tpm_addr {
2005-06-25 14:55:39 -07:00
TPM_SUPERIO_ADDR = 0x2E ,
2005-06-23 22:01:48 -07:00
TPM_ADDR = 0x4E ,
} ;
2005-06-23 22:02:00 -07:00
extern ssize_t tpm_show_pubek ( struct device * , struct device_attribute * attr ,
char * ) ;
extern ssize_t tpm_show_pcrs ( struct device * , struct device_attribute * attr ,
char * ) ;
extern ssize_t tpm_show_caps ( struct device * , struct device_attribute * attr ,
char * ) ;
2006-04-22 02:37:50 -07:00
extern ssize_t tpm_show_caps_1_2 ( struct device * , struct device_attribute * attr ,
char * ) ;
2005-06-23 22:02:00 -07:00
extern ssize_t tpm_store_cancel ( struct device * , struct device_attribute * attr ,
const char * , size_t ) ;
2006-04-22 02:37:50 -07:00
extern ssize_t tpm_show_enabled ( struct device * , struct device_attribute * attr ,
char * ) ;
extern ssize_t tpm_show_active ( struct device * , struct device_attribute * attr ,
char * ) ;
extern ssize_t tpm_show_owned ( struct device * , struct device_attribute * attr ,
char * ) ;
extern ssize_t tpm_show_temp_deactivated ( struct device * ,
struct device_attribute * attr , char * ) ;
2005-04-16 15:20:36 -07:00
struct tpm_chip ;
struct tpm_vendor_specific {
2006-04-22 02:37:26 -07:00
const u8 req_complete_mask ;
const u8 req_complete_val ;
const u8 req_canceled ;
2005-11-13 16:07:41 -08:00
void __iomem * iobase ; /* ioremapped address */
unsigned long base ; /* TPM base address */
2006-04-22 02:38:03 -07:00
int irq ;
2005-11-13 16:07:41 -08:00
int region_size ;
int have_region ;
2005-04-16 15:20:36 -07:00
int ( * recv ) ( struct tpm_chip * , u8 * , size_t ) ;
int ( * send ) ( struct tpm_chip * , u8 * , size_t ) ;
void ( * cancel ) ( struct tpm_chip * ) ;
2005-10-30 15:03:23 -08:00
u8 ( * status ) ( struct tpm_chip * ) ;
2008-02-06 01:37:02 -08:00
void ( * release ) ( struct device * ) ;
2005-04-16 15:20:36 -07:00
struct miscdevice miscdev ;
2005-06-23 22:02:00 -07:00
struct attribute_group * attr_group ;
2006-04-22 02:38:03 -07:00
struct list_head list ;
int locality ;
2006-04-22 02:38:19 -07:00
unsigned long timeout_a , timeout_b , timeout_c , timeout_d ; /* jiffies */
unsigned long duration [ 3 ] ; /* jiffies */
2006-04-22 02:38:03 -07:00
wait_queue_head_t read_queue ;
wait_queue_head_t int_queue ;
2005-04-16 15:20:36 -07:00
} ;
struct tpm_chip {
2005-10-30 15:03:24 -08:00
struct device * dev ; /* Device stuff */
2005-04-16 15:20:36 -07:00
int dev_num ; /* /dev/tpm# */
int num_opens ; /* only one allowed */
int time_expired ;
/* Data passed to and from the tpm via the read/write calls */
u8 * data_buffer ;
atomic_t data_pending ;
2007-05-08 00:32:02 -07:00
struct mutex buffer_mutex ;
2005-04-16 15:20:36 -07:00
struct timer_list user_read_timer ; /* user needs to claim result */
2005-11-13 16:07:43 -08:00
struct work_struct work ;
2007-05-08 00:32:02 -07:00
struct mutex tpm_mutex ; /* tpm is processing */
2005-04-16 15:20:36 -07:00
2006-04-22 02:37:15 -07:00
struct tpm_vendor_specific vendor ;
2005-04-16 15:20:36 -07:00
2006-01-08 01:03:15 -08:00
struct dentry * * bios_dir ;
2005-04-16 15:20:36 -07:00
struct list_head list ;
2008-02-06 01:37:02 -08:00
void ( * release ) ( struct device * ) ;
2005-04-16 15:20:36 -07:00
} ;
2006-04-22 02:38:03 -07:00
# define to_tpm_chip(n) container_of(n, struct tpm_chip, vendor)
2005-06-25 14:55:39 -07:00
static inline int tpm_read_index ( int base , int index )
2005-04-16 15:20:36 -07:00
{
2005-06-25 14:55:39 -07:00
outb ( index , base ) ;
return inb ( base + 1 ) & 0xFF ;
2005-04-16 15:20:36 -07:00
}
2005-06-25 14:55:39 -07:00
static inline void tpm_write_index ( int base , int index , int value )
2005-04-16 15:20:36 -07:00
{
2005-06-25 14:55:39 -07:00
outb ( index , base ) ;
outb ( value & 0xFF , base + 1 ) ;
2005-04-16 15:20:36 -07:00
}
2006-04-22 02:37:50 -07:00
extern void tpm_get_timeouts ( struct tpm_chip * ) ;
extern void tpm_gen_interrupt ( struct tpm_chip * ) ;
extern void tpm_continue_selftest ( struct tpm_chip * ) ;
2006-04-22 02:37:38 -07:00
extern unsigned long tpm_calc_ordinal_duration ( struct tpm_chip * , u32 ) ;
2006-04-22 02:37:26 -07:00
extern struct tpm_chip * tpm_register_hardware ( struct device * ,
const struct tpm_vendor_specific * ) ;
2005-04-16 15:20:36 -07:00
extern int tpm_open ( struct inode * , struct file * ) ;
extern int tpm_release ( struct inode * , struct file * ) ;
extern ssize_t tpm_write ( struct file * , const char __user * , size_t ,
loff_t * ) ;
extern ssize_t tpm_read ( struct file * , char __user * , size_t , loff_t * ) ;
2005-10-30 15:03:24 -08:00
extern void tpm_remove_hardware ( struct device * ) ;
2005-10-30 15:03:25 -08:00
extern int tpm_pm_suspend ( struct device * , pm_message_t ) ;
extern int tpm_pm_resume ( struct device * ) ;
2006-01-08 01:03:15 -08:00
# ifdef CONFIG_ACPI
extern struct dentry * * tpm_bios_log_setup ( char * ) ;
extern void tpm_bios_log_teardown ( struct dentry * * ) ;
# else
2006-05-15 09:44:27 -07:00
static inline struct dentry * * tpm_bios_log_setup ( char * name )
2006-01-08 01:03:15 -08:00
{
return NULL ;
}
static inline void tpm_bios_log_teardown ( struct dentry * * dir )
{
}
# endif