提交 0dc55365 编写于 作者: J Jarkko Sakkinen 提交者: Peter Huewe

tpm: fix raciness of PPI interface lookup

Traversal of the ACPI device tree was not done right. PPI interface
should be looked up only from the ACPI device that is the platform
device for the TPM. This could cause problems with systems with
two TPM chips such as 4th gen Intel systems.

In addition, added the missing license and copyright platter to
the tpm_ppi.c.
Signed-off-by: NJarkko Sakkinen <jarkko.sakkinen@linux.intel.com>
Reviewed-by: NJasob Gunthorpe <jason.gunthorpe@obsidianresearch.com>
Reviewed-by: NStefan Berger <stefanb@linux.vnet.ibm.com>
Tested-by: NScot Doyle <lkml14@scotdoyle.com>
Signed-off-by: NPeter Huewe <peterhuewe@gmx.de>
上级 afb5abc2
...@@ -147,7 +147,7 @@ int tpm_chip_register(struct tpm_chip *chip) ...@@ -147,7 +147,7 @@ int tpm_chip_register(struct tpm_chip *chip)
if (rc) if (rc)
goto del_misc; goto del_misc;
rc = tpm_add_ppi(&chip->dev->kobj); rc = tpm_add_ppi(chip);
if (rc) if (rc)
goto del_sysfs; goto del_sysfs;
...@@ -191,7 +191,7 @@ void tpm_chip_unregister(struct tpm_chip *chip) ...@@ -191,7 +191,7 @@ void tpm_chip_unregister(struct tpm_chip *chip)
if (chip->bios_dir) if (chip->bios_dir)
tpm_bios_log_teardown(chip->bios_dir); tpm_bios_log_teardown(chip->bios_dir);
tpm_remove_ppi(&chip->dev->kobj); tpm_remove_ppi(chip);
tpm_sysfs_del_device(chip); tpm_sysfs_del_device(chip);
tpm_dev_del_device(chip); tpm_dev_del_device(chip);
......
...@@ -27,6 +27,7 @@ ...@@ -27,6 +27,7 @@
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include <linux/io.h> #include <linux/io.h>
#include <linux/tpm.h> #include <linux/tpm.h>
#include <linux/acpi.h>
enum tpm_const { enum tpm_const {
TPM_MINOR = 224, /* officially assigned */ TPM_MINOR = 224, /* officially assigned */
...@@ -94,8 +95,11 @@ struct tpm_vendor_specific { ...@@ -94,8 +95,11 @@ struct tpm_vendor_specific {
#define TPM_VID_WINBOND 0x1050 #define TPM_VID_WINBOND 0x1050
#define TPM_VID_STM 0x104A #define TPM_VID_STM 0x104A
#define TPM_PPI_VERSION_LEN 3
enum tpm_chip_flags { enum tpm_chip_flags {
TPM_CHIP_FLAG_REGISTERED = BIT(0), TPM_CHIP_FLAG_REGISTERED = BIT(0),
TPM_CHIP_FLAG_PPI = BIT(1),
}; };
struct tpm_chip { struct tpm_chip {
...@@ -114,6 +118,11 @@ struct tpm_chip { ...@@ -114,6 +118,11 @@ struct tpm_chip {
struct dentry **bios_dir; struct dentry **bios_dir;
#ifdef CONFIG_ACPI
acpi_handle acpi_dev_handle;
char ppi_version[TPM_PPI_VERSION_LEN + 1];
#endif /* CONFIG_ACPI */
struct list_head list; struct list_head list;
}; };
...@@ -345,15 +354,15 @@ void tpm_sysfs_del_device(struct tpm_chip *chip); ...@@ -345,15 +354,15 @@ void tpm_sysfs_del_device(struct tpm_chip *chip);
int tpm_pcr_read_dev(struct tpm_chip *chip, int pcr_idx, u8 *res_buf); int tpm_pcr_read_dev(struct tpm_chip *chip, int pcr_idx, u8 *res_buf);
#ifdef CONFIG_ACPI #ifdef CONFIG_ACPI
extern int tpm_add_ppi(struct kobject *); extern int tpm_add_ppi(struct tpm_chip *chip);
extern void tpm_remove_ppi(struct kobject *); extern void tpm_remove_ppi(struct tpm_chip *chip);
#else #else
static inline int tpm_add_ppi(struct kobject *parent) static inline int tpm_add_ppi(struct tpm_chip *chip)
{ {
return 0; return 0;
} }
static inline void tpm_remove_ppi(struct kobject *parent) static inline void tpm_remove_ppi(struct tpm_chip *chip)
{ {
} }
#endif #endif
/*
* Copyright (C) 2012-2014 Intel Corporation
*
* Authors:
* Xiaoyan Zhang <xiaoyan.zhang@intel.com>
* Jiang Liu <jiang.liu@linux.intel.com>
* Jarkko Sakkinen <jarkko.sakkinen@linux.intel.com>
*
* Maintained by: <tpmdd-devel@lists.sourceforge.net>
*
* This file contains implementation of the sysfs interface for PPI.
*
* 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/acpi.h> #include <linux/acpi.h>
#include "tpm.h" #include "tpm.h"
...@@ -12,7 +31,6 @@ ...@@ -12,7 +31,6 @@
#define PPI_TPM_REQ_MAX 22 #define PPI_TPM_REQ_MAX 22
#define PPI_VS_REQ_START 128 #define PPI_VS_REQ_START 128
#define PPI_VS_REQ_END 255 #define PPI_VS_REQ_END 255
#define PPI_VERSION_LEN 3
static const u8 tpm_ppi_uuid[] = { static const u8 tpm_ppi_uuid[] = {
0xA6, 0xFA, 0xDD, 0x3D, 0xA6, 0xFA, 0xDD, 0x3D,
...@@ -22,45 +40,22 @@ static const u8 tpm_ppi_uuid[] = { ...@@ -22,45 +40,22 @@ static const u8 tpm_ppi_uuid[] = {
0x8D, 0x10, 0x08, 0x9D, 0x16, 0x53 0x8D, 0x10, 0x08, 0x9D, 0x16, 0x53
}; };
static char tpm_ppi_version[PPI_VERSION_LEN + 1];
static acpi_handle tpm_ppi_handle;
static acpi_status ppi_callback(acpi_handle handle, u32 level, void *context,
void **return_value)
{
union acpi_object *obj;
if (!acpi_check_dsm(handle, tpm_ppi_uuid, TPM_PPI_REVISION_ID,
1 << TPM_PPI_FN_VERSION))
return AE_OK;
/* 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);
}
*return_value = handle;
return AE_CTRL_TERMINATE;
}
static inline union acpi_object * static inline union acpi_object *
tpm_eval_dsm(int func, acpi_object_type type, union acpi_object *argv4) tpm_eval_dsm(acpi_handle ppi_handle, int func, acpi_object_type type,
union acpi_object *argv4)
{ {
BUG_ON(!tpm_ppi_handle); BUG_ON(!ppi_handle);
return acpi_evaluate_dsm_typed(tpm_ppi_handle, tpm_ppi_uuid, return acpi_evaluate_dsm_typed(ppi_handle, tpm_ppi_uuid,
TPM_PPI_REVISION_ID, func, argv4, type); TPM_PPI_REVISION_ID,
func, argv4, type);
} }
static ssize_t tpm_show_ppi_version(struct device *dev, static ssize_t tpm_show_ppi_version(struct device *dev,
struct device_attribute *attr, char *buf) struct device_attribute *attr, char *buf)
{ {
return scnprintf(buf, PAGE_SIZE, "%s\n", tpm_ppi_version); struct tpm_chip *chip = dev_get_drvdata(dev);
return scnprintf(buf, PAGE_SIZE, "%s\n", chip->ppi_version);
} }
static ssize_t tpm_show_ppi_request(struct device *dev, static ssize_t tpm_show_ppi_request(struct device *dev,
...@@ -68,8 +63,10 @@ static ssize_t tpm_show_ppi_request(struct device *dev, ...@@ -68,8 +63,10 @@ static ssize_t tpm_show_ppi_request(struct device *dev,
{ {
ssize_t size = -EINVAL; ssize_t size = -EINVAL;
union acpi_object *obj; union acpi_object *obj;
struct tpm_chip *chip = dev_get_drvdata(dev);
obj = tpm_eval_dsm(TPM_PPI_FN_GETREQ, ACPI_TYPE_PACKAGE, NULL); obj = tpm_eval_dsm(chip->acpi_dev_handle, TPM_PPI_FN_GETREQ,
ACPI_TYPE_PACKAGE, NULL);
if (!obj) if (!obj)
return -ENXIO; return -ENXIO;
...@@ -103,14 +100,15 @@ static ssize_t tpm_store_ppi_request(struct device *dev, ...@@ -103,14 +100,15 @@ static ssize_t tpm_store_ppi_request(struct device *dev,
int func = TPM_PPI_FN_SUBREQ; int func = TPM_PPI_FN_SUBREQ;
union acpi_object *obj, tmp; union acpi_object *obj, tmp;
union acpi_object argv4 = ACPI_INIT_DSM_ARGV4(1, &tmp); union acpi_object argv4 = ACPI_INIT_DSM_ARGV4(1, &tmp);
struct tpm_chip *chip = dev_get_drvdata(dev);
/* /*
* the function to submit TPM operation request to pre-os environment * the function to submit TPM operation request to pre-os environment
* is updated with function index from SUBREQ to SUBREQ2 since PPI * is updated with function index from SUBREQ to SUBREQ2 since PPI
* version 1.1 * version 1.1
*/ */
if (acpi_check_dsm(tpm_ppi_handle, tpm_ppi_uuid, TPM_PPI_REVISION_ID, if (acpi_check_dsm(chip->acpi_dev_handle, tpm_ppi_uuid,
1 << TPM_PPI_FN_SUBREQ2)) TPM_PPI_REVISION_ID, 1 << TPM_PPI_FN_SUBREQ2))
func = TPM_PPI_FN_SUBREQ2; func = TPM_PPI_FN_SUBREQ2;
/* /*
...@@ -119,7 +117,7 @@ static ssize_t tpm_store_ppi_request(struct device *dev, ...@@ -119,7 +117,7 @@ static ssize_t tpm_store_ppi_request(struct device *dev,
* string/package type. For PPI version 1.0 and 1.1, use buffer type * 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. * for compatibility, and use package type since 1.2 according to spec.
*/ */
if (strcmp(tpm_ppi_version, "1.2") < 0) { if (strcmp(chip->ppi_version, "1.2") < 0) {
if (sscanf(buf, "%d", &req) != 1) if (sscanf(buf, "%d", &req) != 1)
return -EINVAL; return -EINVAL;
argv4.type = ACPI_TYPE_BUFFER; argv4.type = ACPI_TYPE_BUFFER;
...@@ -131,7 +129,8 @@ static ssize_t tpm_store_ppi_request(struct device *dev, ...@@ -131,7 +129,8 @@ static ssize_t tpm_store_ppi_request(struct device *dev,
return -EINVAL; return -EINVAL;
} }
obj = tpm_eval_dsm(func, ACPI_TYPE_INTEGER, &argv4); obj = tpm_eval_dsm(chip->acpi_dev_handle, func, ACPI_TYPE_INTEGER,
&argv4);
if (!obj) { if (!obj) {
return -ENXIO; return -ENXIO;
} else { } else {
...@@ -157,6 +156,7 @@ static ssize_t tpm_show_ppi_transition_action(struct device *dev, ...@@ -157,6 +156,7 @@ static ssize_t tpm_show_ppi_transition_action(struct device *dev,
.buffer.length = 0, .buffer.length = 0,
.buffer.pointer = NULL .buffer.pointer = NULL
}; };
struct tpm_chip *chip = dev_get_drvdata(dev);
static char *info[] = { static char *info[] = {
"None", "None",
...@@ -171,9 +171,10 @@ static ssize_t tpm_show_ppi_transition_action(struct device *dev, ...@@ -171,9 +171,10 @@ static ssize_t tpm_show_ppi_transition_action(struct device *dev,
* (e.g. Capella with PPI 1.0) need integer/string/buffer type, so for * (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 * compatibility, define params[3].type as buffer, if PPI version < 1.2
*/ */
if (strcmp(tpm_ppi_version, "1.2") < 0) if (strcmp(chip->ppi_version, "1.2") < 0)
obj = &tmp; obj = &tmp;
obj = tpm_eval_dsm(TPM_PPI_FN_GETACT, ACPI_TYPE_INTEGER, obj); obj = tpm_eval_dsm(chip->acpi_dev_handle, TPM_PPI_FN_GETACT,
ACPI_TYPE_INTEGER, obj);
if (!obj) { if (!obj) {
return -ENXIO; return -ENXIO;
} else { } else {
...@@ -196,8 +197,10 @@ static ssize_t tpm_show_ppi_response(struct device *dev, ...@@ -196,8 +197,10 @@ static ssize_t tpm_show_ppi_response(struct device *dev,
acpi_status status = -EINVAL; acpi_status status = -EINVAL;
union acpi_object *obj, *ret_obj; union acpi_object *obj, *ret_obj;
u64 req, res; u64 req, res;
struct tpm_chip *chip = dev_get_drvdata(dev);
obj = tpm_eval_dsm(TPM_PPI_FN_GETRSP, ACPI_TYPE_PACKAGE, NULL); obj = tpm_eval_dsm(chip->acpi_dev_handle, TPM_PPI_FN_GETRSP,
ACPI_TYPE_PACKAGE, NULL);
if (!obj) if (!obj)
return -ENXIO; return -ENXIO;
...@@ -248,7 +251,8 @@ static ssize_t tpm_show_ppi_response(struct device *dev, ...@@ -248,7 +251,8 @@ static ssize_t tpm_show_ppi_response(struct device *dev,
return status; return status;
} }
static ssize_t show_ppi_operations(char *buf, u32 start, u32 end) static ssize_t show_ppi_operations(acpi_handle dev_handle, char *buf, u32 start,
u32 end)
{ {
int i; int i;
u32 ret; u32 ret;
...@@ -264,14 +268,15 @@ static ssize_t show_ppi_operations(char *buf, u32 start, u32 end) ...@@ -264,14 +268,15 @@ static ssize_t show_ppi_operations(char *buf, u32 start, u32 end)
"User not required", "User not required",
}; };
if (!acpi_check_dsm(tpm_ppi_handle, tpm_ppi_uuid, TPM_PPI_REVISION_ID, if (!acpi_check_dsm(dev_handle, tpm_ppi_uuid, TPM_PPI_REVISION_ID,
1 << TPM_PPI_FN_GETOPR)) 1 << TPM_PPI_FN_GETOPR))
return -EPERM; return -EPERM;
tmp.integer.type = ACPI_TYPE_INTEGER; tmp.integer.type = ACPI_TYPE_INTEGER;
for (i = start; i <= end; i++) { for (i = start; i <= end; i++) {
tmp.integer.value = i; tmp.integer.value = i;
obj = tpm_eval_dsm(TPM_PPI_FN_GETOPR, ACPI_TYPE_INTEGER, &argv); obj = tpm_eval_dsm(dev_handle, TPM_PPI_FN_GETOPR,
ACPI_TYPE_INTEGER, &argv);
if (!obj) { if (!obj) {
return -ENOMEM; return -ENOMEM;
} else { } else {
...@@ -291,14 +296,20 @@ static ssize_t tpm_show_ppi_tcg_operations(struct device *dev, ...@@ -291,14 +296,20 @@ static ssize_t tpm_show_ppi_tcg_operations(struct device *dev,
struct device_attribute *attr, struct device_attribute *attr,
char *buf) char *buf)
{ {
return show_ppi_operations(buf, 0, PPI_TPM_REQ_MAX); struct tpm_chip *chip = dev_get_drvdata(dev);
return show_ppi_operations(chip->acpi_dev_handle, buf, 0,
PPI_TPM_REQ_MAX);
} }
static ssize_t tpm_show_ppi_vs_operations(struct device *dev, static ssize_t tpm_show_ppi_vs_operations(struct device *dev,
struct device_attribute *attr, struct device_attribute *attr,
char *buf) char *buf)
{ {
return show_ppi_operations(buf, PPI_VS_REQ_START, PPI_VS_REQ_END); struct tpm_chip *chip = dev_get_drvdata(dev);
return show_ppi_operations(chip->acpi_dev_handle, buf, PPI_VS_REQ_START,
PPI_VS_REQ_END);
} }
static DEVICE_ATTR(version, S_IRUGO, tpm_show_ppi_version, NULL); static DEVICE_ATTR(version, S_IRUGO, tpm_show_ppi_version, NULL);
...@@ -323,16 +334,38 @@ static struct attribute_group ppi_attr_grp = { ...@@ -323,16 +334,38 @@ static struct attribute_group ppi_attr_grp = {
.attrs = ppi_attrs .attrs = ppi_attrs
}; };
int tpm_add_ppi(struct kobject *parent) int tpm_add_ppi(struct tpm_chip *chip)
{ {
/* Cache TPM ACPI handle and version string */ union acpi_object *obj;
acpi_walk_namespace(ACPI_TYPE_DEVICE, ACPI_ROOT_OBJECT, ACPI_UINT32_MAX, int rc;
ppi_callback, NULL, NULL, &tpm_ppi_handle);
return tpm_ppi_handle ? sysfs_create_group(parent, &ppi_attr_grp) : 0; if (!chip->acpi_dev_handle)
return 0;
if (!acpi_check_dsm(chip->acpi_dev_handle, tpm_ppi_uuid,
TPM_PPI_REVISION_ID, 1 << TPM_PPI_FN_VERSION))
return 0;
/* Cache PPI version string. */
obj = acpi_evaluate_dsm_typed(chip->acpi_dev_handle, tpm_ppi_uuid,
TPM_PPI_REVISION_ID, TPM_PPI_FN_VERSION,
NULL, ACPI_TYPE_STRING);
if (obj) {
strlcpy(chip->ppi_version, obj->string.pointer,
sizeof(chip->ppi_version));
ACPI_FREE(obj);
}
rc = sysfs_create_group(&chip->dev->kobj, &ppi_attr_grp);
if (!rc)
chip->flags |= TPM_CHIP_FLAG_PPI;
return rc;
} }
void tpm_remove_ppi(struct kobject *parent) void tpm_remove_ppi(struct tpm_chip *chip)
{ {
if (tpm_ppi_handle) if (chip->flags & TPM_CHIP_FLAG_PPI)
sysfs_remove_group(parent, &ppi_attr_grp); sysfs_remove_group(&chip->dev->kobj, &ppi_attr_grp);
} }
...@@ -580,8 +580,9 @@ static void tpm_tis_remove(struct tpm_chip *chip) ...@@ -580,8 +580,9 @@ static void tpm_tis_remove(struct tpm_chip *chip)
release_locality(chip, chip->vendor.locality, 1); release_locality(chip, chip->vendor.locality, 1);
} }
static int tpm_tis_init(struct device *dev, resource_size_t start, static int tpm_tis_init(struct device *dev, acpi_handle acpi_dev_handle,
resource_size_t len, unsigned int irq) resource_size_t start, resource_size_t len,
unsigned int irq)
{ {
u32 vendor, intfcaps, intmask; u32 vendor, intfcaps, intmask;
int rc, i, irq_s, irq_e, probe; int rc, i, irq_s, irq_e, probe;
...@@ -597,6 +598,7 @@ static int tpm_tis_init(struct device *dev, resource_size_t start, ...@@ -597,6 +598,7 @@ static int tpm_tis_init(struct device *dev, resource_size_t start,
return PTR_ERR(chip); return PTR_ERR(chip);
chip->vendor.priv = priv; chip->vendor.priv = priv;
chip->acpi_dev_handle = acpi_dev_handle;
chip->vendor.iobase = devm_ioremap(dev, start, len); chip->vendor.iobase = devm_ioremap(dev, start, len);
if (!chip->vendor.iobase) if (!chip->vendor.iobase)
...@@ -827,6 +829,7 @@ static int tpm_tis_pnp_init(struct pnp_dev *pnp_dev, ...@@ -827,6 +829,7 @@ static int tpm_tis_pnp_init(struct pnp_dev *pnp_dev,
{ {
resource_size_t start, len; resource_size_t start, len;
unsigned int irq = 0; unsigned int irq = 0;
acpi_handle acpi_dev_handle = NULL;
start = pnp_mem_start(pnp_dev, 0); start = pnp_mem_start(pnp_dev, 0);
len = pnp_mem_len(pnp_dev, 0); len = pnp_mem_len(pnp_dev, 0);
...@@ -839,7 +842,10 @@ static int tpm_tis_pnp_init(struct pnp_dev *pnp_dev, ...@@ -839,7 +842,10 @@ static int tpm_tis_pnp_init(struct pnp_dev *pnp_dev,
if (is_itpm(pnp_dev)) if (is_itpm(pnp_dev))
itpm = true; itpm = true;
return tpm_tis_init(&pnp_dev->dev, start, len, irq); if (pnp_acpi_device(pnp_dev))
acpi_dev_handle = pnp_acpi_device(pnp_dev)->handle;
return tpm_tis_init(&pnp_dev->dev, acpi_dev_handle, start, len, irq);
} }
static struct pnp_device_id tpm_pnp_tbl[] = { static struct pnp_device_id tpm_pnp_tbl[] = {
...@@ -907,7 +913,7 @@ static int __init init_tis(void) ...@@ -907,7 +913,7 @@ static int __init init_tis(void)
rc = PTR_ERR(pdev); rc = PTR_ERR(pdev);
goto err_dev; goto err_dev;
} }
rc = tpm_tis_init(&pdev->dev, TIS_MEM_BASE, TIS_MEM_LEN, 0); rc = tpm_tis_init(&pdev->dev, NULL, TIS_MEM_BASE, TIS_MEM_LEN, 0);
if (rc) if (rc)
goto err_init; goto err_init;
return 0; return 0;
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册