blob: e04e8f547baf0694910c4bde1cd0ada9441df77d [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 */
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
Simon Glasse56e20e2015-08-22 18:31:29 -0600110static int tpm_tis_i2c_write_generic(u8 addr, u8 *buffer, size_t len,
111 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/*
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 Glasse56e20e2015-08-22 18:31:29 -0600149 return tpm_tis_i2c_write_generic(addr, buffer, len, SLEEP_DURATION,
Rong Changf6267992013-04-12 10:44:57 +0000150 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 */
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 Glasse56e20e2015-08-22 18:31:29 -0600159 return tpm_tis_i2c_write_generic(addr, buffer, len, SLEEP_DURATION_LONG,
Rong Changf6267992013-04-12 10:44:57 +0000160 MAX_COUNT_LONG);
161}
162
Simon Glasse56e20e2015-08-22 18:31:29 -0600163static int tpm_tis_i2c_check_locality(struct tpm_chip *chip, int loc)
Rong Changf6267992013-04-12 10:44:57 +0000164{
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
Simon Glasse56e20e2015-08-22 18:31:29 -0600169 rc = tpm_tis_i2c_read(TPM_ACCESS(loc), &buf, 1);
Rong Changf6267992013-04-12 10:44:57 +0000170 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
Simon Glasse56e20e2015-08-22 18:31:29 -0600181static void tpm_tis_i2c_release_locality(struct tpm_chip *chip, int loc,
182 int force)
Rong Changf6267992013-04-12 10:44:57 +0000183{
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000184 const u8 mask = TPM_ACCESS_REQUEST_PENDING | TPM_ACCESS_VALID;
Rong Changf6267992013-04-12 10:44:57 +0000185 u8 buf;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000186
Simon Glasse56e20e2015-08-22 18:31:29 -0600187 if (tpm_tis_i2c_read(TPM_ACCESS(loc), &buf, 1) < 0)
Rong Changf6267992013-04-12 10:44:57 +0000188 return;
189
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000190 if (force || (buf & mask) == mask) {
Rong Changf6267992013-04-12 10:44:57 +0000191 buf = TPM_ACCESS_ACTIVE_LOCALITY;
Simon Glasse56e20e2015-08-22 18:31:29 -0600192 tpm_tis_i2c_write(TPM_ACCESS(loc), &buf, 1);
Rong Changf6267992013-04-12 10:44:57 +0000193 }
194}
195
Simon Glasse56e20e2015-08-22 18:31:29 -0600196static int tpm_tis_i2c_request_locality(struct tpm_chip *chip, int loc)
Rong Changf6267992013-04-12 10:44:57 +0000197{
198 unsigned long start, stop;
199 u8 buf = TPM_ACCESS_REQUEST_USE;
Simon Glassf90acf12015-05-04 11:30:59 -0600200 int rc;
Rong Changf6267992013-04-12 10:44:57 +0000201
Simon Glasse56e20e2015-08-22 18:31:29 -0600202 if (tpm_tis_i2c_check_locality(chip, loc) >= 0)
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000203 return loc; /* We already have the locality */
Rong Changf6267992013-04-12 10:44:57 +0000204
Simon Glasse56e20e2015-08-22 18:31:29 -0600205 rc = tpm_tis_i2c_write(TPM_ACCESS(loc), &buf, 1);
Simon Glassf90acf12015-05-04 11:30:59 -0600206 if (rc)
207 return rc;
Rong Changf6267992013-04-12 10:44:57 +0000208
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000209 /* Wait for burstcount */
Rong Changf6267992013-04-12 10:44:57 +0000210 start = get_timer(0);
Simon Glass7c735372015-08-22 18:31:24 -0600211 stop = chip->timeout_a;
Rong Changf6267992013-04-12 10:44:57 +0000212 do {
Simon Glasse56e20e2015-08-22 18:31:29 -0600213 if (tpm_tis_i2c_check_locality(chip, loc) >= 0)
Rong Changf6267992013-04-12 10:44:57 +0000214 return loc;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000215 udelay(TPM_TIMEOUT * 1000);
Rong Changf6267992013-04-12 10:44:57 +0000216 } while (get_timer(start) < stop);
217
218 return -1;
219}
220
221static u8 tpm_tis_i2c_status(struct tpm_chip *chip)
222{
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000223 /* NOTE: Since i2c read may fail, return 0 in this case --> time-out */
Rong Changf6267992013-04-12 10:44:57 +0000224 u8 buf;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000225
Simon Glasse56e20e2015-08-22 18:31:29 -0600226 if (tpm_tis_i2c_read(TPM_STS(chip->locality), &buf, 1) < 0)
Rong Changf6267992013-04-12 10:44:57 +0000227 return 0;
228 else
229 return buf;
230}
231
232static void tpm_tis_i2c_ready(struct tpm_chip *chip)
233{
Simon Glassf90acf12015-05-04 11:30:59 -0600234 int rc;
235
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000236 /* This causes the current command to be aborted */
Rong Changf6267992013-04-12 10:44:57 +0000237 u8 buf = TPM_STS_COMMAND_READY;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000238
Simon Glassf90acf12015-05-04 11:30:59 -0600239 debug("%s\n", __func__);
Simon Glasse56e20e2015-08-22 18:31:29 -0600240 rc = tpm_tis_i2c_write_long(TPM_STS(chip->locality), &buf, 1);
Simon Glassf90acf12015-05-04 11:30:59 -0600241 if (rc)
242 debug("%s: rc=%d\n", __func__, rc);
Rong Changf6267992013-04-12 10:44:57 +0000243}
244
Simon Glasse56e20e2015-08-22 18:31:29 -0600245static ssize_t tpm_tis_i2c_get_burstcount(struct tpm_chip *chip)
Rong Changf6267992013-04-12 10:44:57 +0000246{
247 unsigned long start, stop;
248 ssize_t burstcnt;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000249 u8 addr, buf[3];
Rong Changf6267992013-04-12 10:44:57 +0000250
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000251 /* Wait for burstcount */
252 /* XXX: Which timeout value? Spec has 2 answers (c & d) */
Rong Changf6267992013-04-12 10:44:57 +0000253 start = get_timer(0);
Simon Glass7c735372015-08-22 18:31:24 -0600254 stop = chip->timeout_d;
Rong Changf6267992013-04-12 10:44:57 +0000255 do {
256 /* Note: STS is little endian */
Simon Glass7c735372015-08-22 18:31:24 -0600257 addr = TPM_STS(chip->locality) + 1;
Simon Glasse56e20e2015-08-22 18:31:29 -0600258 if (tpm_tis_i2c_read(addr, buf, 3) < 0)
Rong Changf6267992013-04-12 10:44:57 +0000259 burstcnt = 0;
260 else
261 burstcnt = (buf[2] << 16) + (buf[1] << 8) + buf[0];
262
263 if (burstcnt)
264 return burstcnt;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000265 udelay(TPM_TIMEOUT * 1000);
Rong Changf6267992013-04-12 10:44:57 +0000266 } while (get_timer(start) < stop);
267
268 return -EBUSY;
269}
270
Simon Glasse56e20e2015-08-22 18:31:29 -0600271static int tpm_tis_i2c_wait_for_stat(struct tpm_chip *chip, u8 mask,
272 unsigned long timeout, int *status)
Rong Changf6267992013-04-12 10:44:57 +0000273{
274 unsigned long start, stop;
275
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000276 /* Check current status */
Rong Changf6267992013-04-12 10:44:57 +0000277 *status = tpm_tis_i2c_status(chip);
278 if ((*status & mask) == mask)
279 return 0;
280
281 start = get_timer(0);
282 stop = timeout;
283 do {
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000284 udelay(TPM_TIMEOUT * 1000);
Rong Changf6267992013-04-12 10:44:57 +0000285 *status = tpm_tis_i2c_status(chip);
286 if ((*status & mask) == mask)
287 return 0;
Rong Changf6267992013-04-12 10:44:57 +0000288 } while (get_timer(start) < stop);
289
290 return -ETIME;
291}
292
Simon Glasse56e20e2015-08-22 18:31:29 -0600293static int tpm_tis_i2c_recv_data(struct tpm_chip *chip, u8 *buf, size_t count)
Rong Changf6267992013-04-12 10:44:57 +0000294{
295 size_t size = 0;
296 ssize_t burstcnt;
297 int rc;
298
299 while (size < count) {
Simon Glasse56e20e2015-08-22 18:31:29 -0600300 burstcnt = tpm_tis_i2c_get_burstcount(chip);
Rong Changf6267992013-04-12 10:44:57 +0000301
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000302 /* burstcount < 0 -> tpm is busy */
Rong Changf6267992013-04-12 10:44:57 +0000303 if (burstcnt < 0)
304 return burstcnt;
305
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000306 /* Limit received data to max left */
Rong Changf6267992013-04-12 10:44:57 +0000307 if (burstcnt > (count - size))
308 burstcnt = count - size;
309
Simon Glasse56e20e2015-08-22 18:31:29 -0600310 rc = tpm_tis_i2c_read(TPM_DATA_FIFO(chip->locality),
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000311 &(buf[size]), burstcnt);
Rong Changf6267992013-04-12 10:44:57 +0000312 if (rc == 0)
313 size += burstcnt;
314 }
315
316 return size;
317}
318
319static int tpm_tis_i2c_recv(struct tpm_chip *chip, u8 *buf, size_t count)
320{
321 int size = 0;
322 int expected, status;
323
324 if (count < TPM_HEADER_SIZE) {
325 size = -EIO;
326 goto out;
327 }
328
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000329 /* Read first 10 bytes, including tag, paramsize, and result */
Simon Glasse56e20e2015-08-22 18:31:29 -0600330 size = tpm_tis_i2c_recv_data(chip, buf, TPM_HEADER_SIZE);
Rong Changf6267992013-04-12 10:44:57 +0000331 if (size < TPM_HEADER_SIZE) {
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000332 error("Unable to read header\n");
Rong Changf6267992013-04-12 10:44:57 +0000333 goto out;
334 }
335
336 expected = get_unaligned_be32(buf + TPM_RSP_SIZE_BYTE);
337 if ((size_t)expected > count) {
Simon Glassf90acf12015-05-04 11:30:59 -0600338 error("Error size=%x, expected=%x, count=%x\n", size, expected,
339 count);
Rong Changf6267992013-04-12 10:44:57 +0000340 size = -EIO;
341 goto out;
342 }
343
Simon Glasse56e20e2015-08-22 18:31:29 -0600344 size += tpm_tis_i2c_recv_data(chip, &buf[TPM_HEADER_SIZE],
345 expected - TPM_HEADER_SIZE);
Rong Changf6267992013-04-12 10:44:57 +0000346 if (size < expected) {
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000347 error("Unable to read remainder of result\n");
Rong Changf6267992013-04-12 10:44:57 +0000348 size = -ETIME;
349 goto out;
350 }
351
Simon Glasse56e20e2015-08-22 18:31:29 -0600352 tpm_tis_i2c_wait_for_stat(chip, TPM_STS_VALID, chip->timeout_c,
353 &status);
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000354 if (status & TPM_STS_DATA_AVAIL) { /* Retry? */
355 error("Error left over data\n");
Rong Changf6267992013-04-12 10:44:57 +0000356 size = -EIO;
357 goto out;
358 }
359
360out:
361 tpm_tis_i2c_ready(chip);
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000362 /*
363 * The TPM needs some time to clean up here,
Rong Changf6267992013-04-12 10:44:57 +0000364 * so we sleep rather than keeping the bus busy
365 */
366 udelay(2000);
Simon Glasse56e20e2015-08-22 18:31:29 -0600367 tpm_tis_i2c_release_locality(chip, chip->locality, 0);
Rong Changf6267992013-04-12 10:44:57 +0000368
369 return size;
370}
371
372static int tpm_tis_i2c_send(struct tpm_chip *chip, u8 *buf, size_t len)
373{
374 int rc, status;
Simon Glassf90acf12015-05-04 11:30:59 -0600375 size_t burstcnt;
Rong Changf6267992013-04-12 10:44:57 +0000376 size_t count = 0;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000377 int retry = 0;
Rong Changf6267992013-04-12 10:44:57 +0000378 u8 sts = TPM_STS_GO;
379
Simon Glassf90acf12015-05-04 11:30:59 -0600380 debug("%s: len=%d\n", __func__, len);
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000381 if (len > TPM_DEV_BUFSIZE)
382 return -E2BIG; /* Command is too long for our tpm, sorry */
Rong Changf6267992013-04-12 10:44:57 +0000383
Simon Glasse56e20e2015-08-22 18:31:29 -0600384 if (tpm_tis_i2c_request_locality(chip, 0) < 0)
Rong Changf6267992013-04-12 10:44:57 +0000385 return -EBUSY;
386
387 status = tpm_tis_i2c_status(chip);
388 if ((status & TPM_STS_COMMAND_READY) == 0) {
389 tpm_tis_i2c_ready(chip);
Simon Glasse56e20e2015-08-22 18:31:29 -0600390 if (tpm_tis_i2c_wait_for_stat(chip, TPM_STS_COMMAND_READY,
391 chip->timeout_b, &status) < 0) {
Rong Changf6267992013-04-12 10:44:57 +0000392 rc = -ETIME;
393 goto out_err;
394 }
395 }
396
Simon Glasse56e20e2015-08-22 18:31:29 -0600397 burstcnt = tpm_tis_i2c_get_burstcount(chip);
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000398
399 /* burstcount < 0 -> tpm is busy */
400 if (burstcnt < 0)
401 return burstcnt;
402
Simon Glassf90acf12015-05-04 11:30:59 -0600403 while (count < len) {
404 udelay(300);
405 if (burstcnt > len - count)
406 burstcnt = len - count;
Rong Changf6267992013-04-12 10:44:57 +0000407
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000408#ifdef CONFIG_TPM_TIS_I2C_BURST_LIMITATION
409 if (retry && burstcnt > CONFIG_TPM_TIS_I2C_BURST_LIMITATION)
410 burstcnt = CONFIG_TPM_TIS_I2C_BURST_LIMITATION;
411#endif /* CONFIG_TPM_TIS_I2C_BURST_LIMITATION */
Rong Changf6267992013-04-12 10:44:57 +0000412
Simon Glasse56e20e2015-08-22 18:31:29 -0600413 rc = tpm_tis_i2c_write(TPM_DATA_FIFO(chip->locality),
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000414 &(buf[count]), burstcnt);
Rong Changf6267992013-04-12 10:44:57 +0000415 if (rc == 0)
416 count += burstcnt;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000417 else {
Simon Glassf90acf12015-05-04 11:30:59 -0600418 debug("%s: error\n", __func__);
419 if (retry++ > 10) {
420 rc = -EIO;
421 goto out_err;
422 }
Simon Glasse56e20e2015-08-22 18:31:29 -0600423 rc = tpm_tis_i2c_wait_for_stat(chip, TPM_STS_VALID,
424 chip->timeout_c,
425 &status);
Simon Glassf90acf12015-05-04 11:30:59 -0600426 if (rc)
427 goto out_err;
Rong Changf6267992013-04-12 10:44:57 +0000428
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000429 if ((status & TPM_STS_DATA_EXPECT) == 0) {
430 rc = -EIO;
431 goto out_err;
432 }
Rong Changf6267992013-04-12 10:44:57 +0000433 }
434 }
435
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000436 /* Go and do it */
Simon Glasse56e20e2015-08-22 18:31:29 -0600437 tpm_tis_i2c_write(TPM_STS(chip->locality), &sts, 1);
Simon Glassf90acf12015-05-04 11:30:59 -0600438 debug("done\n");
Rong Changf6267992013-04-12 10:44:57 +0000439
440 return len;
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000441
Rong Changf6267992013-04-12 10:44:57 +0000442out_err:
Simon Glassf90acf12015-05-04 11:30:59 -0600443 debug("%s: out_err\n", __func__);
Rong Changf6267992013-04-12 10:44:57 +0000444 tpm_tis_i2c_ready(chip);
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000445 /*
446 * The TPM needs some time to clean up here,
Rong Changf6267992013-04-12 10:44:57 +0000447 * so we sleep rather than keeping the bus busy
448 */
449 udelay(2000);
Simon Glasse56e20e2015-08-22 18:31:29 -0600450 tpm_tis_i2c_release_locality(chip, chip->locality, 0);
Rong Changf6267992013-04-12 10:44:57 +0000451
452 return rc;
453}
454
Simon Glass13932b02015-08-22 18:31:25 -0600455static enum i2c_chip_type tpm_tis_i2c_chip_type(void)
Vincent Palatinec34fa52013-04-12 11:04:36 +0000456{
Masahiro Yamada0f925822015-08-12 07:31:55 +0900457#if CONFIG_IS_ENABLED(OF_CONTROL)
Vincent Palatinec34fa52013-04-12 11:04:36 +0000458 const void *blob = gd->fdt_blob;
459
460 if (fdtdec_next_compatible(blob, 0, COMPAT_INFINEON_SLB9645_TPM) >= 0)
461 return SLB9645;
462
463 if (fdtdec_next_compatible(blob, 0, COMPAT_INFINEON_SLB9635_TPM) >= 0)
464 return SLB9635;
465#endif
466 return UNKNOWN;
467}
468
Simon Glass13932b02015-08-22 18:31:25 -0600469static int tpm_tis_i2c_init(struct udevice *dev)
Rong Changf6267992013-04-12 10:44:57 +0000470{
Simon Glass7c735372015-08-22 18:31:24 -0600471 struct tpm_chip *chip = &g_chip;
Rong Changf6267992013-04-12 10:44:57 +0000472 u32 vendor;
Vincent Palatinec34fa52013-04-12 11:04:36 +0000473 u32 expected_did_vid;
Rong Changf6267992013-04-12 10:44:57 +0000474
Simon Glass13932b02015-08-22 18:31:25 -0600475 g_chip.dev = dev;
476 g_chip.chip_type = tpm_tis_i2c_chip_type();
Simon Glass7c735372015-08-22 18:31:24 -0600477 chip->is_open = 1;
Rong Changf6267992013-04-12 10:44:57 +0000478
479 /* Disable interrupts (not supported) */
Simon Glass7c735372015-08-22 18:31:24 -0600480 chip->irq = 0;
Rong Changf6267992013-04-12 10:44:57 +0000481
Simon Glasse56e20e2015-08-22 18:31:29 -0600482 /* Default timeouts - these could move to the device tree */
Simon Glass7c735372015-08-22 18:31:24 -0600483 chip->timeout_a = TIS_SHORT_TIMEOUT;
484 chip->timeout_b = TIS_LONG_TIMEOUT;
485 chip->timeout_c = TIS_SHORT_TIMEOUT;
486 chip->timeout_d = TIS_SHORT_TIMEOUT;
487 chip->req_complete_mask = TPM_STS_DATA_AVAIL | TPM_STS_VALID;
488 chip->req_complete_val = TPM_STS_DATA_AVAIL | TPM_STS_VALID;
489 chip->req_canceled = TPM_STS_COMMAND_READY;
Rong Changf6267992013-04-12 10:44:57 +0000490
Simon Glasse56e20e2015-08-22 18:31:29 -0600491 if (tpm_tis_i2c_request_locality(chip, 0) < 0)
Simon Glassf90acf12015-05-04 11:30:59 -0600492 return -ENODEV;
Rong Changf6267992013-04-12 10:44:57 +0000493
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000494 /* Read four bytes from DID_VID register */
Simon Glasse56e20e2015-08-22 18:31:29 -0600495 if (tpm_tis_i2c_read(TPM_DID_VID(0), (uchar *)&vendor, 4) < 0) {
496 tpm_tis_i2c_release_locality(chip, 0, 1);
Simon Glassf90acf12015-05-04 11:30:59 -0600497 return -EIO;
Rong Changf6267992013-04-12 10:44:57 +0000498 }
499
Simon Glass13932b02015-08-22 18:31:25 -0600500 if (g_chip.chip_type == SLB9635) {
Vincent Palatinec34fa52013-04-12 11:04:36 +0000501 vendor = be32_to_cpu(vendor);
502 expected_did_vid = TPM_TIS_I2C_DID_VID_9635;
503 } else {
504 /* device id and byte order has changed for newer i2c tpms */
505 expected_did_vid = TPM_TIS_I2C_DID_VID_9645;
506 }
Rong Changf6267992013-04-12 10:44:57 +0000507
Simon Glass13932b02015-08-22 18:31:25 -0600508 if (g_chip.chip_type != UNKNOWN && vendor != expected_did_vid) {
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000509 error("Vendor id did not match! ID was %08x\n", vendor);
Simon Glassf90acf12015-05-04 11:30:59 -0600510 return -ENODEV;
Rong Changf6267992013-04-12 10:44:57 +0000511 }
512
Tom Wai-Hong Tam1b393db2013-04-12 11:04:37 +0000513 debug("1.2 TPM (chip type %s device-id 0x%X)\n",
Simon Glass13932b02015-08-22 18:31:25 -0600514 chip_name[g_chip.chip_type], vendor >> 16);
Rong Changf6267992013-04-12 10:44:57 +0000515
516 /*
517 * A timeout query to TPM can be placed here.
518 * Standard timeout values are used so far
519 */
520
521 return 0;
Simon Glassf90acf12015-05-04 11:30:59 -0600522}
Rong Changf6267992013-04-12 10:44:57 +0000523
Simon Glass4cd7b782015-08-22 18:31:22 -0600524/* Returns max number of milliseconds to wait */
Simon Glasse56e20e2015-08-22 18:31:29 -0600525static unsigned long tpm_tis_i2c_calc_ordinal_duration(struct tpm_chip *chip,
526 u32 ordinal)
Simon Glass4cd7b782015-08-22 18:31:22 -0600527{
528 int duration_idx = TPM_UNDEFINED;
529 int duration = 0;
530
531 if (ordinal < TPM_MAX_ORDINAL) {
532 duration_idx = tpm_ordinal_duration[ordinal];
533 } else if ((ordinal & TPM_PROTECTED_ORDINAL_MASK) <
534 TPM_MAX_PROTECTED_ORDINAL) {
535 duration_idx = tpm_protected_ordinal_duration[
536 ordinal & TPM_PROTECTED_ORDINAL_MASK];
537 }
538
539 if (duration_idx != TPM_UNDEFINED)
Simon Glass7c735372015-08-22 18:31:24 -0600540 duration = chip->duration[duration_idx];
Simon Glass4cd7b782015-08-22 18:31:22 -0600541
542 if (duration <= 0)
543 return 2 * 60 * HZ; /* Two minutes timeout */
544 else
545 return duration;
546}
547
Simon Glasse56e20e2015-08-22 18:31:29 -0600548static ssize_t tpm_tis_i2c_transmit(const unsigned char *buf, size_t bufsiz)
Simon Glass4cd7b782015-08-22 18:31:22 -0600549{
550 int rc;
551 u32 count, ordinal;
552 unsigned long start, stop;
553
554 struct tpm_chip *chip = &g_chip;
555
556 /* switch endianess: big->little */
557 count = get_unaligned_be32(buf + TPM_CMD_COUNT_BYTE);
558 ordinal = get_unaligned_be32(buf + TPM_CMD_ORDINAL_BYTE);
559
560 if (count == 0) {
561 error("no data\n");
562 return -ENODATA;
563 }
564 if (count > bufsiz) {
565 error("invalid count value %x %zx\n", count, bufsiz);
566 return -E2BIG;
567 }
568
569 debug("Calling send\n");
Simon Glassb382e022015-08-22 18:31:23 -0600570 rc = tpm_tis_i2c_send(chip, (u8 *)buf, count);
Simon Glass4cd7b782015-08-22 18:31:22 -0600571 debug(" ... done calling send\n");
572 if (rc < 0) {
573 error("tpm_transmit: tpm_send: error %d\n", rc);
574 goto out;
575 }
576
Simon Glass7c735372015-08-22 18:31:24 -0600577 if (chip->irq)
Simon Glass4cd7b782015-08-22 18:31:22 -0600578 goto out_recv;
579
580 start = get_timer(0);
Simon Glasse56e20e2015-08-22 18:31:29 -0600581 stop = tpm_tis_i2c_calc_ordinal_duration(chip, ordinal);
Simon Glass4cd7b782015-08-22 18:31:22 -0600582 do {
583 debug("waiting for status... %ld %ld\n", start, stop);
Simon Glassb382e022015-08-22 18:31:23 -0600584 u8 status = tpm_tis_i2c_status(chip);
Simon Glass7c735372015-08-22 18:31:24 -0600585 if ((status & chip->req_complete_mask) ==
586 chip->req_complete_val) {
Simon Glass4cd7b782015-08-22 18:31:22 -0600587 debug("...got it;\n");
588 goto out_recv;
589 }
590
Simon Glass7c735372015-08-22 18:31:24 -0600591 if (status == chip->req_canceled) {
Simon Glass4cd7b782015-08-22 18:31:22 -0600592 error("Operation Canceled\n");
593 rc = -ECANCELED;
594 goto out;
595 }
596 udelay(TPM_TIMEOUT * 1000);
597 } while (get_timer(start) < stop);
598
Simon Glassb382e022015-08-22 18:31:23 -0600599 tpm_tis_i2c_ready(chip);
Simon Glass4cd7b782015-08-22 18:31:22 -0600600 error("Operation Timed out\n");
601 rc = -ETIME;
602 goto out;
603
604out_recv:
605 debug("out_recv: reading response...\n");
Simon Glassb382e022015-08-22 18:31:23 -0600606 rc = tpm_tis_i2c_recv(chip, (u8 *)buf, TPM_BUFSIZE);
Simon Glass4cd7b782015-08-22 18:31:22 -0600607 if (rc < 0)
608 error("tpm_transmit: tpm_recv: error %d\n", rc);
609
610out:
611 return rc;
612}
613
Simon Glass4cd7b782015-08-22 18:31:22 -0600614/**
615 * Decode TPM configuration.
616 *
617 * @param dev Returns a configuration of TPM device
618 * @return 0 if ok, -1 on error
619 */
Simon Glasse56e20e2015-08-22 18:31:29 -0600620static int tpm_tis_i2c_decode_config(struct tpm_chip *chip)
Simon Glass4cd7b782015-08-22 18:31:22 -0600621{
622 const void *blob = gd->fdt_blob;
623 struct udevice *bus;
624 int chip_addr;
625 int parent;
626 int node;
627 int ret;
628
629 node = fdtdec_next_compatible(blob, 0, COMPAT_INFINEON_SLB9635_TPM);
630 if (node < 0) {
631 node = fdtdec_next_compatible(blob, 0,
632 COMPAT_INFINEON_SLB9645_TPM);
633 }
634 if (node < 0) {
635 debug("%s: Node not found\n", __func__);
636 return -1;
637 }
638 parent = fdt_parent_offset(blob, node);
639 if (parent < 0) {
640 debug("%s: Cannot find node parent\n", __func__);
641 return -1;
642 }
643
644 /*
645 * TODO(sjg@chromium.org): Remove this when driver model supports
646 * TPMs
647 */
648 ret = uclass_get_device_by_of_offset(UCLASS_I2C, parent, &bus);
649 if (ret) {
650 debug("Cannot find bus for node '%s: ret=%d'\n",
651 fdt_get_name(blob, parent, NULL), ret);
652 return ret;
653 }
654
655 chip_addr = fdtdec_get_int(blob, node, "reg", -1);
656 if (chip_addr == -1) {
657 debug("Cannot find reg property for node '%s: ret=%d'\n",
658 fdt_get_name(blob, node, NULL), ret);
659 return ret;
660 }
661 /*
662 * TODO(sjg@chromium.org): Older TPMs will need to use the older method
Simon Glasse56e20e2015-08-22 18:31:29 -0600663 * in tpm_tis_i2c_read() so the offset length needs to be 0 here.
Simon Glass4cd7b782015-08-22 18:31:22 -0600664 */
Simon Glass605152a2015-08-22 18:31:26 -0600665 ret = i2c_get_chip(bus, chip_addr, 1, &chip->dev);
Simon Glass4cd7b782015-08-22 18:31:22 -0600666 if (ret) {
667 debug("Cannot find device for node '%s: ret=%d'\n",
668 fdt_get_name(blob, node, NULL), ret);
669 return ret;
670 }
671
672 return 0;
673}
674
Simon Glass4cd7b782015-08-22 18:31:22 -0600675int tis_init(void)
676{
Simon Glass605152a2015-08-22 18:31:26 -0600677 if (g_chip.inited)
Simon Glass4cd7b782015-08-22 18:31:22 -0600678 return 0;
679
Simon Glasse56e20e2015-08-22 18:31:29 -0600680 if (tpm_tis_i2c_decode_config(&g_chip))
Simon Glass4cd7b782015-08-22 18:31:22 -0600681 return -1;
682
683 debug("%s: done\n", __func__);
684
Simon Glass605152a2015-08-22 18:31:26 -0600685 g_chip.inited = 1;
Simon Glass4cd7b782015-08-22 18:31:22 -0600686
687 return 0;
688}
689
690int tis_open(void)
691{
692 int rc;
693
Simon Glass605152a2015-08-22 18:31:26 -0600694 if (!g_chip.inited)
Simon Glass4cd7b782015-08-22 18:31:22 -0600695 return -1;
696
Simon Glassa53b79a2015-08-22 18:31:28 -0600697 debug("%s: start\n", __func__);
698 if (g_chip.is_open)
699 return -EBUSY;
700 rc = tpm_tis_i2c_init(g_chip.dev);
701 if (rc < 0)
702 g_chip.is_open = 0;
Simon Glass4cd7b782015-08-22 18:31:22 -0600703
704 return rc;
705}
706
707int tis_close(void)
708{
Simon Glass605152a2015-08-22 18:31:26 -0600709 if (!g_chip.inited)
Simon Glass4cd7b782015-08-22 18:31:22 -0600710 return -1;
711
Simon Glassa53b79a2015-08-22 18:31:28 -0600712 if (g_chip.is_open) {
Simon Glasse56e20e2015-08-22 18:31:29 -0600713 tpm_tis_i2c_release_locality(&g_chip, g_chip.locality, 1);
Simon Glassa53b79a2015-08-22 18:31:28 -0600714 g_chip.is_open = 0;
715 }
Simon Glass4cd7b782015-08-22 18:31:22 -0600716
717 return 0;
718}
719
720int tis_sendrecv(const uint8_t *sendbuf, size_t sbuf_size,
721 uint8_t *recvbuf, size_t *rbuf_len)
722{
723 int len;
724 uint8_t buf[4096];
725
Simon Glass605152a2015-08-22 18:31:26 -0600726 if (!g_chip.inited)
Simon Glass4cd7b782015-08-22 18:31:22 -0600727 return -1;
728
729 if (sizeof(buf) < sbuf_size)
730 return -1;
731
732 memcpy(buf, sendbuf, sbuf_size);
733
Simon Glasse56e20e2015-08-22 18:31:29 -0600734 len = tpm_tis_i2c_transmit(buf, sbuf_size);
Simon Glass4cd7b782015-08-22 18:31:22 -0600735
736 if (len < 10) {
737 *rbuf_len = 0;
738 return -1;
739 }
740
741 memcpy(recvbuf, buf, len);
742 *rbuf_len = len;
743
744 return 0;
745}