blob: b66bcfc4233ac1cedadbb3b6ce374d6bd6c67ab6 [file] [log] [blame]
// SPDX-License-Identifier: GPL-2.0
/*
* Copyright (C) 2022 MediaTek Inc. All Rights Reserved.
*
* Author: SkyLake.Huang <skylake.huang@mediatek.com>
*/
#include <clk.h>
#include <cpu_func.h>
#include <div64.h>
#include <dm.h>
#include <spi.h>
#include <spi-mem.h>
#include <stdbool.h>
#include <watchdog.h>
#include <dm/device.h>
#include <dm/device_compat.h>
#include <dm/devres.h>
#include <dm/pinctrl.h>
#include <linux/bitops.h>
#include <linux/dma-mapping.h>
#include <linux/io.h>
#include <linux/iopoll.h>
#include <linux/sizes.h>
#define SPI_CFG0_REG 0x0000
#define SPI_CFG1_REG 0x0004
#define SPI_TX_SRC_REG 0x0008
#define SPI_RX_DST_REG 0x000c
#define SPI_TX_DATA_REG 0x0010
#define SPI_RX_DATA_REG 0x0014
#define SPI_CMD_REG 0x0018
#define SPI_IRQ_REG 0x001c
#define SPI_STATUS_REG 0x0020
#define SPI_PAD_SEL_REG 0x0024
#define SPI_CFG2_REG 0x0028
#define SPI_TX_SRC_REG_64 0x002c
#define SPI_RX_DST_REG_64 0x0030
#define SPI_CFG3_IPM_REG 0x0040
#define SPI_CFG0_SCK_HIGH_OFFSET 0
#define SPI_CFG0_SCK_LOW_OFFSET 8
#define SPI_CFG0_CS_HOLD_OFFSET 16
#define SPI_CFG0_CS_SETUP_OFFSET 24
#define SPI_ADJUST_CFG0_CS_HOLD_OFFSET 0
#define SPI_ADJUST_CFG0_CS_SETUP_OFFSET 16
#define SPI_CFG1_CS_IDLE_OFFSET 0
#define SPI_CFG1_PACKET_LOOP_OFFSET 8
#define SPI_CFG1_PACKET_LENGTH_OFFSET 16
#define SPI_CFG1_GET_TICKDLY_OFFSET 29
#define SPI_CFG1_GET_TICKDLY_MASK GENMASK(31, 29)
#define SPI_CFG1_CS_IDLE_MASK 0xff
#define SPI_CFG1_PACKET_LOOP_MASK 0xff00
#define SPI_CFG1_PACKET_LENGTH_MASK 0x3ff0000
#define SPI_CFG1_IPM_PACKET_LENGTH_MASK GENMASK(31, 16)
#define SPI_CFG2_SCK_HIGH_OFFSET 0
#define SPI_CFG2_SCK_LOW_OFFSET 16
#define SPI_CFG2_SCK_HIGH_MASK GENMASK(15, 0)
#define SPI_CFG2_SCK_LOW_MASK GENMASK(31, 16)
#define SPI_CMD_ACT BIT(0)
#define SPI_CMD_RESUME BIT(1)
#define SPI_CMD_RST BIT(2)
#define SPI_CMD_PAUSE_EN BIT(4)
#define SPI_CMD_DEASSERT BIT(5)
#define SPI_CMD_SAMPLE_SEL BIT(6)
#define SPI_CMD_CS_POL BIT(7)
#define SPI_CMD_CPHA BIT(8)
#define SPI_CMD_CPOL BIT(9)
#define SPI_CMD_RX_DMA BIT(10)
#define SPI_CMD_TX_DMA BIT(11)
#define SPI_CMD_TXMSBF BIT(12)
#define SPI_CMD_RXMSBF BIT(13)
#define SPI_CMD_RX_ENDIAN BIT(14)
#define SPI_CMD_TX_ENDIAN BIT(15)
#define SPI_CMD_FINISH_IE BIT(16)
#define SPI_CMD_PAUSE_IE BIT(17)
#define SPI_CMD_IPM_NONIDLE_MODE BIT(19)
#define SPI_CMD_IPM_SPIM_LOOP BIT(21)
#define SPI_CMD_IPM_GET_TICKDLY_OFFSET 22
#define SPI_CMD_IPM_GET_TICKDLY_MASK GENMASK(24, 22)
#define PIN_MODE_CFG(x) ((x) / 2)
#define SPI_CFG3_IPM_PIN_MODE_OFFSET 0
#define SPI_CFG3_IPM_HALF_DUPLEX_DIR BIT(2)
#define SPI_CFG3_IPM_HALF_DUPLEX_EN BIT(3)
#define SPI_CFG3_IPM_XMODE_EN BIT(4)
#define SPI_CFG3_IPM_NODATA_FLAG BIT(5)
#define SPI_CFG3_IPM_CMD_BYTELEN_OFFSET 8
#define SPI_CFG3_IPM_ADDR_BYTELEN_OFFSET 12
#define SPI_CFG3_IPM_DUMMY_BYTELEN_OFFSET 16
#define SPI_CFG3_IPM_CMD_PIN_MODE_MASK GENMASK(1, 0)
#define SPI_CFG3_IPM_CMD_BYTELEN_MASK GENMASK(11, 8)
#define SPI_CFG3_IPM_ADDR_BYTELEN_MASK GENMASK(15, 12)
#define SPI_CFG3_IPM_DUMMY_BYTELEN_MASK GENMASK(19, 16)
#define MT8173_SPI_MAX_PAD_SEL 3
#define MTK_SPI_PAUSE_INT_STATUS 0x2
#define MTK_SPI_IDLE 0
#define MTK_SPI_PAUSED 1
#define MTK_SPI_MAX_FIFO_SIZE 32U
#define MTK_SPI_PACKET_SIZE 1024
#define MTK_SPI_IPM_PACKET_SIZE SZ_64K
#define MTK_SPI_IPM_PACKET_LOOP SZ_256
#define MTK_SPI_32BITS_MASK 0xffffffff
#define DMA_ADDR_EXT_BITS 36
#define DMA_ADDR_DEF_BITS 32
#define CLK_TO_US(freq, clkcnt) DIV_ROUND_UP((clkcnt), (freq) / 1000000)
/* struct mtk_spim_capability
* @enhance_timing: Some IC design adjust cfg register to enhance time accuracy
* @dma_ext: Some IC support DMA addr extension
* @ipm_design: The IPM IP design improves some features, and supports dual/quad mode
* @support_quad: Whether quad mode is supported
*/
struct mtk_spim_capability {
bool enhance_timing;
bool dma_ext;
bool ipm_design;
bool support_quad;
};
/* struct mtk_spim_priv
* @base: Base address of the spi controller
* @state: Controller state
* @sel_clk: Pad clock
* @spi_clk: Core clock
* @parent_clk: Parent clock (needed for mediatek,spi-ipm, upstream DTSI)
* @hclk: HCLK clock (needed for mediatek,spi-ipm, upstream DTSI)
* @pll_clk_rate: Controller's PLL source clock rate, which is different
* from SPI bus clock rate
* @xfer_len: Current length of data for transfer
* @hw_cap: Controller capabilities
* @tick_dly: Used to postpone SPI sampling time
* @sample_sel: Sample edge of MISO
* @dev: udevice of this spi controller
* @tx_dma: Tx DMA address
* @rx_dma: Rx DMA address
*/
struct mtk_spim_priv {
void __iomem *base;
u32 state;
struct clk sel_clk, spi_clk;
struct clk parent_clk, hclk;
u32 pll_clk_rate;
u32 xfer_len;
struct mtk_spim_capability hw_cap;
u32 tick_dly;
u32 sample_sel;
struct device *dev;
dma_addr_t tx_dma;
dma_addr_t rx_dma;
};
static void mtk_spim_reset(struct mtk_spim_priv *priv)
{
/* set the software reset bit in SPI_CMD_REG. */
setbits_le32(priv->base + SPI_CMD_REG, SPI_CMD_RST);
clrbits_le32(priv->base + SPI_CMD_REG, SPI_CMD_RST);
}
static int mtk_spim_hw_init(struct spi_slave *slave)
{
struct udevice *bus = dev_get_parent(slave->dev);
struct mtk_spim_priv *priv = dev_get_priv(bus);
u16 cpha, cpol;
u32 reg_val;
cpha = slave->mode & SPI_CPHA ? 1 : 0;
cpol = slave->mode & SPI_CPOL ? 1 : 0;
if (priv->hw_cap.enhance_timing) {
if (priv->hw_cap.ipm_design) {
/* CFG3 reg only used for spi-mem,
* here write to default value
*/
writel(0x0, priv->base + SPI_CFG3_IPM_REG);
clrsetbits_le32(priv->base + SPI_CMD_REG,
SPI_CMD_IPM_GET_TICKDLY_MASK,
priv->tick_dly <<
SPI_CMD_IPM_GET_TICKDLY_OFFSET);
} else {
clrsetbits_le32(priv->base + SPI_CFG1_REG,
SPI_CFG1_GET_TICKDLY_MASK,
priv->tick_dly <<
SPI_CFG1_GET_TICKDLY_OFFSET);
}
}
reg_val = readl(priv->base + SPI_CMD_REG);
if (priv->hw_cap.ipm_design) {
/* SPI transfer without idle time until packet length done */
reg_val |= SPI_CMD_IPM_NONIDLE_MODE;
if (slave->mode & SPI_LOOP)
reg_val |= SPI_CMD_IPM_SPIM_LOOP;
else
reg_val &= ~SPI_CMD_IPM_SPIM_LOOP;
}
if (cpha)
reg_val |= SPI_CMD_CPHA;
else
reg_val &= ~SPI_CMD_CPHA;
if (cpol)
reg_val |= SPI_CMD_CPOL;
else
reg_val &= ~SPI_CMD_CPOL;
/* set the mlsbx and mlsbtx */
if (slave->mode & SPI_LSB_FIRST) {
reg_val &= ~SPI_CMD_TXMSBF;
reg_val &= ~SPI_CMD_RXMSBF;
} else {
reg_val |= SPI_CMD_TXMSBF;
reg_val |= SPI_CMD_RXMSBF;
}
/* do not reverse tx/rx endian */
reg_val &= ~SPI_CMD_TX_ENDIAN;
reg_val &= ~SPI_CMD_RX_ENDIAN;
if (priv->hw_cap.enhance_timing) {
/* set CS polarity */
if (slave->mode & SPI_CS_HIGH)
reg_val |= SPI_CMD_CS_POL;
else
reg_val &= ~SPI_CMD_CS_POL;
if (priv->sample_sel)
reg_val |= SPI_CMD_SAMPLE_SEL;
else
reg_val &= ~SPI_CMD_SAMPLE_SEL;
}
/* Disable interrupt enable for pause mode & normal mode */
reg_val &= ~(SPI_CMD_PAUSE_IE | SPI_CMD_FINISH_IE);
/* disable dma mode */
reg_val &= ~(SPI_CMD_TX_DMA | SPI_CMD_RX_DMA);
/* disable deassert mode */
reg_val &= ~SPI_CMD_DEASSERT;
writel(reg_val, priv->base + SPI_CMD_REG);
return 0;
}
static void mtk_spim_prepare_transfer(struct mtk_spim_priv *priv,
u32 speed_hz)
{
u32 div, sck_time, cs_time, reg_val;
if (speed_hz <= priv->pll_clk_rate / 4)
div = DIV_ROUND_UP(priv->pll_clk_rate, speed_hz);
else
div = 4;
sck_time = (div + 1) / 2;
cs_time = sck_time * 2;
if (priv->hw_cap.enhance_timing) {
reg_val = ((sck_time - 1) & 0xffff)
<< SPI_CFG2_SCK_HIGH_OFFSET;
reg_val |= ((sck_time - 1) & 0xffff)
<< SPI_CFG2_SCK_LOW_OFFSET;
writel(reg_val, priv->base + SPI_CFG2_REG);
reg_val = ((cs_time - 1) & 0xffff)
<< SPI_ADJUST_CFG0_CS_HOLD_OFFSET;
reg_val |= ((cs_time - 1) & 0xffff)
<< SPI_ADJUST_CFG0_CS_SETUP_OFFSET;
writel(reg_val, priv->base + SPI_CFG0_REG);
} else {
reg_val = ((sck_time - 1) & 0xff)
<< SPI_CFG0_SCK_HIGH_OFFSET;
reg_val |= ((sck_time - 1) & 0xff) << SPI_CFG0_SCK_LOW_OFFSET;
reg_val |= ((cs_time - 1) & 0xff) << SPI_CFG0_CS_HOLD_OFFSET;
reg_val |= ((cs_time - 1) & 0xff) << SPI_CFG0_CS_SETUP_OFFSET;
writel(reg_val, priv->base + SPI_CFG0_REG);
}
reg_val = readl(priv->base + SPI_CFG1_REG);
reg_val &= ~SPI_CFG1_CS_IDLE_MASK;
reg_val |= ((cs_time - 1) & 0xff) << SPI_CFG1_CS_IDLE_OFFSET;
writel(reg_val, priv->base + SPI_CFG1_REG);
}
/**
* mtk_spim_setup_packet() - setup packet format.
* @priv: controller priv
*
* This controller sents/receives data in packets. The packet size is
* configurable.
*
* This function calculates the maximum packet size available for current
* data, and calculates the number of packets required to sent/receive data
* as much as possible.
*/
static void mtk_spim_setup_packet(struct mtk_spim_priv *priv)
{
u32 packet_size, packet_loop, reg_val;
/* Calculate maximum packet size */
if (priv->hw_cap.ipm_design)
packet_size = min_t(u32,
priv->xfer_len,
MTK_SPI_IPM_PACKET_SIZE);
else
packet_size = min_t(u32,
priv->xfer_len,
MTK_SPI_PACKET_SIZE);
/* Calculates number of packets to sent/receive */
packet_loop = priv->xfer_len / packet_size;
reg_val = readl(priv->base + SPI_CFG1_REG);
if (priv->hw_cap.ipm_design)
reg_val &= ~SPI_CFG1_IPM_PACKET_LENGTH_MASK;
else
reg_val &= ~SPI_CFG1_PACKET_LENGTH_MASK;
reg_val |= (packet_size - 1) << SPI_CFG1_PACKET_LENGTH_OFFSET;
reg_val &= ~SPI_CFG1_PACKET_LOOP_MASK;
reg_val |= (packet_loop - 1) << SPI_CFG1_PACKET_LOOP_OFFSET;
writel(reg_val, priv->base + SPI_CFG1_REG);
}
static void mtk_spim_enable_transfer(struct mtk_spim_priv *priv)
{
u32 cmd;
cmd = readl(priv->base + SPI_CMD_REG);
if (priv->state == MTK_SPI_IDLE)
cmd |= SPI_CMD_ACT;
else
cmd |= SPI_CMD_RESUME;
writel(cmd, priv->base + SPI_CMD_REG);
}
static bool mtk_spim_supports_op(struct spi_slave *slave,
const struct spi_mem_op *op)
{
struct udevice *bus = dev_get_parent(slave->dev);
struct mtk_spim_priv *priv = dev_get_priv(bus);
if (op->cmd.buswidth == 0 || op->cmd.buswidth > 4 ||
op->addr.buswidth > 4 || op->dummy.buswidth > 4 ||
op->data.buswidth > 4)
return false;
if (!priv->hw_cap.support_quad && (op->cmd.buswidth > 2 ||
op->addr.buswidth > 2 || op->dummy.buswidth > 2 ||
op->data.buswidth > 2))
return false;
if (op->addr.nbytes && op->dummy.nbytes &&
op->addr.buswidth != op->dummy.buswidth)
return false;
if (op->addr.nbytes + op->dummy.nbytes > 16)
return false;
if (op->data.nbytes > MTK_SPI_IPM_PACKET_SIZE) {
if (op->data.nbytes / MTK_SPI_IPM_PACKET_SIZE >
MTK_SPI_IPM_PACKET_LOOP ||
op->data.nbytes % MTK_SPI_IPM_PACKET_SIZE != 0)
return false;
}
return true;
}
static void mtk_spim_setup_dma_xfer(struct mtk_spim_priv *priv,
const struct spi_mem_op *op)
{
writel((u32)(priv->tx_dma & MTK_SPI_32BITS_MASK),
priv->base + SPI_TX_SRC_REG);
if (priv->hw_cap.dma_ext)
writel((u32)(priv->tx_dma >> 32),
priv->base + SPI_TX_SRC_REG_64);
if (op->data.dir == SPI_MEM_DATA_IN) {
writel((u32)(priv->rx_dma & MTK_SPI_32BITS_MASK),
priv->base + SPI_RX_DST_REG);
if (priv->hw_cap.dma_ext)
writel((u32)(priv->rx_dma >> 32),
priv->base + SPI_RX_DST_REG_64);
}
}
static int mtk_spim_transfer_wait(struct spi_slave *slave,
const struct spi_mem_op *op)
{
struct udevice *bus = dev_get_parent(slave->dev);
struct mtk_spim_priv *priv = dev_get_priv(bus);
u32 pll_clk, sck_l, sck_h, clk_count, reg;
ulong us = 1;
int ret = 0;
if (op->data.dir == SPI_MEM_NO_DATA)
clk_count = 32;
else
clk_count = op->data.nbytes;
pll_clk = priv->pll_clk_rate;
sck_l = readl(priv->base + SPI_CFG2_REG) >> SPI_CFG2_SCK_LOW_OFFSET;
sck_h = readl(priv->base + SPI_CFG2_REG) & SPI_CFG2_SCK_HIGH_MASK;
do_div(pll_clk, sck_l + sck_h + 2);
us = CLK_TO_US(pll_clk, clk_count * 8);
us += 1000 * 1000; /* 1s tolerance */
if (us > UINT_MAX)
us = UINT_MAX;
ret = readl_poll_timeout(priv->base + SPI_STATUS_REG, reg,
reg & 0x1, us);
if (ret < 0) {
dev_err(priv->dev, "transfer timeout, val: 0x%lx\n", us);
return -ETIMEDOUT;
}
return 0;
}
static int mtk_spim_exec_op(struct spi_slave *slave,
const struct spi_mem_op *op)
{
struct udevice *bus = dev_get_parent(slave->dev);
struct mtk_spim_priv *priv = dev_get_priv(bus);
u32 reg_val, nio = 1, tx_size;
char *tx_tmp_buf;
char *rx_tmp_buf;
int i, ret = 0;
mtk_spim_reset(priv);
mtk_spim_hw_init(slave);
mtk_spim_prepare_transfer(priv, slave->max_hz);
reg_val = readl(priv->base + SPI_CFG3_IPM_REG);
/* opcode byte len */
reg_val &= ~SPI_CFG3_IPM_CMD_BYTELEN_MASK;
reg_val |= 1 << SPI_CFG3_IPM_CMD_BYTELEN_OFFSET;
/* addr & dummy byte len */
if (op->addr.nbytes || op->dummy.nbytes)
reg_val |= (op->addr.nbytes + op->dummy.nbytes) <<
SPI_CFG3_IPM_ADDR_BYTELEN_OFFSET;
/* data byte len */
if (!op->data.nbytes) {
reg_val |= SPI_CFG3_IPM_NODATA_FLAG;
writel(0, priv->base + SPI_CFG1_REG);
} else {
reg_val &= ~SPI_CFG3_IPM_NODATA_FLAG;
priv->xfer_len = op->data.nbytes;
mtk_spim_setup_packet(priv);
}
if (op->addr.nbytes || op->dummy.nbytes) {
if (op->addr.buswidth == 1 || op->dummy.buswidth == 1)
reg_val |= SPI_CFG3_IPM_XMODE_EN;
else
reg_val &= ~SPI_CFG3_IPM_XMODE_EN;
}
if (op->addr.buswidth == 2 ||
op->dummy.buswidth == 2 ||
op->data.buswidth == 2)
nio = 2;
else if (op->addr.buswidth == 4 ||
op->dummy.buswidth == 4 ||
op->data.buswidth == 4)
nio = 4;
reg_val &= ~SPI_CFG3_IPM_CMD_PIN_MODE_MASK;
reg_val |= PIN_MODE_CFG(nio) << SPI_CFG3_IPM_PIN_MODE_OFFSET;
reg_val |= SPI_CFG3_IPM_HALF_DUPLEX_EN;
if (op->data.dir == SPI_MEM_DATA_IN)
reg_val |= SPI_CFG3_IPM_HALF_DUPLEX_DIR;
else
reg_val &= ~SPI_CFG3_IPM_HALF_DUPLEX_DIR;
writel(reg_val, priv->base + SPI_CFG3_IPM_REG);
tx_size = 1 + op->addr.nbytes + op->dummy.nbytes;
if (op->data.dir == SPI_MEM_DATA_OUT)
tx_size += op->data.nbytes;
tx_size = max(tx_size, (u32)32);
/* Fill up tx data */
tx_tmp_buf = kzalloc(tx_size, GFP_KERNEL);
if (!tx_tmp_buf) {
ret = -ENOMEM;
goto exit;
}
tx_tmp_buf[0] = op->cmd.opcode;
if (op->addr.nbytes) {
for (i = 0; i < op->addr.nbytes; i++)
tx_tmp_buf[i + 1] = op->addr.val >>
(8 * (op->addr.nbytes - i - 1));
}
if (op->dummy.nbytes)
memset(tx_tmp_buf + op->addr.nbytes + 1, 0xff,
op->dummy.nbytes);
if (op->data.nbytes && op->data.dir == SPI_MEM_DATA_OUT)
memcpy(tx_tmp_buf + op->dummy.nbytes + op->addr.nbytes + 1,
op->data.buf.out, op->data.nbytes);
/* Finish filling up tx data */
priv->tx_dma = dma_map_single(tx_tmp_buf, tx_size, DMA_TO_DEVICE);
if (dma_mapping_error(priv->dev, priv->tx_dma)) {
ret = -ENOMEM;
goto tx_free;
}
if (op->data.dir == SPI_MEM_DATA_IN) {
if (!IS_ALIGNED((size_t)op->data.buf.in, 4)) {
rx_tmp_buf = kzalloc(op->data.nbytes, GFP_KERNEL);
if (!rx_tmp_buf) {
ret = -ENOMEM;
goto tx_unmap;
}
} else {
rx_tmp_buf = op->data.buf.in;
}
priv->rx_dma = dma_map_single(rx_tmp_buf, op->data.nbytes,
DMA_FROM_DEVICE);
if (dma_mapping_error(priv->dev, priv->rx_dma)) {
ret = -ENOMEM;
goto rx_free;
}
}
reg_val = readl(priv->base + SPI_CMD_REG);
reg_val |= SPI_CMD_TX_DMA;
if (op->data.dir == SPI_MEM_DATA_IN)
reg_val |= SPI_CMD_RX_DMA;
writel(reg_val, priv->base + SPI_CMD_REG);
mtk_spim_setup_dma_xfer(priv, op);
mtk_spim_enable_transfer(priv);
/* Wait for the interrupt. */
ret = mtk_spim_transfer_wait(slave, op);
if (ret)
goto rx_unmap;
if (op->data.dir == SPI_MEM_DATA_IN &&
!IS_ALIGNED((size_t)op->data.buf.in, 4))
memcpy(op->data.buf.in, rx_tmp_buf, op->data.nbytes);
rx_unmap:
/* spi disable dma */
reg_val = readl(priv->base + SPI_CMD_REG);
reg_val &= ~SPI_CMD_TX_DMA;
if (op->data.dir == SPI_MEM_DATA_IN)
reg_val &= ~SPI_CMD_RX_DMA;
writel(reg_val, priv->base + SPI_CMD_REG);
writel(0, priv->base + SPI_TX_SRC_REG);
writel(0, priv->base + SPI_RX_DST_REG);
if (op->data.dir == SPI_MEM_DATA_IN)
dma_unmap_single(priv->rx_dma,
op->data.nbytes, DMA_FROM_DEVICE);
rx_free:
if (op->data.dir == SPI_MEM_DATA_IN &&
!IS_ALIGNED((size_t)op->data.buf.in, 4))
kfree(rx_tmp_buf);
tx_unmap:
dma_unmap_single(priv->tx_dma,
tx_size, DMA_TO_DEVICE);
tx_free:
kfree(tx_tmp_buf);
exit:
return ret;
}
static int mtk_spim_adjust_op_size(struct spi_slave *slave,
struct spi_mem_op *op)
{
int opcode_len;
if (!op->data.nbytes)
return 0;
if (op->data.dir != SPI_MEM_NO_DATA) {
opcode_len = 1 + op->addr.nbytes + op->dummy.nbytes;
if (opcode_len + op->data.nbytes > MTK_SPI_IPM_PACKET_SIZE) {
op->data.nbytes = MTK_SPI_IPM_PACKET_SIZE - opcode_len;
/* force data buffer dma-aligned. */
op->data.nbytes -= op->data.nbytes % 4;
}
}
return 0;
}
static int mtk_spim_get_attr(struct mtk_spim_priv *priv, struct udevice *dev)
{
int ret;
priv->hw_cap.enhance_timing = dev_read_bool(dev, "enhance_timing");
priv->hw_cap.dma_ext = dev_read_bool(dev, "dma_ext");
priv->hw_cap.ipm_design = dev_read_bool(dev, "ipm_design");
priv->hw_cap.support_quad = dev_read_bool(dev, "support_quad");
ret = dev_read_u32(dev, "tick_dly", &priv->tick_dly);
if (ret < 0)
dev_err(priv->dev, "tick dly not set.\n");
ret = dev_read_u32(dev, "sample_sel", &priv->sample_sel);
if (ret < 0)
dev_err(priv->dev, "sample sel not set.\n");
return ret;
}
static int mtk_spim_probe(struct udevice *dev)
{
struct mtk_spim_priv *priv = dev_get_priv(dev);
int ret;
priv->base = devfdt_get_addr_ptr(dev);
if (!priv->base)
return -EINVAL;
/*
* Upstream linux driver for ipm design enable all the modes
* and setup the calibrarion values directly in the driver with
* standard values.
*/
if (device_is_compatible(dev, "mediatek,spi-ipm")) {
priv->hw_cap.enhance_timing = true;
priv->hw_cap.dma_ext = true;
priv->hw_cap.ipm_design = true;
priv->hw_cap.support_quad = true;
priv->sample_sel = 0;
priv->tick_dly = 2;
} else {
mtk_spim_get_attr(priv, dev);
}
ret = clk_get_by_name(dev, "sel-clk", &priv->sel_clk);
if (ret < 0) {
dev_err(dev, "failed to get sel-clk\n");
return ret;
}
ret = clk_get_by_name(dev, "spi-clk", &priv->spi_clk);
if (ret < 0) {
dev_err(dev, "failed to get spi-clk\n");
return ret;
}
/*
* Upstream DTSI use a different compatible that provide additional
* clock instead of the assigned-clock implementation.
*/
if (device_is_compatible(dev, "mediatek,spi-ipm")) {
ret = clk_get_by_name(dev, "parent-clk", &priv->parent_clk);
if (ret < 0) {
dev_err(dev, "failed to get parent-clk\n");
return ret;
}
ret = clk_get_by_name(dev, "hclk", &priv->hclk);
if (ret < 0) {
dev_err(dev, "failed to get hclk\n");
return ret;
}
clk_enable(&priv->parent_clk);
clk_set_parent(&priv->sel_clk, &priv->parent_clk);
clk_enable(&priv->hclk);
}
clk_enable(&priv->spi_clk);
clk_enable(&priv->sel_clk);
priv->pll_clk_rate = clk_get_rate(&priv->spi_clk);
if (priv->pll_clk_rate == 0)
return -EINVAL;
return 0;
}
static int mtk_spim_set_speed(struct udevice *dev, uint speed)
{
return 0;
}
static int mtk_spim_set_mode(struct udevice *dev, uint mode)
{
return 0;
}
static const struct spi_controller_mem_ops mtk_spim_mem_ops = {
.adjust_op_size = mtk_spim_adjust_op_size,
.supports_op = mtk_spim_supports_op,
.exec_op = mtk_spim_exec_op
};
static const struct dm_spi_ops mtk_spim_ops = {
.mem_ops = &mtk_spim_mem_ops,
.set_speed = mtk_spim_set_speed,
.set_mode = mtk_spim_set_mode,
};
static const struct udevice_id mtk_spim_ids[] = {
{ .compatible = "mediatek,ipm-spi" },
{ .compatible = "mediatek,spi-ipm", },
{}
};
U_BOOT_DRIVER(mtk_spim) = {
.name = "mtk_spim",
.id = UCLASS_SPI,
.of_match = mtk_spim_ids,
.ops = &mtk_spim_ops,
.priv_auto = sizeof(struct mtk_spim_priv),
.probe = mtk_spim_probe,
};