drivers/spi: Pass flash parameters from coreboot to payload

A payload may want to run erase operations on SPI NOR flash without
re-probing the device to get its properties. This patch passes up
three properties of flash to achieve that:
- The size of the flash device
- The sector size, i.e., the granularity of erase
- The command used for erase
The patch sends the parameters through coreboot and then libpayload.
The patch also includes a minor refactoring of the flash erase code.
Parameters are sent up for just one flash device. If multiple SPI
flash devices are probed, the second one will "win" and its
parameters will be sent up to the payload.

TEST=Observed parameters to be passed up to depthcharge through
libpayload and be used to correctly initialize flash and do an erase.
TEST=Winbond and Gigadevices spi flash drivers compile with the changes;
others don't, for seemingly unrelated reasons.
BRANCH=none
BUG=chromium:446377

Change-Id: Ib8be86494b5a3d1cfe1d23d3492e3b5cba5f99c6
Signed-off-by: Patrick Georgi <pgeorgi@chromium.org>
Original-Commit-Id: 988c8c68bbfcdfa69d497ea5f806567bc80f8126
Original-Change-Id: Ie2b3a7f5b6e016d212f4f9bac3fabd80daf2ce72
Original-Signed-off-by: Dan Ehrenberg <dehrenberg@chromium.org>
Original-Reviewed-on: https://chromium-review.googlesource.com/239570
Original-Reviewed-by: Vadim Bendebury <vbendeb@chromium.org>
Reviewed-on: http://review.coreboot.org/9726
Tested-by: build bot (Jenkins)
Reviewed-by: Stefan Reinauer <stefan.reinauer@coreboot.org>
This commit is contained in:
Dan Ehrenberg 2015-01-08 10:29:19 -08:00 committed by Patrick Georgi
parent f9b49e8782
commit a5aac76ac6
15 changed files with 84 additions and 66 deletions

View File

@ -145,11 +145,6 @@ out:
return ret; return ret;
} }
static int adesto_erase(struct spi_flash *flash, u32 offset, size_t len)
{
return spi_flash_cmd_erase(flash, CMD_AT25DF_SE, offset, len);
}
struct spi_flash *spi_flash_probe_adesto(struct spi_slave *spi, u8 *idcode) struct spi_flash *spi_flash_probe_adesto(struct spi_slave *spi, u8 *idcode)
{ {
const struct adesto_spi_flash_params *params; const struct adesto_spi_flash_params *params;
@ -183,7 +178,7 @@ struct spi_flash *spi_flash_probe_adesto(struct spi_slave *spi, u8 *idcode)
page_size = 1 << params->l2_page_size; page_size = 1 << params->l2_page_size;
stm->flash.write = adesto_write; stm->flash.write = adesto_write;
stm->flash.erase = adesto_erase; stm->flash.erase = spi_flash_cmd_erase;
#if CONFIG_SPI_FLASH_NO_FAST_READ #if CONFIG_SPI_FLASH_NO_FAST_READ
stm->flash.read = spi_flash_cmd_read_slow; stm->flash.read = spi_flash_cmd_read_slow;
#else #else
@ -194,6 +189,7 @@ struct spi_flash *spi_flash_probe_adesto(struct spi_slave *spi, u8 *idcode)
stm->flash.size = page_size * params->pages_per_sector stm->flash.size = page_size * params->pages_per_sector
* params->sectors_per_block * params->sectors_per_block
* params->nr_blocks; * params->nr_blocks;
stm->flash.erase_cmd = CMD_AT25DF_SE;
return &stm->flash; return &stm->flash;
} }

View File

@ -127,11 +127,6 @@ out:
return ret; return ret;
} }
static int amic_erase(struct spi_flash *flash, u32 offset, size_t len)
{
return spi_flash_cmd_erase(flash, CMD_A25_SE, offset, len);
}
struct spi_flash *spi_flash_probe_amic(struct spi_slave *spi, u8 *idcode) struct spi_flash *spi_flash_probe_amic(struct spi_slave *spi, u8 *idcode)
{ {
const struct amic_spi_flash_params *params; const struct amic_spi_flash_params *params;
@ -165,7 +160,7 @@ struct spi_flash *spi_flash_probe_amic(struct spi_slave *spi, u8 *idcode)
page_size = 1 << params->l2_page_size; page_size = 1 << params->l2_page_size;
amic->flash.write = amic_write; amic->flash.write = amic_write;
amic->flash.erase = amic_erase; amic->flash.erase = spi_flash_cmd_erase;
#if CONFIG_SPI_FLASH_NO_FAST_READ #if CONFIG_SPI_FLASH_NO_FAST_READ
amic->flash.read = spi_flash_cmd_read_slow; amic->flash.read = spi_flash_cmd_read_slow;
#else #else
@ -176,6 +171,7 @@ struct spi_flash *spi_flash_probe_amic(struct spi_slave *spi, u8 *idcode)
amic->flash.size = page_size * params->pages_per_sector amic->flash.size = page_size * params->pages_per_sector
* params->sectors_per_block * params->sectors_per_block
* params->nr_blocks; * params->nr_blocks;
amic->flash.erase_cmd = CMD_A25_SE;
return &amic->flash; return &amic->flash;
} }

View File

@ -172,11 +172,6 @@ out:
return ret; return ret;
} }
static int atmel_erase(struct spi_flash *flash, u32 offset, size_t len)
{
return spi_flash_cmd_erase(flash, CMD_AT25_SE, offset, len);
}
struct spi_flash *spi_flash_probe_atmel(struct spi_slave *spi, u8 *idcode) struct spi_flash *spi_flash_probe_atmel(struct spi_slave *spi, u8 *idcode)
{ {
const struct atmel_spi_flash_params *params; const struct atmel_spi_flash_params *params;
@ -210,7 +205,7 @@ struct spi_flash *spi_flash_probe_atmel(struct spi_slave *spi, u8 *idcode)
page_size = 1 << params->l2_page_size; page_size = 1 << params->l2_page_size;
stm->flash.write = atmel_write; stm->flash.write = atmel_write;
stm->flash.erase = atmel_erase; stm->flash.erase = spi_flash_cmd_erase;
#if CONFIG_SPI_FLASH_NO_FAST_READ #if CONFIG_SPI_FLASH_NO_FAST_READ
stm->flash.read = spi_flash_cmd_read_slow; stm->flash.read = spi_flash_cmd_read_slow;
#else #else
@ -221,6 +216,7 @@ struct spi_flash *spi_flash_probe_atmel(struct spi_slave *spi, u8 *idcode)
stm->flash.size = page_size * params->pages_per_sector stm->flash.size = page_size * params->pages_per_sector
* params->sectors_per_block * params->sectors_per_block
* params->nr_blocks; * params->nr_blocks;
stm->flash.erase_cmd = CMD_AT25_SE;
return &stm->flash; return &stm->flash;
} }

View File

@ -138,11 +138,6 @@ out:
return ret; return ret;
} }
static int eon_erase(struct spi_flash *flash, u32 offset, size_t len)
{
return spi_flash_cmd_erase(flash, CMD_EN25_SE, offset, len);
}
struct spi_flash *spi_flash_probe_eon(struct spi_slave *spi, u8 *idcode) struct spi_flash *spi_flash_probe_eon(struct spi_slave *spi, u8 *idcode)
{ {
const struct eon_spi_flash_params *params; const struct eon_spi_flash_params *params;
@ -172,11 +167,12 @@ struct spi_flash *spi_flash_probe_eon(struct spi_slave *spi, u8 *idcode)
eon->flash.name = params->name; eon->flash.name = params->name;
eon->flash.write = eon_write; eon->flash.write = eon_write;
eon->flash.erase = eon_erase; eon->flash.erase = spi_flash_cmd_erase;
eon->flash.read = spi_flash_cmd_read_fast; eon->flash.read = spi_flash_cmd_read_fast;
eon->flash.sector_size = params->page_size * params->pages_per_sector; eon->flash.sector_size = params->page_size * params->pages_per_sector;
eon->flash.size = params->page_size * params->pages_per_sector eon->flash.size = params->page_size * params->pages_per_sector
* params->nr_sectors; * params->nr_sectors;
eon->flash.erase_cmd = CMD_EN25_SE;
return &eon->flash; return &eon->flash;
} }

View File

@ -192,11 +192,6 @@ out:
return ret; return ret;
} }
static int gigadevice_erase(struct spi_flash *flash, u32 offset, size_t len)
{
return spi_flash_cmd_erase(flash, CMD_GD25_SE, offset, len);
}
static struct gigadevice_spi_flash stm; static struct gigadevice_spi_flash stm;
struct spi_flash *spi_flash_probe_gigadevice(struct spi_slave *spi, u8 *idcode) struct spi_flash *spi_flash_probe_gigadevice(struct spi_slave *spi, u8 *idcode)
@ -226,7 +221,7 @@ struct spi_flash *spi_flash_probe_gigadevice(struct spi_slave *spi, u8 *idcode)
page_size = 1 << params->l2_page_size; page_size = 1 << params->l2_page_size;
stm.flash.write = gigadevice_write; stm.flash.write = gigadevice_write;
stm.flash.erase = gigadevice_erase; stm.flash.erase = spi_flash_cmd_erase;
#if CONFIG_SPI_FLASH_NO_FAST_READ #if CONFIG_SPI_FLASH_NO_FAST_READ
stm.flash.read = spi_flash_cmd_read_slow; stm.flash.read = spi_flash_cmd_read_slow;
#else #else
@ -237,6 +232,7 @@ struct spi_flash *spi_flash_probe_gigadevice(struct spi_slave *spi, u8 *idcode)
stm.flash.size = page_size * params->pages_per_sector stm.flash.size = page_size * params->pages_per_sector
* params->sectors_per_block * params->sectors_per_block
* params->nr_blocks; * params->nr_blocks;
stm.flash.erase_cmd = CMD_GD25_SE;
return &stm.flash; return &stm.flash;
} }

View File

@ -212,11 +212,6 @@ static int macronix_write(struct spi_flash *flash,
return ret; return ret;
} }
static int macronix_erase(struct spi_flash *flash, u32 offset, size_t len)
{
return spi_flash_cmd_erase(flash, CMD_MX25XX_SE, offset, len);
}
struct spi_flash *spi_flash_probe_macronix(struct spi_slave *spi, u8 *idcode) struct spi_flash *spi_flash_probe_macronix(struct spi_slave *spi, u8 *idcode)
{ {
const struct macronix_spi_flash_params *params; const struct macronix_spi_flash_params *params;
@ -246,7 +241,7 @@ struct spi_flash *spi_flash_probe_macronix(struct spi_slave *spi, u8 *idcode)
mcx->flash.name = params->name; mcx->flash.name = params->name;
mcx->flash.write = macronix_write; mcx->flash.write = macronix_write;
mcx->flash.erase = macronix_erase; mcx->flash.erase = spi_flash_cmd_erase;
#if CONFIG_SPI_FLASH_NO_FAST_READ #if CONFIG_SPI_FLASH_NO_FAST_READ
mcx->flash.read = spi_flash_cmd_read_slow; mcx->flash.read = spi_flash_cmd_read_slow;
#else #else
@ -255,6 +250,7 @@ struct spi_flash *spi_flash_probe_macronix(struct spi_slave *spi, u8 *idcode)
mcx->flash.sector_size = params->page_size * params->pages_per_sector; mcx->flash.sector_size = params->page_size * params->pages_per_sector;
mcx->flash.size = mcx->flash.sector_size * params->sectors_per_block * mcx->flash.size = mcx->flash.sector_size * params->sectors_per_block *
params->nr_blocks; params->nr_blocks;
mcx->flash.erase_cmd = CMD_MX25XX_SE;
return &mcx->flash; return &mcx->flash;
} }

View File

@ -267,11 +267,6 @@ static int spansion_write(struct spi_flash *flash,
return ret; return ret;
} }
static int spansion_erase(struct spi_flash *flash, u32 offset, size_t len)
{
return spi_flash_cmd_erase(flash, CMD_S25FLXX_SE, offset, len);
}
static struct spansion_spi_flash spsn_flash; static struct spansion_spi_flash spsn_flash;
struct spi_flash *spi_flash_probe_spansion(struct spi_slave *spi, u8 *idcode) struct spi_flash *spi_flash_probe_spansion(struct spi_slave *spi, u8 *idcode)
@ -300,10 +295,11 @@ struct spi_flash *spi_flash_probe_spansion(struct spi_slave *spi, u8 *idcode)
spsn->flash.name = params->name; spsn->flash.name = params->name;
spsn->flash.write = spansion_write; spsn->flash.write = spansion_write;
spsn->flash.erase = spansion_erase; spsn->flash.erase = spi_flash_cmd_erase;
spsn->flash.read = spi_flash_cmd_read_slow; spsn->flash.read = spi_flash_cmd_read_slow;
spsn->flash.sector_size = params->page_size * params->pages_per_sector; spsn->flash.sector_size = params->page_size * params->pages_per_sector;
spsn->flash.size = spsn->flash.sector_size * params->nr_sectors; spsn->flash.size = spsn->flash.sector_size * params->nr_sectors;
spsn->flash.erase_cmd = CMD_S25FLXX_SE;
return &spsn->flash; return &spsn->flash;
} }

View File

@ -7,6 +7,7 @@
* Licensed under the GPL-2 or later. * Licensed under the GPL-2 or later.
*/ */
#include <cbfs.h>
#include <cpu/x86/smm.h> #include <cpu/x86/smm.h>
#include <delay.h> #include <delay.h>
#include <stdlib.h> #include <stdlib.h>
@ -17,6 +18,8 @@
#include "spi_flash_internal.h" #include "spi_flash_internal.h"
#include <timer.h> #include <timer.h>
static struct spi_flash *spi_flash_dev = NULL;
static void spi_flash_addr(u32 addr, u8 *cmd) static void spi_flash_addr(u32 addr, u8 *cmd)
{ {
/* cmd[0] is actual command */ /* cmd[0] is actual command */
@ -186,8 +189,7 @@ int spi_flash_cmd_wait_ready(struct spi_flash *flash, unsigned long timeout)
CMD_READ_STATUS, STATUS_WIP); CMD_READ_STATUS, STATUS_WIP);
} }
int spi_flash_cmd_erase(struct spi_flash *flash, u8 erase_cmd, int spi_flash_cmd_erase(struct spi_flash *flash, u32 offset, size_t len)
u32 offset, size_t len)
{ {
u32 start, end, erase_size; u32 start, end, erase_size;
int ret; int ret;
@ -201,7 +203,7 @@ int spi_flash_cmd_erase(struct spi_flash *flash, u8 erase_cmd,
flash->spi->rw = SPI_WRITE_FLAG; flash->spi->rw = SPI_WRITE_FLAG;
cmd[0] = erase_cmd; cmd[0] = flash->erase_cmd;
start = offset; start = offset;
end = start + len; end = start + len;
@ -376,9 +378,45 @@ flash_detected:
printk(BIOS_INFO, "SF: Detected %s with page size %x, total %x\n", printk(BIOS_INFO, "SF: Detected %s with page size %x, total %x\n",
flash->name, flash->sector_size, flash->size); flash->name, flash->sector_size, flash->size);
spi_flash_dev = flash;
return flash; return flash;
err_manufacturer_probe: err_manufacturer_probe:
err_read_id: err_read_id:
return NULL; return NULL;
} }
/* Only the RAM stage will build in the lb_new_record symbol
* so only define this function if we are after that stage */
#ifdef __RAMSTAGE__
void lb_spi_flash(struct lb_header *header)
{
struct lb_spi_flash *flash;
flash = (struct lb_spi_flash *)lb_new_record(header);
flash->tag = LB_TAG_SPI_FLASH;
flash->size = sizeof(*flash);
/* Try to get the flash device if not loaded yet */
if (!spi_flash_dev) {
struct cbfs_media media;
init_default_cbfs_media(&media);
}
if (spi_flash_dev) {
flash->flash_size = spi_flash_dev->size;
flash->sector_size = spi_flash_dev->sector_size;
flash->erase_cmd = spi_flash_dev->erase_cmd;
} else {
flash->flash_size = CONFIG_ROM_SIZE;
/* Default 64k erase command should work on most flash.
* Uniform 4k erase only works on certain devices. */
flash->sector_size = 64 * KiB;
flash->erase_cmd = CMD_BLOCK_ERASE;
}
}
#endif

View File

@ -26,6 +26,8 @@
#define CMD_READ_STATUS 0x05 #define CMD_READ_STATUS 0x05
#define CMD_WRITE_ENABLE 0x06 #define CMD_WRITE_ENABLE 0x06
#define CMD_BLOCK_ERASE 0xD8
/* Common status */ /* Common status */
#define STATUS_WIP 0x01 #define STATUS_WIP 0x01
@ -56,8 +58,7 @@ int spi_flash_cmd_poll_bit(struct spi_flash *flash, unsigned long timeout,
int spi_flash_cmd_wait_ready(struct spi_flash *flash, unsigned long timeout); int spi_flash_cmd_wait_ready(struct spi_flash *flash, unsigned long timeout);
/* Erase sectors. */ /* Erase sectors. */
int spi_flash_cmd_erase(struct spi_flash *flash, u8 erase_cmd, int spi_flash_cmd_erase(struct spi_flash *flash, u32 offset, size_t len);
u32 offset, size_t len);
/* Manufacturer-specific probe functions */ /* Manufacturer-specific probe functions */
struct spi_flash *spi_flash_probe_spansion(struct spi_slave *spi, u8 *idcode); struct spi_flash *spi_flash_probe_spansion(struct spi_slave *spi, u8 *idcode);

View File

@ -207,10 +207,6 @@ done:
return ret; return ret;
} }
static int sst_erase(struct spi_flash *flash, u32 offset, size_t len)
{
return spi_flash_cmd_erase(flash, CMD_SST_SE, offset, len);
}
static int static int
sst_unlock(struct spi_flash *flash) sst_unlock(struct spi_flash *flash)
@ -262,10 +258,11 @@ spi_flash_probe_sst(struct spi_slave *spi, u8 *idcode)
stm->flash.name = params->name; stm->flash.name = params->name;
stm->flash.write = sst_write; stm->flash.write = sst_write;
stm->flash.erase = sst_erase; stm->flash.erase = spi_flash_cmd_erase;
stm->flash.read = spi_flash_cmd_read_fast; stm->flash.read = spi_flash_cmd_read_fast;
stm->flash.sector_size = SST_SECTOR_SIZE; stm->flash.sector_size = SST_SECTOR_SIZE;
stm->flash.size = stm->flash.sector_size * params->nr_sectors; stm->flash.size = stm->flash.sector_size * params->nr_sectors;
stm->flash.erase_cmd = CMD_SST_SE;
/* Flash powers up read-only, so clear BP# bits */ /* Flash powers up read-only, so clear BP# bits */
sst_unlock(&stm->flash); sst_unlock(&stm->flash);

View File

@ -221,13 +221,6 @@ out:
return ret; return ret;
} }
static int stmicro_erase(struct spi_flash *flash, u32 offset, size_t len)
{
struct stmicro_spi_flash *stm = to_stmicro_spi_flash(flash);
return spi_flash_cmd_erase(flash, stm->params->op_erase, offset, len);
}
struct spi_flash *spi_flash_probe_stmicro(struct spi_slave *spi, u8 * idcode) struct spi_flash *spi_flash_probe_stmicro(struct spi_slave *spi, u8 * idcode)
{ {
const struct stmicro_spi_flash_params *params; const struct stmicro_spi_flash_params *params;
@ -275,10 +268,11 @@ struct spi_flash *spi_flash_probe_stmicro(struct spi_slave *spi, u8 * idcode)
stm->flash.name = params->name; stm->flash.name = params->name;
stm->flash.write = stmicro_write; stm->flash.write = stmicro_write;
stm->flash.erase = stmicro_erase; stm->flash.erase = spi_flash_cmd_erase;
stm->flash.read = spi_flash_cmd_read_fast; stm->flash.read = spi_flash_cmd_read_fast;
stm->flash.sector_size = params->page_size * params->pages_per_sector; stm->flash.sector_size = params->page_size * params->pages_per_sector;
stm->flash.size = stm->flash.sector_size * params->nr_sectors; stm->flash.size = stm->flash.sector_size * params->nr_sectors;
stm->flash.erase_cmd = params->op_erase;
return &stm->flash; return &stm->flash;
} }

View File

@ -189,11 +189,6 @@ out:
return ret; return ret;
} }
static int winbond_erase(struct spi_flash *flash, u32 offset, size_t len)
{
return spi_flash_cmd_erase(flash, CMD_W25_SE, offset, len);
}
static struct winbond_spi_flash stm; static struct winbond_spi_flash stm;
struct spi_flash *spi_flash_probe_winbond(struct spi_slave *spi, u8 *idcode) struct spi_flash *spi_flash_probe_winbond(struct spi_slave *spi, u8 *idcode)
@ -222,7 +217,7 @@ struct spi_flash *spi_flash_probe_winbond(struct spi_slave *spi, u8 *idcode)
page_size = 1 << params->l2_page_size; page_size = 1 << params->l2_page_size;
stm.flash.write = winbond_write; stm.flash.write = winbond_write;
stm.flash.erase = winbond_erase; stm.flash.erase = spi_flash_cmd_erase;
#if CONFIG_SPI_FLASH_NO_FAST_READ #if CONFIG_SPI_FLASH_NO_FAST_READ
stm.flash.read = spi_flash_cmd_read_slow; stm.flash.read = spi_flash_cmd_read_slow;
#else #else
@ -233,6 +228,7 @@ struct spi_flash *spi_flash_probe_winbond(struct spi_slave *spi, u8 *idcode)
stm.flash.size = page_size * params->pages_per_sector stm.flash.size = page_size * params->pages_per_sector
* params->sectors_per_block * params->sectors_per_block
* params->nr_blocks; * params->nr_blocks;
stm.flash.erase_cmd = CMD_W25_SE;
return &stm.flash; return &stm.flash;
} }

View File

@ -286,6 +286,15 @@ struct lb_ram_code {
uint32_t ram_code; uint32_t ram_code;
}; };
#define LB_TAG_SPI_FLASH 0x0029
struct lb_spi_flash {
uint32_t tag;
uint32_t size;
uint32_t flash_size;
uint32_t sector_size;
uint32_t erase_cmd;
};
/* The following structures are for the cmos definitions table */ /* The following structures are for the cmos definitions table */
#define LB_TAG_CMOS_OPTION_TABLE 200 #define LB_TAG_CMOS_OPTION_TABLE 200
/* cmos header record */ /* cmos header record */

View File

@ -27,6 +27,7 @@
#include <stddef.h> #include <stddef.h>
#include <console/console.h> #include <console/console.h>
#include <spi-generic.h> #include <spi-generic.h>
#include <boot/coreboot_tables.h>
struct spi_flash { struct spi_flash {
struct spi_slave *spi; struct spi_slave *spi;
@ -37,6 +38,8 @@ struct spi_flash {
u32 sector_size; u32 sector_size;
u8 erase_cmd;
int (*read)(struct spi_flash *flash, u32 offset, int (*read)(struct spi_flash *flash, u32 offset,
size_t len, void *buf); size_t len, void *buf);
int (*write)(struct spi_flash *flash, u32 offset, int (*write)(struct spi_flash *flash, u32 offset,
@ -47,4 +50,6 @@ struct spi_flash {
struct spi_flash *spi_flash_probe(unsigned int bus, unsigned int cs); struct spi_flash *spi_flash_probe(unsigned int bus, unsigned int cs);
void lb_spi_flash(struct lb_header *header);
#endif /* _SPI_FLASH_H_ */ #endif /* _SPI_FLASH_H_ */

View File

@ -32,6 +32,7 @@
#include <cbfs.h> #include <cbfs.h>
#include <cbmem.h> #include <cbmem.h>
#include <bootmem.h> #include <bootmem.h>
#include <spi_flash.h>
#if CONFIG_CHROMEOS #if CONFIG_CHROMEOS
#if CONFIG_HAVE_ACPI_TABLES #if CONFIG_HAVE_ACPI_TABLES
#include <arch/acpi.h> #include <arch/acpi.h>
@ -468,6 +469,11 @@ unsigned long write_coreboot_table(
/* Add RAM config if available */ /* Add RAM config if available */
lb_ram_code(head); lb_ram_code(head);
#if IS_ENABLED(CONFIG_SPI_FLASH)
/* Add SPI flash description if available */
lb_spi_flash(head);
#endif
add_cbmem_pointers(head); add_cbmem_pointers(head);
/* Add board-specific table entries, if any. */ /* Add board-specific table entries, if any. */