| .. | .. |
|---|
| 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: |
|---|