linux/drivers/spi/spi-sg2044-nor.c
Mark Brown 48303ef31d
spi: sg2044-nor: fix a couple static checker bugs
Merge series from Dan Carpenter <dan.carpenter@linaro.org>:

Here are two fixes for Smatch warnings.

Dan Carpenter (2):
  spi: sg2044-nor: fix signedness bug in sg2044_spifmc_write()
  spi: sg2044-nor: Fix uninitialized variable in probe

 drivers/spi/spi-sg2044-nor.c | 7 +++----
 1 file changed, 3 insertions(+), 4 deletions(-)

--
2.47.2
2025-03-18 14:37:39 +00:00

489 lines
13 KiB
C

// SPDX-License-Identifier: GPL-2.0-only
/*
* SG2044 SPI NOR controller driver
*
* Copyright (c) 2025 Longbin Li <looong.bin@gmail.com>
*/
#include <linux/bitfield.h>
#include <linux/clk.h>
#include <linux/iopoll.h>
#include <linux/module.h>
#include <linux/of.h>
#include <linux/platform_device.h>
#include <linux/spi/spi-mem.h>
/* Hardware register definitions */
#define SPIFMC_CTRL 0x00
#define SPIFMC_CTRL_CPHA BIT(12)
#define SPIFMC_CTRL_CPOL BIT(13)
#define SPIFMC_CTRL_HOLD_OL BIT(14)
#define SPIFMC_CTRL_WP_OL BIT(15)
#define SPIFMC_CTRL_LSBF BIT(20)
#define SPIFMC_CTRL_SRST BIT(21)
#define SPIFMC_CTRL_SCK_DIV_SHIFT 0
#define SPIFMC_CTRL_FRAME_LEN_SHIFT 16
#define SPIFMC_CTRL_SCK_DIV_MASK 0x7FF
#define SPIFMC_CE_CTRL 0x04
#define SPIFMC_CE_CTRL_CEMANUAL BIT(0)
#define SPIFMC_CE_CTRL_CEMANUAL_EN BIT(1)
#define SPIFMC_DLY_CTRL 0x08
#define SPIFMC_CTRL_FM_INTVL_MASK 0x000f
#define SPIFMC_CTRL_FM_INTVL BIT(0)
#define SPIFMC_CTRL_CET_MASK 0x0f00
#define SPIFMC_CTRL_CET BIT(8)
#define SPIFMC_DMMR 0x0c
#define SPIFMC_TRAN_CSR 0x10
#define SPIFMC_TRAN_CSR_TRAN_MODE_MASK GENMASK(1, 0)
#define SPIFMC_TRAN_CSR_TRAN_MODE_RX BIT(0)
#define SPIFMC_TRAN_CSR_TRAN_MODE_TX BIT(1)
#define SPIFMC_TRAN_CSR_FAST_MODE BIT(3)
#define SPIFMC_TRAN_CSR_BUS_WIDTH_1_BIT (0x00 << 4)
#define SPIFMC_TRAN_CSR_BUS_WIDTH_2_BIT (0x01 << 4)
#define SPIFMC_TRAN_CSR_BUS_WIDTH_4_BIT (0x02 << 4)
#define SPIFMC_TRAN_CSR_DMA_EN BIT(6)
#define SPIFMC_TRAN_CSR_MISO_LEVEL BIT(7)
#define SPIFMC_TRAN_CSR_ADDR_BYTES_MASK GENMASK(10, 8)
#define SPIFMC_TRAN_CSR_ADDR_BYTES_SHIFT 8
#define SPIFMC_TRAN_CSR_WITH_CMD BIT(11)
#define SPIFMC_TRAN_CSR_FIFO_TRG_LVL_MASK GENMASK(13, 12)
#define SPIFMC_TRAN_CSR_FIFO_TRG_LVL_1_BYTE (0x00 << 12)
#define SPIFMC_TRAN_CSR_FIFO_TRG_LVL_2_BYTE (0x01 << 12)
#define SPIFMC_TRAN_CSR_FIFO_TRG_LVL_4_BYTE (0x02 << 12)
#define SPIFMC_TRAN_CSR_FIFO_TRG_LVL_8_BYTE (0x03 << 12)
#define SPIFMC_TRAN_CSR_GO_BUSY BIT(15)
#define SPIFMC_TRAN_CSR_ADDR4B_SHIFT 20
#define SPIFMC_TRAN_CSR_CMD4B_SHIFT 21
#define SPIFMC_TRAN_NUM 0x14
#define SPIFMC_FIFO_PORT 0x18
#define SPIFMC_FIFO_PT 0x20
#define SPIFMC_INT_STS 0x28
#define SPIFMC_INT_TRAN_DONE BIT(0)
#define SPIFMC_INT_RD_FIFO BIT(2)
#define SPIFMC_INT_WR_FIFO BIT(3)
#define SPIFMC_INT_RX_FRAME BIT(4)
#define SPIFMC_INT_TX_FRAME BIT(5)
#define SPIFMC_INT_EN 0x2c
#define SPIFMC_INT_TRAN_DONE_EN BIT(0)
#define SPIFMC_INT_RD_FIFO_EN BIT(2)
#define SPIFMC_INT_WR_FIFO_EN BIT(3)
#define SPIFMC_INT_RX_FRAME_EN BIT(4)
#define SPIFMC_INT_TX_FRAME_EN BIT(5)
#define SPIFMC_OPT 0x030
#define SPIFMC_OPT_DISABLE_FIFO_FLUSH BIT(1)
#define SPIFMC_MAX_FIFO_DEPTH 8
#define SPIFMC_MAX_READ_SIZE 0x10000
struct sg2044_spifmc {
struct spi_controller *ctrl;
void __iomem *io_base;
struct device *dev;
struct mutex lock;
struct clk *clk;
};
static int sg2044_spifmc_wait_int(struct sg2044_spifmc *spifmc, u8 int_type)
{
u32 stat;
return readl_poll_timeout(spifmc->io_base + SPIFMC_INT_STS, stat,
(stat & int_type), 0, 1000000);
}
static int sg2044_spifmc_wait_xfer_size(struct sg2044_spifmc *spifmc,
int xfer_size)
{
u8 stat;
return readl_poll_timeout(spifmc->io_base + SPIFMC_FIFO_PT, stat,
((stat & 0xf) == xfer_size), 1, 1000000);
}
static u32 sg2044_spifmc_init_reg(struct sg2044_spifmc *spifmc)
{
u32 reg;
reg = readl(spifmc->io_base + SPIFMC_TRAN_CSR);
reg &= ~(SPIFMC_TRAN_CSR_TRAN_MODE_MASK |
SPIFMC_TRAN_CSR_FAST_MODE |
SPIFMC_TRAN_CSR_BUS_WIDTH_2_BIT |
SPIFMC_TRAN_CSR_BUS_WIDTH_4_BIT |
SPIFMC_TRAN_CSR_DMA_EN |
SPIFMC_TRAN_CSR_ADDR_BYTES_MASK |
SPIFMC_TRAN_CSR_WITH_CMD |
SPIFMC_TRAN_CSR_FIFO_TRG_LVL_MASK);
writel(reg, spifmc->io_base + SPIFMC_TRAN_CSR);
return reg;
}
static ssize_t sg2044_spifmc_read_64k(struct sg2044_spifmc *spifmc,
const struct spi_mem_op *op, loff_t from,
size_t len, u_char *buf)
{
int xfer_size, offset;
u32 reg;
int ret;
int i;
reg = sg2044_spifmc_init_reg(spifmc);
reg |= (op->addr.nbytes + op->dummy.nbytes) << SPIFMC_TRAN_CSR_ADDR_BYTES_SHIFT;
reg |= SPIFMC_TRAN_CSR_FIFO_TRG_LVL_8_BYTE;
reg |= SPIFMC_TRAN_CSR_WITH_CMD;
reg |= SPIFMC_TRAN_CSR_TRAN_MODE_RX;
writel(0, spifmc->io_base + SPIFMC_FIFO_PT);
writeb(op->cmd.opcode, spifmc->io_base + SPIFMC_FIFO_PORT);
for (i = op->addr.nbytes - 1; i >= 0; i--)
writeb((from >> i * 8) & 0xff, spifmc->io_base + SPIFMC_FIFO_PORT);
for (i = 0; i < op->dummy.nbytes; i++)
writeb(0xff, spifmc->io_base + SPIFMC_FIFO_PORT);
writel(len, spifmc->io_base + SPIFMC_TRAN_NUM);
writel(0, spifmc->io_base + SPIFMC_INT_STS);
reg |= SPIFMC_TRAN_CSR_GO_BUSY;
writel(reg, spifmc->io_base + SPIFMC_TRAN_CSR);
ret = sg2044_spifmc_wait_int(spifmc, SPIFMC_INT_RD_FIFO);
if (ret < 0)
return ret;
offset = 0;
while (offset < len) {
xfer_size = min_t(size_t, SPIFMC_MAX_FIFO_DEPTH, len - offset);
ret = sg2044_spifmc_wait_xfer_size(spifmc, xfer_size);
if (ret < 0)
return ret;
for (i = 0; i < xfer_size; i++)
buf[i + offset] = readb(spifmc->io_base + SPIFMC_FIFO_PORT);
offset += xfer_size;
}
ret = sg2044_spifmc_wait_int(spifmc, SPIFMC_INT_TRAN_DONE);
if (ret < 0)
return ret;
writel(0, spifmc->io_base + SPIFMC_FIFO_PT);
return len;
}
static ssize_t sg2044_spifmc_read(struct sg2044_spifmc *spifmc,
const struct spi_mem_op *op)
{
size_t xfer_size;
size_t offset;
loff_t from = op->addr.val;
size_t len = op->data.nbytes;
int ret;
u8 *din = op->data.buf.in;
offset = 0;
while (offset < len) {
xfer_size = min_t(size_t, SPIFMC_MAX_READ_SIZE, len - offset);
ret = sg2044_spifmc_read_64k(spifmc, op, from, xfer_size, din);
if (ret < 0)
return ret;
offset += xfer_size;
din += xfer_size;
from += xfer_size;
}
return 0;
}
static ssize_t sg2044_spifmc_write(struct sg2044_spifmc *spifmc,
const struct spi_mem_op *op)
{
size_t xfer_size;
const u8 *dout = op->data.buf.out;
int i, offset;
int ret;
u32 reg;
reg = sg2044_spifmc_init_reg(spifmc);
reg |= (op->addr.nbytes + op->dummy.nbytes) << SPIFMC_TRAN_CSR_ADDR_BYTES_SHIFT;
reg |= SPIFMC_TRAN_CSR_FIFO_TRG_LVL_8_BYTE;
reg |= SPIFMC_TRAN_CSR_WITH_CMD;
reg |= SPIFMC_TRAN_CSR_TRAN_MODE_TX;
writel(0, spifmc->io_base + SPIFMC_FIFO_PT);
writeb(op->cmd.opcode, spifmc->io_base + SPIFMC_FIFO_PORT);
for (i = op->addr.nbytes - 1; i >= 0; i--)
writeb((op->addr.val >> i * 8) & 0xff, spifmc->io_base + SPIFMC_FIFO_PORT);
for (i = 0; i < op->dummy.nbytes; i++)
writeb(0xff, spifmc->io_base + SPIFMC_FIFO_PORT);
writel(0, spifmc->io_base + SPIFMC_INT_STS);
writel(op->data.nbytes, spifmc->io_base + SPIFMC_TRAN_NUM);
reg |= SPIFMC_TRAN_CSR_GO_BUSY;
writel(reg, spifmc->io_base + SPIFMC_TRAN_CSR);
ret = sg2044_spifmc_wait_xfer_size(spifmc, 0);
if (ret < 0)
return ret;
writel(0, spifmc->io_base + SPIFMC_FIFO_PT);
offset = 0;
while (offset < op->data.nbytes) {
xfer_size = min_t(size_t, SPIFMC_MAX_FIFO_DEPTH, op->data.nbytes - offset);
ret = sg2044_spifmc_wait_xfer_size(spifmc, 0);
if (ret < 0)
return ret;
for (i = 0; i < xfer_size; i++)
writeb(dout[i + offset], spifmc->io_base + SPIFMC_FIFO_PORT);
offset += xfer_size;
}
ret = sg2044_spifmc_wait_int(spifmc, SPIFMC_INT_TRAN_DONE);
if (ret < 0)
return ret;
writel(0, spifmc->io_base + SPIFMC_FIFO_PT);
return 0;
}
static ssize_t sg2044_spifmc_tran_cmd(struct sg2044_spifmc *spifmc,
const struct spi_mem_op *op)
{
int i, ret;
u32 reg;
reg = sg2044_spifmc_init_reg(spifmc);
reg |= (op->addr.nbytes + op->dummy.nbytes) << SPIFMC_TRAN_CSR_ADDR_BYTES_SHIFT;
reg |= SPIFMC_TRAN_CSR_FIFO_TRG_LVL_1_BYTE;
reg |= SPIFMC_TRAN_CSR_WITH_CMD;
writel(0, spifmc->io_base + SPIFMC_FIFO_PT);
writeb(op->cmd.opcode, spifmc->io_base + SPIFMC_FIFO_PORT);
for (i = op->addr.nbytes - 1; i >= 0; i--)
writeb((op->addr.val >> i * 8) & 0xff, spifmc->io_base + SPIFMC_FIFO_PORT);
for (i = 0; i < op->dummy.nbytes; i++)
writeb(0xff, spifmc->io_base + SPIFMC_FIFO_PORT);
writel(0, spifmc->io_base + SPIFMC_INT_STS);
reg |= SPIFMC_TRAN_CSR_GO_BUSY;
writel(reg, spifmc->io_base + SPIFMC_TRAN_CSR);
ret = sg2044_spifmc_wait_int(spifmc, SPIFMC_INT_TRAN_DONE);
if (ret < 0)
return ret;
writel(0, spifmc->io_base + SPIFMC_FIFO_PT);
return 0;
}
static void sg2044_spifmc_trans(struct sg2044_spifmc *spifmc,
const struct spi_mem_op *op)
{
if (op->data.dir == SPI_MEM_DATA_IN)
sg2044_spifmc_read(spifmc, op);
else if (op->data.dir == SPI_MEM_DATA_OUT)
sg2044_spifmc_write(spifmc, op);
else
sg2044_spifmc_tran_cmd(spifmc, op);
}
static ssize_t sg2044_spifmc_trans_reg(struct sg2044_spifmc *spifmc,
const struct spi_mem_op *op)
{
const u8 *dout = NULL;
u8 *din = NULL;
size_t len = op->data.nbytes;
int ret, i;
u32 reg;
if (op->data.dir == SPI_MEM_DATA_IN)
din = op->data.buf.in;
else
dout = op->data.buf.out;
reg = sg2044_spifmc_init_reg(spifmc);
reg |= SPIFMC_TRAN_CSR_FIFO_TRG_LVL_1_BYTE;
reg |= SPIFMC_TRAN_CSR_WITH_CMD;
if (din) {
reg |= SPIFMC_TRAN_CSR_BUS_WIDTH_1_BIT;
reg |= SPIFMC_TRAN_CSR_TRAN_MODE_RX;
reg |= SPIFMC_TRAN_CSR_TRAN_MODE_TX;
writel(SPIFMC_OPT_DISABLE_FIFO_FLUSH, spifmc->io_base + SPIFMC_OPT);
} else {
/*
* If write values to the Status Register,
* configure TRAN_CSR register as the same as
* sg2044_spifmc_read_reg.
*/
if (op->cmd.opcode == 0x01) {
reg |= SPIFMC_TRAN_CSR_TRAN_MODE_RX;
reg |= SPIFMC_TRAN_CSR_TRAN_MODE_TX;
writel(len, spifmc->io_base + SPIFMC_TRAN_NUM);
}
}
writel(0, spifmc->io_base + SPIFMC_FIFO_PT);
writeb(op->cmd.opcode, spifmc->io_base + SPIFMC_FIFO_PORT);
for (i = 0; i < len; i++) {
if (din)
writeb(0xff, spifmc->io_base + SPIFMC_FIFO_PORT);
else
writeb(dout[i], spifmc->io_base + SPIFMC_FIFO_PORT);
}
writel(0, spifmc->io_base + SPIFMC_INT_STS);
writel(len, spifmc->io_base + SPIFMC_TRAN_NUM);
reg |= SPIFMC_TRAN_CSR_GO_BUSY;
writel(reg, spifmc->io_base + SPIFMC_TRAN_CSR);
ret = sg2044_spifmc_wait_int(spifmc, SPIFMC_INT_TRAN_DONE);
if (ret < 0)
return ret;
if (din) {
while (len--)
*din++ = readb(spifmc->io_base + SPIFMC_FIFO_PORT);
}
writel(0, spifmc->io_base + SPIFMC_FIFO_PT);
return 0;
}
static int sg2044_spifmc_exec_op(struct spi_mem *mem,
const struct spi_mem_op *op)
{
struct sg2044_spifmc *spifmc;
spifmc = spi_controller_get_devdata(mem->spi->controller);
mutex_lock(&spifmc->lock);
if (op->addr.nbytes == 0)
sg2044_spifmc_trans_reg(spifmc, op);
else
sg2044_spifmc_trans(spifmc, op);
mutex_unlock(&spifmc->lock);
return 0;
}
static const struct spi_controller_mem_ops sg2044_spifmc_mem_ops = {
.exec_op = sg2044_spifmc_exec_op,
};
static void sg2044_spifmc_init(struct sg2044_spifmc *spifmc)
{
u32 tran_csr;
u32 reg;
writel(0, spifmc->io_base + SPIFMC_DMMR);
reg = readl(spifmc->io_base + SPIFMC_CTRL);
reg |= SPIFMC_CTRL_SRST;
reg &= ~(SPIFMC_CTRL_SCK_DIV_MASK);
reg |= 1;
writel(reg, spifmc->io_base + SPIFMC_CTRL);
writel(0, spifmc->io_base + SPIFMC_CE_CTRL);
tran_csr = readl(spifmc->io_base + SPIFMC_TRAN_CSR);
tran_csr |= (0 << SPIFMC_TRAN_CSR_ADDR_BYTES_SHIFT);
tran_csr |= SPIFMC_TRAN_CSR_FIFO_TRG_LVL_4_BYTE;
tran_csr |= SPIFMC_TRAN_CSR_WITH_CMD;
writel(tran_csr, spifmc->io_base + SPIFMC_TRAN_CSR);
}
static int sg2044_spifmc_probe(struct platform_device *pdev)
{
struct device *dev = &pdev->dev;
struct spi_controller *ctrl;
struct sg2044_spifmc *spifmc;
int ret;
ctrl = devm_spi_alloc_host(&pdev->dev, sizeof(*spifmc));
if (!ctrl)
return -ENOMEM;
spifmc = spi_controller_get_devdata(ctrl);
spifmc->clk = devm_clk_get_enabled(&pdev->dev, NULL);
if (IS_ERR(spifmc->clk))
return dev_err_probe(dev, PTR_ERR(spifmc->clk), "Cannot get and enable AHB clock\n");
spifmc->dev = &pdev->dev;
spifmc->ctrl = ctrl;
spifmc->io_base = devm_platform_ioremap_resource(pdev, 0);
if (IS_ERR(spifmc->io_base))
return PTR_ERR(spifmc->io_base);
ctrl->num_chipselect = 1;
ctrl->dev.of_node = pdev->dev.of_node;
ctrl->bits_per_word_mask = SPI_BPW_MASK(8);
ctrl->auto_runtime_pm = false;
ctrl->mem_ops = &sg2044_spifmc_mem_ops;
ctrl->mode_bits = SPI_RX_DUAL | SPI_TX_DUAL | SPI_RX_QUAD | SPI_TX_QUAD;
ret = devm_mutex_init(dev, &spifmc->lock);
if (ret)
return ret;
sg2044_spifmc_init(spifmc);
sg2044_spifmc_init_reg(spifmc);
ret = devm_spi_register_controller(&pdev->dev, ctrl);
if (ret)
return dev_err_probe(dev, ret, "spi_register_controller failed\n");
return 0;
}
static const struct of_device_id sg2044_spifmc_match[] = {
{ .compatible = "sophgo,sg2044-spifmc-nor" },
{ /* sentinel */ }
};
MODULE_DEVICE_TABLE(of, sg2044_spifmc_match);
static struct platform_driver sg2044_nor_driver = {
.driver = {
.name = "sg2044,spifmc-nor",
.of_match_table = sg2044_spifmc_match,
},
.probe = sg2044_spifmc_probe,
};
module_platform_driver(sg2044_nor_driver);
MODULE_DESCRIPTION("SG2044 SPI NOR controller driver");
MODULE_AUTHOR("Longbin Li <looong.bin@gmail.com>");
MODULE_LICENSE("GPL");