.. | .. |
---|
| 1 | +// SPDX-License-Identifier: GPL-2.0-only |
---|
1 | 2 | /* |
---|
2 | 3 | * Copyright (c) 2015 MediaTek Inc. |
---|
3 | 4 | * Author: Leilk Liu <leilk.liu@mediatek.com> |
---|
4 | | - * |
---|
5 | | - * This program is free software; you can redistribute it and/or modify |
---|
6 | | - * it under the terms of the GNU General Public License version 2 as |
---|
7 | | - * published by the Free Software Foundation. |
---|
8 | | - * |
---|
9 | | - * This program is distributed in the hope that it will be useful, |
---|
10 | | - * but WITHOUT ANY WARRANTY; without even the implied warranty of |
---|
11 | | - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
---|
12 | | - * GNU General Public License for more details. |
---|
13 | 5 | */ |
---|
14 | 6 | |
---|
15 | 7 | #include <linux/clk.h> |
---|
.. | .. |
---|
25 | 17 | #include <linux/platform_data/spi-mt65xx.h> |
---|
26 | 18 | #include <linux/pm_runtime.h> |
---|
27 | 19 | #include <linux/spi/spi.h> |
---|
| 20 | +#include <linux/dma-mapping.h> |
---|
28 | 21 | |
---|
29 | 22 | #define SPI_CFG0_REG 0x0000 |
---|
30 | 23 | #define SPI_CFG1_REG 0x0004 |
---|
.. | .. |
---|
36 | 29 | #define SPI_STATUS0_REG 0x001c |
---|
37 | 30 | #define SPI_PAD_SEL_REG 0x0024 |
---|
38 | 31 | #define SPI_CFG2_REG 0x0028 |
---|
| 32 | +#define SPI_TX_SRC_REG_64 0x002c |
---|
| 33 | +#define SPI_RX_DST_REG_64 0x0030 |
---|
39 | 34 | |
---|
40 | 35 | #define SPI_CFG0_SCK_HIGH_OFFSET 0 |
---|
41 | 36 | #define SPI_CFG0_SCK_LOW_OFFSET 8 |
---|
.. | .. |
---|
82 | 77 | |
---|
83 | 78 | #define MTK_SPI_MAX_FIFO_SIZE 32U |
---|
84 | 79 | #define MTK_SPI_PACKET_SIZE 1024 |
---|
| 80 | +#define MTK_SPI_32BITS_MASK (0xffffffff) |
---|
| 81 | + |
---|
| 82 | +#define DMA_ADDR_EXT_BITS (36) |
---|
| 83 | +#define DMA_ADDR_DEF_BITS (32) |
---|
85 | 84 | |
---|
86 | 85 | struct mtk_spi_compatible { |
---|
87 | 86 | bool need_pad_sel; |
---|
.. | .. |
---|
89 | 88 | bool must_tx; |
---|
90 | 89 | /* some IC design adjust cfg register to enhance time accuracy */ |
---|
91 | 90 | bool enhance_timing; |
---|
| 91 | + /* some IC support DMA addr extension */ |
---|
| 92 | + bool dma_ext; |
---|
92 | 93 | }; |
---|
93 | 94 | |
---|
94 | 95 | struct mtk_spi { |
---|
.. | .. |
---|
111 | 112 | .must_tx = true, |
---|
112 | 113 | }; |
---|
113 | 114 | |
---|
| 115 | +static const struct mtk_spi_compatible mt6765_compat = { |
---|
| 116 | + .need_pad_sel = true, |
---|
| 117 | + .must_tx = true, |
---|
| 118 | + .enhance_timing = true, |
---|
| 119 | + .dma_ext = true, |
---|
| 120 | +}; |
---|
| 121 | + |
---|
114 | 122 | static const struct mtk_spi_compatible mt7622_compat = { |
---|
115 | 123 | .must_tx = true, |
---|
116 | 124 | .enhance_timing = true, |
---|
.. | .. |
---|
121 | 129 | .must_tx = true, |
---|
122 | 130 | }; |
---|
123 | 131 | |
---|
| 132 | +static const struct mtk_spi_compatible mt8183_compat = { |
---|
| 133 | + .need_pad_sel = true, |
---|
| 134 | + .must_tx = true, |
---|
| 135 | + .enhance_timing = true, |
---|
| 136 | +}; |
---|
| 137 | + |
---|
124 | 138 | /* |
---|
125 | 139 | * A piece of default chip info unless the platform |
---|
126 | 140 | * supplies it. |
---|
127 | 141 | */ |
---|
128 | 142 | static const struct mtk_chip_config mtk_default_chip_info = { |
---|
129 | | - .rx_mlsb = 1, |
---|
130 | | - .tx_mlsb = 1, |
---|
131 | | - .cs_pol = 0, |
---|
132 | 143 | .sample_sel = 0, |
---|
133 | 144 | }; |
---|
134 | 145 | |
---|
.. | .. |
---|
142 | 153 | { .compatible = "mediatek,mt6589-spi", |
---|
143 | 154 | .data = (void *)&mtk_common_compat, |
---|
144 | 155 | }, |
---|
| 156 | + { .compatible = "mediatek,mt6765-spi", |
---|
| 157 | + .data = (void *)&mt6765_compat, |
---|
| 158 | + }, |
---|
145 | 159 | { .compatible = "mediatek,mt7622-spi", |
---|
| 160 | + .data = (void *)&mt7622_compat, |
---|
| 161 | + }, |
---|
| 162 | + { .compatible = "mediatek,mt7629-spi", |
---|
146 | 163 | .data = (void *)&mt7622_compat, |
---|
147 | 164 | }, |
---|
148 | 165 | { .compatible = "mediatek,mt8135-spi", |
---|
.. | .. |
---|
150 | 167 | }, |
---|
151 | 168 | { .compatible = "mediatek,mt8173-spi", |
---|
152 | 169 | .data = (void *)&mt8173_compat, |
---|
| 170 | + }, |
---|
| 171 | + { .compatible = "mediatek,mt8183-spi", |
---|
| 172 | + .data = (void *)&mt8183_compat, |
---|
| 173 | + }, |
---|
| 174 | + { .compatible = "mediatek,mt8192-spi", |
---|
| 175 | + .data = (void *)&mt6765_compat, |
---|
153 | 176 | }, |
---|
154 | 177 | {} |
---|
155 | 178 | }; |
---|
.. | .. |
---|
192 | 215 | reg_val &= ~SPI_CMD_CPOL; |
---|
193 | 216 | |
---|
194 | 217 | /* set the mlsbx and mlsbtx */ |
---|
195 | | - if (chip_config->tx_mlsb) |
---|
196 | | - reg_val |= SPI_CMD_TXMSBF; |
---|
197 | | - else |
---|
| 218 | + if (spi->mode & SPI_LSB_FIRST) { |
---|
198 | 219 | reg_val &= ~SPI_CMD_TXMSBF; |
---|
199 | | - if (chip_config->rx_mlsb) |
---|
200 | | - reg_val |= SPI_CMD_RXMSBF; |
---|
201 | | - else |
---|
202 | 220 | reg_val &= ~SPI_CMD_RXMSBF; |
---|
| 221 | + } else { |
---|
| 222 | + reg_val |= SPI_CMD_TXMSBF; |
---|
| 223 | + reg_val |= SPI_CMD_RXMSBF; |
---|
| 224 | + } |
---|
203 | 225 | |
---|
204 | 226 | /* set the tx/rx endian */ |
---|
205 | 227 | #ifdef __LITTLE_ENDIAN |
---|
.. | .. |
---|
211 | 233 | #endif |
---|
212 | 234 | |
---|
213 | 235 | if (mdata->dev_comp->enhance_timing) { |
---|
214 | | - if (chip_config->cs_pol) |
---|
| 236 | + /* set CS polarity */ |
---|
| 237 | + if (spi->mode & SPI_CS_HIGH) |
---|
215 | 238 | reg_val |= SPI_CMD_CS_POL; |
---|
216 | 239 | else |
---|
217 | 240 | reg_val &= ~SPI_CMD_CS_POL; |
---|
| 241 | + |
---|
218 | 242 | if (chip_config->sample_sel) |
---|
219 | 243 | reg_val |= SPI_CMD_SAMPLE_SEL; |
---|
220 | 244 | else |
---|
.. | .. |
---|
244 | 268 | { |
---|
245 | 269 | u32 reg_val; |
---|
246 | 270 | struct mtk_spi *mdata = spi_master_get_devdata(spi->master); |
---|
| 271 | + |
---|
| 272 | + if (spi->mode & SPI_CS_HIGH) |
---|
| 273 | + enable = !enable; |
---|
247 | 274 | |
---|
248 | 275 | reg_val = readl(mdata->base + SPI_CMD_REG); |
---|
249 | 276 | if (!enable) { |
---|
.. | .. |
---|
371 | 398 | { |
---|
372 | 399 | struct mtk_spi *mdata = spi_master_get_devdata(master); |
---|
373 | 400 | |
---|
374 | | - if (mdata->tx_sgl) |
---|
375 | | - writel(xfer->tx_dma, mdata->base + SPI_TX_SRC_REG); |
---|
376 | | - if (mdata->rx_sgl) |
---|
377 | | - writel(xfer->rx_dma, mdata->base + SPI_RX_DST_REG); |
---|
| 401 | + if (mdata->tx_sgl) { |
---|
| 402 | + writel((u32)(xfer->tx_dma & MTK_SPI_32BITS_MASK), |
---|
| 403 | + mdata->base + SPI_TX_SRC_REG); |
---|
| 404 | +#ifdef CONFIG_ARCH_DMA_ADDR_T_64BIT |
---|
| 405 | + if (mdata->dev_comp->dma_ext) |
---|
| 406 | + writel((u32)(xfer->tx_dma >> 32), |
---|
| 407 | + mdata->base + SPI_TX_SRC_REG_64); |
---|
| 408 | +#endif |
---|
| 409 | + } |
---|
| 410 | + |
---|
| 411 | + if (mdata->rx_sgl) { |
---|
| 412 | + writel((u32)(xfer->rx_dma & MTK_SPI_32BITS_MASK), |
---|
| 413 | + mdata->base + SPI_RX_DST_REG); |
---|
| 414 | +#ifdef CONFIG_ARCH_DMA_ADDR_T_64BIT |
---|
| 415 | + if (mdata->dev_comp->dma_ext) |
---|
| 416 | + writel((u32)(xfer->rx_dma >> 32), |
---|
| 417 | + mdata->base + SPI_RX_DST_REG_64); |
---|
| 418 | +#endif |
---|
| 419 | + } |
---|
378 | 420 | } |
---|
379 | 421 | |
---|
380 | 422 | static int mtk_spi_fifo_transfer(struct spi_master *master, |
---|
.. | .. |
---|
586 | 628 | struct spi_master *master; |
---|
587 | 629 | struct mtk_spi *mdata; |
---|
588 | 630 | const struct of_device_id *of_id; |
---|
589 | | - struct resource *res; |
---|
590 | | - int i, irq, ret; |
---|
| 631 | + int i, irq, ret, addr_bits; |
---|
591 | 632 | |
---|
592 | 633 | master = spi_alloc_master(&pdev->dev, sizeof(*mdata)); |
---|
593 | 634 | if (!master) { |
---|
.. | .. |
---|
597 | 638 | |
---|
598 | 639 | master->auto_runtime_pm = true; |
---|
599 | 640 | master->dev.of_node = pdev->dev.of_node; |
---|
600 | | - master->mode_bits = SPI_CPOL | SPI_CPHA; |
---|
| 641 | + master->mode_bits = SPI_CPOL | SPI_CPHA | SPI_LSB_FIRST; |
---|
601 | 642 | |
---|
602 | 643 | master->set_cs = mtk_spi_set_cs; |
---|
603 | 644 | master->prepare_message = mtk_spi_prepare_message; |
---|
.. | .. |
---|
614 | 655 | |
---|
615 | 656 | mdata = spi_master_get_devdata(master); |
---|
616 | 657 | mdata->dev_comp = of_id->data; |
---|
| 658 | + |
---|
| 659 | + if (mdata->dev_comp->enhance_timing) |
---|
| 660 | + master->mode_bits |= SPI_CS_HIGH; |
---|
| 661 | + |
---|
617 | 662 | if (mdata->dev_comp->must_tx) |
---|
618 | 663 | master->flags = SPI_MASTER_MUST_TX; |
---|
619 | 664 | |
---|
.. | .. |
---|
649 | 694 | } |
---|
650 | 695 | |
---|
651 | 696 | platform_set_drvdata(pdev, master); |
---|
652 | | - |
---|
653 | | - res = platform_get_resource(pdev, IORESOURCE_MEM, 0); |
---|
654 | | - if (!res) { |
---|
655 | | - ret = -ENODEV; |
---|
656 | | - dev_err(&pdev->dev, "failed to determine base address\n"); |
---|
657 | | - goto err_put_master; |
---|
658 | | - } |
---|
659 | | - |
---|
660 | | - mdata->base = devm_ioremap_resource(&pdev->dev, res); |
---|
| 697 | + mdata->base = devm_platform_ioremap_resource(pdev, 0); |
---|
661 | 698 | if (IS_ERR(mdata->base)) { |
---|
662 | 699 | ret = PTR_ERR(mdata->base); |
---|
663 | 700 | goto err_put_master; |
---|
.. | .. |
---|
665 | 702 | |
---|
666 | 703 | irq = platform_get_irq(pdev, 0); |
---|
667 | 704 | if (irq < 0) { |
---|
668 | | - dev_err(&pdev->dev, "failed to get irq (%d)\n", irq); |
---|
669 | 705 | ret = irq; |
---|
670 | 706 | goto err_put_master; |
---|
671 | 707 | } |
---|
.. | .. |
---|
754 | 790 | } |
---|
755 | 791 | } |
---|
756 | 792 | |
---|
| 793 | + if (mdata->dev_comp->dma_ext) |
---|
| 794 | + addr_bits = DMA_ADDR_EXT_BITS; |
---|
| 795 | + else |
---|
| 796 | + addr_bits = DMA_ADDR_DEF_BITS; |
---|
| 797 | + ret = dma_set_mask(&pdev->dev, DMA_BIT_MASK(addr_bits)); |
---|
| 798 | + if (ret) |
---|
| 799 | + dev_notice(&pdev->dev, "SPI dma_set_mask(%d) failed, ret:%d\n", |
---|
| 800 | + addr_bits, ret); |
---|
| 801 | + |
---|
757 | 802 | return 0; |
---|
758 | 803 | |
---|
759 | 804 | err_disable_runtime_pm: |
---|