| // SPDX-License-Identifier: GPL-2.0+ |
| /* |
| * uniphier_spi.c - Socionext UniPhier SPI driver |
| * Copyright 2019 Socionext, Inc. |
| */ |
| |
| #include <clk.h> |
| #include <common.h> |
| #include <dm.h> |
| #include <log.h> |
| #include <time.h> |
| #include <dm/device_compat.h> |
| #include <linux/bitfield.h> |
| #include <linux/bitops.h> |
| #include <linux/delay.h> |
| #include <linux/io.h> |
| #include <spi.h> |
| #include <wait_bit.h> |
| |
| DECLARE_GLOBAL_DATA_PTR; |
| |
| #define SSI_CTL 0x00 |
| #define SSI_CTL_EN BIT(0) |
| |
| #define SSI_CKS 0x04 |
| #define SSI_CKS_CKRAT_MASK GENMASK(7, 0) |
| #define SSI_CKS_CKPHS BIT(14) |
| #define SSI_CKS_CKINIT BIT(13) |
| #define SSI_CKS_CKDLY BIT(12) |
| |
| #define SSI_TXWDS 0x08 |
| #define SSI_TXWDS_WDLEN_MASK GENMASK(13, 8) |
| #define SSI_TXWDS_TDTF_MASK GENMASK(7, 6) |
| #define SSI_TXWDS_DTLEN_MASK GENMASK(5, 0) |
| |
| #define SSI_RXWDS 0x0c |
| #define SSI_RXWDS_RDTF_MASK GENMASK(7, 6) |
| #define SSI_RXWDS_DTLEN_MASK GENMASK(5, 0) |
| |
| #define SSI_FPS 0x10 |
| #define SSI_FPS_FSPOL BIT(15) |
| #define SSI_FPS_FSTRT BIT(14) |
| |
| #define SSI_SR 0x14 |
| #define SSI_SR_BUSY BIT(7) |
| #define SSI_SR_TNF BIT(5) |
| #define SSI_SR_RNE BIT(0) |
| |
| #define SSI_IE 0x18 |
| |
| #define SSI_IC 0x1c |
| #define SSI_IC_TCIC BIT(4) |
| #define SSI_IC_RCIC BIT(3) |
| #define SSI_IC_RORIC BIT(0) |
| |
| #define SSI_FC 0x20 |
| #define SSI_FC_TXFFL BIT(12) |
| #define SSI_FC_TXFTH_MASK GENMASK(11, 8) |
| #define SSI_FC_RXFFL BIT(4) |
| #define SSI_FC_RXFTH_MASK GENMASK(3, 0) |
| |
| #define SSI_XDR 0x24 /* TXDR for write, RXDR for read */ |
| |
| #define SSI_FIFO_DEPTH 8U |
| |
| #define SSI_REG_TIMEOUT (CONFIG_SYS_HZ / 100) /* 10 ms */ |
| #define SSI_XFER_TIMEOUT (CONFIG_SYS_HZ) /* 1 sec */ |
| |
| #define SSI_CLK 50000000 /* internal I/O clock: 50MHz */ |
| |
| struct uniphier_spi_plat { |
| void __iomem *base; |
| u32 frequency; /* input frequency */ |
| u32 speed_hz; |
| uint deactivate_delay_us; /* Delay to wait after deactivate */ |
| uint activate_delay_us; /* Delay to wait after activate */ |
| }; |
| |
| struct uniphier_spi_priv { |
| void __iomem *base; |
| u8 mode; |
| u8 fifo_depth; |
| u8 bits_per_word; |
| ulong last_transaction_us; /* Time of last transaction end */ |
| }; |
| |
| static void uniphier_spi_enable(struct uniphier_spi_priv *priv, int enable) |
| { |
| u32 val; |
| |
| val = readl(priv->base + SSI_CTL); |
| if (enable) |
| val |= SSI_CTL_EN; |
| else |
| val &= ~SSI_CTL_EN; |
| writel(val, priv->base + SSI_CTL); |
| } |
| |
| static void uniphier_spi_regdump(struct uniphier_spi_priv *priv) |
| { |
| pr_debug("CTL %08x\n", readl(priv->base + SSI_CTL)); |
| pr_debug("CKS %08x\n", readl(priv->base + SSI_CKS)); |
| pr_debug("TXWDS %08x\n", readl(priv->base + SSI_TXWDS)); |
| pr_debug("RXWDS %08x\n", readl(priv->base + SSI_RXWDS)); |
| pr_debug("FPS %08x\n", readl(priv->base + SSI_FPS)); |
| pr_debug("SR %08x\n", readl(priv->base + SSI_SR)); |
| pr_debug("IE %08x\n", readl(priv->base + SSI_IE)); |
| pr_debug("IC %08x\n", readl(priv->base + SSI_IC)); |
| pr_debug("FC %08x\n", readl(priv->base + SSI_FC)); |
| pr_debug("XDR %08x\n", readl(priv->base + SSI_XDR)); |
| } |
| |
| static void spi_cs_activate(struct udevice *dev) |
| { |
| struct udevice *bus = dev->parent; |
| struct uniphier_spi_plat *plat = bus->plat; |
| struct uniphier_spi_priv *priv = dev_get_priv(bus); |
| ulong delay_us; /* The delay completed so far */ |
| u32 val; |
| |
| /* If it's too soon to do another transaction, wait */ |
| if (plat->deactivate_delay_us && priv->last_transaction_us) { |
| delay_us = timer_get_us() - priv->last_transaction_us; |
| if (delay_us < plat->deactivate_delay_us) |
| udelay(plat->deactivate_delay_us - delay_us); |
| } |
| |
| val = readl(priv->base + SSI_FPS); |
| if (priv->mode & SPI_CS_HIGH) |
| val |= SSI_FPS_FSPOL; |
| else |
| val &= ~SSI_FPS_FSPOL; |
| writel(val, priv->base + SSI_FPS); |
| |
| if (plat->activate_delay_us) |
| udelay(plat->activate_delay_us); |
| } |
| |
| static void spi_cs_deactivate(struct udevice *dev) |
| { |
| struct udevice *bus = dev->parent; |
| struct uniphier_spi_plat *plat = bus->plat; |
| struct uniphier_spi_priv *priv = dev_get_priv(bus); |
| u32 val; |
| |
| val = readl(priv->base + SSI_FPS); |
| if (priv->mode & SPI_CS_HIGH) |
| val &= ~SSI_FPS_FSPOL; |
| else |
| val |= SSI_FPS_FSPOL; |
| writel(val, priv->base + SSI_FPS); |
| |
| /* Remember time of this transaction so we can honour the bus delay */ |
| if (plat->deactivate_delay_us) |
| priv->last_transaction_us = timer_get_us(); |
| } |
| |
| static int uniphier_spi_claim_bus(struct udevice *dev) |
| { |
| struct udevice *bus = dev->parent; |
| struct uniphier_spi_priv *priv = dev_get_priv(bus); |
| u32 val, size; |
| |
| uniphier_spi_enable(priv, false); |
| |
| /* disable interrupts */ |
| writel(0, priv->base + SSI_IE); |
| |
| /* bits_per_word */ |
| size = priv->bits_per_word; |
| val = readl(priv->base + SSI_TXWDS); |
| val &= ~(SSI_TXWDS_WDLEN_MASK | SSI_TXWDS_DTLEN_MASK); |
| val |= FIELD_PREP(SSI_TXWDS_WDLEN_MASK, size); |
| val |= FIELD_PREP(SSI_TXWDS_DTLEN_MASK, size); |
| writel(val, priv->base + SSI_TXWDS); |
| |
| val = readl(priv->base + SSI_RXWDS); |
| val &= ~SSI_RXWDS_DTLEN_MASK; |
| val |= FIELD_PREP(SSI_RXWDS_DTLEN_MASK, size); |
| writel(val, priv->base + SSI_RXWDS); |
| |
| /* reset FIFOs */ |
| val = SSI_FC_TXFFL | SSI_FC_RXFFL; |
| writel(val, priv->base + SSI_FC); |
| |
| /* FIFO threthold */ |
| val = readl(priv->base + SSI_FC); |
| val &= ~(SSI_FC_TXFTH_MASK | SSI_FC_RXFTH_MASK); |
| val |= FIELD_PREP(SSI_FC_TXFTH_MASK, priv->fifo_depth); |
| val |= FIELD_PREP(SSI_FC_RXFTH_MASK, priv->fifo_depth); |
| writel(val, priv->base + SSI_FC); |
| |
| /* clear interrupts */ |
| writel(SSI_IC_TCIC | SSI_IC_RCIC | SSI_IC_RORIC, |
| priv->base + SSI_IC); |
| |
| uniphier_spi_enable(priv, true); |
| |
| return 0; |
| } |
| |
| static int uniphier_spi_release_bus(struct udevice *dev) |
| { |
| struct udevice *bus = dev->parent; |
| struct uniphier_spi_priv *priv = dev_get_priv(bus); |
| |
| uniphier_spi_enable(priv, false); |
| |
| return 0; |
| } |
| |
| static int uniphier_spi_xfer(struct udevice *dev, unsigned int bitlen, |
| const void *dout, void *din, unsigned long flags) |
| { |
| struct udevice *bus = dev->parent; |
| struct uniphier_spi_priv *priv = dev_get_priv(bus); |
| const u8 *tx_buf = dout; |
| u8 *rx_buf = din, buf; |
| u32 len = bitlen / 8; |
| u32 tx_len, rx_len; |
| u32 ts, status; |
| int ret = 0; |
| |
| if (bitlen % 8) { |
| dev_err(dev, "Non byte aligned SPI transfer\n"); |
| return -EINVAL; |
| } |
| |
| if (flags & SPI_XFER_BEGIN) |
| spi_cs_activate(dev); |
| |
| uniphier_spi_enable(priv, true); |
| |
| ts = get_timer(0); |
| tx_len = len; |
| rx_len = len; |
| |
| uniphier_spi_regdump(priv); |
| |
| while (tx_len || rx_len) { |
| ret = wait_for_bit_le32(priv->base + SSI_SR, SSI_SR_BUSY, false, |
| SSI_REG_TIMEOUT * 1000, false); |
| if (ret) { |
| if (ret == -ETIMEDOUT) |
| dev_err(dev, "access timeout\n"); |
| break; |
| } |
| |
| status = readl(priv->base + SSI_SR); |
| /* write the data into TX */ |
| if (tx_len && (status & SSI_SR_TNF)) { |
| buf = tx_buf ? *tx_buf++ : 0; |
| writel(buf, priv->base + SSI_XDR); |
| tx_len--; |
| } |
| |
| /* read the data from RX */ |
| if (rx_len && (status & SSI_SR_RNE)) { |
| buf = readl(priv->base + SSI_XDR); |
| if (rx_buf) |
| *rx_buf++ = buf; |
| rx_len--; |
| } |
| |
| if (get_timer(ts) >= SSI_XFER_TIMEOUT) { |
| dev_err(dev, "transfer timeout\n"); |
| ret = -ETIMEDOUT; |
| break; |
| } |
| } |
| |
| if (flags & SPI_XFER_END) |
| spi_cs_deactivate(dev); |
| |
| uniphier_spi_enable(priv, false); |
| |
| return ret; |
| } |
| |
| static int uniphier_spi_set_speed(struct udevice *bus, uint speed) |
| { |
| struct uniphier_spi_plat *plat = bus->plat; |
| struct uniphier_spi_priv *priv = dev_get_priv(bus); |
| u32 val, ckdiv; |
| |
| if (speed > plat->frequency) |
| speed = plat->frequency; |
| |
| /* baudrate */ |
| ckdiv = DIV_ROUND_UP(SSI_CLK, speed); |
| ckdiv = round_up(ckdiv, 2); |
| |
| val = readl(priv->base + SSI_CKS); |
| val &= ~SSI_CKS_CKRAT_MASK; |
| val |= ckdiv & SSI_CKS_CKRAT_MASK; |
| writel(val, priv->base + SSI_CKS); |
| |
| return 0; |
| } |
| |
| static int uniphier_spi_set_mode(struct udevice *bus, uint mode) |
| { |
| struct uniphier_spi_priv *priv = dev_get_priv(bus); |
| u32 val1, val2; |
| |
| /* |
| * clock setting |
| * CKPHS capture timing. 0:rising edge, 1:falling edge |
| * CKINIT clock initial level. 0:low, 1:high |
| * CKDLY clock delay. 0:no delay, 1:delay depending on FSTRT |
| * (FSTRT=0: 1 clock, FSTRT=1: 0.5 clock) |
| * |
| * frame setting |
| * FSPOL frame signal porarity. 0: low, 1: high |
| * FSTRT start frame timing |
| * 0: rising edge of clock, 1: falling edge of clock |
| */ |
| val1 = readl(priv->base + SSI_CKS); |
| val2 = readl(priv->base + SSI_FPS); |
| |
| switch (mode & (SPI_CPOL | SPI_CPHA)) { |
| case SPI_MODE_0: |
| /* CKPHS=1, CKINIT=0, CKDLY=1, FSTRT=0 */ |
| val1 |= SSI_CKS_CKPHS | SSI_CKS_CKDLY; |
| val1 &= ~SSI_CKS_CKINIT; |
| val2 &= ~SSI_FPS_FSTRT; |
| break; |
| case SPI_MODE_1: |
| /* CKPHS=0, CKINIT=0, CKDLY=0, FSTRT=1 */ |
| val1 &= ~(SSI_CKS_CKPHS | SSI_CKS_CKINIT | SSI_CKS_CKDLY); |
| val2 |= SSI_FPS_FSTRT; |
| break; |
| case SPI_MODE_2: |
| /* CKPHS=0, CKINIT=1, CKDLY=1, FSTRT=1 */ |
| val1 |= SSI_CKS_CKINIT | SSI_CKS_CKDLY; |
| val1 &= ~SSI_CKS_CKPHS; |
| val2 |= SSI_FPS_FSTRT; |
| break; |
| case SPI_MODE_3: |
| /* CKPHS=1, CKINIT=1, CKDLY=0, FSTRT=0 */ |
| val1 |= SSI_CKS_CKPHS | SSI_CKS_CKINIT; |
| val1 &= ~SSI_CKS_CKDLY; |
| val2 &= ~SSI_FPS_FSTRT; |
| break; |
| } |
| |
| writel(val1, priv->base + SSI_CKS); |
| writel(val2, priv->base + SSI_FPS); |
| |
| /* format */ |
| val1 = readl(priv->base + SSI_TXWDS); |
| val2 = readl(priv->base + SSI_RXWDS); |
| if (mode & SPI_LSB_FIRST) { |
| val1 |= FIELD_PREP(SSI_TXWDS_TDTF_MASK, 1); |
| val2 |= FIELD_PREP(SSI_RXWDS_RDTF_MASK, 1); |
| } |
| writel(val1, priv->base + SSI_TXWDS); |
| writel(val2, priv->base + SSI_RXWDS); |
| |
| priv->mode = mode; |
| |
| return 0; |
| } |
| |
| static int uniphier_spi_of_to_plat(struct udevice *bus) |
| { |
| struct uniphier_spi_plat *plat = bus->plat; |
| const void *blob = gd->fdt_blob; |
| int node = dev_of_offset(bus); |
| |
| plat->base = dev_read_addr_ptr(bus); |
| |
| plat->frequency = |
| fdtdec_get_int(blob, node, "spi-max-frequency", 12500000); |
| plat->deactivate_delay_us = |
| fdtdec_get_int(blob, node, "spi-deactivate-delay", 0); |
| plat->activate_delay_us = |
| fdtdec_get_int(blob, node, "spi-activate-delay", 0); |
| plat->speed_hz = plat->frequency / 2; |
| |
| return 0; |
| } |
| |
| static int uniphier_spi_probe(struct udevice *bus) |
| { |
| struct uniphier_spi_plat *plat = dev_get_plat(bus); |
| struct uniphier_spi_priv *priv = dev_get_priv(bus); |
| |
| priv->base = plat->base; |
| priv->fifo_depth = SSI_FIFO_DEPTH; |
| priv->bits_per_word = 8; |
| |
| return 0; |
| } |
| |
| static const struct dm_spi_ops uniphier_spi_ops = { |
| .claim_bus = uniphier_spi_claim_bus, |
| .release_bus = uniphier_spi_release_bus, |
| .xfer = uniphier_spi_xfer, |
| .set_speed = uniphier_spi_set_speed, |
| .set_mode = uniphier_spi_set_mode, |
| }; |
| |
| static const struct udevice_id uniphier_spi_ids[] = { |
| { .compatible = "socionext,uniphier-scssi" }, |
| { /* Sentinel */ } |
| }; |
| |
| U_BOOT_DRIVER(uniphier_spi) = { |
| .name = "uniphier_spi", |
| .id = UCLASS_SPI, |
| .of_match = uniphier_spi_ids, |
| .ops = &uniphier_spi_ops, |
| .of_to_plat = uniphier_spi_of_to_plat, |
| .plat_auto = sizeof(struct uniphier_spi_plat), |
| .priv_auto = sizeof(struct uniphier_spi_priv), |
| .probe = uniphier_spi_probe, |
| }; |