blob: 4b2ef94aa7d2780f459f6fca19c4b8d548b202d9 [file] [log] [blame]
Rong Changf6267992013-04-12 10:44:57 +00001/*
2 * Copyright (C) 2011 Infineon Technologies
3 *
4 * Authors:
5 * Peter Huewe <huewe.external@infineon.com>
6 *
7 * Description:
8 * Device driver for TCG/TCPA TPM (trusted platform module).
9 * Specifications at www.trustedcomputinggroup.org
10 *
11 * This device driver implements the TPM interface as defined in
12 * the TCG TPM Interface Spec version 1.2, revision 1.0 and the
13 * Infineon I2C Protocol Stack Specification v0.20.
14 *
15 * It is based on the Linux kernel driver tpm.c from Leendert van
16 * Dorn, Dave Safford, Reiner Sailer, and Kyleen Hall.
17 *
18 * Version: 2.1.1
19 *
Simon Glass07470d62015-08-22 18:31:21 -060020 * SPDX-License-Identifier: GPL-2.0
Rong Changf6267992013-04-12 10:44:57 +000021 */
22
23#include <common.h>
Simon Glassf90acf12015-05-04 11:30:59 -060024#include <dm.h>
Vincent Palatinec34fa52013-04-12 11:04:36 +000025#include <fdtdec.h>
Masahiro Yamadaafc366f2014-11-26 16:00:58 +090026#include <linux/compiler.h>
Rong Changf6267992013-04-12 10:44:57 +000027#include <i2c.h>
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +000028#include <tpm.h>
29#include <asm-generic/errno.h>
Rong Changf6267992013-04-12 10:44:57 +000030#include <linux/types.h>
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +000031#include <linux/unaligned/be_byteshift.h>
Rong Changf6267992013-04-12 10:44:57 +000032
Simon Glass4cd7b782015-08-22 18:31:22 -060033#include "tpm_tis_i2c.h"
Rong Changf6267992013-04-12 10:44:57 +000034
Vincent Palatinec34fa52013-04-12 11:04:36 +000035DECLARE_GLOBAL_DATA_PTR;
36
Vincent Palatinec34fa52013-04-12 11:04:36 +000037static const char * const chip_name[] = {
38 [SLB9635] = "slb9635tt",
39 [SLB9645] = "slb9645tt",
40 [UNKNOWN] = "unknown/fallback to slb9635",
41};
Rong Changf6267992013-04-12 10:44:57 +000042
Simon Glass4cd7b782015-08-22 18:31:22 -060043static struct tpm_chip g_chip;
44
Rong Changf6267992013-04-12 10:44:57 +000045/*
Simon Glasse56e20e2015-08-22 18:31:29 -060046 * tpm_tis_i2c_read() - read from TPM register
Rong Changf6267992013-04-12 10:44:57 +000047 * @addr: register address to read from
48 * @buffer: provided by caller
49 * @len: number of bytes to read
50 *
51 * Read len bytes from TPM register and put them into
52 * buffer (little-endian format, i.e. first byte is put into buffer[0]).
53 *
54 * NOTE: TPM is big-endian for multi-byte values. Multi-byte
55 * values have to be swapped.
56 *
57 * Return -EIO on error, 0 on success.
58 */
Simon Glasse56e20e2015-08-22 18:31:29 -060059static int tpm_tis_i2c_read(u8 addr, u8 *buffer, size_t len)
Rong Changf6267992013-04-12 10:44:57 +000060{
61 int rc;
62 int count;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +000063 uint32_t addrbuf = addr;
Rong Changf6267992013-04-12 10:44:57 +000064
Simon Glass13932b02015-08-22 18:31:25 -060065 if ((g_chip.chip_type == SLB9635) || (g_chip.chip_type == UNKNOWN)) {
Vincent Palatinec34fa52013-04-12 11:04:36 +000066 /* slb9635 protocol should work in both cases */
67 for (count = 0; count < MAX_COUNT; count++) {
Simon Glass13932b02015-08-22 18:31:25 -060068 rc = dm_i2c_write(g_chip.dev, 0, (uchar *)&addrbuf, 1);
Vincent Palatinec34fa52013-04-12 11:04:36 +000069 if (rc == 0)
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +000070 break; /* Success, break to skip sleep */
Simon Glass42c8ec52015-08-22 18:31:30 -060071 udelay(SLEEP_DURATION_US);
Vincent Palatinec34fa52013-04-12 11:04:36 +000072 }
Vincent Palatinec34fa52013-04-12 11:04:36 +000073 if (rc)
74 return -rc;
75
76 /* After the TPM has successfully received the register address
77 * it needs some time, thus we're sleeping here again, before
78 * retrieving the data
79 */
80 for (count = 0; count < MAX_COUNT; count++) {
Simon Glass42c8ec52015-08-22 18:31:30 -060081 udelay(SLEEP_DURATION_US);
Simon Glass13932b02015-08-22 18:31:25 -060082 rc = dm_i2c_read(g_chip.dev, 0, buffer, len);
Vincent Palatinec34fa52013-04-12 11:04:36 +000083 if (rc == 0)
84 break; /* success, break to skip sleep */
85 }
86 } else {
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +000087 /*
88 * Use a combined read for newer chips.
89 * Unfortunately the smbus functions are not suitable due to
Vincent Palatinec34fa52013-04-12 11:04:36 +000090 * the 32 byte limit of the smbus.
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +000091 * Retries should usually not be needed, but are kept just to
Vincent Palatinec34fa52013-04-12 11:04:36 +000092 * be safe on the safe side.
93 */
94 for (count = 0; count < MAX_COUNT; count++) {
Simon Glass13932b02015-08-22 18:31:25 -060095 rc = dm_i2c_read(g_chip.dev, addr, buffer, len);
Vincent Palatinec34fa52013-04-12 11:04:36 +000096 if (rc == 0)
97 break; /* break here to skip sleep */
Simon Glass42c8ec52015-08-22 18:31:30 -060098 udelay(SLEEP_DURATION_US);
Vincent Palatinec34fa52013-04-12 11:04:36 +000099 }
Rong Changf6267992013-04-12 10:44:57 +0000100 }
101
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000102 /* Take care of 'guard time' */
Simon Glass42c8ec52015-08-22 18:31:30 -0600103 udelay(SLEEP_DURATION_US);
Rong Changf6267992013-04-12 10:44:57 +0000104 if (rc)
105 return -rc;
106
107 return 0;
108}
109
Simon Glasse56e20e2015-08-22 18:31:29 -0600110static int tpm_tis_i2c_write_generic(u8 addr, u8 *buffer, size_t len,
Simon Glass42c8ec52015-08-22 18:31:30 -0600111 unsigned int sleep_time_us, u8 max_count)
Rong Changf6267992013-04-12 10:44:57 +0000112{
113 int rc = 0;
114 int count;
115
Rong Changf6267992013-04-12 10:44:57 +0000116 for (count = 0; count < max_count; count++) {
Simon Glass13932b02015-08-22 18:31:25 -0600117 rc = dm_i2c_write(g_chip.dev, addr, buffer, len);
Rong Changf6267992013-04-12 10:44:57 +0000118 if (rc == 0)
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000119 break; /* Success, break to skip sleep */
Simon Glass42c8ec52015-08-22 18:31:30 -0600120 udelay(sleep_time_us);
Rong Changf6267992013-04-12 10:44:57 +0000121 }
122
Vincent Palatinec34fa52013-04-12 11:04:36 +0000123 /* take care of 'guard time' */
Simon Glass42c8ec52015-08-22 18:31:30 -0600124 udelay(sleep_time_us);
Rong Changf6267992013-04-12 10:44:57 +0000125 if (rc)
126 return -rc;
127
128 return 0;
129}
130
131/*
Simon Glasse56e20e2015-08-22 18:31:29 -0600132 * tpm_tis_i2c_write() - write to TPM register
Rong Changf6267992013-04-12 10:44:57 +0000133 * @addr: register address to write to
134 * @buffer: containing data to be written
135 * @len: number of bytes to write
136 *
137 * Write len bytes from provided buffer to TPM register (little
138 * endian format, i.e. buffer[0] is written as first byte).
139 *
140 * NOTE: TPM is big-endian for multi-byte values. Multi-byte
141 * values have to be swapped.
142 *
Simon Glasse56e20e2015-08-22 18:31:29 -0600143 * NOTE: use this function instead of the tpm_tis_i2c_write_generic function.
Rong Changf6267992013-04-12 10:44:57 +0000144 *
145 * Return -EIO on error, 0 on success
146 */
Simon Glasse56e20e2015-08-22 18:31:29 -0600147static int tpm_tis_i2c_write(u8 addr, u8 *buffer, size_t len)
Rong Changf6267992013-04-12 10:44:57 +0000148{
Simon Glass42c8ec52015-08-22 18:31:30 -0600149 return tpm_tis_i2c_write_generic(addr, buffer, len, SLEEP_DURATION_US,
150 MAX_COUNT);
Rong Changf6267992013-04-12 10:44:57 +0000151}
152
153/*
154 * This function is needed especially for the cleanup situation after
155 * sending TPM_READY
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000156 */
Simon Glasse56e20e2015-08-22 18:31:29 -0600157static int tpm_tis_i2c_write_long(u8 addr, u8 *buffer, size_t len)
Rong Changf6267992013-04-12 10:44:57 +0000158{
Simon Glass42c8ec52015-08-22 18:31:30 -0600159 return tpm_tis_i2c_write_generic(addr, buffer, len,
160 SLEEP_DURATION_LONG_US,
161 MAX_COUNT_LONG);
Rong Changf6267992013-04-12 10:44:57 +0000162}
163
Simon Glasse56e20e2015-08-22 18:31:29 -0600164static int tpm_tis_i2c_check_locality(struct tpm_chip *chip, int loc)
Rong Changf6267992013-04-12 10:44:57 +0000165{
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000166 const u8 mask = TPM_ACCESS_ACTIVE_LOCALITY | TPM_ACCESS_VALID;
Rong Changf6267992013-04-12 10:44:57 +0000167 u8 buf;
168 int rc;
169
Simon Glasse56e20e2015-08-22 18:31:29 -0600170 rc = tpm_tis_i2c_read(TPM_ACCESS(loc), &buf, 1);
Rong Changf6267992013-04-12 10:44:57 +0000171 if (rc < 0)
172 return rc;
173
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000174 if ((buf & mask) == mask) {
Simon Glass7c735372015-08-22 18:31:24 -0600175 chip->locality = loc;
Rong Changf6267992013-04-12 10:44:57 +0000176 return loc;
177 }
178
179 return -1;
180}
181
Simon Glasse56e20e2015-08-22 18:31:29 -0600182static void tpm_tis_i2c_release_locality(struct tpm_chip *chip, int loc,
183 int force)
Rong Changf6267992013-04-12 10:44:57 +0000184{
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000185 const u8 mask = TPM_ACCESS_REQUEST_PENDING | TPM_ACCESS_VALID;
Rong Changf6267992013-04-12 10:44:57 +0000186 u8 buf;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000187
Simon Glasse56e20e2015-08-22 18:31:29 -0600188 if (tpm_tis_i2c_read(TPM_ACCESS(loc), &buf, 1) < 0)
Rong Changf6267992013-04-12 10:44:57 +0000189 return;
190
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000191 if (force || (buf & mask) == mask) {
Rong Changf6267992013-04-12 10:44:57 +0000192 buf = TPM_ACCESS_ACTIVE_LOCALITY;
Simon Glasse56e20e2015-08-22 18:31:29 -0600193 tpm_tis_i2c_write(TPM_ACCESS(loc), &buf, 1);
Rong Changf6267992013-04-12 10:44:57 +0000194 }
195}
196
Simon Glasse56e20e2015-08-22 18:31:29 -0600197static int tpm_tis_i2c_request_locality(struct tpm_chip *chip, int loc)
Rong Changf6267992013-04-12 10:44:57 +0000198{
199 unsigned long start, stop;
200 u8 buf = TPM_ACCESS_REQUEST_USE;
Simon Glassf90acf12015-05-04 11:30:59 -0600201 int rc;
Rong Changf6267992013-04-12 10:44:57 +0000202
Simon Glasse56e20e2015-08-22 18:31:29 -0600203 if (tpm_tis_i2c_check_locality(chip, loc) >= 0)
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000204 return loc; /* We already have the locality */
Rong Changf6267992013-04-12 10:44:57 +0000205
Simon Glasse56e20e2015-08-22 18:31:29 -0600206 rc = tpm_tis_i2c_write(TPM_ACCESS(loc), &buf, 1);
Simon Glassf90acf12015-05-04 11:30:59 -0600207 if (rc)
208 return rc;
Rong Changf6267992013-04-12 10:44:57 +0000209
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000210 /* Wait for burstcount */
Rong Changf6267992013-04-12 10:44:57 +0000211 start = get_timer(0);
Simon Glass7c735372015-08-22 18:31:24 -0600212 stop = chip->timeout_a;
Rong Changf6267992013-04-12 10:44:57 +0000213 do {
Simon Glasse56e20e2015-08-22 18:31:29 -0600214 if (tpm_tis_i2c_check_locality(chip, loc) >= 0)
Rong Changf6267992013-04-12 10:44:57 +0000215 return loc;
Simon Glass42c8ec52015-08-22 18:31:30 -0600216 mdelay(TPM_TIMEOUT_MS);
Rong Changf6267992013-04-12 10:44:57 +0000217 } while (get_timer(start) < stop);
218
219 return -1;
220}
221
222static u8 tpm_tis_i2c_status(struct tpm_chip *chip)
223{
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000224 /* NOTE: Since i2c read may fail, return 0 in this case --> time-out */
Rong Changf6267992013-04-12 10:44:57 +0000225 u8 buf;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000226
Simon Glasse56e20e2015-08-22 18:31:29 -0600227 if (tpm_tis_i2c_read(TPM_STS(chip->locality), &buf, 1) < 0)
Rong Changf6267992013-04-12 10:44:57 +0000228 return 0;
229 else
230 return buf;
231}
232
233static void tpm_tis_i2c_ready(struct tpm_chip *chip)
234{
Simon Glassf90acf12015-05-04 11:30:59 -0600235 int rc;
236
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000237 /* This causes the current command to be aborted */
Rong Changf6267992013-04-12 10:44:57 +0000238 u8 buf = TPM_STS_COMMAND_READY;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000239
Simon Glassf90acf12015-05-04 11:30:59 -0600240 debug("%s\n", __func__);
Simon Glasse56e20e2015-08-22 18:31:29 -0600241 rc = tpm_tis_i2c_write_long(TPM_STS(chip->locality), &buf, 1);
Simon Glassf90acf12015-05-04 11:30:59 -0600242 if (rc)
243 debug("%s: rc=%d\n", __func__, rc);
Rong Changf6267992013-04-12 10:44:57 +0000244}
245
Simon Glasse56e20e2015-08-22 18:31:29 -0600246static ssize_t tpm_tis_i2c_get_burstcount(struct tpm_chip *chip)
Rong Changf6267992013-04-12 10:44:57 +0000247{
248 unsigned long start, stop;
249 ssize_t burstcnt;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000250 u8 addr, buf[3];
Rong Changf6267992013-04-12 10:44:57 +0000251
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000252 /* Wait for burstcount */
253 /* XXX: Which timeout value? Spec has 2 answers (c & d) */
Rong Changf6267992013-04-12 10:44:57 +0000254 start = get_timer(0);
Simon Glass7c735372015-08-22 18:31:24 -0600255 stop = chip->timeout_d;
Rong Changf6267992013-04-12 10:44:57 +0000256 do {
257 /* Note: STS is little endian */
Simon Glass7c735372015-08-22 18:31:24 -0600258 addr = TPM_STS(chip->locality) + 1;
Simon Glasse56e20e2015-08-22 18:31:29 -0600259 if (tpm_tis_i2c_read(addr, buf, 3) < 0)
Rong Changf6267992013-04-12 10:44:57 +0000260 burstcnt = 0;
261 else
262 burstcnt = (buf[2] << 16) + (buf[1] << 8) + buf[0];
263
264 if (burstcnt)
265 return burstcnt;
Simon Glass42c8ec52015-08-22 18:31:30 -0600266 mdelay(TPM_TIMEOUT_MS);
Rong Changf6267992013-04-12 10:44:57 +0000267 } while (get_timer(start) < stop);
268
269 return -EBUSY;
270}
271
Simon Glasse56e20e2015-08-22 18:31:29 -0600272static int tpm_tis_i2c_wait_for_stat(struct tpm_chip *chip, u8 mask,
273 unsigned long timeout, int *status)
Rong Changf6267992013-04-12 10:44:57 +0000274{
275 unsigned long start, stop;
276
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000277 /* Check current status */
Rong Changf6267992013-04-12 10:44:57 +0000278 *status = tpm_tis_i2c_status(chip);
279 if ((*status & mask) == mask)
280 return 0;
281
282 start = get_timer(0);
283 stop = timeout;
284 do {
Simon Glass42c8ec52015-08-22 18:31:30 -0600285 mdelay(TPM_TIMEOUT_MS);
Rong Changf6267992013-04-12 10:44:57 +0000286 *status = tpm_tis_i2c_status(chip);
287 if ((*status & mask) == mask)
288 return 0;
Rong Changf6267992013-04-12 10:44:57 +0000289 } while (get_timer(start) < stop);
290
291 return -ETIME;
292}
293
Simon Glasse56e20e2015-08-22 18:31:29 -0600294static int tpm_tis_i2c_recv_data(struct tpm_chip *chip, u8 *buf, size_t count)
Rong Changf6267992013-04-12 10:44:57 +0000295{
296 size_t size = 0;
297 ssize_t burstcnt;
298 int rc;
299
300 while (size < count) {
Simon Glasse56e20e2015-08-22 18:31:29 -0600301 burstcnt = tpm_tis_i2c_get_burstcount(chip);
Rong Changf6267992013-04-12 10:44:57 +0000302
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000303 /* burstcount < 0 -> tpm is busy */
Rong Changf6267992013-04-12 10:44:57 +0000304 if (burstcnt < 0)
305 return burstcnt;
306
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000307 /* Limit received data to max left */
Rong Changf6267992013-04-12 10:44:57 +0000308 if (burstcnt > (count - size))
309 burstcnt = count - size;
310
Simon Glasse56e20e2015-08-22 18:31:29 -0600311 rc = tpm_tis_i2c_read(TPM_DATA_FIFO(chip->locality),
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000312 &(buf[size]), burstcnt);
Rong Changf6267992013-04-12 10:44:57 +0000313 if (rc == 0)
314 size += burstcnt;
315 }
316
317 return size;
318}
319
320static int tpm_tis_i2c_recv(struct tpm_chip *chip, u8 *buf, size_t count)
321{
322 int size = 0;
323 int expected, status;
324
325 if (count < TPM_HEADER_SIZE) {
326 size = -EIO;
327 goto out;
328 }
329
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000330 /* Read first 10 bytes, including tag, paramsize, and result */
Simon Glasse56e20e2015-08-22 18:31:29 -0600331 size = tpm_tis_i2c_recv_data(chip, buf, TPM_HEADER_SIZE);
Rong Changf6267992013-04-12 10:44:57 +0000332 if (size < TPM_HEADER_SIZE) {
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000333 error("Unable to read header\n");
Rong Changf6267992013-04-12 10:44:57 +0000334 goto out;
335 }
336
337 expected = get_unaligned_be32(buf + TPM_RSP_SIZE_BYTE);
338 if ((size_t)expected > count) {
Simon Glassf90acf12015-05-04 11:30:59 -0600339 error("Error size=%x, expected=%x, count=%x\n", size, expected,
340 count);
Rong Changf6267992013-04-12 10:44:57 +0000341 size = -EIO;
342 goto out;
343 }
344
Simon Glasse56e20e2015-08-22 18:31:29 -0600345 size += tpm_tis_i2c_recv_data(chip, &buf[TPM_HEADER_SIZE],
346 expected - TPM_HEADER_SIZE);
Rong Changf6267992013-04-12 10:44:57 +0000347 if (size < expected) {
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000348 error("Unable to read remainder of result\n");
Rong Changf6267992013-04-12 10:44:57 +0000349 size = -ETIME;
350 goto out;
351 }
352
Simon Glasse56e20e2015-08-22 18:31:29 -0600353 tpm_tis_i2c_wait_for_stat(chip, TPM_STS_VALID, chip->timeout_c,
354 &status);
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000355 if (status & TPM_STS_DATA_AVAIL) { /* Retry? */
356 error("Error left over data\n");
Rong Changf6267992013-04-12 10:44:57 +0000357 size = -EIO;
358 goto out;
359 }
360
361out:
362 tpm_tis_i2c_ready(chip);
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000363 /*
364 * The TPM needs some time to clean up here,
Rong Changf6267992013-04-12 10:44:57 +0000365 * so we sleep rather than keeping the bus busy
366 */
Simon Glass42c8ec52015-08-22 18:31:30 -0600367 mdelay(2);
Simon Glasse56e20e2015-08-22 18:31:29 -0600368 tpm_tis_i2c_release_locality(chip, chip->locality, 0);
Rong Changf6267992013-04-12 10:44:57 +0000369
370 return size;
371}
372
373static int tpm_tis_i2c_send(struct tpm_chip *chip, u8 *buf, size_t len)
374{
375 int rc, status;
Simon Glassf90acf12015-05-04 11:30:59 -0600376 size_t burstcnt;
Rong Changf6267992013-04-12 10:44:57 +0000377 size_t count = 0;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000378 int retry = 0;
Rong Changf6267992013-04-12 10:44:57 +0000379 u8 sts = TPM_STS_GO;
380
Simon Glassf90acf12015-05-04 11:30:59 -0600381 debug("%s: len=%d\n", __func__, len);
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000382 if (len > TPM_DEV_BUFSIZE)
383 return -E2BIG; /* Command is too long for our tpm, sorry */
Rong Changf6267992013-04-12 10:44:57 +0000384
Simon Glasse56e20e2015-08-22 18:31:29 -0600385 if (tpm_tis_i2c_request_locality(chip, 0) < 0)
Rong Changf6267992013-04-12 10:44:57 +0000386 return -EBUSY;
387
388 status = tpm_tis_i2c_status(chip);
389 if ((status & TPM_STS_COMMAND_READY) == 0) {
390 tpm_tis_i2c_ready(chip);
Simon Glasse56e20e2015-08-22 18:31:29 -0600391 if (tpm_tis_i2c_wait_for_stat(chip, TPM_STS_COMMAND_READY,
392 chip->timeout_b, &status) < 0) {
Rong Changf6267992013-04-12 10:44:57 +0000393 rc = -ETIME;
394 goto out_err;
395 }
396 }
397
Simon Glasse56e20e2015-08-22 18:31:29 -0600398 burstcnt = tpm_tis_i2c_get_burstcount(chip);
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000399
400 /* burstcount < 0 -> tpm is busy */
401 if (burstcnt < 0)
402 return burstcnt;
403
Simon Glassf90acf12015-05-04 11:30:59 -0600404 while (count < len) {
405 udelay(300);
406 if (burstcnt > len - count)
407 burstcnt = len - count;
Rong Changf6267992013-04-12 10:44:57 +0000408
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000409#ifdef CONFIG_TPM_TIS_I2C_BURST_LIMITATION
410 if (retry && burstcnt > CONFIG_TPM_TIS_I2C_BURST_LIMITATION)
411 burstcnt = CONFIG_TPM_TIS_I2C_BURST_LIMITATION;
412#endif /* CONFIG_TPM_TIS_I2C_BURST_LIMITATION */
Rong Changf6267992013-04-12 10:44:57 +0000413
Simon Glasse56e20e2015-08-22 18:31:29 -0600414 rc = tpm_tis_i2c_write(TPM_DATA_FIFO(chip->locality),
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000415 &(buf[count]), burstcnt);
Rong Changf6267992013-04-12 10:44:57 +0000416 if (rc == 0)
417 count += burstcnt;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000418 else {
Simon Glassf90acf12015-05-04 11:30:59 -0600419 debug("%s: error\n", __func__);
420 if (retry++ > 10) {
421 rc = -EIO;
422 goto out_err;
423 }
Simon Glasse56e20e2015-08-22 18:31:29 -0600424 rc = tpm_tis_i2c_wait_for_stat(chip, TPM_STS_VALID,
425 chip->timeout_c,
426 &status);
Simon Glassf90acf12015-05-04 11:30:59 -0600427 if (rc)
428 goto out_err;
Rong Changf6267992013-04-12 10:44:57 +0000429
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000430 if ((status & TPM_STS_DATA_EXPECT) == 0) {
431 rc = -EIO;
432 goto out_err;
433 }
Rong Changf6267992013-04-12 10:44:57 +0000434 }
435 }
436
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000437 /* Go and do it */
Simon Glasse56e20e2015-08-22 18:31:29 -0600438 tpm_tis_i2c_write(TPM_STS(chip->locality), &sts, 1);
Simon Glassf90acf12015-05-04 11:30:59 -0600439 debug("done\n");
Rong Changf6267992013-04-12 10:44:57 +0000440
441 return len;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000442
Rong Changf6267992013-04-12 10:44:57 +0000443out_err:
Simon Glassf90acf12015-05-04 11:30:59 -0600444 debug("%s: out_err\n", __func__);
Rong Changf6267992013-04-12 10:44:57 +0000445 tpm_tis_i2c_ready(chip);
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000446 /*
447 * The TPM needs some time to clean up here,
Rong Changf6267992013-04-12 10:44:57 +0000448 * so we sleep rather than keeping the bus busy
449 */
Simon Glass42c8ec52015-08-22 18:31:30 -0600450 mdelay(2);
Simon Glasse56e20e2015-08-22 18:31:29 -0600451 tpm_tis_i2c_release_locality(chip, chip->locality, 0);
Rong Changf6267992013-04-12 10:44:57 +0000452
453 return rc;
454}
455
Simon Glass13932b02015-08-22 18:31:25 -0600456static enum i2c_chip_type tpm_tis_i2c_chip_type(void)
Vincent Palatinec34fa52013-04-12 11:04:36 +0000457{
Masahiro Yamada0f925822015-08-12 07:31:55 +0900458#if CONFIG_IS_ENABLED(OF_CONTROL)
Vincent Palatinec34fa52013-04-12 11:04:36 +0000459 const void *blob = gd->fdt_blob;
460
461 if (fdtdec_next_compatible(blob, 0, COMPAT_INFINEON_SLB9645_TPM) >= 0)
462 return SLB9645;
463
464 if (fdtdec_next_compatible(blob, 0, COMPAT_INFINEON_SLB9635_TPM) >= 0)
465 return SLB9635;
466#endif
467 return UNKNOWN;
468}
469
Simon Glass13932b02015-08-22 18:31:25 -0600470static int tpm_tis_i2c_init(struct udevice *dev)
Rong Changf6267992013-04-12 10:44:57 +0000471{
Simon Glass7c735372015-08-22 18:31:24 -0600472 struct tpm_chip *chip = &g_chip;
Rong Changf6267992013-04-12 10:44:57 +0000473 u32 vendor;
Vincent Palatinec34fa52013-04-12 11:04:36 +0000474 u32 expected_did_vid;
Rong Changf6267992013-04-12 10:44:57 +0000475
Simon Glass13932b02015-08-22 18:31:25 -0600476 g_chip.dev = dev;
477 g_chip.chip_type = tpm_tis_i2c_chip_type();
Simon Glass7c735372015-08-22 18:31:24 -0600478 chip->is_open = 1;
Rong Changf6267992013-04-12 10:44:57 +0000479
480 /* Disable interrupts (not supported) */
Simon Glass7c735372015-08-22 18:31:24 -0600481 chip->irq = 0;
Rong Changf6267992013-04-12 10:44:57 +0000482
Simon Glasse56e20e2015-08-22 18:31:29 -0600483 /* Default timeouts - these could move to the device tree */
Simon Glass42c8ec52015-08-22 18:31:30 -0600484 chip->timeout_a = TIS_SHORT_TIMEOUT_MS;
485 chip->timeout_b = TIS_LONG_TIMEOUT_MS;
486 chip->timeout_c = TIS_SHORT_TIMEOUT_MS;
487 chip->timeout_d = TIS_SHORT_TIMEOUT_MS;
Simon Glass7c735372015-08-22 18:31:24 -0600488 chip->req_complete_mask = TPM_STS_DATA_AVAIL | TPM_STS_VALID;
489 chip->req_complete_val = TPM_STS_DATA_AVAIL | TPM_STS_VALID;
490 chip->req_canceled = TPM_STS_COMMAND_READY;
Rong Changf6267992013-04-12 10:44:57 +0000491
Simon Glasse56e20e2015-08-22 18:31:29 -0600492 if (tpm_tis_i2c_request_locality(chip, 0) < 0)
Simon Glassf90acf12015-05-04 11:30:59 -0600493 return -ENODEV;
Rong Changf6267992013-04-12 10:44:57 +0000494
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000495 /* Read four bytes from DID_VID register */
Simon Glasse56e20e2015-08-22 18:31:29 -0600496 if (tpm_tis_i2c_read(TPM_DID_VID(0), (uchar *)&vendor, 4) < 0) {
497 tpm_tis_i2c_release_locality(chip, 0, 1);
Simon Glassf90acf12015-05-04 11:30:59 -0600498 return -EIO;
Rong Changf6267992013-04-12 10:44:57 +0000499 }
500
Simon Glass13932b02015-08-22 18:31:25 -0600501 if (g_chip.chip_type == SLB9635) {
Vincent Palatinec34fa52013-04-12 11:04:36 +0000502 vendor = be32_to_cpu(vendor);
503 expected_did_vid = TPM_TIS_I2C_DID_VID_9635;
504 } else {
505 /* device id and byte order has changed for newer i2c tpms */
506 expected_did_vid = TPM_TIS_I2C_DID_VID_9645;
507 }
Rong Changf6267992013-04-12 10:44:57 +0000508
Simon Glass13932b02015-08-22 18:31:25 -0600509 if (g_chip.chip_type != UNKNOWN && vendor != expected_did_vid) {
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000510 error("Vendor id did not match! ID was %08x\n", vendor);
Simon Glassf90acf12015-05-04 11:30:59 -0600511 return -ENODEV;
Rong Changf6267992013-04-12 10:44:57 +0000512 }
513
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000514 debug("1.2 TPM (chip type %s device-id 0x%X)\n",
Simon Glass13932b02015-08-22 18:31:25 -0600515 chip_name[g_chip.chip_type], vendor >> 16);
Rong Changf6267992013-04-12 10:44:57 +0000516
517 /*
518 * A timeout query to TPM can be placed here.
519 * Standard timeout values are used so far
520 */
521
522 return 0;
Simon Glassf90acf12015-05-04 11:30:59 -0600523}
Rong Changf6267992013-04-12 10:44:57 +0000524
Simon Glass4cd7b782015-08-22 18:31:22 -0600525/* Returns max number of milliseconds to wait */
Simon Glasse56e20e2015-08-22 18:31:29 -0600526static unsigned long tpm_tis_i2c_calc_ordinal_duration(struct tpm_chip *chip,
527 u32 ordinal)
Simon Glass4cd7b782015-08-22 18:31:22 -0600528{
529 int duration_idx = TPM_UNDEFINED;
530 int duration = 0;
531
532 if (ordinal < TPM_MAX_ORDINAL) {
533 duration_idx = tpm_ordinal_duration[ordinal];
534 } else if ((ordinal & TPM_PROTECTED_ORDINAL_MASK) <
535 TPM_MAX_PROTECTED_ORDINAL) {
536 duration_idx = tpm_protected_ordinal_duration[
537 ordinal & TPM_PROTECTED_ORDINAL_MASK];
538 }
539
540 if (duration_idx != TPM_UNDEFINED)
Simon Glass7c735372015-08-22 18:31:24 -0600541 duration = chip->duration[duration_idx];
Simon Glass4cd7b782015-08-22 18:31:22 -0600542
543 if (duration <= 0)
544 return 2 * 60 * HZ; /* Two minutes timeout */
545 else
546 return duration;
547}
548
Simon Glasse56e20e2015-08-22 18:31:29 -0600549static ssize_t tpm_tis_i2c_transmit(const unsigned char *buf, size_t bufsiz)
Simon Glass4cd7b782015-08-22 18:31:22 -0600550{
551 int rc;
552 u32 count, ordinal;
553 unsigned long start, stop;
554
555 struct tpm_chip *chip = &g_chip;
556
557 /* switch endianess: big->little */
558 count = get_unaligned_be32(buf + TPM_CMD_COUNT_BYTE);
559 ordinal = get_unaligned_be32(buf + TPM_CMD_ORDINAL_BYTE);
560
561 if (count == 0) {
562 error("no data\n");
563 return -ENODATA;
564 }
565 if (count > bufsiz) {
566 error("invalid count value %x %zx\n", count, bufsiz);
567 return -E2BIG;
568 }
569
570 debug("Calling send\n");
Simon Glassb382e022015-08-22 18:31:23 -0600571 rc = tpm_tis_i2c_send(chip, (u8 *)buf, count);
Simon Glass4cd7b782015-08-22 18:31:22 -0600572 debug(" ... done calling send\n");
573 if (rc < 0) {
574 error("tpm_transmit: tpm_send: error %d\n", rc);
575 goto out;
576 }
577
Simon Glass7c735372015-08-22 18:31:24 -0600578 if (chip->irq)
Simon Glass4cd7b782015-08-22 18:31:22 -0600579 goto out_recv;
580
581 start = get_timer(0);
Simon Glasse56e20e2015-08-22 18:31:29 -0600582 stop = tpm_tis_i2c_calc_ordinal_duration(chip, ordinal);
Simon Glass4cd7b782015-08-22 18:31:22 -0600583 do {
584 debug("waiting for status... %ld %ld\n", start, stop);
Simon Glassb382e022015-08-22 18:31:23 -0600585 u8 status = tpm_tis_i2c_status(chip);
Simon Glass7c735372015-08-22 18:31:24 -0600586 if ((status & chip->req_complete_mask) ==
587 chip->req_complete_val) {
Simon Glass4cd7b782015-08-22 18:31:22 -0600588 debug("...got it;\n");
589 goto out_recv;
590 }
591
Simon Glass7c735372015-08-22 18:31:24 -0600592 if (status == chip->req_canceled) {
Simon Glass4cd7b782015-08-22 18:31:22 -0600593 error("Operation Canceled\n");
594 rc = -ECANCELED;
595 goto out;
596 }
Simon Glass42c8ec52015-08-22 18:31:30 -0600597 mdelay(TPM_TIMEOUT_MS);
Simon Glass4cd7b782015-08-22 18:31:22 -0600598 } while (get_timer(start) < stop);
599
Simon Glassb382e022015-08-22 18:31:23 -0600600 tpm_tis_i2c_ready(chip);
Simon Glass4cd7b782015-08-22 18:31:22 -0600601 error("Operation Timed out\n");
602 rc = -ETIME;
603 goto out;
604
605out_recv:
606 debug("out_recv: reading response...\n");
Simon Glassb382e022015-08-22 18:31:23 -0600607 rc = tpm_tis_i2c_recv(chip, (u8 *)buf, TPM_BUFSIZE);
Simon Glass4cd7b782015-08-22 18:31:22 -0600608 if (rc < 0)
609 error("tpm_transmit: tpm_recv: error %d\n", rc);
610
611out:
612 return rc;
613}
614
Simon Glass4cd7b782015-08-22 18:31:22 -0600615/**
616 * Decode TPM configuration.
617 *
618 * @param dev Returns a configuration of TPM device
619 * @return 0 if ok, -1 on error
620 */
Simon Glasse56e20e2015-08-22 18:31:29 -0600621static int tpm_tis_i2c_decode_config(struct tpm_chip *chip)
Simon Glass4cd7b782015-08-22 18:31:22 -0600622{
623 const void *blob = gd->fdt_blob;
624 struct udevice *bus;
625 int chip_addr;
626 int parent;
627 int node;
628 int ret;
629
630 node = fdtdec_next_compatible(blob, 0, COMPAT_INFINEON_SLB9635_TPM);
631 if (node < 0) {
632 node = fdtdec_next_compatible(blob, 0,
633 COMPAT_INFINEON_SLB9645_TPM);
634 }
635 if (node < 0) {
636 debug("%s: Node not found\n", __func__);
637 return -1;
638 }
639 parent = fdt_parent_offset(blob, node);
640 if (parent < 0) {
641 debug("%s: Cannot find node parent\n", __func__);
642 return -1;
643 }
644
645 /*
646 * TODO(sjg@chromium.org): Remove this when driver model supports
647 * TPMs
648 */
649 ret = uclass_get_device_by_of_offset(UCLASS_I2C, parent, &bus);
650 if (ret) {
651 debug("Cannot find bus for node '%s: ret=%d'\n",
652 fdt_get_name(blob, parent, NULL), ret);
653 return ret;
654 }
655
656 chip_addr = fdtdec_get_int(blob, node, "reg", -1);
657 if (chip_addr == -1) {
658 debug("Cannot find reg property for node '%s: ret=%d'\n",
659 fdt_get_name(blob, node, NULL), ret);
660 return ret;
661 }
662 /*
663 * TODO(sjg@chromium.org): Older TPMs will need to use the older method
Simon Glasse56e20e2015-08-22 18:31:29 -0600664 * in tpm_tis_i2c_read() so the offset length needs to be 0 here.
Simon Glass4cd7b782015-08-22 18:31:22 -0600665 */
Simon Glass605152a2015-08-22 18:31:26 -0600666 ret = i2c_get_chip(bus, chip_addr, 1, &chip->dev);
Simon Glass4cd7b782015-08-22 18:31:22 -0600667 if (ret) {
668 debug("Cannot find device for node '%s: ret=%d'\n",
669 fdt_get_name(blob, node, NULL), ret);
670 return ret;
671 }
672
673 return 0;
674}
675
Simon Glass4cd7b782015-08-22 18:31:22 -0600676int tis_init(void)
677{
Simon Glass605152a2015-08-22 18:31:26 -0600678 if (g_chip.inited)
Simon Glass4cd7b782015-08-22 18:31:22 -0600679 return 0;
680
Simon Glasse56e20e2015-08-22 18:31:29 -0600681 if (tpm_tis_i2c_decode_config(&g_chip))
Simon Glass4cd7b782015-08-22 18:31:22 -0600682 return -1;
683
684 debug("%s: done\n", __func__);
685
Simon Glass605152a2015-08-22 18:31:26 -0600686 g_chip.inited = 1;
Simon Glass4cd7b782015-08-22 18:31:22 -0600687
688 return 0;
689}
690
691int tis_open(void)
692{
693 int rc;
694
Simon Glass605152a2015-08-22 18:31:26 -0600695 if (!g_chip.inited)
Simon Glass4cd7b782015-08-22 18:31:22 -0600696 return -1;
697
Simon Glassa53b79a2015-08-22 18:31:28 -0600698 debug("%s: start\n", __func__);
699 if (g_chip.is_open)
700 return -EBUSY;
701 rc = tpm_tis_i2c_init(g_chip.dev);
702 if (rc < 0)
703 g_chip.is_open = 0;
Simon Glass4cd7b782015-08-22 18:31:22 -0600704
705 return rc;
706}
707
708int tis_close(void)
709{
Simon Glass605152a2015-08-22 18:31:26 -0600710 if (!g_chip.inited)
Simon Glass4cd7b782015-08-22 18:31:22 -0600711 return -1;
712
Simon Glassa53b79a2015-08-22 18:31:28 -0600713 if (g_chip.is_open) {
Simon Glasse56e20e2015-08-22 18:31:29 -0600714 tpm_tis_i2c_release_locality(&g_chip, g_chip.locality, 1);
Simon Glassa53b79a2015-08-22 18:31:28 -0600715 g_chip.is_open = 0;
716 }
Simon Glass4cd7b782015-08-22 18:31:22 -0600717
718 return 0;
719}
720
721int tis_sendrecv(const uint8_t *sendbuf, size_t sbuf_size,
722 uint8_t *recvbuf, size_t *rbuf_len)
723{
724 int len;
725 uint8_t buf[4096];
726
Simon Glass605152a2015-08-22 18:31:26 -0600727 if (!g_chip.inited)
Simon Glass4cd7b782015-08-22 18:31:22 -0600728 return -1;
729
730 if (sizeof(buf) < sbuf_size)
731 return -1;
732
733 memcpy(buf, sendbuf, sbuf_size);
734
Simon Glasse56e20e2015-08-22 18:31:29 -0600735 len = tpm_tis_i2c_transmit(buf, sbuf_size);
Simon Glass4cd7b782015-08-22 18:31:22 -0600736
737 if (len < 10) {
738 *rbuf_len = 0;
739 return -1;
740 }
741
742 memcpy(recvbuf, buf, len);
743 *rbuf_len = len;
744
745 return 0;
746}