123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359 |
- // SPDX-License-Identifier: GPL-2.0
- /*
- * Copyright (c) 2018 MediaTek, Inc.
- * Author : Guochun.Mao@mediatek.com
- */
- #include <common.h>
- #include <dm.h>
- #include <malloc.h>
- #include <spi.h>
- #include <asm/io.h>
- #include <linux/iopoll.h>
- #include <linux/ioport.h>
- /* Register Offset */
- struct mtk_qspi_regs {
- u32 cmd;
- u32 cnt;
- u32 rdsr;
- u32 rdata;
- u32 radr[3];
- u32 wdata;
- u32 prgdata[6];
- u32 shreg[10];
- u32 cfg[2];
- u32 shreg10;
- u32 mode_mon;
- u32 status[4];
- u32 flash_time;
- u32 flash_cfg;
- u32 reserved_0[3];
- u32 sf_time;
- u32 pp_dw_data;
- u32 reserved_1;
- u32 delsel_0[2];
- u32 intrstus;
- u32 intren;
- u32 reserved_2;
- u32 cfg3;
- u32 reserved_3;
- u32 chksum;
- u32 aaicmd;
- u32 wrprot;
- u32 radr3;
- u32 dual;
- u32 delsel_1[3];
- };
- struct mtk_qspi_platdata {
- fdt_addr_t reg_base;
- fdt_addr_t mem_base;
- };
- struct mtk_qspi_priv {
- struct mtk_qspi_regs *regs;
- unsigned long *mem_base;
- u8 op;
- u8 tx[3]; /* only record max 3 bytes paras, when it's address. */
- u32 txlen; /* dout buffer length - op code length */
- u8 *rx;
- u32 rxlen;
- };
- #define MTK_QSPI_CMD_POLLINGREG_US 500000
- #define MTK_QSPI_WRBUF_SIZE 256
- #define MTK_QSPI_COMMAND_ENABLE 0x30
- /* NOR flash controller commands */
- #define MTK_QSPI_RD_TRIGGER BIT(0)
- #define MTK_QSPI_READSTATUS BIT(1)
- #define MTK_QSPI_PRG_CMD BIT(2)
- #define MTK_QSPI_WR_TRIGGER BIT(4)
- #define MTK_QSPI_WRITESTATUS BIT(5)
- #define MTK_QSPI_AUTOINC BIT(7)
- #define MTK_QSPI_MAX_RX_TX_SHIFT 0x6
- #define MTK_QSPI_MAX_SHIFT 0x8
- #define MTK_QSPI_WR_BUF_ENABLE 0x1
- #define MTK_QSPI_WR_BUF_DISABLE 0x0
- static int mtk_qspi_execute_cmd(struct mtk_qspi_priv *priv, u8 cmd)
- {
- u8 tmp;
- u8 val = cmd & ~MTK_QSPI_AUTOINC;
- writeb(cmd, &priv->regs->cmd);
- return readb_poll_timeout(&priv->regs->cmd, tmp, !(val & tmp),
- MTK_QSPI_CMD_POLLINGREG_US);
- }
- static int mtk_qspi_tx_rx(struct mtk_qspi_priv *priv)
- {
- int len = 1 + priv->txlen + priv->rxlen;
- int i, ret, idx;
- if (len > MTK_QSPI_MAX_SHIFT)
- return -ERR_INVAL;
- writeb(len * 8, &priv->regs->cnt);
- /* start at PRGDATA5, go down to PRGDATA0 */
- idx = MTK_QSPI_MAX_RX_TX_SHIFT - 1;
- /* opcode */
- writeb(priv->op, &priv->regs->prgdata[idx]);
- idx--;
- /* program TX data */
- for (i = 0; i < priv->txlen; i++, idx--)
- writeb(priv->tx[i], &priv->regs->prgdata[idx]);
- /* clear out rest of TX registers */
- while (idx >= 0) {
- writeb(0, &priv->regs->prgdata[idx]);
- idx--;
- }
- ret = mtk_qspi_execute_cmd(priv, MTK_QSPI_PRG_CMD);
- if (ret)
- return ret;
- /* restart at first RX byte */
- idx = priv->rxlen - 1;
- /* read out RX data */
- for (i = 0; i < priv->rxlen; i++, idx--)
- priv->rx[i] = readb(&priv->regs->shreg[idx]);
- return 0;
- }
- static int mtk_qspi_read(struct mtk_qspi_priv *priv,
- u32 addr, u8 *buf, u32 len)
- {
- memcpy(buf, (u8 *)priv->mem_base + addr, len);
- return 0;
- }
- static void mtk_qspi_set_addr(struct mtk_qspi_priv *priv, u32 addr)
- {
- int i;
- for (i = 0; i < 3; i++) {
- writeb(addr & 0xff, &priv->regs->radr[i]);
- addr >>= 8;
- }
- }
- static int mtk_qspi_write_single_byte(struct mtk_qspi_priv *priv,
- u32 addr, u32 length, const u8 *data)
- {
- int i, ret;
- mtk_qspi_set_addr(priv, addr);
- for (i = 0; i < length; i++) {
- writeb(*data++, &priv->regs->wdata);
- ret = mtk_qspi_execute_cmd(priv, MTK_QSPI_WR_TRIGGER);
- if (ret < 0)
- return ret;
- }
- return 0;
- }
- static int mtk_qspi_write_buffer(struct mtk_qspi_priv *priv, u32 addr,
- const u8 *buf)
- {
- int i, data;
- mtk_qspi_set_addr(priv, addr);
- for (i = 0; i < MTK_QSPI_WRBUF_SIZE; i += 4) {
- data = buf[i + 3] << 24 | buf[i + 2] << 16 |
- buf[i + 1] << 8 | buf[i];
- writel(data, &priv->regs->pp_dw_data);
- }
- return mtk_qspi_execute_cmd(priv, MTK_QSPI_WR_TRIGGER);
- }
- static int mtk_qspi_write(struct mtk_qspi_priv *priv,
- u32 addr, const u8 *buf, u32 len)
- {
- int ret;
- /* setting pre-fetch buffer for page program */
- writel(MTK_QSPI_WR_BUF_ENABLE, &priv->regs->cfg[1]);
- while (len >= MTK_QSPI_WRBUF_SIZE) {
- ret = mtk_qspi_write_buffer(priv, addr, buf);
- if (ret < 0)
- return ret;
- len -= MTK_QSPI_WRBUF_SIZE;
- addr += MTK_QSPI_WRBUF_SIZE;
- buf += MTK_QSPI_WRBUF_SIZE;
- }
- /* disable pre-fetch buffer for page program */
- writel(MTK_QSPI_WR_BUF_DISABLE, &priv->regs->cfg[1]);
- if (len)
- return mtk_qspi_write_single_byte(priv, addr, len, buf);
- return 0;
- }
- static int mtk_qspi_claim_bus(struct udevice *dev)
- {
- /* nothing to do */
- return 0;
- }
- static int mtk_qspi_release_bus(struct udevice *dev)
- {
- /* nothing to do */
- return 0;
- }
- static int mtk_qspi_transfer(struct mtk_qspi_priv *priv, unsigned int bitlen,
- const void *dout, void *din, unsigned long flags)
- {
- u32 bytes = DIV_ROUND_UP(bitlen, 8);
- u32 addr;
- if (!bytes)
- return -ERR_INVAL;
- if (dout) {
- if (flags & SPI_XFER_BEGIN) {
- /* parse op code and potential paras first */
- priv->op = *(u8 *)dout;
- if (bytes > 1)
- memcpy(priv->tx, (u8 *)dout + 1,
- bytes <= 4 ? bytes - 1 : 3);
- priv->txlen = bytes - 1;
- }
- if (flags == SPI_XFER_ONCE) {
- /* operations without receiving or sending data.
- * for example: erase, write flash register or write
- * enable...
- */
- priv->rx = NULL;
- priv->rxlen = 0;
- return mtk_qspi_tx_rx(priv);
- }
- if (flags & SPI_XFER_END) {
- /* here, dout should be data to be written.
- * and priv->tx should be filled 3Bytes address.
- */
- addr = priv->tx[0] << 16 | priv->tx[1] << 8 |
- priv->tx[2];
- return mtk_qspi_write(priv, addr, (u8 *)dout, bytes);
- }
- }
- if (din) {
- if (priv->txlen >= 3) {
- /* if run to here, priv->tx[] should be the address
- * where read data from,
- * and, din is the buf to receive data.
- */
- addr = priv->tx[0] << 16 | priv->tx[1] << 8 |
- priv->tx[2];
- return mtk_qspi_read(priv, addr, (u8 *)din, bytes);
- }
- /* should be reading flash's register */
- priv->rx = (u8 *)din;
- priv->rxlen = bytes;
- return mtk_qspi_tx_rx(priv);
- }
- return 0;
- }
- static int mtk_qspi_xfer(struct udevice *dev, unsigned int bitlen,
- const void *dout, void *din, unsigned long flags)
- {
- struct udevice *bus = dev->parent;
- struct mtk_qspi_priv *priv = dev_get_priv(bus);
- return mtk_qspi_transfer(priv, bitlen, dout, din, flags);
- }
- static int mtk_qspi_set_speed(struct udevice *bus, uint speed)
- {
- /* nothing to do */
- return 0;
- }
- static int mtk_qspi_set_mode(struct udevice *bus, uint mode)
- {
- /* nothing to do */
- return 0;
- }
- static int mtk_qspi_ofdata_to_platdata(struct udevice *bus)
- {
- struct resource res_reg, res_mem;
- struct mtk_qspi_platdata *plat = bus->platdata;
- int ret;
- ret = dev_read_resource_byname(bus, "reg_base", &res_reg);
- if (ret) {
- debug("can't get reg_base resource(ret = %d)\n", ret);
- return -ENOMEM;
- }
- ret = dev_read_resource_byname(bus, "mem_base", &res_mem);
- if (ret) {
- debug("can't get map_base resource(ret = %d)\n", ret);
- return -ENOMEM;
- }
- plat->mem_base = res_mem.start;
- plat->reg_base = res_reg.start;
- return 0;
- }
- static int mtk_qspi_probe(struct udevice *bus)
- {
- struct mtk_qspi_platdata *plat = dev_get_platdata(bus);
- struct mtk_qspi_priv *priv = dev_get_priv(bus);
- priv->regs = (struct mtk_qspi_regs *)plat->reg_base;
- priv->mem_base = (unsigned long *)plat->mem_base;
- writel(MTK_QSPI_COMMAND_ENABLE, &priv->regs->wrprot);
- return 0;
- }
- static const struct dm_spi_ops mtk_qspi_ops = {
- .claim_bus = mtk_qspi_claim_bus,
- .release_bus = mtk_qspi_release_bus,
- .xfer = mtk_qspi_xfer,
- .set_speed = mtk_qspi_set_speed,
- .set_mode = mtk_qspi_set_mode,
- };
- static const struct udevice_id mtk_qspi_ids[] = {
- { .compatible = "mediatek,mt7629-qspi" },
- { }
- };
- U_BOOT_DRIVER(mtk_qspi) = {
- .name = "mtk_qspi",
- .id = UCLASS_SPI,
- .of_match = mtk_qspi_ids,
- .ops = &mtk_qspi_ops,
- .ofdata_to_platdata = mtk_qspi_ofdata_to_platdata,
- .platdata_auto_alloc_size = sizeof(struct mtk_qspi_platdata),
- .priv_auto_alloc_size = sizeof(struct mtk_qspi_priv),
- .probe = mtk_qspi_probe,
- };
|