mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git/
synced 2025-04-19 20:58:31 +09:00
spi: sophgo: add SG2044 SPI NOR controller driver
Add support for SG2044 SPI NOR controller in Sophgo SoC. Signed-off-by: Longbin Li <looong.bin@gmail.com> Link: https://patch.msgid.link/20250304083548.10101-3-looong.bin@gmail.com Signed-off-by: Mark Brown <broonie@kernel.org>
This commit is contained in:
parent
9f95e2dff3
commit
de16c322ee
@ -1034,6 +1034,15 @@ config SPI_SN_F_OSPI
|
||||
for connecting an SPI Flash memory over up to 8-bit wide bus.
|
||||
It supports indirect access mode only.
|
||||
|
||||
config SPI_SG2044_NOR
|
||||
tristate "SG2044 SPI NOR Controller"
|
||||
depends on ARCH_SOPHGO || COMPILE_TEST
|
||||
help
|
||||
This enables support for the SG2044 SPI NOR controller,
|
||||
which supports Dual/Quad read and write operations while
|
||||
also supporting 3Byte address devices and 4Byte address
|
||||
devices.
|
||||
|
||||
config SPI_SPRD
|
||||
tristate "Spreadtrum SPI controller"
|
||||
depends on ARCH_SPRD || COMPILE_TEST
|
||||
|
@ -136,6 +136,7 @@ obj-$(CONFIG_SPI_SH_SCI) += spi-sh-sci.o
|
||||
obj-$(CONFIG_SPI_SIFIVE) += spi-sifive.o
|
||||
obj-$(CONFIG_SPI_SLAVE_MT27XX) += spi-slave-mt27xx.o
|
||||
obj-$(CONFIG_SPI_SN_F_OSPI) += spi-sn-f-ospi.o
|
||||
obj-$(CONFIG_SPI_SG2044_NOR) += spi-sg2044-nor.o
|
||||
obj-$(CONFIG_SPI_SPRD) += spi-sprd.o
|
||||
obj-$(CONFIG_SPI_SPRD_ADI) += spi-sprd-adi.o
|
||||
obj-$(CONFIG_SPI_STM32) += spi-stm32.o
|
||||
|
500
drivers/spi/spi-sg2044-nor.c
Normal file
500
drivers/spi/spi-sg2044-nor.c
Normal file
@ -0,0 +1,500 @@
|
||||
// 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;
|
||||
size_t 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 spi_controller *ctrl;
|
||||
struct sg2044_spifmc *spifmc;
|
||||
void __iomem *base;
|
||||
int ret;
|
||||
|
||||
ctrl = devm_spi_alloc_host(&pdev->dev, sizeof(*spifmc));
|
||||
if (!ctrl)
|
||||
return -ENOMEM;
|
||||
|
||||
spifmc = spi_controller_get_devdata(ctrl);
|
||||
dev_set_drvdata(&pdev->dev, ctrl);
|
||||
|
||||
spifmc->clk = devm_clk_get_enabled(&pdev->dev, NULL);
|
||||
if (IS_ERR(spifmc->clk))
|
||||
return dev_err_probe(&pdev->dev, PTR_ERR(spifmc->clk),
|
||||
"%s: Cannot get and enable AHB clock\n",
|
||||
__func__);
|
||||
|
||||
spifmc->dev = &pdev->dev;
|
||||
spifmc->ctrl = ctrl;
|
||||
|
||||
spifmc->io_base = devm_platform_ioremap_resource(pdev, 0);
|
||||
if (IS_ERR(base))
|
||||
return PTR_ERR(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;
|
||||
|
||||
mutex_init(&spifmc->lock);
|
||||
|
||||
sg2044_spifmc_init(spifmc);
|
||||
sg2044_spifmc_init_reg(spifmc);
|
||||
|
||||
ret = devm_spi_register_controller(&pdev->dev, ctrl);
|
||||
if (ret) {
|
||||
mutex_destroy(&spifmc->lock);
|
||||
dev_err(&pdev->dev, "spi_register_controller failed\n");
|
||||
return ret;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void sg2044_spifmc_remove(struct platform_device *pdev)
|
||||
{
|
||||
struct sg2044_spifmc *spifmc = platform_get_drvdata(pdev);
|
||||
|
||||
mutex_destroy(&spifmc->lock);
|
||||
}
|
||||
|
||||
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,
|
||||
.remove = sg2044_spifmc_remove,
|
||||
};
|
||||
module_platform_driver(sg2044_nor_driver);
|
||||
|
||||
MODULE_DESCRIPTION("SG2044 SPI NOR controller driver");
|
||||
MODULE_AUTHOR("Longbin Li <looong.bin@gmail.com>");
|
||||
MODULE_LICENSE("GPL");
|
Loading…
x
Reference in New Issue
Block a user