blob: 9dd40dd3cf51ba4637e209192b25e32ba77f0d57 [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/*
46 * iic_tpm_read() - read from TPM register
47 * @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 */
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +000059static int iic_tpm_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 */
Vincent Palatinec34fa52013-04-12 11:04:36 +000071 udelay(SLEEP_DURATION);
72 }
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++) {
81 udelay(SLEEP_DURATION);
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 */
98 udelay(SLEEP_DURATION);
99 }
Rong Changf6267992013-04-12 10:44:57 +0000100 }
101
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000102 /* Take care of 'guard time' */
Vincent Palatinec34fa52013-04-12 11:04:36 +0000103 udelay(SLEEP_DURATION);
Rong Changf6267992013-04-12 10:44:57 +0000104 if (rc)
105 return -rc;
106
107 return 0;
108}
109
110static int iic_tpm_write_generic(u8 addr, u8 *buffer, size_t len,
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000111 unsigned int sleep_time, 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 */
Rong Changf6267992013-04-12 10:44:57 +0000120 udelay(sleep_time);
121 }
122
Vincent Palatinec34fa52013-04-12 11:04:36 +0000123 /* take care of 'guard time' */
Simon Glassf90acf12015-05-04 11:30:59 -0600124 udelay(sleep_time);
Rong Changf6267992013-04-12 10:44:57 +0000125 if (rc)
126 return -rc;
127
128 return 0;
129}
130
131/*
132 * iic_tpm_write() - write to TPM register
133 * @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 *
143 * NOTE: use this function instead of the iic_tpm_write_generic function.
144 *
145 * Return -EIO on error, 0 on success
146 */
147static int iic_tpm_write(u8 addr, u8 *buffer, size_t len)
148{
149 return iic_tpm_write_generic(addr, buffer, len, SLEEP_DURATION,
150 MAX_COUNT);
151}
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 */
Rong Changf6267992013-04-12 10:44:57 +0000157static int iic_tpm_write_long(u8 addr, u8 *buffer, size_t len)
158{
159 return iic_tpm_write_generic(addr, buffer, len, SLEEP_DURATION_LONG,
160 MAX_COUNT_LONG);
161}
162
Rong Changf6267992013-04-12 10:44:57 +0000163static int check_locality(struct tpm_chip *chip, int loc)
164{
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000165 const u8 mask = TPM_ACCESS_ACTIVE_LOCALITY | TPM_ACCESS_VALID;
Rong Changf6267992013-04-12 10:44:57 +0000166 u8 buf;
167 int rc;
168
169 rc = iic_tpm_read(TPM_ACCESS(loc), &buf, 1);
170 if (rc < 0)
171 return rc;
172
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000173 if ((buf & mask) == mask) {
Simon Glass7c735372015-08-22 18:31:24 -0600174 chip->locality = loc;
Rong Changf6267992013-04-12 10:44:57 +0000175 return loc;
176 }
177
178 return -1;
179}
180
181static void release_locality(struct tpm_chip *chip, int loc, int force)
182{
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000183 const u8 mask = TPM_ACCESS_REQUEST_PENDING | TPM_ACCESS_VALID;
Rong Changf6267992013-04-12 10:44:57 +0000184 u8 buf;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000185
Rong Changf6267992013-04-12 10:44:57 +0000186 if (iic_tpm_read(TPM_ACCESS(loc), &buf, 1) < 0)
187 return;
188
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000189 if (force || (buf & mask) == mask) {
Rong Changf6267992013-04-12 10:44:57 +0000190 buf = TPM_ACCESS_ACTIVE_LOCALITY;
191 iic_tpm_write(TPM_ACCESS(loc), &buf, 1);
192 }
193}
194
195static int request_locality(struct tpm_chip *chip, int loc)
196{
197 unsigned long start, stop;
198 u8 buf = TPM_ACCESS_REQUEST_USE;
Simon Glassf90acf12015-05-04 11:30:59 -0600199 int rc;
Rong Changf6267992013-04-12 10:44:57 +0000200
201 if (check_locality(chip, loc) >= 0)
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000202 return loc; /* We already have the locality */
Rong Changf6267992013-04-12 10:44:57 +0000203
Simon Glassf90acf12015-05-04 11:30:59 -0600204 rc = iic_tpm_write(TPM_ACCESS(loc), &buf, 1);
205 if (rc)
206 return rc;
Rong Changf6267992013-04-12 10:44:57 +0000207
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000208 /* Wait for burstcount */
Rong Changf6267992013-04-12 10:44:57 +0000209 start = get_timer(0);
Simon Glass7c735372015-08-22 18:31:24 -0600210 stop = chip->timeout_a;
Rong Changf6267992013-04-12 10:44:57 +0000211 do {
212 if (check_locality(chip, loc) >= 0)
213 return loc;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000214 udelay(TPM_TIMEOUT * 1000);
Rong Changf6267992013-04-12 10:44:57 +0000215 } while (get_timer(start) < stop);
216
217 return -1;
218}
219
220static u8 tpm_tis_i2c_status(struct tpm_chip *chip)
221{
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000222 /* NOTE: Since i2c read may fail, return 0 in this case --> time-out */
Rong Changf6267992013-04-12 10:44:57 +0000223 u8 buf;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000224
Simon Glass7c735372015-08-22 18:31:24 -0600225 if (iic_tpm_read(TPM_STS(chip->locality), &buf, 1) < 0)
Rong Changf6267992013-04-12 10:44:57 +0000226 return 0;
227 else
228 return buf;
229}
230
231static void tpm_tis_i2c_ready(struct tpm_chip *chip)
232{
Simon Glassf90acf12015-05-04 11:30:59 -0600233 int rc;
234
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000235 /* This causes the current command to be aborted */
Rong Changf6267992013-04-12 10:44:57 +0000236 u8 buf = TPM_STS_COMMAND_READY;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000237
Simon Glassf90acf12015-05-04 11:30:59 -0600238 debug("%s\n", __func__);
Simon Glass7c735372015-08-22 18:31:24 -0600239 rc = iic_tpm_write_long(TPM_STS(chip->locality), &buf, 1);
Simon Glassf90acf12015-05-04 11:30:59 -0600240 if (rc)
241 debug("%s: rc=%d\n", __func__, rc);
Rong Changf6267992013-04-12 10:44:57 +0000242}
243
244static ssize_t get_burstcount(struct tpm_chip *chip)
245{
246 unsigned long start, stop;
247 ssize_t burstcnt;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000248 u8 addr, buf[3];
Rong Changf6267992013-04-12 10:44:57 +0000249
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000250 /* Wait for burstcount */
251 /* XXX: Which timeout value? Spec has 2 answers (c & d) */
Rong Changf6267992013-04-12 10:44:57 +0000252 start = get_timer(0);
Simon Glass7c735372015-08-22 18:31:24 -0600253 stop = chip->timeout_d;
Rong Changf6267992013-04-12 10:44:57 +0000254 do {
255 /* Note: STS is little endian */
Simon Glass7c735372015-08-22 18:31:24 -0600256 addr = TPM_STS(chip->locality) + 1;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000257 if (iic_tpm_read(addr, buf, 3) < 0)
Rong Changf6267992013-04-12 10:44:57 +0000258 burstcnt = 0;
259 else
260 burstcnt = (buf[2] << 16) + (buf[1] << 8) + buf[0];
261
262 if (burstcnt)
263 return burstcnt;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000264 udelay(TPM_TIMEOUT * 1000);
Rong Changf6267992013-04-12 10:44:57 +0000265 } while (get_timer(start) < stop);
266
267 return -EBUSY;
268}
269
270static int wait_for_stat(struct tpm_chip *chip, u8 mask, unsigned long timeout,
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000271 int *status)
Rong Changf6267992013-04-12 10:44:57 +0000272{
273 unsigned long start, stop;
274
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000275 /* Check current status */
Rong Changf6267992013-04-12 10:44:57 +0000276 *status = tpm_tis_i2c_status(chip);
277 if ((*status & mask) == mask)
278 return 0;
279
280 start = get_timer(0);
281 stop = timeout;
282 do {
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000283 udelay(TPM_TIMEOUT * 1000);
Rong Changf6267992013-04-12 10:44:57 +0000284 *status = tpm_tis_i2c_status(chip);
285 if ((*status & mask) == mask)
286 return 0;
Rong Changf6267992013-04-12 10:44:57 +0000287 } while (get_timer(start) < stop);
288
289 return -ETIME;
290}
291
292static int recv_data(struct tpm_chip *chip, u8 *buf, size_t count)
293{
294 size_t size = 0;
295 ssize_t burstcnt;
296 int rc;
297
298 while (size < count) {
299 burstcnt = get_burstcount(chip);
300
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000301 /* burstcount < 0 -> tpm is busy */
Rong Changf6267992013-04-12 10:44:57 +0000302 if (burstcnt < 0)
303 return burstcnt;
304
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000305 /* Limit received data to max left */
Rong Changf6267992013-04-12 10:44:57 +0000306 if (burstcnt > (count - size))
307 burstcnt = count - size;
308
Simon Glass7c735372015-08-22 18:31:24 -0600309 rc = iic_tpm_read(TPM_DATA_FIFO(chip->locality),
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000310 &(buf[size]), burstcnt);
Rong Changf6267992013-04-12 10:44:57 +0000311 if (rc == 0)
312 size += burstcnt;
313 }
314
315 return size;
316}
317
318static int tpm_tis_i2c_recv(struct tpm_chip *chip, u8 *buf, size_t count)
319{
320 int size = 0;
321 int expected, status;
322
323 if (count < TPM_HEADER_SIZE) {
324 size = -EIO;
325 goto out;
326 }
327
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000328 /* Read first 10 bytes, including tag, paramsize, and result */
Rong Changf6267992013-04-12 10:44:57 +0000329 size = recv_data(chip, buf, TPM_HEADER_SIZE);
330 if (size < TPM_HEADER_SIZE) {
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000331 error("Unable to read header\n");
Rong Changf6267992013-04-12 10:44:57 +0000332 goto out;
333 }
334
335 expected = get_unaligned_be32(buf + TPM_RSP_SIZE_BYTE);
336 if ((size_t)expected > count) {
Simon Glassf90acf12015-05-04 11:30:59 -0600337 error("Error size=%x, expected=%x, count=%x\n", size, expected,
338 count);
Rong Changf6267992013-04-12 10:44:57 +0000339 size = -EIO;
340 goto out;
341 }
342
343 size += recv_data(chip, &buf[TPM_HEADER_SIZE],
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000344 expected - TPM_HEADER_SIZE);
Rong Changf6267992013-04-12 10:44:57 +0000345 if (size < expected) {
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000346 error("Unable to read remainder of result\n");
Rong Changf6267992013-04-12 10:44:57 +0000347 size = -ETIME;
348 goto out;
349 }
350
Simon Glass7c735372015-08-22 18:31:24 -0600351 wait_for_stat(chip, TPM_STS_VALID, chip->timeout_c, &status);
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000352 if (status & TPM_STS_DATA_AVAIL) { /* Retry? */
353 error("Error left over data\n");
Rong Changf6267992013-04-12 10:44:57 +0000354 size = -EIO;
355 goto out;
356 }
357
358out:
359 tpm_tis_i2c_ready(chip);
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000360 /*
361 * The TPM needs some time to clean up here,
Rong Changf6267992013-04-12 10:44:57 +0000362 * so we sleep rather than keeping the bus busy
363 */
364 udelay(2000);
Simon Glass7c735372015-08-22 18:31:24 -0600365 release_locality(chip, chip->locality, 0);
Rong Changf6267992013-04-12 10:44:57 +0000366
367 return size;
368}
369
370static int tpm_tis_i2c_send(struct tpm_chip *chip, u8 *buf, size_t len)
371{
372 int rc, status;
Simon Glassf90acf12015-05-04 11:30:59 -0600373 size_t burstcnt;
Rong Changf6267992013-04-12 10:44:57 +0000374 size_t count = 0;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000375 int retry = 0;
Rong Changf6267992013-04-12 10:44:57 +0000376 u8 sts = TPM_STS_GO;
377
Simon Glassf90acf12015-05-04 11:30:59 -0600378 debug("%s: len=%d\n", __func__, len);
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000379 if (len > TPM_DEV_BUFSIZE)
380 return -E2BIG; /* Command is too long for our tpm, sorry */
Rong Changf6267992013-04-12 10:44:57 +0000381
382 if (request_locality(chip, 0) < 0)
383 return -EBUSY;
384
385 status = tpm_tis_i2c_status(chip);
386 if ((status & TPM_STS_COMMAND_READY) == 0) {
387 tpm_tis_i2c_ready(chip);
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000388 if (wait_for_stat(chip, TPM_STS_COMMAND_READY,
Simon Glass7c735372015-08-22 18:31:24 -0600389 chip->timeout_b, &status) < 0) {
Rong Changf6267992013-04-12 10:44:57 +0000390 rc = -ETIME;
391 goto out_err;
392 }
393 }
394
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000395 burstcnt = get_burstcount(chip);
396
397 /* burstcount < 0 -> tpm is busy */
398 if (burstcnt < 0)
399 return burstcnt;
400
Simon Glassf90acf12015-05-04 11:30:59 -0600401 while (count < len) {
402 udelay(300);
403 if (burstcnt > len - count)
404 burstcnt = len - count;
Rong Changf6267992013-04-12 10:44:57 +0000405
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000406#ifdef CONFIG_TPM_TIS_I2C_BURST_LIMITATION
407 if (retry && burstcnt > CONFIG_TPM_TIS_I2C_BURST_LIMITATION)
408 burstcnt = CONFIG_TPM_TIS_I2C_BURST_LIMITATION;
409#endif /* CONFIG_TPM_TIS_I2C_BURST_LIMITATION */
Rong Changf6267992013-04-12 10:44:57 +0000410
Simon Glass7c735372015-08-22 18:31:24 -0600411 rc = iic_tpm_write(TPM_DATA_FIFO(chip->locality),
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000412 &(buf[count]), burstcnt);
Rong Changf6267992013-04-12 10:44:57 +0000413 if (rc == 0)
414 count += burstcnt;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000415 else {
Simon Glassf90acf12015-05-04 11:30:59 -0600416 debug("%s: error\n", __func__);
417 if (retry++ > 10) {
418 rc = -EIO;
419 goto out_err;
420 }
421 rc = wait_for_stat(chip, TPM_STS_VALID,
Simon Glass7c735372015-08-22 18:31:24 -0600422 chip->timeout_c, &status);
Simon Glassf90acf12015-05-04 11:30:59 -0600423 if (rc)
424 goto out_err;
Rong Changf6267992013-04-12 10:44:57 +0000425
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000426 if ((status & TPM_STS_DATA_EXPECT) == 0) {
427 rc = -EIO;
428 goto out_err;
429 }
Rong Changf6267992013-04-12 10:44:57 +0000430 }
431 }
432
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000433 /* Go and do it */
Simon Glass7c735372015-08-22 18:31:24 -0600434 iic_tpm_write(TPM_STS(chip->locality), &sts, 1);
Simon Glassf90acf12015-05-04 11:30:59 -0600435 debug("done\n");
Rong Changf6267992013-04-12 10:44:57 +0000436
437 return len;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000438
Rong Changf6267992013-04-12 10:44:57 +0000439out_err:
Simon Glassf90acf12015-05-04 11:30:59 -0600440 debug("%s: out_err\n", __func__);
Rong Changf6267992013-04-12 10:44:57 +0000441 tpm_tis_i2c_ready(chip);
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000442 /*
443 * The TPM needs some time to clean up here,
Rong Changf6267992013-04-12 10:44:57 +0000444 * so we sleep rather than keeping the bus busy
445 */
446 udelay(2000);
Simon Glass7c735372015-08-22 18:31:24 -0600447 release_locality(chip, chip->locality, 0);
Rong Changf6267992013-04-12 10:44:57 +0000448
449 return rc;
450}
451
Simon Glass13932b02015-08-22 18:31:25 -0600452static enum i2c_chip_type tpm_tis_i2c_chip_type(void)
Vincent Palatinec34fa52013-04-12 11:04:36 +0000453{
Masahiro Yamada0f925822015-08-12 07:31:55 +0900454#if CONFIG_IS_ENABLED(OF_CONTROL)
Vincent Palatinec34fa52013-04-12 11:04:36 +0000455 const void *blob = gd->fdt_blob;
456
457 if (fdtdec_next_compatible(blob, 0, COMPAT_INFINEON_SLB9645_TPM) >= 0)
458 return SLB9645;
459
460 if (fdtdec_next_compatible(blob, 0, COMPAT_INFINEON_SLB9635_TPM) >= 0)
461 return SLB9635;
462#endif
463 return UNKNOWN;
464}
465
Simon Glass13932b02015-08-22 18:31:25 -0600466static int tpm_tis_i2c_init(struct udevice *dev)
Rong Changf6267992013-04-12 10:44:57 +0000467{
Simon Glass7c735372015-08-22 18:31:24 -0600468 struct tpm_chip *chip = &g_chip;
Rong Changf6267992013-04-12 10:44:57 +0000469 u32 vendor;
Vincent Palatinec34fa52013-04-12 11:04:36 +0000470 u32 expected_did_vid;
Rong Changf6267992013-04-12 10:44:57 +0000471
Simon Glass13932b02015-08-22 18:31:25 -0600472 g_chip.dev = dev;
473 g_chip.chip_type = tpm_tis_i2c_chip_type();
Simon Glass7c735372015-08-22 18:31:24 -0600474 chip->is_open = 1;
Rong Changf6267992013-04-12 10:44:57 +0000475
476 /* Disable interrupts (not supported) */
Simon Glass7c735372015-08-22 18:31:24 -0600477 chip->irq = 0;
Rong Changf6267992013-04-12 10:44:57 +0000478
479 /* Default timeouts */
Simon Glass7c735372015-08-22 18:31:24 -0600480 chip->timeout_a = TIS_SHORT_TIMEOUT;
481 chip->timeout_b = TIS_LONG_TIMEOUT;
482 chip->timeout_c = TIS_SHORT_TIMEOUT;
483 chip->timeout_d = TIS_SHORT_TIMEOUT;
484 chip->req_complete_mask = TPM_STS_DATA_AVAIL | TPM_STS_VALID;
485 chip->req_complete_val = TPM_STS_DATA_AVAIL | TPM_STS_VALID;
486 chip->req_canceled = TPM_STS_COMMAND_READY;
Rong Changf6267992013-04-12 10:44:57 +0000487
Simon Glassf90acf12015-05-04 11:30:59 -0600488 if (request_locality(chip, 0) < 0)
489 return -ENODEV;
Rong Changf6267992013-04-12 10:44:57 +0000490
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000491 /* Read four bytes from DID_VID register */
Rong Changf6267992013-04-12 10:44:57 +0000492 if (iic_tpm_read(TPM_DID_VID(0), (uchar *)&vendor, 4) < 0) {
Simon Glassf90acf12015-05-04 11:30:59 -0600493 release_locality(chip, 0, 1);
494 return -EIO;
Rong Changf6267992013-04-12 10:44:57 +0000495 }
496
Simon Glass13932b02015-08-22 18:31:25 -0600497 if (g_chip.chip_type == SLB9635) {
Vincent Palatinec34fa52013-04-12 11:04:36 +0000498 vendor = be32_to_cpu(vendor);
499 expected_did_vid = TPM_TIS_I2C_DID_VID_9635;
500 } else {
501 /* device id and byte order has changed for newer i2c tpms */
502 expected_did_vid = TPM_TIS_I2C_DID_VID_9645;
503 }
Rong Changf6267992013-04-12 10:44:57 +0000504
Simon Glass13932b02015-08-22 18:31:25 -0600505 if (g_chip.chip_type != UNKNOWN && vendor != expected_did_vid) {
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000506 error("Vendor id did not match! ID was %08x\n", vendor);
Simon Glassf90acf12015-05-04 11:30:59 -0600507 return -ENODEV;
Rong Changf6267992013-04-12 10:44:57 +0000508 }
509
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000510 debug("1.2 TPM (chip type %s device-id 0x%X)\n",
Simon Glass13932b02015-08-22 18:31:25 -0600511 chip_name[g_chip.chip_type], vendor >> 16);
Rong Changf6267992013-04-12 10:44:57 +0000512
513 /*
514 * A timeout query to TPM can be placed here.
515 * Standard timeout values are used so far
516 */
517
518 return 0;
Simon Glassf90acf12015-05-04 11:30:59 -0600519}
Rong Changf6267992013-04-12 10:44:57 +0000520
Simon Glass4cd7b782015-08-22 18:31:22 -0600521/* Returns max number of milliseconds to wait */
522static unsigned long tpm_calc_ordinal_duration(struct tpm_chip *chip,
523 u32 ordinal)
524{
525 int duration_idx = TPM_UNDEFINED;
526 int duration = 0;
527
528 if (ordinal < TPM_MAX_ORDINAL) {
529 duration_idx = tpm_ordinal_duration[ordinal];
530 } else if ((ordinal & TPM_PROTECTED_ORDINAL_MASK) <
531 TPM_MAX_PROTECTED_ORDINAL) {
532 duration_idx = tpm_protected_ordinal_duration[
533 ordinal & TPM_PROTECTED_ORDINAL_MASK];
534 }
535
536 if (duration_idx != TPM_UNDEFINED)
Simon Glass7c735372015-08-22 18:31:24 -0600537 duration = chip->duration[duration_idx];
Simon Glass4cd7b782015-08-22 18:31:22 -0600538
539 if (duration <= 0)
540 return 2 * 60 * HZ; /* Two minutes timeout */
541 else
542 return duration;
543}
544
545static ssize_t tpm_transmit(const unsigned char *buf, size_t bufsiz)
546{
547 int rc;
548 u32 count, ordinal;
549 unsigned long start, stop;
550
551 struct tpm_chip *chip = &g_chip;
552
553 /* switch endianess: big->little */
554 count = get_unaligned_be32(buf + TPM_CMD_COUNT_BYTE);
555 ordinal = get_unaligned_be32(buf + TPM_CMD_ORDINAL_BYTE);
556
557 if (count == 0) {
558 error("no data\n");
559 return -ENODATA;
560 }
561 if (count > bufsiz) {
562 error("invalid count value %x %zx\n", count, bufsiz);
563 return -E2BIG;
564 }
565
566 debug("Calling send\n");
Simon Glassb382e022015-08-22 18:31:23 -0600567 rc = tpm_tis_i2c_send(chip, (u8 *)buf, count);
Simon Glass4cd7b782015-08-22 18:31:22 -0600568 debug(" ... done calling send\n");
569 if (rc < 0) {
570 error("tpm_transmit: tpm_send: error %d\n", rc);
571 goto out;
572 }
573
Simon Glass7c735372015-08-22 18:31:24 -0600574 if (chip->irq)
Simon Glass4cd7b782015-08-22 18:31:22 -0600575 goto out_recv;
576
577 start = get_timer(0);
578 stop = tpm_calc_ordinal_duration(chip, ordinal);
579 do {
580 debug("waiting for status... %ld %ld\n", start, stop);
Simon Glassb382e022015-08-22 18:31:23 -0600581 u8 status = tpm_tis_i2c_status(chip);
Simon Glass7c735372015-08-22 18:31:24 -0600582 if ((status & chip->req_complete_mask) ==
583 chip->req_complete_val) {
Simon Glass4cd7b782015-08-22 18:31:22 -0600584 debug("...got it;\n");
585 goto out_recv;
586 }
587
Simon Glass7c735372015-08-22 18:31:24 -0600588 if (status == chip->req_canceled) {
Simon Glass4cd7b782015-08-22 18:31:22 -0600589 error("Operation Canceled\n");
590 rc = -ECANCELED;
591 goto out;
592 }
593 udelay(TPM_TIMEOUT * 1000);
594 } while (get_timer(start) < stop);
595
Simon Glassb382e022015-08-22 18:31:23 -0600596 tpm_tis_i2c_ready(chip);
Simon Glass4cd7b782015-08-22 18:31:22 -0600597 error("Operation Timed out\n");
598 rc = -ETIME;
599 goto out;
600
601out_recv:
602 debug("out_recv: reading response...\n");
Simon Glassb382e022015-08-22 18:31:23 -0600603 rc = tpm_tis_i2c_recv(chip, (u8 *)buf, TPM_BUFSIZE);
Simon Glass4cd7b782015-08-22 18:31:22 -0600604 if (rc < 0)
605 error("tpm_transmit: tpm_recv: error %d\n", rc);
606
607out:
608 return rc;
609}
610
Simon Glass4cd7b782015-08-22 18:31:22 -0600611/**
612 * Decode TPM configuration.
613 *
614 * @param dev Returns a configuration of TPM device
615 * @return 0 if ok, -1 on error
616 */
Simon Glass605152a2015-08-22 18:31:26 -0600617static int tpm_decode_config(struct tpm_chip *chip)
Simon Glass4cd7b782015-08-22 18:31:22 -0600618{
619 const void *blob = gd->fdt_blob;
620 struct udevice *bus;
621 int chip_addr;
622 int parent;
623 int node;
624 int ret;
625
626 node = fdtdec_next_compatible(blob, 0, COMPAT_INFINEON_SLB9635_TPM);
627 if (node < 0) {
628 node = fdtdec_next_compatible(blob, 0,
629 COMPAT_INFINEON_SLB9645_TPM);
630 }
631 if (node < 0) {
632 debug("%s: Node not found\n", __func__);
633 return -1;
634 }
635 parent = fdt_parent_offset(blob, node);
636 if (parent < 0) {
637 debug("%s: Cannot find node parent\n", __func__);
638 return -1;
639 }
640
641 /*
642 * TODO(sjg@chromium.org): Remove this when driver model supports
643 * TPMs
644 */
645 ret = uclass_get_device_by_of_offset(UCLASS_I2C, parent, &bus);
646 if (ret) {
647 debug("Cannot find bus for node '%s: ret=%d'\n",
648 fdt_get_name(blob, parent, NULL), ret);
649 return ret;
650 }
651
652 chip_addr = fdtdec_get_int(blob, node, "reg", -1);
653 if (chip_addr == -1) {
654 debug("Cannot find reg property for node '%s: ret=%d'\n",
655 fdt_get_name(blob, node, NULL), ret);
656 return ret;
657 }
658 /*
659 * TODO(sjg@chromium.org): Older TPMs will need to use the older method
660 * in iic_tpm_read() so the offset length needs to be 0 here.
661 */
Simon Glass605152a2015-08-22 18:31:26 -0600662 ret = i2c_get_chip(bus, chip_addr, 1, &chip->dev);
Simon Glass4cd7b782015-08-22 18:31:22 -0600663 if (ret) {
664 debug("Cannot find device for node '%s: ret=%d'\n",
665 fdt_get_name(blob, node, NULL), ret);
666 return ret;
667 }
668
669 return 0;
670}
671
Simon Glass4cd7b782015-08-22 18:31:22 -0600672int tis_init(void)
673{
Simon Glass605152a2015-08-22 18:31:26 -0600674 if (g_chip.inited)
Simon Glass4cd7b782015-08-22 18:31:22 -0600675 return 0;
676
Simon Glass605152a2015-08-22 18:31:26 -0600677 if (tpm_decode_config(&g_chip))
Simon Glass4cd7b782015-08-22 18:31:22 -0600678 return -1;
679
680 debug("%s: done\n", __func__);
681
Simon Glass605152a2015-08-22 18:31:26 -0600682 g_chip.inited = 1;
Simon Glass4cd7b782015-08-22 18:31:22 -0600683
684 return 0;
685}
686
687int tis_open(void)
688{
689 int rc;
690
Simon Glass605152a2015-08-22 18:31:26 -0600691 if (!g_chip.inited)
Simon Glass4cd7b782015-08-22 18:31:22 -0600692 return -1;
693
Simon Glassa53b79a2015-08-22 18:31:28 -0600694 debug("%s: start\n", __func__);
695 if (g_chip.is_open)
696 return -EBUSY;
697 rc = tpm_tis_i2c_init(g_chip.dev);
698 if (rc < 0)
699 g_chip.is_open = 0;
Simon Glass4cd7b782015-08-22 18:31:22 -0600700
701 return rc;
702}
703
704int tis_close(void)
705{
Simon Glass605152a2015-08-22 18:31:26 -0600706 if (!g_chip.inited)
Simon Glass4cd7b782015-08-22 18:31:22 -0600707 return -1;
708
Simon Glassa53b79a2015-08-22 18:31:28 -0600709 if (g_chip.is_open) {
710 release_locality(&g_chip, g_chip.locality, 1);
711 g_chip.is_open = 0;
712 }
Simon Glass4cd7b782015-08-22 18:31:22 -0600713
714 return 0;
715}
716
717int tis_sendrecv(const uint8_t *sendbuf, size_t sbuf_size,
718 uint8_t *recvbuf, size_t *rbuf_len)
719{
720 int len;
721 uint8_t buf[4096];
722
Simon Glass605152a2015-08-22 18:31:26 -0600723 if (!g_chip.inited)
Simon Glass4cd7b782015-08-22 18:31:22 -0600724 return -1;
725
726 if (sizeof(buf) < sbuf_size)
727 return -1;
728
729 memcpy(buf, sendbuf, sbuf_size);
730
731 len = tpm_transmit(buf, sbuf_size);
732
733 if (len < 10) {
734 *rbuf_len = 0;
735 return -1;
736 }
737
738 memcpy(recvbuf, buf, len);
739 *rbuf_len = len;
740
741 return 0;
742}