drivers/spi/tpm: Poll TPM_VALID bit until valid

In case the TPM is doing a long crypto operation the initial probe
could be very delayed.  Rather than end up in recovery make the delay
long enough to accommodate the (current) long crypto times. This would
add a maximum of 30 seconds to boot time.

Mirroring changes done on i2c side in CL:756918

BUG=b:65867313, b:68729265
BRANCH=None
TEST=Make sure fizz boots up

Change-Id: Ie944bfb6fe33d6e9ee794439165716ab624be491
Signed-off-by: Shelley Chen <shchen@chromium.org>
Reviewed-on: https://review.coreboot.org/22370
Tested-by: build bot (Jenkins) <no-reply@coreboot.org>
Reviewed-by: Aaron Durbin <adurbin@chromium.org>
This commit is contained in:
Shelley Chen 2017-11-07 14:24:19 -08:00 committed by Shelley Chen
parent 067031e0e1
commit 85eb0311d2
1 changed files with 10 additions and 0 deletions

View File

@ -37,6 +37,8 @@
#define TPM_RID_REG (TPM_LOCALITY_0_SPI_BASE + 0xf04) #define TPM_RID_REG (TPM_LOCALITY_0_SPI_BASE + 0xf04)
#define TPM_FW_VER (TPM_LOCALITY_0_SPI_BASE + 0xf90) #define TPM_FW_VER (TPM_LOCALITY_0_SPI_BASE + 0xf90)
#define CR50_TIMEOUT_INIT_MS 30000 /* Very long timeout for TPM init */
/* SPI slave structure for TPM device. */ /* SPI slave structure for TPM device. */
static struct spi_slave g_spi_slave CAR_GLOBAL; static struct spi_slave g_spi_slave CAR_GLOBAL;
@ -342,6 +344,7 @@ static void tpm2_write_access_reg(uint8_t cmd)
static int tpm2_claim_locality(void) static int tpm2_claim_locality(void)
{ {
uint8_t access; uint8_t access;
struct stopwatch sw;
access = tpm2_read_access_reg(); access = tpm2_read_access_reg();
/* /*
@ -353,6 +356,13 @@ static int tpm2_claim_locality(void)
access = tpm2_read_access_reg(); access = tpm2_read_access_reg();
} }
/*
* If cr50 is doing a long crypto operation, it can take up to
* 30 seconds to get a valid status value back
*/
stopwatch_init_msecs_expire(&sw, CR50_TIMEOUT_INIT_MS);
while (!stopwatch_expired(&sw) && access != TPM_ACCESS_VALID)
access = tpm2_read_access_reg();
if (access != TPM_ACCESS_VALID) { if (access != TPM_ACCESS_VALID) {
printk(BIOS_ERR, "Invalid reset status: %#x\n", access); printk(BIOS_ERR, "Invalid reset status: %#x\n", access);
return 0; return 0;