mtd: spi-nor: move all xilinx specifics into xilinx.c
Mechanically move all the xilinx functions to its own module. Then register the new flash specific ready() function. Signed-off-by: Michael Walle <michael@walle.cc> Signed-off-by: Tudor Ambarus <tudor.ambarus@microchip.com> Reviewed-by: Pratyush Yadav <p.yadav@ti.com> Link: https://lore.kernel.org/r/20220223134358.1914798-22-michael@walle.cc
This commit is contained in:
parent
9fb4beb1b0
commit
8b4195cd6d
@ -598,57 +598,6 @@ int spi_nor_write_ear(struct spi_nor *nor, u8 ear)
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* spi_nor_xread_sr() - Read the Status Register on S3AN flashes.
|
|
||||||
* @nor: pointer to 'struct spi_nor'.
|
|
||||||
* @sr: pointer to a DMA-able buffer where the value of the
|
|
||||||
* Status Register will be written.
|
|
||||||
*
|
|
||||||
* Return: 0 on success, -errno otherwise.
|
|
||||||
*/
|
|
||||||
int spi_nor_xread_sr(struct spi_nor *nor, u8 *sr)
|
|
||||||
{
|
|
||||||
int ret;
|
|
||||||
|
|
||||||
if (nor->spimem) {
|
|
||||||
struct spi_mem_op op =
|
|
||||||
SPI_MEM_OP(SPI_MEM_OP_CMD(SPINOR_OP_XRDSR, 0),
|
|
||||||
SPI_MEM_OP_NO_ADDR,
|
|
||||||
SPI_MEM_OP_NO_DUMMY,
|
|
||||||
SPI_MEM_OP_DATA_IN(1, sr, 0));
|
|
||||||
|
|
||||||
spi_nor_spimem_setup_op(nor, &op, nor->reg_proto);
|
|
||||||
|
|
||||||
ret = spi_mem_exec_op(nor->spimem, &op);
|
|
||||||
} else {
|
|
||||||
ret = spi_nor_controller_ops_read_reg(nor, SPINOR_OP_XRDSR, sr,
|
|
||||||
1);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (ret)
|
|
||||||
dev_dbg(nor->dev, "error %d reading XRDSR\n", ret);
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* spi_nor_xsr_ready() - Query the Status Register of the S3AN flash to see if
|
|
||||||
* the flash is ready for new commands.
|
|
||||||
* @nor: pointer to 'struct spi_nor'.
|
|
||||||
*
|
|
||||||
* Return: 1 if ready, 0 if not ready, -errno on errors.
|
|
||||||
*/
|
|
||||||
static int spi_nor_xsr_ready(struct spi_nor *nor)
|
|
||||||
{
|
|
||||||
int ret;
|
|
||||||
|
|
||||||
ret = spi_nor_xread_sr(nor, nor->bouncebuf);
|
|
||||||
if (ret)
|
|
||||||
return ret;
|
|
||||||
|
|
||||||
return !!(nor->bouncebuf[0] & XSR_RDY);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* spi_nor_clear_sr() - Clear the Status Register.
|
* spi_nor_clear_sr() - Clear the Status Register.
|
||||||
* @nor: pointer to 'struct spi_nor'.
|
* @nor: pointer to 'struct spi_nor'.
|
||||||
@ -798,10 +747,7 @@ static int spi_nor_ready(struct spi_nor *nor)
|
|||||||
if (nor->params->ready)
|
if (nor->params->ready)
|
||||||
return nor->params->ready(nor);
|
return nor->params->ready(nor);
|
||||||
|
|
||||||
if (nor->flags & SNOR_F_READY_XSR_RDY)
|
sr = spi_nor_sr_ready(nor);
|
||||||
sr = spi_nor_xsr_ready(nor);
|
|
||||||
else
|
|
||||||
sr = spi_nor_sr_ready(nor);
|
|
||||||
if (sr < 0)
|
if (sr < 0)
|
||||||
return sr;
|
return sr;
|
||||||
fsr = nor->flags & SNOR_F_USE_FSR ? spi_nor_fsr_ready(nor) : 1;
|
fsr = nor->flags & SNOR_F_USE_FSR ? spi_nor_fsr_ready(nor) : 1;
|
||||||
@ -2677,14 +2623,6 @@ static void spi_nor_init_flags(struct spi_nor *nor)
|
|||||||
|
|
||||||
if (flags & USE_FSR)
|
if (flags & USE_FSR)
|
||||||
nor->flags |= SNOR_F_USE_FSR;
|
nor->flags |= SNOR_F_USE_FSR;
|
||||||
|
|
||||||
/*
|
|
||||||
* Make sure the XSR_RDY flag is set before calling
|
|
||||||
* spi_nor_wait_till_ready(). Xilinx S3AN share MFR
|
|
||||||
* with Atmel SPI NOR.
|
|
||||||
*/
|
|
||||||
if (flags & SPI_NOR_XSR_RDY)
|
|
||||||
nor->flags |= SNOR_F_READY_XSR_RDY;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -15,7 +15,6 @@ enum spi_nor_option_flags {
|
|||||||
SNOR_F_USE_FSR = BIT(0),
|
SNOR_F_USE_FSR = BIT(0),
|
||||||
SNOR_F_HAS_SR_TB = BIT(1),
|
SNOR_F_HAS_SR_TB = BIT(1),
|
||||||
SNOR_F_NO_OP_CHIP_ERASE = BIT(2),
|
SNOR_F_NO_OP_CHIP_ERASE = BIT(2),
|
||||||
SNOR_F_READY_XSR_RDY = BIT(3),
|
|
||||||
SNOR_F_USE_CLSR = BIT(4),
|
SNOR_F_USE_CLSR = BIT(4),
|
||||||
SNOR_F_BROKEN_RESET = BIT(5),
|
SNOR_F_BROKEN_RESET = BIT(5),
|
||||||
SNOR_F_4B_OPCODES = BIT(6),
|
SNOR_F_4B_OPCODES = BIT(6),
|
||||||
@ -351,8 +350,6 @@ struct spi_nor_fixups {
|
|||||||
* SPI_NOR_NO_FR: can't do fastread.
|
* SPI_NOR_NO_FR: can't do fastread.
|
||||||
* USE_CLSR: use CLSR command.
|
* USE_CLSR: use CLSR command.
|
||||||
* USE_FSR: use flag status register
|
* USE_FSR: use flag status register
|
||||||
* SPI_NOR_XSR_RDY: S3AN flashes have specific opcode to read the
|
|
||||||
* status register.
|
|
||||||
*
|
*
|
||||||
* @no_sfdp_flags: flags that indicate support that can be discovered via SFDP.
|
* @no_sfdp_flags: flags that indicate support that can be discovered via SFDP.
|
||||||
* Used when SFDP tables are not defined in the flash. These
|
* Used when SFDP tables are not defined in the flash. These
|
||||||
@ -405,7 +402,6 @@ struct flash_info {
|
|||||||
#define SPI_NOR_NO_FR BIT(8)
|
#define SPI_NOR_NO_FR BIT(8)
|
||||||
#define USE_CLSR BIT(9)
|
#define USE_CLSR BIT(9)
|
||||||
#define USE_FSR BIT(10)
|
#define USE_FSR BIT(10)
|
||||||
#define SPI_NOR_XSR_RDY BIT(11)
|
|
||||||
|
|
||||||
u8 no_sfdp_flags;
|
u8 no_sfdp_flags;
|
||||||
#define SPI_NOR_SKIP_SFDP BIT(0)
|
#define SPI_NOR_SKIP_SFDP BIT(0)
|
||||||
@ -462,19 +458,6 @@ struct flash_info {
|
|||||||
.addr_width = (_addr_width), \
|
.addr_width = (_addr_width), \
|
||||||
.flags = SPI_NOR_NO_ERASE | SPI_NOR_NO_FR, \
|
.flags = SPI_NOR_NO_ERASE | SPI_NOR_NO_FR, \
|
||||||
|
|
||||||
#define S3AN_INFO(_jedec_id, _n_sectors, _page_size) \
|
|
||||||
.id = { \
|
|
||||||
((_jedec_id) >> 16) & 0xff, \
|
|
||||||
((_jedec_id) >> 8) & 0xff, \
|
|
||||||
(_jedec_id) & 0xff \
|
|
||||||
}, \
|
|
||||||
.id_len = 3, \
|
|
||||||
.sector_size = (8 * (_page_size)), \
|
|
||||||
.n_sectors = (_n_sectors), \
|
|
||||||
.page_size = (_page_size), \
|
|
||||||
.addr_width = 3, \
|
|
||||||
.flags = SPI_NOR_NO_FR | SPI_NOR_XSR_RDY,
|
|
||||||
|
|
||||||
#define OTP_INFO(_len, _n_regions, _base, _offset) \
|
#define OTP_INFO(_len, _n_regions, _base, _offset) \
|
||||||
.otp_org = { \
|
.otp_org = { \
|
||||||
.len = (_len), \
|
.len = (_len), \
|
||||||
@ -564,7 +547,6 @@ int spi_nor_write_sr(struct spi_nor *nor, const u8 *sr, size_t len);
|
|||||||
int spi_nor_write_sr_and_check(struct spi_nor *nor, u8 sr1);
|
int spi_nor_write_sr_and_check(struct spi_nor *nor, u8 sr1);
|
||||||
int spi_nor_write_16bit_cr_and_check(struct spi_nor *nor, u8 cr);
|
int spi_nor_write_16bit_cr_and_check(struct spi_nor *nor, u8 cr);
|
||||||
|
|
||||||
int spi_nor_xread_sr(struct spi_nor *nor, u8 *sr);
|
|
||||||
ssize_t spi_nor_read_data(struct spi_nor *nor, loff_t from, size_t len,
|
ssize_t spi_nor_read_data(struct spi_nor *nor, loff_t from, size_t len,
|
||||||
u8 *buf);
|
u8 *buf);
|
||||||
ssize_t spi_nor_write_data(struct spi_nor *nor, loff_t to, size_t len,
|
ssize_t spi_nor_write_data(struct spi_nor *nor, loff_t to, size_t len,
|
||||||
|
@ -8,6 +8,27 @@
|
|||||||
|
|
||||||
#include "core.h"
|
#include "core.h"
|
||||||
|
|
||||||
|
#define SPINOR_OP_XSE 0x50 /* Sector erase */
|
||||||
|
#define SPINOR_OP_XPP 0x82 /* Page program */
|
||||||
|
#define SPINOR_OP_XRDSR 0xd7 /* Read status register */
|
||||||
|
|
||||||
|
#define XSR_PAGESIZE BIT(0) /* Page size in Po2 or Linear */
|
||||||
|
#define XSR_RDY BIT(7) /* Ready */
|
||||||
|
|
||||||
|
#define S3AN_INFO(_jedec_id, _n_sectors, _page_size) \
|
||||||
|
.id = { \
|
||||||
|
((_jedec_id) >> 16) & 0xff, \
|
||||||
|
((_jedec_id) >> 8) & 0xff, \
|
||||||
|
(_jedec_id) & 0xff \
|
||||||
|
}, \
|
||||||
|
.id_len = 3, \
|
||||||
|
.sector_size = (8 * (_page_size)), \
|
||||||
|
.n_sectors = (_n_sectors), \
|
||||||
|
.page_size = (_page_size), \
|
||||||
|
.addr_width = 3, \
|
||||||
|
.flags = SPI_NOR_NO_FR
|
||||||
|
|
||||||
|
/* Xilinx S3AN share MFR with Atmel SPI NOR */
|
||||||
static const struct flash_info xilinx_nor_parts[] = {
|
static const struct flash_info xilinx_nor_parts[] = {
|
||||||
/* Xilinx S3AN Internal Flash */
|
/* Xilinx S3AN Internal Flash */
|
||||||
{ "3S50AN", S3AN_INFO(0x1f2200, 64, 264) },
|
{ "3S50AN", S3AN_INFO(0x1f2200, 64, 264) },
|
||||||
@ -38,6 +59,57 @@ static u32 s3an_nor_convert_addr(struct spi_nor *nor, u32 addr)
|
|||||||
return page | offset;
|
return page | offset;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* spi_nor_xread_sr() - Read the Status Register on S3AN flashes.
|
||||||
|
* @nor: pointer to 'struct spi_nor'.
|
||||||
|
* @sr: pointer to a DMA-able buffer where the value of the
|
||||||
|
* Status Register will be written.
|
||||||
|
*
|
||||||
|
* Return: 0 on success, -errno otherwise.
|
||||||
|
*/
|
||||||
|
static int spi_nor_xread_sr(struct spi_nor *nor, u8 *sr)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
if (nor->spimem) {
|
||||||
|
struct spi_mem_op op =
|
||||||
|
SPI_MEM_OP(SPI_MEM_OP_CMD(SPINOR_OP_XRDSR, 0),
|
||||||
|
SPI_MEM_OP_NO_ADDR,
|
||||||
|
SPI_MEM_OP_NO_DUMMY,
|
||||||
|
SPI_MEM_OP_DATA_IN(1, sr, 0));
|
||||||
|
|
||||||
|
spi_nor_spimem_setup_op(nor, &op, nor->reg_proto);
|
||||||
|
|
||||||
|
ret = spi_mem_exec_op(nor->spimem, &op);
|
||||||
|
} else {
|
||||||
|
ret = spi_nor_controller_ops_read_reg(nor, SPINOR_OP_XRDSR, sr,
|
||||||
|
1);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ret)
|
||||||
|
dev_dbg(nor->dev, "error %d reading XRDSR\n", ret);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* spi_nor_xsr_ready() - Query the Status Register of the S3AN flash to see if
|
||||||
|
* the flash is ready for new commands.
|
||||||
|
* @nor: pointer to 'struct spi_nor'.
|
||||||
|
*
|
||||||
|
* Return: 1 if ready, 0 if not ready, -errno on errors.
|
||||||
|
*/
|
||||||
|
static int spi_nor_xsr_ready(struct spi_nor *nor)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
ret = spi_nor_xread_sr(nor, nor->bouncebuf);
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
return !!(nor->bouncebuf[0] & XSR_RDY);
|
||||||
|
}
|
||||||
|
|
||||||
static int xilinx_nor_setup(struct spi_nor *nor,
|
static int xilinx_nor_setup(struct spi_nor *nor,
|
||||||
const struct spi_nor_hwcaps *hwcaps)
|
const struct spi_nor_hwcaps *hwcaps)
|
||||||
{
|
{
|
||||||
@ -83,6 +155,7 @@ static int xilinx_nor_setup(struct spi_nor *nor,
|
|||||||
static void xilinx_nor_late_init(struct spi_nor *nor)
|
static void xilinx_nor_late_init(struct spi_nor *nor)
|
||||||
{
|
{
|
||||||
nor->params->setup = xilinx_nor_setup;
|
nor->params->setup = xilinx_nor_setup;
|
||||||
|
nor->params->ready = spi_nor_xsr_ready;
|
||||||
}
|
}
|
||||||
|
|
||||||
static const struct spi_nor_fixups xilinx_nor_fixups = {
|
static const struct spi_nor_fixups xilinx_nor_fixups = {
|
||||||
|
@ -86,15 +86,6 @@
|
|||||||
#define SPINOR_OP_BP 0x02 /* Byte program */
|
#define SPINOR_OP_BP 0x02 /* Byte program */
|
||||||
#define SPINOR_OP_AAI_WP 0xad /* Auto address increment word program */
|
#define SPINOR_OP_AAI_WP 0xad /* Auto address increment word program */
|
||||||
|
|
||||||
/* Used for S3AN flashes only */
|
|
||||||
#define SPINOR_OP_XSE 0x50 /* Sector erase */
|
|
||||||
#define SPINOR_OP_XPP 0x82 /* Page program */
|
|
||||||
#define SPINOR_OP_XRDSR 0xd7 /* Read status register */
|
|
||||||
|
|
||||||
#define XSR_PAGESIZE BIT(0) /* Page size in Po2 or Linear */
|
|
||||||
#define XSR_RDY BIT(7) /* Ready */
|
|
||||||
|
|
||||||
|
|
||||||
/* Used for Macronix and Winbond flashes. */
|
/* Used for Macronix and Winbond flashes. */
|
||||||
#define SPINOR_OP_EN4B 0xb7 /* Enter 4-byte mode */
|
#define SPINOR_OP_EN4B 0xb7 /* Enter 4-byte mode */
|
||||||
#define SPINOR_OP_EX4B 0xe9 /* Exit 4-byte mode */
|
#define SPINOR_OP_EX4B 0xe9 /* Exit 4-byte mode */
|
||||||
|
Loading…
Reference in New Issue
Block a user