[PATCH v1 1/1] spi: spi-mt65xx: add cs timing support
Leilk.Liu
leilk.liu at mediatek.com
Tue Jan 5 02:57:20 EST 2021
On Thu, 2020-12-31 at 17:57 +0800, Mason.Zhang at mediatek.com wrote:
> From: mtk22786 <Mason.Zhang at mediatek.com>
>
> spi user can set cs_setup, cs_hold, cs_inactive by call function:
> spi_set_cs_timing. If user have not call this, spi master will use
> default cs timing.
>
> Change-Id: I0189bab9a066d384e1f265846dac46816bf4561c
> Signed-off-by: Mason.Zhang
> ---
> drivers/spi/spi-mt65xx.c | 74 ++++++++++++++++++++++++++++++++++------
> 1 file changed, 63 insertions(+), 11 deletions(-)
>
> diff --git a/drivers/spi/spi-mt65xx.c b/drivers/spi/spi-mt65xx.c
> index 5d643051bf3d..23d94eb9a465 100644
> --- a/drivers/spi/spi-mt65xx.c
> +++ b/drivers/spi/spi-mt65xx.c
> @@ -284,11 +284,38 @@ static void mtk_spi_set_cs(struct spi_device *spi, bool enable)
> }
> }
>
> +static int mtk_set_cs_timing(struct spi_device *spi,
> + struct spi_delay *setup,
> + struct spi_delay *hold,
> + struct spi_delay *inactive)
> +{
> + u32 len = sizeof(struct spi_delay);
> +
> + if (setup)
> + memcpy(&spi->controller->cs_setup, setup, len);
> + else
> + memset(&spi->controller->cs_setup, 0, len);
> +
> + if (hold)
> + memcpy(&spi->controller->cs_hold, hold, len);
> + else
> + memset(&spi->controller->cs_hold, 0, len);
> +
> + if (inactive)
> + memcpy(&spi->controller->cs_inactive, inactive, len);
> + else
> + memset(&spi->controller->cs_inactive, 0, len);
> +
> + return 0;
> +}
> +
> static void mtk_spi_prepare_transfer(struct spi_master *master,
> - struct spi_transfer *xfer)
> + struct spi_transfer *xfer, struct spi_device *spi)
> {
> - u32 spi_clk_hz, div, sck_time, cs_time, reg_val;
> + u32 spi_clk_hz, div, sck_time, cs_time, reg_val, delay = 0;
> struct mtk_spi *mdata = spi_master_get_devdata(master);
> + u32 cs_setuptime, cs_holdtime, cs_idletime = 0;
> + u32 clk_period_ns = 0;
>
> spi_clk_hz = clk_get_rate(mdata->spi_clk);
> if (xfer->speed_hz < spi_clk_hz / 2)
> @@ -297,31 +324,55 @@ static void mtk_spi_prepare_transfer(struct spi_master *master,
> div = 1;
>
> sck_time = (div + 1) / 2;
> + /* set default cs timing */
> cs_time = sck_time * 2;
>
> + clk_period_ns = 1000000000 / xfer->speed_hz;
> +
> + delay = spi_delay_to_ns(&master->cs_setup, xfer);
> + cs_setuptime = delay / clk_period_ns;
> +
> + delay = spi_delay_to_ns(&master->cs_hold, xfer);
> + cs_holdtime = delay / clk_period_ns;
> +
> + delay = spi_delay_to_ns(&master->cs_inactive, xfer);
> + cs_idletime = delay / clk_period_ns;
> +
> + if (cs_setuptime < cs_time)
> + cs_setuptime = cs_time;
> +
> + if (cs_holdtime < cs_time)
> + cs_holdtime = cs_time;
> +
> + if (cs_idletime < cs_time)
> + cs_idletime = cs_time;
> +
> if (mdata->dev_comp->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, mdata->base + SPI_CFG2_REG);
> - reg_val = (((cs_time - 1) & 0xffff)
> + reg_val = (((cs_holdtime - 1) & 0xffff)
> << SPI_ADJUST_CFG0_CS_HOLD_OFFSET);
> - reg_val |= (((cs_time - 1) & 0xffff)
> + reg_val |= (((cs_setuptime - 1) & 0xffff)
> << SPI_ADJUST_CFG0_CS_SETUP_OFFSET);
> writel(reg_val, mdata->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);
> + << SPI_CFG0_SCK_HIGH_OFFSET);
> + reg_val |= (((sck_time - 1) & 0xff)
> + << SPI_CFG0_SCK_LOW_OFFSET);
> + reg_val |= (((cs_holdtime - 1) & 0xff)
> + << SPI_CFG0_CS_HOLD_OFFSET);
> + reg_val |= (((cs_setuptime - 1) & 0xff)
> + << SPI_CFG0_CS_SETUP_OFFSET);
> writel(reg_val, mdata->base + SPI_CFG0_REG);
> }
>
> reg_val = readl(mdata->base + SPI_CFG1_REG);
> reg_val &= ~SPI_CFG1_CS_IDLE_MASK;
> - reg_val |= (((cs_time - 1) & 0xff) << SPI_CFG1_CS_IDLE_OFFSET);
> + reg_val |= (((cs_idletime - 1) & 0xff) << SPI_CFG1_CS_IDLE_OFFSET);
> writel(reg_val, mdata->base + SPI_CFG1_REG);
> }
>
> @@ -430,7 +481,7 @@ static int mtk_spi_fifo_transfer(struct spi_master *master,
> mdata->cur_transfer = xfer;
> mdata->xfer_len = min(MTK_SPI_MAX_FIFO_SIZE, xfer->len);
> mdata->num_xfered = 0;
> - mtk_spi_prepare_transfer(master, xfer);
> + mtk_spi_prepare_transfer(master, xfer, spi);
> mtk_spi_setup_packet(master);
>
> cnt = xfer->len / 4;
> @@ -462,7 +513,7 @@ static int mtk_spi_dma_transfer(struct spi_master *master,
> mdata->cur_transfer = xfer;
> mdata->num_xfered = 0;
>
> - mtk_spi_prepare_transfer(master, xfer);
> + mtk_spi_prepare_transfer(master, xfer, spi);
>
> cmd = readl(mdata->base + SPI_CMD_REG);
> if (xfer->tx_buf)
> @@ -644,6 +695,7 @@ static int mtk_spi_probe(struct platform_device *pdev)
> master->transfer_one = mtk_spi_transfer_one;
> master->can_dma = mtk_spi_can_dma;
> master->setup = mtk_spi_setup;
> + master->set_cs_timing = mtk_set_cs_timing;
while add set_cs_timing callback, this driver can't support multiple
devices on one bus(use "cs-gpios" property) with gpio cs timing.
>
> of_id = of_match_node(mtk_spi_of_match, pdev->dev.of_node);
> if (!of_id) {
More information about the Linux-mediatek
mailing list