soc/amd/stoneyridge: use new host controller programming

The SPI controller on stoneyridge apparently has a large fifo
and an alternate method for programming the controller. The
fifo is directly accessible as well as the rx and tx pointer
in addition to the execute bit. Remove the unneeded #defines
and program the host controller with the above changes in mind.

In addition, add debug hooks to the driver so one can dump the
state of the controller when in operation.

The time it took to read 4KiB of flash in the elog driver went
from 20593 microseconds to 5693 microseconds on cdx03/kahlee.

BUG=b:65485690

Change-Id: Ie7ea9d18cef5511686700ad9b2b9fdfeb6d5685b
Signed-off-by: Aaron Durbin <adurbin@chromium.org>
Reviewed-on: https://review.coreboot.org/23493
Reviewed-by: Furquan Shaikh <furquan@google.com>
Reviewed-by: Justin TerAvest <teravest@chromium.org>
Tested-by: build bot (Jenkins) <no-reply@coreboot.org>
This commit is contained in:
Aaron Durbin 2018-01-26 16:39:04 -07:00
parent 063c00c2e0
commit 0d2d77a597
2 changed files with 65 additions and 78 deletions

View file

@ -193,8 +193,6 @@
#define SPI_FROM_HOST_PREFETCH_EN BIT(0) #define SPI_FROM_HOST_PREFETCH_EN BIT(0)
/* SPI Controller */ /* SPI Controller */
#define SPI_FIFO_DEPTH 8
#define SPI_CNTRL0 0x00 #define SPI_CNTRL0 0x00
#define SPI_BUSY BIT(31) #define SPI_BUSY BIT(31)
#define SPI_READ_MODE_MASK (BIT(30) | BIT(29) | BIT(18)) #define SPI_READ_MODE_MASK (BIT(30) | BIT(29) | BIT(18))
@ -209,13 +207,21 @@
#define SPI_FIFO_PTR_CLR BIT(20) #define SPI_FIFO_PTR_CLR BIT(20)
#define SPI_ARB_ENABLE BIT(19) #define SPI_ARB_ENABLE BIT(19)
#define EXEC_OPCODE BIT(16) #define EXEC_OPCODE BIT(16)
#define SPI_REG_CNTRL01 0x01
#define SPI_REG_CNTRL02 0x02
#define SPI_FIFO_PTR_CLR02 (SPI_FIFO_PTR_CLR >> 16)
#define SPI_CNTRL1 0x0c #define SPI_CNTRL1 0x0c
#define SPI_FIFO_PTR_MASK (BIT(8) | BIT(9) | BIT(10)) #define SPI_CMD_CODE 0x45
#define SPI_CNTRL11 0x0d #define SPI_CMD_TRIGGER 0x47
#define SPI_FIFO_PTR_MASK11 (SPI_FIFO_PTR_MASK >> 8) #define SPI_CMD_TRIGGER_EXECUTE (BIT(7))
#define SPI_TX_BYTE_COUNT 0x48
#define SPI_RX_BYTE_COUNT 0x4B
#define SPI_STATUS 0x4c
#define SPI_DONE_BYTE_COUNT_SHIFT 0
#define SPI_DONE_BYTE_COUNT_MASK 0xff
#define SPI_FIFO_WR_PTR_SHIFT 8
#define SPI_FIFO_WR_PTR_MASK 0x7f
#define SPI_FIFO_RD_PTR_SHIFT 16
#define SPI_FIFO_RD_PTR_MASK 0x7f
#define SPI_FIFO 0x80
#define SPI_FIFO_DEPTH (0xc7 - SPI_FIFO)
#define SPI100_SPEED_CONFIG 0x22 #define SPI100_SPEED_CONFIG 0x22
/* Use SPI_SPEED_16M-SPI_SPEED_66M below for the southbridge */ /* Use SPI_SPEED_16M-SPI_SPEED_66M below for the southbridge */
@ -223,17 +229,6 @@
#define SPI_NORM_SPEED_SH 12 #define SPI_NORM_SPEED_SH 12
#define SPI_FAST_SPEED_SH 8 #define SPI_FAST_SPEED_SH 8
#define SPI_EXT_INDEX 0x1e
#define SPI_EXT_DATA 0x1f
#define SPI_DDR_CMD 0x0
#define SPI_QDR_CMD 0x1
#define SPI_DPR_CMD 0x2
#define SPI_QPR_CMD 0x3
#define SPI_MODE_BYTE 0x4
#define SPI_TX_BYTE_COUNT 0x5
#define SPI_RX_BYTE_COUNT 0x6
#define SPI_SPI_DATA_FIFO_PTR 0x7
#define SPI100_ENABLE 0x20 #define SPI100_ENABLE 0x20
#define SPI_USE_SPI100 BIT(0) #define SPI_USE_SPI100 BIT(0)

View file

@ -18,6 +18,7 @@
#include <string.h> #include <string.h>
#include <arch/io.h> #include <arch/io.h>
#include <arch/early_variables.h> #include <arch/early_variables.h>
#include <lib.h>
#include <timer.h> #include <timer.h>
#include <console/console.h> #include <console/console.h>
#include <commonlib/helpers.h> #include <commonlib/helpers.h>
@ -30,6 +31,8 @@
#include <soc/pci_devs.h> #include <soc/pci_devs.h>
#include <soc/imc.h> #include <soc/imc.h>
#define SPI_DEBUG_DRIVER IS_ENABLED(CONFIG_DEBUG_SPI_FLASH)
static uintptr_t spibar CAR_GLOBAL; static uintptr_t spibar CAR_GLOBAL;
static uintptr_t get_spibase(void) static uintptr_t get_spibase(void)
@ -62,46 +65,48 @@ static inline void spi_write32(uint8_t reg, uint32_t val)
write32((void *)(get_spibase() + reg), val); write32((void *)(get_spibase() + reg), val);
} }
static int reset_internal_fifo_pointer(void) static void dump_state(const char *str)
{
if (!SPI_DEBUG_DRIVER)
return;
printk(BIOS_DEBUG, "SPI: %s\n", str);
printk(BIOS_DEBUG, "Cntrl0: %x\n", spi_read32(SPI_CNTRL0));
printk(BIOS_DEBUG, "Status: %x\n", spi_read32(SPI_STATUS));
printk(BIOS_DEBUG, "TxByteCount: %x\n", spi_read8(SPI_TX_BYTE_COUNT));
printk(BIOS_DEBUG, "RxByteCount: %x\n", spi_read8(SPI_RX_BYTE_COUNT));
printk(BIOS_DEBUG, "CmdCode: %x\n", spi_read8(SPI_CMD_CODE));
hexdump((void *)(get_spibase() + SPI_FIFO), SPI_FIFO_DEPTH);
}
static int wait_for_ready(void)
{ {
uint8_t reg;
const uint32_t timeout_ms = 500; const uint32_t timeout_ms = 500;
struct stopwatch sw; struct stopwatch sw;
stopwatch_init_msecs_expire(&sw, timeout_ms); stopwatch_init_msecs_expire(&sw, timeout_ms);
do { do {
reg = spi_read8(SPI_REG_CNTRL02); if (!(spi_read32(SPI_STATUS) & SPI_BUSY))
reg |= SPI_FIFO_PTR_CLR02;
spi_write8(SPI_REG_CNTRL02, reg);
/* wait for ptr=0 */
if (!(spi_read8(SPI_CNTRL11) & (SPI_FIFO_PTR_MASK11)))
return 0; return 0;
} while (!stopwatch_expired(&sw)); } while (!stopwatch_expired(&sw));
printk(BIOS_DEBUG, "FCH SPI Error: FIFO reset failed\n");
return -1; return -1;
} }
static int execute_command(void) static int execute_command(void)
{ {
uint32_t reg; dump_state("Before Execute");
const uint32_t timeout_ms = 500;
struct stopwatch sw;
stopwatch_init_msecs_expire(&sw, timeout_ms); spi_write8(SPI_CMD_TRIGGER, SPI_CMD_TRIGGER_EXECUTE);
reg = spi_read32(SPI_CNTRL0); if (wait_for_ready())
reg |= EXEC_OPCODE; printk(BIOS_DEBUG,
spi_write32(SPI_CNTRL0, reg); "FCH SPI Error: Timeout executing command\n");
do { dump_state("Transaction finished");
if (!(spi_read32(SPI_CNTRL0) & (EXEC_OPCODE | SPI_BUSY)))
return 0;
} while (!stopwatch_expired(&sw));
printk(BIOS_DEBUG, "FCH SPI Error: Timeout executing command\n"); return 0;
return -1;
} }
void spi_init(void) void spi_init(void)
@ -113,45 +118,21 @@ void spi_init(void)
set_spibar(bar); set_spibar(bar);
} }
static int do_command(uint8_t cmd, const void *dout,
size_t bytesout, uint8_t **din, size_t *bytesin)
{
size_t count;
size_t max_in = MIN(*bytesin, SPI_FIFO_DEPTH);
spi_write8(SPI_EXT_INDEX, SPI_TX_BYTE_COUNT);
spi_write8(SPI_EXT_DATA, bytesout);
spi_write8(SPI_EXT_INDEX, SPI_RX_BYTE_COUNT);
spi_write8(SPI_EXT_DATA, max_in);
spi_write8(SPI_CNTRL0, cmd);
if (reset_internal_fifo_pointer())
return -1;
for (count = 0; count < bytesout; count++, dout++)
spi_write8(SPI_CNTRL1, *(uint8_t *)dout);
if (execute_command())
return -1;
if (reset_internal_fifo_pointer())
return -1;
for (count = 0; count < bytesout; count++)
spi_read8(SPI_CNTRL1); /* skip the bytes we sent */
for (count = 0; count < max_in; count++, (*din)++)
**din = spi_read8(SPI_CNTRL1);
*bytesin -= max_in;
return 0;
}
static int spi_ctrlr_xfer(const struct spi_slave *slave, const void *dout, static int spi_ctrlr_xfer(const struct spi_slave *slave, const void *dout,
size_t bytesout, void *din, size_t bytesin) size_t bytesout, void *din, size_t bytesin)
{ {
size_t count;
uint8_t cmd; uint8_t cmd;
uint8_t *bufin = din;
const uint8_t *bufout = dout;
if (SPI_DEBUG_DRIVER)
printk(BIOS_DEBUG, "%s(%lx, %lx)\n", __func__, bytesout,
bytesin);
/* First byte is cmd which cannot be sent through FIFO */ /* First byte is cmd which cannot be sent through FIFO */
cmd = *(uint8_t *)dout++; cmd = bufout[0];
bufout++;
bytesout--; bytesout--;
/* /*
@ -161,16 +142,27 @@ static int spi_ctrlr_xfer(const struct spi_slave *slave, const void *dout,
* and followed by other SPI commands, and this sequence is controlled * and followed by other SPI commands, and this sequence is controlled
* by the SPI chip driver. * by the SPI chip driver.
*/ */
if (bytesout > SPI_FIFO_DEPTH) { if (bytesout + bytesin > SPI_FIFO_DEPTH) {
printk(BIOS_DEBUG, "FCH SPI: Too much to write. Does your SPI" printk(BIOS_DEBUG, "FCH SPI: Too much to write. Does your SPI"
" chip driver use spi_crop_chunk()?\n"); " chip driver use spi_crop_chunk()?\n");
return -1; return -1;
} }
do { if (wait_for_ready())
if (do_command(cmd, dout, bytesout, (uint8_t **)&din, &bytesin)) return -1;
return -1;
} while (bytesin); spi_write8(SPI_CMD_CODE, cmd);
spi_write8(SPI_TX_BYTE_COUNT, bytesout);
spi_write8(SPI_RX_BYTE_COUNT, bytesin);
for (count = 0; count < bytesout; count++)
spi_write8(SPI_FIFO + count, bufout[count]);
if (execute_command())
return -1;
for (count = 0; count < bytesin; count++)
bufin[count] = spi_read8(SPI_FIFO + count + bytesout);
return 0; return 0;
} }