2005-04-17 05:20:36 +07:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2004 IBM Corporation
|
2015-10-11 16:26:58 +07:00
|
|
|
* Copyright (C) 2015 Intel Corporation
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
|
|
|
* 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-23 04:01:04 +07:00
|
|
|
* Maintained by: <tpmdd-devel@lists.sourceforge.net>
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
|
|
|
* Device driver for TCG/TCPA TPM (trusted platform module).
|
2015-01-17 23:03:30 +07:00
|
|
|
* Specifications at www.trustedcomputinggroup.org
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
|
|
|
* 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.
|
2015-01-17 23:03:30 +07:00
|
|
|
*
|
2005-04-17 05:20:36 +07:00
|
|
|
*/
|
2016-05-05 02:06:22 +07:00
|
|
|
|
|
|
|
#ifndef __TPM_H__
|
|
|
|
#define __TPM_H__
|
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/delay.h>
|
|
|
|
#include <linux/fs.h>
|
2017-11-17 20:24:03 +07:00
|
|
|
#include <linux/hw_random.h>
|
2007-05-08 14:32:02 +07:00
|
|
|
#include <linux/mutex.h>
|
2006-10-19 00:55:46 +07:00
|
|
|
#include <linux/sched.h>
|
2005-11-01 22:14:05 +07:00
|
|
|
#include <linux/platform_device.h>
|
2006-03-25 18:07:35 +07:00
|
|
|
#include <linux/io.h>
|
2009-02-03 00:23:44 +07:00
|
|
|
#include <linux/tpm.h>
|
2014-12-13 02:46:35 +07:00
|
|
|
#include <linux/acpi.h>
|
2014-12-13 02:46:37 +07:00
|
|
|
#include <linux/cdev.h>
|
2015-10-11 16:26:58 +07:00
|
|
|
#include <linux/highmem.h>
|
2017-09-20 15:13:36 +07:00
|
|
|
#include <linux/tpm_eventlog.h>
|
2017-01-30 16:59:41 +07:00
|
|
|
#include <crypto/hash_info.h>
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2017-06-19 09:17:59 +07:00
|
|
|
#ifdef CONFIG_X86
|
|
|
|
#include <asm/intel-family.h>
|
|
|
|
#endif
|
|
|
|
|
2012-06-08 01:47:14 +07:00
|
|
|
enum tpm_const {
|
|
|
|
TPM_MINOR = 224, /* officially assigned */
|
|
|
|
TPM_BUFSIZE = 4096,
|
2016-02-29 20:53:02 +07:00
|
|
|
TPM_NUM_DEVICES = 65536,
|
2013-03-18 04:56:39 +07:00
|
|
|
TPM_RETRY = 50, /* 5 seconds */
|
2016-11-14 17:00:50 +07:00
|
|
|
TPM_NUM_EVENT_LOG_FILES = 3,
|
2012-06-08 01:47:14 +07:00
|
|
|
};
|
|
|
|
|
2005-06-24 12:01:48 +07:00
|
|
|
enum tpm_timeout {
|
|
|
|
TPM_TIMEOUT = 5, /* msecs */
|
2017-08-15 01:09:16 +07:00
|
|
|
TPM_TIMEOUT_RETRY = 100, /* msecs */
|
2018-05-07 23:07:32 +07:00
|
|
|
TPM_TIMEOUT_RANGE_US = 300, /* usecs */
|
tpm: reduce polling time to usecs for even finer granularity
The TPM burstcount and status commands are supposed to return very
quickly [2][3]. This patch further reduces the TPM poll sleep time to usecs
in get_burstcount() and wait_for_tpm_stat() by calling usleep_range()
directly.
After this change, performance on a system[1] with a TPM 1.2 with an 8 byte
burstcount for 1000 extends improved from ~10.7 sec to ~7 sec.
[1] All tests are performed on an x86 based, locked down, single purpose
closed system. It has Infineon TPM 1.2 using LPC Bus.
[2] From the TCG Specification "TCG PC Client Specific TPM Interface
Specification (TIS), Family 1.2":
"NOTE : It takes roughly 330 ns per byte transfer on LPC. 256 bytes would
take 84 us, which is a long time to stall the CPU. Chipsets may not be
designed to post this much data to LPC; therefore, the CPU itself is
stalled for much of this time. Sending 1 kB would take 350 μs. Therefore,
even if the TPM_STS_x.burstCount field is a high value, software SHOULD
be interruptible during this period."
[3] From the TCG Specification 2.0, "TCG PC Client Platform TPM Profile
(PTP) Specification":
"It takes roughly 330 ns per byte transfer on LPC. 256 bytes would take
84 us. Chipsets may not be designed to post this much data to LPC;
therefore, the CPU itself is stalled for much of this time. Sending 1 kB
would take 350 us. Therefore, even if the TPM_STS_x.burstCount field is a
high value, software should be interruptible during this period. For SPI,
assuming 20MHz clock and 64-byte transfers, it would take about 120 usec
to move 256B of data. Sending 1kB would take about 500 usec. If the
transactions are done using 4 bytes at a time, then it would take about
1 msec. to transfer 1kB of data."
Signed-off-by: Nayna Jain <nayna@linux.vnet.ibm.com>
Reviewed-by: Mimi Zohar <zohar@linux.vnet.ibm.com>
Reviewed-by: Jarkko Sakkinen <jarkko.sakkinen@linux.intel.com>
Acked-by: Jay Freyensee <why2jjj.linux@gmail.com>
Tested-by: Jarkko Sakkinen <jarkko.sakkinen@linux.intel.com>
Signed-off-by: Jarkko Sakkinen <jarkko.sakkinen@linux.intel.com>
2018-05-16 12:51:25 +07:00
|
|
|
TPM_TIMEOUT_POLL = 1, /* msecs */
|
|
|
|
TPM_TIMEOUT_USECS_MIN = 100, /* usecs */
|
|
|
|
TPM_TIMEOUT_USECS_MAX = 500 /* usecs */
|
2005-06-24 12:01:48 +07:00
|
|
|
};
|
2005-04-17 05:20:36 +07:00
|
|
|
|
|
|
|
/* TPM addresses */
|
2005-06-24 12:01:48 +07:00
|
|
|
enum tpm_addr {
|
2005-06-26 04:55:39 +07:00
|
|
|
TPM_SUPERIO_ADDR = 0x2E,
|
2005-06-24 12:01:48 +07:00
|
|
|
TPM_ADDR = 0x4E,
|
|
|
|
};
|
|
|
|
|
2013-11-27 03:30:41 +07:00
|
|
|
/* Indexes the duration array */
|
|
|
|
enum tpm_duration {
|
|
|
|
TPM_SHORT = 0,
|
|
|
|
TPM_MEDIUM = 1,
|
|
|
|
TPM_LONG = 2,
|
2018-03-10 22:15:45 +07:00
|
|
|
TPM_LONG_LONG = 3,
|
2013-11-27 03:30:41 +07:00
|
|
|
TPM_UNDEFINED,
|
2018-03-10 22:15:45 +07:00
|
|
|
TPM_NUM_DURATIONS = TPM_UNDEFINED,
|
2013-11-27 03:30:41 +07:00
|
|
|
};
|
|
|
|
|
2013-03-18 04:56:39 +07:00
|
|
|
#define TPM_WARN_RETRY 0x800
|
2011-11-12 00:57:04 +07:00
|
|
|
#define TPM_WARN_DOING_SELFTEST 0x802
|
2012-01-18 10:07:30 +07:00
|
|
|
#define TPM_ERR_DEACTIVATED 0x6
|
|
|
|
#define TPM_ERR_DISABLED 0x7
|
2012-11-22 03:54:33 +07:00
|
|
|
#define TPM_ERR_INVALID_POSTINIT 38
|
2012-01-18 10:07:30 +07:00
|
|
|
|
2011-11-02 02:00:52 +07:00
|
|
|
#define TPM_HEADER_SIZE 10
|
2014-12-13 02:46:38 +07:00
|
|
|
|
|
|
|
enum tpm2_const {
|
2018-03-10 22:15:45 +07:00
|
|
|
TPM2_PLATFORM_PCR = 24,
|
|
|
|
TPM2_PCR_SELECT_MIN = ((TPM2_PLATFORM_PCR + 7) / 8),
|
|
|
|
};
|
|
|
|
|
|
|
|
enum tpm2_timeouts {
|
|
|
|
TPM2_TIMEOUT_A = 750,
|
|
|
|
TPM2_TIMEOUT_B = 2000,
|
|
|
|
TPM2_TIMEOUT_C = 200,
|
|
|
|
TPM2_TIMEOUT_D = 30,
|
|
|
|
TPM2_DURATION_SHORT = 20,
|
|
|
|
TPM2_DURATION_MEDIUM = 750,
|
|
|
|
TPM2_DURATION_LONG = 2000,
|
|
|
|
TPM2_DURATION_LONG_LONG = 300000,
|
|
|
|
TPM2_DURATION_DEFAULT = 120000,
|
2014-12-13 02:46:38 +07:00
|
|
|
};
|
|
|
|
|
|
|
|
enum tpm2_structures {
|
|
|
|
TPM2_ST_NO_SESSIONS = 0x8001,
|
|
|
|
TPM2_ST_SESSIONS = 0x8002,
|
|
|
|
};
|
|
|
|
|
2017-11-30 14:39:07 +07:00
|
|
|
/* Indicates from what layer of the software stack the error comes from */
|
|
|
|
#define TSS2_RC_LAYER_SHIFT 16
|
|
|
|
#define TSS2_RESMGR_TPM_RC_LAYER (11 << TSS2_RC_LAYER_SHIFT)
|
|
|
|
|
2014-12-13 02:46:38 +07:00
|
|
|
enum tpm2_return_codes {
|
2017-01-06 19:03:45 +07:00
|
|
|
TPM2_RC_SUCCESS = 0x0000,
|
2015-11-06 02:43:06 +07:00
|
|
|
TPM2_RC_HASH = 0x0083, /* RC_FMT1 */
|
2017-01-06 19:03:45 +07:00
|
|
|
TPM2_RC_HANDLE = 0x008B,
|
2015-11-06 02:43:06 +07:00
|
|
|
TPM2_RC_INITIALIZE = 0x0100, /* RC_VER1 */
|
2018-03-22 22:32:20 +07:00
|
|
|
TPM2_RC_FAILURE = 0x0101,
|
2014-12-13 02:46:38 +07:00
|
|
|
TPM2_RC_DISABLED = 0x0120,
|
2017-11-30 14:39:07 +07:00
|
|
|
TPM2_RC_COMMAND_CODE = 0x0143,
|
2015-11-06 02:43:06 +07:00
|
|
|
TPM2_RC_TESTING = 0x090A, /* RC_WARN */
|
2017-01-06 19:03:45 +07:00
|
|
|
TPM2_RC_REFERENCE_H0 = 0x0910,
|
2018-03-22 01:43:48 +07:00
|
|
|
TPM2_RC_RETRY = 0x0922,
|
2014-12-13 02:46:38 +07:00
|
|
|
};
|
|
|
|
|
|
|
|
enum tpm2_algorithms {
|
2017-01-30 16:59:40 +07:00
|
|
|
TPM2_ALG_ERROR = 0x0000,
|
2014-12-13 02:46:38 +07:00
|
|
|
TPM2_ALG_SHA1 = 0x0004,
|
2015-05-30 12:09:04 +07:00
|
|
|
TPM2_ALG_KEYEDHASH = 0x0008,
|
|
|
|
TPM2_ALG_SHA256 = 0x000B,
|
2015-11-06 02:43:06 +07:00
|
|
|
TPM2_ALG_SHA384 = 0x000C,
|
|
|
|
TPM2_ALG_SHA512 = 0x000D,
|
|
|
|
TPM2_ALG_NULL = 0x0010,
|
|
|
|
TPM2_ALG_SM3_256 = 0x0012,
|
2014-12-13 02:46:38 +07:00
|
|
|
};
|
|
|
|
|
|
|
|
enum tpm2_command_codes {
|
2018-10-20 01:22:48 +07:00
|
|
|
TPM2_CC_FIRST = 0x011F,
|
|
|
|
TPM2_CC_HIERARCHY_CONTROL = 0x0121,
|
|
|
|
TPM2_CC_HIERARCHY_CHANGE_AUTH = 0x0129,
|
|
|
|
TPM2_CC_CREATE_PRIMARY = 0x0131,
|
|
|
|
TPM2_CC_SEQUENCE_COMPLETE = 0x013E,
|
|
|
|
TPM2_CC_SELF_TEST = 0x0143,
|
|
|
|
TPM2_CC_STARTUP = 0x0144,
|
|
|
|
TPM2_CC_SHUTDOWN = 0x0145,
|
|
|
|
TPM2_CC_NV_READ = 0x014E,
|
|
|
|
TPM2_CC_CREATE = 0x0153,
|
|
|
|
TPM2_CC_LOAD = 0x0157,
|
|
|
|
TPM2_CC_SEQUENCE_UPDATE = 0x015C,
|
|
|
|
TPM2_CC_UNSEAL = 0x015E,
|
|
|
|
TPM2_CC_CONTEXT_LOAD = 0x0161,
|
|
|
|
TPM2_CC_CONTEXT_SAVE = 0x0162,
|
|
|
|
TPM2_CC_FLUSH_CONTEXT = 0x0165,
|
|
|
|
TPM2_CC_VERIFY_SIGNATURE = 0x0177,
|
|
|
|
TPM2_CC_GET_CAPABILITY = 0x017A,
|
|
|
|
TPM2_CC_GET_RANDOM = 0x017B,
|
|
|
|
TPM2_CC_PCR_READ = 0x017E,
|
|
|
|
TPM2_CC_PCR_EXTEND = 0x0182,
|
|
|
|
TPM2_CC_EVENT_SEQUENCE_COMPLETE = 0x0185,
|
|
|
|
TPM2_CC_HASH_SEQUENCE_START = 0x0186,
|
|
|
|
TPM2_CC_CREATE_LOADED = 0x0191,
|
|
|
|
TPM2_CC_LAST = 0x0193, /* Spec 1.36 */
|
2014-12-13 02:46:38 +07:00
|
|
|
};
|
|
|
|
|
|
|
|
enum tpm2_permanent_handles {
|
|
|
|
TPM2_RS_PW = 0x40000009,
|
|
|
|
};
|
|
|
|
|
|
|
|
enum tpm2_capabilities {
|
2017-01-06 19:03:45 +07:00
|
|
|
TPM2_CAP_HANDLES = 1,
|
2016-11-11 11:42:07 +07:00
|
|
|
TPM2_CAP_COMMANDS = 2,
|
2017-01-30 16:59:40 +07:00
|
|
|
TPM2_CAP_PCRS = 5,
|
2014-12-13 02:46:38 +07:00
|
|
|
TPM2_CAP_TPM_PROPERTIES = 6,
|
|
|
|
};
|
|
|
|
|
2016-11-11 11:42:07 +07:00
|
|
|
enum tpm2_properties {
|
|
|
|
TPM_PT_TOTAL_COMMANDS = 0x0129,
|
|
|
|
};
|
|
|
|
|
2014-12-13 02:46:38 +07:00
|
|
|
enum tpm2_startup_types {
|
|
|
|
TPM2_SU_CLEAR = 0x0000,
|
|
|
|
TPM2_SU_STATE = 0x0001,
|
|
|
|
};
|
|
|
|
|
2016-11-11 11:42:07 +07:00
|
|
|
enum tpm2_cc_attrs {
|
|
|
|
TPM2_CC_ATTR_CHANDLES = 25,
|
|
|
|
TPM2_CC_ATTR_RHANDLE = 28,
|
|
|
|
};
|
|
|
|
|
2012-01-21 00:58:49 +07:00
|
|
|
#define TPM_VID_INTEL 0x8086
|
2013-01-23 02:52:35 +07:00
|
|
|
#define TPM_VID_WINBOND 0x1050
|
|
|
|
#define TPM_VID_STM 0x104A
|
2012-01-21 00:58:49 +07:00
|
|
|
|
2014-12-13 02:46:35 +07:00
|
|
|
#define TPM_PPI_VERSION_LEN 3
|
|
|
|
|
2017-01-06 19:03:45 +07:00
|
|
|
struct tpm_space {
|
|
|
|
u32 context_tbl[3];
|
|
|
|
u8 *context_buf;
|
2017-02-01 06:47:31 +07:00
|
|
|
u32 session_tbl[3];
|
|
|
|
u8 *session_buf;
|
2017-01-06 19:03:45 +07:00
|
|
|
};
|
|
|
|
|
2014-12-13 02:46:34 +07:00
|
|
|
enum tpm_chip_flags {
|
2015-04-14 21:56:48 +07:00
|
|
|
TPM_CHIP_FLAG_TPM2 = BIT(1),
|
2016-04-01 03:56:56 +07:00
|
|
|
TPM_CHIP_FLAG_IRQ = BIT(2),
|
2016-04-19 00:26:14 +07:00
|
|
|
TPM_CHIP_FLAG_VIRTUAL = BIT(3),
|
2016-10-27 05:28:44 +07:00
|
|
|
TPM_CHIP_FLAG_HAVE_TIMEOUTS = BIT(4),
|
2017-06-27 17:27:24 +07:00
|
|
|
TPM_CHIP_FLAG_ALWAYS_POWERED = BIT(5),
|
2014-12-13 02:46:34 +07:00
|
|
|
};
|
|
|
|
|
2017-01-23 14:26:27 +07:00
|
|
|
struct tpm_bios_log {
|
|
|
|
void *bios_event_log;
|
|
|
|
void *bios_event_log_end;
|
|
|
|
};
|
|
|
|
|
2016-11-14 17:00:52 +07:00
|
|
|
struct tpm_chip_seqops {
|
|
|
|
struct tpm_chip *chip;
|
|
|
|
const struct seq_operations *seqops;
|
|
|
|
};
|
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
struct tpm_chip {
|
2014-12-13 02:46:37 +07:00
|
|
|
struct device dev;
|
2017-01-04 00:07:32 +07:00
|
|
|
struct device devs;
|
2014-12-13 02:46:37 +07:00
|
|
|
struct cdev cdev;
|
2017-01-04 00:07:32 +07:00
|
|
|
struct cdev cdevs;
|
2014-12-13 02:46:37 +07:00
|
|
|
|
2016-02-13 10:29:53 +07:00
|
|
|
/* A driver callback under ops cannot be run unless ops_sem is held
|
|
|
|
* (sometimes implicitly, eg for the sysfs code). ops becomes null
|
|
|
|
* when the driver is unregistered, see tpm_try_get_ops.
|
|
|
|
*/
|
|
|
|
struct rw_semaphore ops_sem;
|
2013-11-27 03:30:44 +07:00
|
|
|
const struct tpm_class_ops *ops;
|
2016-02-13 10:29:53 +07:00
|
|
|
|
2016-11-14 17:00:52 +07:00
|
|
|
struct tpm_bios_log log;
|
|
|
|
struct tpm_chip_seqops bin_log_seqops;
|
|
|
|
struct tpm_chip_seqops ascii_log_seqops;
|
|
|
|
|
2014-12-13 02:46:34 +07:00
|
|
|
unsigned int flags;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
|
|
|
int dev_num; /* /dev/tpm# */
|
2008-10-11 05:04:02 +07:00
|
|
|
unsigned long is_open; /* only one allowed */
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2017-11-17 20:24:03 +07:00
|
|
|
char hwrng_name[64];
|
|
|
|
struct hwrng hwrng;
|
|
|
|
|
2007-05-08 14:32:02 +07:00
|
|
|
struct mutex tpm_mutex; /* tpm is processing */
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2016-04-01 03:56:59 +07:00
|
|
|
unsigned long timeout_a; /* jiffies */
|
|
|
|
unsigned long timeout_b; /* jiffies */
|
|
|
|
unsigned long timeout_c; /* jiffies */
|
|
|
|
unsigned long timeout_d; /* jiffies */
|
|
|
|
bool timeout_adjusted;
|
2018-03-10 22:15:45 +07:00
|
|
|
unsigned long duration[TPM_NUM_DURATIONS]; /* jiffies */
|
2016-04-01 03:56:59 +07:00
|
|
|
bool duration_adjusted;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2016-11-14 17:00:50 +07:00
|
|
|
struct dentry *bios_dir[TPM_NUM_EVENT_LOG_FILES];
|
2006-01-08 16:03:15 +07:00
|
|
|
|
2016-04-19 00:26:13 +07:00
|
|
|
const struct attribute_group *groups[3];
|
2015-04-14 21:56:48 +07:00
|
|
|
unsigned int groups_cnt;
|
2017-01-30 16:59:40 +07:00
|
|
|
|
|
|
|
u16 active_banks[7];
|
2016-04-19 00:26:13 +07:00
|
|
|
#ifdef CONFIG_ACPI
|
2014-12-13 02:46:35 +07:00
|
|
|
acpi_handle acpi_dev_handle;
|
|
|
|
char ppi_version[TPM_PPI_VERSION_LEN + 1];
|
|
|
|
#endif /* CONFIG_ACPI */
|
2016-11-11 11:42:07 +07:00
|
|
|
|
2017-01-06 19:03:45 +07:00
|
|
|
struct tpm_space work_space;
|
2016-11-11 11:42:07 +07:00
|
|
|
u32 nr_commands;
|
|
|
|
u32 *cc_attrs_tbl;
|
2017-03-24 16:45:49 +07:00
|
|
|
|
|
|
|
/* active locality */
|
|
|
|
int locality;
|
2005-04-17 05:20:36 +07:00
|
|
|
};
|
|
|
|
|
2015-04-14 21:56:48 +07:00
|
|
|
#define to_tpm_chip(d) container_of(d, struct tpm_chip, dev)
|
2006-04-22 16:38:03 +07:00
|
|
|
|
2009-02-03 00:23:43 +07:00
|
|
|
struct tpm_input_header {
|
|
|
|
__be16 tag;
|
|
|
|
__be32 length;
|
|
|
|
__be32 ordinal;
|
2012-11-22 03:56:45 +07:00
|
|
|
} __packed;
|
2009-02-03 00:23:43 +07:00
|
|
|
|
|
|
|
struct tpm_output_header {
|
|
|
|
__be16 tag;
|
|
|
|
__be32 length;
|
|
|
|
__be32 return_code;
|
2012-11-22 03:56:45 +07:00
|
|
|
} __packed;
|
2009-02-03 00:23:43 +07:00
|
|
|
|
2017-05-03 23:19:09 +07:00
|
|
|
#define TPM_TAG_RQU_COMMAND 193
|
2013-11-27 03:30:41 +07:00
|
|
|
|
2009-02-03 00:23:43 +07:00
|
|
|
struct stclear_flags_t {
|
|
|
|
__be16 tag;
|
|
|
|
u8 deactivated;
|
|
|
|
u8 disableForceClear;
|
|
|
|
u8 physicalPresence;
|
|
|
|
u8 physicalPresenceLock;
|
|
|
|
u8 bGlobalLock;
|
2012-11-22 03:56:45 +07:00
|
|
|
} __packed;
|
2009-02-03 00:23:43 +07:00
|
|
|
|
|
|
|
struct tpm_version_t {
|
|
|
|
u8 Major;
|
|
|
|
u8 Minor;
|
|
|
|
u8 revMajor;
|
|
|
|
u8 revMinor;
|
2012-11-22 03:56:45 +07:00
|
|
|
} __packed;
|
2009-02-03 00:23:43 +07:00
|
|
|
|
|
|
|
struct tpm_version_1_2_t {
|
|
|
|
__be16 tag;
|
|
|
|
u8 Major;
|
|
|
|
u8 Minor;
|
|
|
|
u8 revMajor;
|
|
|
|
u8 revMinor;
|
2012-11-22 03:56:45 +07:00
|
|
|
} __packed;
|
2009-02-03 00:23:43 +07:00
|
|
|
|
|
|
|
struct timeout_t {
|
|
|
|
__be32 a;
|
|
|
|
__be32 b;
|
|
|
|
__be32 c;
|
|
|
|
__be32 d;
|
2012-11-22 03:56:45 +07:00
|
|
|
} __packed;
|
2009-02-03 00:23:43 +07:00
|
|
|
|
|
|
|
struct duration_t {
|
|
|
|
__be32 tpm_short;
|
|
|
|
__be32 tpm_medium;
|
|
|
|
__be32 tpm_long;
|
2012-11-22 03:56:45 +07:00
|
|
|
} __packed;
|
2009-02-03 00:23:43 +07:00
|
|
|
|
|
|
|
struct permanent_flags_t {
|
|
|
|
__be16 tag;
|
|
|
|
u8 disable;
|
|
|
|
u8 ownership;
|
|
|
|
u8 deactivated;
|
|
|
|
u8 readPubek;
|
|
|
|
u8 disableOwnerClear;
|
|
|
|
u8 allowMaintenance;
|
|
|
|
u8 physicalPresenceLifetimeLock;
|
|
|
|
u8 physicalPresenceHWEnable;
|
|
|
|
u8 physicalPresenceCMDEnable;
|
|
|
|
u8 CEKPUsed;
|
|
|
|
u8 TPMpost;
|
|
|
|
u8 TPMpostLock;
|
|
|
|
u8 FIPS;
|
|
|
|
u8 operator;
|
|
|
|
u8 enableRevokeEK;
|
|
|
|
u8 nvLocked;
|
|
|
|
u8 readSRKPub;
|
|
|
|
u8 tpmEstablished;
|
|
|
|
u8 maintenanceDone;
|
|
|
|
u8 disableFullDALogicInfo;
|
2012-11-22 03:56:45 +07:00
|
|
|
} __packed;
|
2009-02-03 00:23:43 +07:00
|
|
|
|
|
|
|
typedef union {
|
|
|
|
struct permanent_flags_t perm_flags;
|
|
|
|
struct stclear_flags_t stclear_flags;
|
2017-02-01 01:43:59 +07:00
|
|
|
__u8 owned;
|
2009-02-03 00:23:43 +07:00
|
|
|
__be32 num_pcrs;
|
|
|
|
struct tpm_version_t tpm_version;
|
|
|
|
struct tpm_version_1_2_t tpm_version_1_2;
|
|
|
|
__be32 manufacturer_id;
|
|
|
|
struct timeout_t timeout;
|
|
|
|
struct duration_t duration;
|
|
|
|
} cap_t;
|
|
|
|
|
2013-11-27 03:30:41 +07:00
|
|
|
enum tpm_capabilities {
|
2016-09-20 03:22:09 +07:00
|
|
|
TPM_CAP_FLAG = 4,
|
|
|
|
TPM_CAP_PROP = 5,
|
|
|
|
TPM_CAP_VERSION_1_1 = 0x06,
|
|
|
|
TPM_CAP_VERSION_1_2 = 0x1A,
|
2013-11-27 03:30:41 +07:00
|
|
|
};
|
|
|
|
|
|
|
|
enum tpm_sub_capabilities {
|
2016-09-20 03:22:09 +07:00
|
|
|
TPM_CAP_PROP_PCR = 0x101,
|
|
|
|
TPM_CAP_PROP_MANUFACTURER = 0x103,
|
|
|
|
TPM_CAP_FLAG_PERM = 0x108,
|
|
|
|
TPM_CAP_FLAG_VOL = 0x109,
|
|
|
|
TPM_CAP_PROP_OWNER = 0x111,
|
|
|
|
TPM_CAP_PROP_TIS_TIMEOUT = 0x115,
|
|
|
|
TPM_CAP_PROP_TIS_DURATION = 0x120,
|
2013-11-27 03:30:41 +07:00
|
|
|
};
|
|
|
|
|
2009-02-03 00:23:43 +07:00
|
|
|
typedef union {
|
|
|
|
struct tpm_input_header in;
|
|
|
|
struct tpm_output_header out;
|
|
|
|
} tpm_cmd_header;
|
|
|
|
|
2009-02-03 00:23:44 +07:00
|
|
|
struct tpm_pcrread_out {
|
|
|
|
u8 pcr_result[TPM_DIGEST_SIZE];
|
2012-11-22 03:56:45 +07:00
|
|
|
} __packed;
|
2009-02-03 00:23:44 +07:00
|
|
|
|
|
|
|
struct tpm_pcrread_in {
|
|
|
|
__be32 pcr_idx;
|
2012-11-22 03:56:45 +07:00
|
|
|
} __packed;
|
2009-02-03 00:23:44 +07:00
|
|
|
|
2012-06-08 01:47:14 +07:00
|
|
|
/* 128 bytes is an arbitrary cap. This could be as large as TPM_BUFSIZE - 18
|
|
|
|
* bytes, but 128 is still a relatively large number of random bytes and
|
|
|
|
* anything much bigger causes users of struct tpm_cmd_t to start getting
|
|
|
|
* compiler warnings about stack frame size. */
|
|
|
|
#define TPM_MAX_RNG_DATA 128
|
|
|
|
|
|
|
|
struct tpm_getrandom_out {
|
|
|
|
__be32 rng_data_len;
|
|
|
|
u8 rng_data[TPM_MAX_RNG_DATA];
|
2012-11-22 03:56:45 +07:00
|
|
|
} __packed;
|
2012-06-08 01:47:14 +07:00
|
|
|
|
|
|
|
struct tpm_getrandom_in {
|
|
|
|
__be32 num_bytes;
|
2012-11-22 03:56:45 +07:00
|
|
|
} __packed;
|
2012-06-08 01:47:14 +07:00
|
|
|
|
2009-02-03 00:23:43 +07:00
|
|
|
typedef union {
|
2009-02-03 00:23:44 +07:00
|
|
|
struct tpm_pcrread_in pcrread_in;
|
|
|
|
struct tpm_pcrread_out pcrread_out;
|
2012-06-08 01:47:14 +07:00
|
|
|
struct tpm_getrandom_in getrandom_in;
|
|
|
|
struct tpm_getrandom_out getrandom_out;
|
2009-02-03 00:23:43 +07:00
|
|
|
} tpm_cmd_params;
|
|
|
|
|
|
|
|
struct tpm_cmd_t {
|
|
|
|
tpm_cmd_header header;
|
|
|
|
tpm_cmd_params params;
|
2012-11-22 03:56:45 +07:00
|
|
|
} __packed;
|
2009-02-03 00:23:43 +07:00
|
|
|
|
2017-01-30 16:59:41 +07:00
|
|
|
|
2015-10-11 16:26:58 +07:00
|
|
|
/* A string buffer type for constructing TPM commands. This is based on the
|
|
|
|
* ideas of string buffer code in security/keys/trusted.h but is heap based
|
|
|
|
* in order to keep the stack usage minimal.
|
|
|
|
*/
|
|
|
|
|
|
|
|
enum tpm_buf_flags {
|
|
|
|
TPM_BUF_OVERFLOW = BIT(0),
|
|
|
|
};
|
|
|
|
|
|
|
|
struct tpm_buf {
|
|
|
|
struct page *data_page;
|
|
|
|
unsigned int flags;
|
|
|
|
u8 *data;
|
|
|
|
};
|
|
|
|
|
2018-03-26 19:14:06 +07:00
|
|
|
static inline void tpm_buf_reset(struct tpm_buf *buf, u16 tag, u32 ordinal)
|
2015-10-11 16:26:58 +07:00
|
|
|
{
|
|
|
|
struct tpm_input_header *head;
|
2018-03-26 19:14:06 +07:00
|
|
|
head = (struct tpm_input_header *)buf->data;
|
|
|
|
head->tag = cpu_to_be16(tag);
|
|
|
|
head->length = cpu_to_be32(sizeof(*head));
|
|
|
|
head->ordinal = cpu_to_be32(ordinal);
|
|
|
|
}
|
2015-10-11 16:26:58 +07:00
|
|
|
|
2018-03-26 19:14:06 +07:00
|
|
|
static inline int tpm_buf_init(struct tpm_buf *buf, u16 tag, u32 ordinal)
|
|
|
|
{
|
2015-10-11 16:26:58 +07:00
|
|
|
buf->data_page = alloc_page(GFP_HIGHUSER);
|
|
|
|
if (!buf->data_page)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
buf->flags = 0;
|
|
|
|
buf->data = kmap(buf->data_page);
|
2018-03-26 19:14:06 +07:00
|
|
|
tpm_buf_reset(buf, tag, ordinal);
|
2015-10-11 16:26:58 +07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void tpm_buf_destroy(struct tpm_buf *buf)
|
|
|
|
{
|
|
|
|
kunmap(buf->data_page);
|
|
|
|
__free_page(buf->data_page);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline u32 tpm_buf_length(struct tpm_buf *buf)
|
|
|
|
{
|
|
|
|
struct tpm_input_header *head = (struct tpm_input_header *) buf->data;
|
|
|
|
|
|
|
|
return be32_to_cpu(head->length);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline u16 tpm_buf_tag(struct tpm_buf *buf)
|
|
|
|
{
|
|
|
|
struct tpm_input_header *head = (struct tpm_input_header *) buf->data;
|
|
|
|
|
|
|
|
return be16_to_cpu(head->tag);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void tpm_buf_append(struct tpm_buf *buf,
|
|
|
|
const unsigned char *new_data,
|
|
|
|
unsigned int new_len)
|
|
|
|
{
|
|
|
|
struct tpm_input_header *head = (struct tpm_input_header *) buf->data;
|
|
|
|
u32 len = tpm_buf_length(buf);
|
|
|
|
|
|
|
|
/* Return silently if overflow has already happened. */
|
|
|
|
if (buf->flags & TPM_BUF_OVERFLOW)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if ((len + new_len) > PAGE_SIZE) {
|
|
|
|
WARN(1, "tpm_buf: overflow\n");
|
|
|
|
buf->flags |= TPM_BUF_OVERFLOW;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(&buf->data[len], new_data, new_len);
|
|
|
|
head->length = cpu_to_be32(len + new_len);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void tpm_buf_append_u8(struct tpm_buf *buf, const u8 value)
|
|
|
|
{
|
|
|
|
tpm_buf_append(buf, &value, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void tpm_buf_append_u16(struct tpm_buf *buf, const u16 value)
|
|
|
|
{
|
|
|
|
__be16 value2 = cpu_to_be16(value);
|
|
|
|
|
|
|
|
tpm_buf_append(buf, (u8 *) &value2, 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void tpm_buf_append_u32(struct tpm_buf *buf, const u32 value)
|
|
|
|
{
|
|
|
|
__be32 value2 = cpu_to_be32(value);
|
|
|
|
|
|
|
|
tpm_buf_append(buf, (u8 *) &value2, 4);
|
|
|
|
}
|
|
|
|
|
2014-12-13 02:46:37 +07:00
|
|
|
extern struct class *tpm_class;
|
2017-01-04 00:07:32 +07:00
|
|
|
extern struct class *tpmrm_class;
|
2014-12-13 02:46:37 +07:00
|
|
|
extern dev_t tpm_devt;
|
|
|
|
extern const struct file_operations tpm_fops;
|
2017-01-04 00:07:32 +07:00
|
|
|
extern const struct file_operations tpmrm_fops;
|
2016-02-29 20:53:02 +07:00
|
|
|
extern struct idr dev_nums_idr;
|
2014-12-13 02:46:37 +07:00
|
|
|
|
2018-06-28 22:13:33 +07:00
|
|
|
/**
|
2018-06-29 19:24:50 +07:00
|
|
|
* enum tpm_transmit_flags - flags for tpm_transmit()
|
2018-06-28 22:13:33 +07:00
|
|
|
*
|
2018-06-29 19:24:50 +07:00
|
|
|
* @TPM_TRANSMIT_UNLOCKED: do not lock the chip
|
|
|
|
* @TPM_TRANSMIT_NESTED: discard setup steps (power management,
|
|
|
|
* locality) including locking (i.e. implicit
|
|
|
|
* UNLOCKED)
|
2018-06-28 22:13:33 +07:00
|
|
|
*/
|
2016-08-17 02:00:38 +07:00
|
|
|
enum tpm_transmit_flags {
|
2018-06-29 19:24:50 +07:00
|
|
|
TPM_TRANSMIT_UNLOCKED = BIT(0),
|
|
|
|
TPM_TRANSMIT_NESTED = BIT(1),
|
2016-08-17 02:00:38 +07:00
|
|
|
};
|
|
|
|
|
2017-01-06 19:03:45 +07:00
|
|
|
ssize_t tpm_transmit(struct tpm_chip *chip, struct tpm_space *space,
|
|
|
|
u8 *buf, size_t bufsiz, unsigned int flags);
|
|
|
|
ssize_t tpm_transmit_cmd(struct tpm_chip *chip, struct tpm_space *space,
|
2018-03-05 19:48:26 +07:00
|
|
|
void *buf, size_t bufsiz,
|
2017-01-06 19:03:45 +07:00
|
|
|
size_t min_rsp_body_length, unsigned int flags,
|
2017-01-19 19:19:12 +07:00
|
|
|
const char *desc);
|
2017-06-21 14:31:34 +07:00
|
|
|
int tpm_startup(struct tpm_chip *chip);
|
2016-09-20 03:22:09 +07:00
|
|
|
ssize_t tpm_getcap(struct tpm_chip *chip, u32 subcap_id, cap_t *cap,
|
2017-01-19 19:19:12 +07:00
|
|
|
const char *desc, size_t min_cap_length);
|
2016-06-26 03:33:09 +07:00
|
|
|
int tpm_get_timeouts(struct tpm_chip *);
|
2016-07-13 00:41:49 +07:00
|
|
|
int tpm1_auto_startup(struct tpm_chip *chip);
|
2016-06-26 03:33:09 +07:00
|
|
|
int tpm_do_selftest(struct tpm_chip *chip);
|
2018-10-20 01:22:50 +07:00
|
|
|
unsigned long tpm1_calc_ordinal_duration(struct tpm_chip *chip, u32 ordinal);
|
2016-06-26 03:33:09 +07:00
|
|
|
int tpm_pm_suspend(struct device *dev);
|
|
|
|
int tpm_pm_resume(struct device *dev);
|
2012-08-22 17:47:22 +07:00
|
|
|
|
2017-08-15 01:09:16 +07:00
|
|
|
static inline void tpm_msleep(unsigned int delay_msec)
|
|
|
|
{
|
2017-10-18 03:32:32 +07:00
|
|
|
usleep_range((delay_msec * 1000) - TPM_TIMEOUT_RANGE_US,
|
|
|
|
delay_msec * 1000);
|
2017-08-15 01:09:16 +07:00
|
|
|
};
|
|
|
|
|
2018-06-26 18:06:15 +07:00
|
|
|
struct tpm_chip *tpm_find_get_ops(struct tpm_chip *chip);
|
2016-02-13 10:29:53 +07:00
|
|
|
__must_check int tpm_try_get_ops(struct tpm_chip *chip);
|
|
|
|
void tpm_put_ops(struct tpm_chip *chip);
|
|
|
|
|
2016-06-26 03:33:09 +07:00
|
|
|
struct tpm_chip *tpm_chip_alloc(struct device *dev,
|
|
|
|
const struct tpm_class_ops *ops);
|
|
|
|
struct tpm_chip *tpmm_chip_alloc(struct device *pdev,
|
|
|
|
const struct tpm_class_ops *ops);
|
|
|
|
int tpm_chip_register(struct tpm_chip *chip);
|
|
|
|
void tpm_chip_unregister(struct tpm_chip *chip);
|
2014-12-13 02:46:34 +07:00
|
|
|
|
2016-04-19 00:26:13 +07:00
|
|
|
void tpm_sysfs_add_device(struct tpm_chip *chip);
|
2013-11-27 03:30:40 +07:00
|
|
|
|
2013-11-27 03:30:41 +07:00
|
|
|
int tpm_pcr_read_dev(struct tpm_chip *chip, int pcr_idx, u8 *res_buf);
|
|
|
|
|
2012-08-22 17:47:22 +07:00
|
|
|
#ifdef CONFIG_ACPI
|
2015-04-14 21:56:48 +07:00
|
|
|
extern void tpm_add_ppi(struct tpm_chip *chip);
|
2012-08-22 17:47:22 +07:00
|
|
|
#else
|
2015-04-14 21:56:48 +07:00
|
|
|
static inline void tpm_add_ppi(struct tpm_chip *chip)
|
2012-10-09 16:35:22 +07:00
|
|
|
{
|
|
|
|
}
|
2012-08-22 17:47:22 +07:00
|
|
|
#endif
|
2014-12-13 02:46:38 +07:00
|
|
|
|
2017-09-26 20:58:57 +07:00
|
|
|
static inline u32 tpm2_rc_value(u32 rc)
|
2017-01-26 04:00:22 +07:00
|
|
|
{
|
|
|
|
return (rc & BIT(7)) ? rc & 0xff : rc;
|
|
|
|
}
|
|
|
|
|
2014-12-13 02:46:38 +07:00
|
|
|
int tpm2_pcr_read(struct tpm_chip *chip, int pcr_idx, u8 *res_buf);
|
2017-01-30 16:59:41 +07:00
|
|
|
int tpm2_pcr_extend(struct tpm_chip *chip, int pcr_idx, u32 count,
|
|
|
|
struct tpm2_digest *digests);
|
2018-03-26 19:14:06 +07:00
|
|
|
int tpm2_get_random(struct tpm_chip *chip, u8 *dest, size_t max);
|
2016-11-26 18:39:35 +07:00
|
|
|
void tpm2_flush_context_cmd(struct tpm_chip *chip, u32 handle,
|
|
|
|
unsigned int flags);
|
2015-05-30 12:09:04 +07:00
|
|
|
int tpm2_seal_trusted(struct tpm_chip *chip,
|
|
|
|
struct trusted_key_payload *payload,
|
|
|
|
struct trusted_key_options *options);
|
|
|
|
int tpm2_unseal_trusted(struct tpm_chip *chip,
|
|
|
|
struct trusted_key_payload *payload,
|
|
|
|
struct trusted_key_options *options);
|
2014-12-13 02:46:38 +07:00
|
|
|
ssize_t tpm2_get_tpm_pt(struct tpm_chip *chip, u32 property_id,
|
|
|
|
u32 *value, const char *desc);
|
|
|
|
|
2016-07-13 00:41:49 +07:00
|
|
|
int tpm2_auto_startup(struct tpm_chip *chip);
|
2016-06-26 03:33:09 +07:00
|
|
|
void tpm2_shutdown(struct tpm_chip *chip, u16 shutdown_type);
|
|
|
|
unsigned long tpm2_calc_ordinal_duration(struct tpm_chip *chip, u32 ordinal);
|
|
|
|
int tpm2_probe(struct tpm_chip *chip);
|
2016-11-11 11:42:07 +07:00
|
|
|
int tpm2_find_cc(struct tpm_chip *chip, u32 cc);
|
2017-01-06 19:03:45 +07:00
|
|
|
int tpm2_init_space(struct tpm_space *space);
|
2017-02-01 06:47:31 +07:00
|
|
|
void tpm2_del_space(struct tpm_chip *chip, struct tpm_space *space);
|
2017-01-06 19:03:45 +07:00
|
|
|
int tpm2_prepare_space(struct tpm_chip *chip, struct tpm_space *space, u32 cc,
|
|
|
|
u8 *cmd);
|
|
|
|
int tpm2_commit_space(struct tpm_chip *chip, struct tpm_space *space,
|
|
|
|
u32 cc, u8 *buf, size_t *bufsiz);
|
2017-09-20 15:13:36 +07:00
|
|
|
|
|
|
|
int tpm_bios_log_setup(struct tpm_chip *chip);
|
|
|
|
void tpm_bios_log_teardown(struct tpm_chip *chip);
|
2018-09-11 00:18:33 +07:00
|
|
|
int tpm_dev_common_init(void);
|
|
|
|
void tpm_dev_common_exit(void);
|
2016-05-05 02:06:22 +07:00
|
|
|
#endif
|