diff options
author | Jiang Liu <jiang.liu@linux.intel.com> | 2013-12-19 07:38:18 -0500 |
---|---|---|
committer | Rafael J. Wysocki <rafael.j.wysocki@intel.com> | 2014-01-05 10:07:15 -0500 |
commit | 1569a4c4cebaf358eff12633830baeaf6507fed0 (patch) | |
tree | 7ee80bb3bdd0ae2c276d762f55953b0a579c1712 /drivers/char/tpm | |
parent | 84b1667dea233d2af29b5138bfd2d70417cfa720 (diff) |
ACPI / TPM: detect PPI features by checking availability of _DSM functions
Detecting physical presence interface features by checking availbility
of corresponding ACPI _DSM functions, it should be more accurate than
checking TPM version number.
Signed-off-by: Jiang Liu <jiang.liu@linux.intel.com>
Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
Diffstat (limited to 'drivers/char/tpm')
-rw-r--r-- | drivers/char/tpm/tpm_ppi.c | 45 |
1 files changed, 19 insertions, 26 deletions
diff --git a/drivers/char/tpm/tpm_ppi.c b/drivers/char/tpm/tpm_ppi.c index d542ac6d0ae0..117fab6daf39 100644 --- a/drivers/char/tpm/tpm_ppi.c +++ b/drivers/char/tpm/tpm_ppi.c | |||
@@ -23,39 +23,31 @@ static const u8 tpm_ppi_uuid[] = { | |||
23 | 0x8D, 0x10, 0x08, 0x9D, 0x16, 0x53 | 23 | 0x8D, 0x10, 0x08, 0x9D, 0x16, 0x53 |
24 | }; | 24 | }; |
25 | 25 | ||
26 | static char *tpm_device_name = "TPM"; | ||
27 | static char tpm_ppi_version[PPI_VERSION_LEN + 1]; | 26 | static char tpm_ppi_version[PPI_VERSION_LEN + 1]; |
28 | static acpi_handle tpm_ppi_handle; | 27 | static acpi_handle tpm_ppi_handle; |
29 | 28 | ||
30 | static acpi_status ppi_callback(acpi_handle handle, u32 level, void *context, | 29 | static acpi_status ppi_callback(acpi_handle handle, u32 level, void *context, |
31 | void **return_value) | 30 | void **return_value) |
32 | { | 31 | { |
33 | acpi_status status = AE_OK; | 32 | union acpi_object *obj; |
34 | struct acpi_buffer buffer = { ACPI_ALLOCATE_BUFFER, NULL }; | ||
35 | 33 | ||
36 | status = acpi_get_name(handle, ACPI_SINGLE_NAME, &buffer); | 34 | if (!acpi_check_dsm(handle, tpm_ppi_uuid, TPM_PPI_REVISION_ID, |
37 | if (ACPI_FAILURE(status)) | 35 | 1 << TPM_PPI_FN_VERSION)) |
38 | return AE_OK; | 36 | return AE_OK; |
39 | 37 | ||
40 | if (strstr(buffer.pointer, context) != NULL) { | 38 | /* Cache version string */ |
41 | union acpi_object *obj; | 39 | obj = acpi_evaluate_dsm_typed(handle, tpm_ppi_uuid, |
42 | 40 | TPM_PPI_REVISION_ID, TPM_PPI_FN_VERSION, | |
43 | /* Cache version string */ | 41 | NULL, ACPI_TYPE_STRING); |
44 | obj = acpi_evaluate_dsm_typed(handle, tpm_ppi_uuid, | 42 | if (obj) { |
45 | TPM_PPI_REVISION_ID, TPM_PPI_FN_VERSION, | 43 | strlcpy(tpm_ppi_version, obj->string.pointer, |
46 | NULL, ACPI_TYPE_STRING); | 44 | PPI_VERSION_LEN + 1); |
47 | if (obj) { | 45 | ACPI_FREE(obj); |
48 | strlcpy(tpm_ppi_version, obj->string.pointer, | ||
49 | PPI_VERSION_LEN + 1); | ||
50 | ACPI_FREE(obj); | ||
51 | } | ||
52 | |||
53 | *return_value = handle; | ||
54 | status = AE_CTRL_TERMINATE; | ||
55 | } | 46 | } |
56 | kfree(buffer.pointer); | ||
57 | 47 | ||
58 | return status; | 48 | *return_value = handle; |
49 | |||
50 | return AE_CTRL_TERMINATE; | ||
59 | } | 51 | } |
60 | 52 | ||
61 | static inline union acpi_object * | 53 | static inline union acpi_object * |
@@ -118,7 +110,8 @@ static ssize_t tpm_store_ppi_request(struct device *dev, | |||
118 | * is updated with function index from SUBREQ to SUBREQ2 since PPI | 110 | * is updated with function index from SUBREQ to SUBREQ2 since PPI |
119 | * version 1.1 | 111 | * version 1.1 |
120 | */ | 112 | */ |
121 | if (strcmp(tpm_ppi_version, "1.1") >= 0) | 113 | if (acpi_check_dsm(tpm_ppi_handle, tpm_ppi_uuid, TPM_PPI_REVISION_ID, |
114 | 1 << TPM_PPI_FN_SUBREQ2)) | ||
122 | func = TPM_PPI_FN_SUBREQ2; | 115 | func = TPM_PPI_FN_SUBREQ2; |
123 | 116 | ||
124 | /* | 117 | /* |
@@ -272,7 +265,8 @@ static ssize_t show_ppi_operations(char *buf, u32 start, u32 end) | |||
272 | "User not required", | 265 | "User not required", |
273 | }; | 266 | }; |
274 | 267 | ||
275 | if (strcmp(tpm_ppi_version, "1.2") < 0) | 268 | if (!acpi_check_dsm(tpm_ppi_handle, tpm_ppi_uuid, TPM_PPI_REVISION_ID, |
269 | 1 << TPM_PPI_FN_GETOPR)) | ||
276 | return -EPERM; | 270 | return -EPERM; |
277 | 271 | ||
278 | tmp.integer.type = ACPI_TYPE_INTEGER; | 272 | tmp.integer.type = ACPI_TYPE_INTEGER; |
@@ -334,8 +328,7 @@ int tpm_add_ppi(struct kobject *parent) | |||
334 | { | 328 | { |
335 | /* Cache TPM ACPI handle and version string */ | 329 | /* Cache TPM ACPI handle and version string */ |
336 | acpi_walk_namespace(ACPI_TYPE_DEVICE, ACPI_ROOT_OBJECT, ACPI_UINT32_MAX, | 330 | acpi_walk_namespace(ACPI_TYPE_DEVICE, ACPI_ROOT_OBJECT, ACPI_UINT32_MAX, |
337 | ppi_callback, NULL, | 331 | ppi_callback, NULL, NULL, &tpm_ppi_handle); |
338 | tpm_device_name, &tpm_ppi_handle); | ||
339 | if (tpm_ppi_handle == NULL) | 332 | if (tpm_ppi_handle == NULL) |
340 | return -ENODEV; | 333 | return -ENODEV; |
341 | 334 | ||