Lines Matching +full:spi +full:- +full:mux
2 * Driver for Amlogic Meson SPI communication controller (SPICC)
7 * SPDX-License-Identifier: GPL-2.0+
12 #include <linux/clk-provider.h>
19 #include <linux/spi/spi.h>
30 * - all transfers are cutted in 16 words burst because the FIFO hangs on
31 * TX underflow, and there is no TX "Half-Empty" interrupt, so we go by
33 * - CS management is dumb, and goes UP between every burst, so is really a
69 #define SPICC_TH_EN BIT(1) /* TX FIFO Half-Full Interrupt */
72 #define SPICC_RH_EN BIT(4) /* RX FIFO Half-Full Interrupt */
89 #define SPICC_TH BIT(1) /* TX FIFO Half-Full Interrupt */
92 #define SPICC_RH BIT(4) /* RX FIFO Half-Full Interrupt */
104 #define SPICC_LBC_RO BIT(13) /* Loop Back Control Read-Only */
105 #define SPICC_LBC_W1 BIT(14) /* Loop Back Control Write-Only */
106 #define SPICC_SWAP_RO BIT(14) /* RX FIFO Data Swap Read-Only */
107 #define SPICC_SWAP_W1 BIT(15) /* RX FIFO Data Swap Write-Only */
108 #define SPICC_DLYCTL_RO_MASK GENMASK(20, 15) /* Delay Control Read-Only */
124 #define SPICC_FIFORST_RO_MASK GENMASK(22, 21) /* FIFO Softreset Read-Only */
125 #define SPICC_FIFORST_W1_MASK GENMASK(23, 22) /* FIFO Softreset Write-Only */
182 if (!spicc->data->has_oen) { in meson_spicc_oen_enable()
184 spicc->pins_idle_high = pinctrl_lookup_state(spicc->pinctrl, in meson_spicc_oen_enable()
185 "idle-high"); in meson_spicc_oen_enable()
186 if (IS_ERR(spicc->pins_idle_high)) { in meson_spicc_oen_enable()
187 dev_warn(&spicc->pdev->dev, "can't get idle-high pinctrl\n"); in meson_spicc_oen_enable()
188 spicc->pins_idle_high = NULL; in meson_spicc_oen_enable()
190 spicc->pins_idle_low = pinctrl_lookup_state(spicc->pinctrl, in meson_spicc_oen_enable()
191 "idle-low"); in meson_spicc_oen_enable()
192 if (IS_ERR(spicc->pins_idle_low)) { in meson_spicc_oen_enable()
193 dev_warn(&spicc->pdev->dev, "can't get idle-low pinctrl\n"); in meson_spicc_oen_enable()
194 spicc->pins_idle_low = NULL; in meson_spicc_oen_enable()
199 conf = readl_relaxed(spicc->base + SPICC_ENH_CTL0) | in meson_spicc_oen_enable()
202 writel_relaxed(conf, spicc->base + SPICC_ENH_CTL0); in meson_spicc_oen_enable()
208 readl_relaxed(spicc->base + SPICC_STATREG)); in meson_spicc_txfull()
214 readl_relaxed(spicc->base + SPICC_STATREG)); in meson_spicc_rxready()
219 unsigned int bytes = spicc->bytes_per_word; in meson_spicc_pull_data()
224 while (bytes--) { in meson_spicc_pull_data()
225 byte = *spicc->tx_buf++; in meson_spicc_pull_data()
230 spicc->tx_remain--; in meson_spicc_pull_data()
237 unsigned int bytes = spicc->bytes_per_word; in meson_spicc_push_data()
241 while (bytes--) { in meson_spicc_push_data()
243 *spicc->rx_buf++ = byte; in meson_spicc_push_data()
247 spicc->rx_remain--; in meson_spicc_push_data()
253 while (spicc->rx_remain && in meson_spicc_rx()
256 readl_relaxed(spicc->base + SPICC_RXDATA)); in meson_spicc_rx()
262 while (spicc->tx_remain && in meson_spicc_tx()
265 spicc->base + SPICC_TXDATA); in meson_spicc_tx()
272 spicc->xfer_remain / in meson_spicc_setup_burst()
273 spicc->bytes_per_word, in meson_spicc_setup_burst()
274 spicc->data->fifo_size); in meson_spicc_setup_burst()
276 spicc->tx_remain = burst_len; in meson_spicc_setup_burst()
277 spicc->rx_remain = burst_len; in meson_spicc_setup_burst()
278 spicc->xfer_remain -= burst_len * spicc->bytes_per_word; in meson_spicc_setup_burst()
283 burst_len - 1), in meson_spicc_setup_burst()
284 spicc->base + SPICC_CONREG); in meson_spicc_setup_burst()
294 writel_bits_relaxed(SPICC_TC, SPICC_TC, spicc->base + SPICC_STATREG); in meson_spicc_irq()
299 if (!spicc->xfer_remain) { in meson_spicc_irq()
301 writel(0, spicc->base + SPICC_INTREG); in meson_spicc_irq()
303 complete(&spicc->done); in meson_spicc_irq()
312 writel_bits_relaxed(SPICC_XCH, SPICC_XCH, spicc->base + SPICC_CONREG); in meson_spicc_irq()
323 if (spicc->data->has_enhance_clk_div) { in meson_spicc_auto_io_delay()
325 readl_relaxed(spicc->base + SPICC_ENH_CTL0)); in meson_spicc_auto_io_delay()
330 readl_relaxed(spicc->base + SPICC_CONREG)); in meson_spicc_auto_io_delay()
337 hz = clk_get_rate(spicc->clk); in meson_spicc_auto_io_delay()
352 conf = readl_relaxed(spicc->base + SPICC_TESTREG); in meson_spicc_auto_io_delay()
357 writel_relaxed(conf, spicc->base + SPICC_TESTREG); in meson_spicc_auto_io_delay()
366 conf = conf_orig = readl_relaxed(spicc->base + SPICC_CONREG); in meson_spicc_setup_xfer()
371 (spicc->bytes_per_word << 3) - 1); in meson_spicc_setup_xfer()
375 writel_relaxed(conf, spicc->base + SPICC_CONREG); in meson_spicc_setup_xfer()
377 clk_set_rate(spicc->clk, xfer->speed_hz); in meson_spicc_setup_xfer()
381 writel_relaxed(0, spicc->base + SPICC_DMAREG); in meson_spicc_setup_xfer()
386 if (spicc->data->has_oen) in meson_spicc_reset_fifo()
389 spicc->base + SPICC_ENH_CTL0); in meson_spicc_reset_fifo()
392 spicc->base + SPICC_TESTREG); in meson_spicc_reset_fifo()
395 readl_relaxed(spicc->base + SPICC_RXDATA); in meson_spicc_reset_fifo()
397 if (spicc->data->has_oen) in meson_spicc_reset_fifo()
399 spicc->base + SPICC_ENH_CTL0); in meson_spicc_reset_fifo()
403 struct spi_device *spi, in meson_spicc_transfer_one() argument
410 spicc->xfer = xfer; in meson_spicc_transfer_one()
413 spicc->tx_buf = (u8 *)xfer->tx_buf; in meson_spicc_transfer_one()
414 spicc->rx_buf = (u8 *)xfer->rx_buf; in meson_spicc_transfer_one()
415 spicc->xfer_remain = xfer->len; in meson_spicc_transfer_one()
417 /* Pre-calculate word size */ in meson_spicc_transfer_one()
418 spicc->bytes_per_word = in meson_spicc_transfer_one()
419 DIV_ROUND_UP(spicc->xfer->bits_per_word, 8); in meson_spicc_transfer_one()
421 if (xfer->len % spicc->bytes_per_word) in meson_spicc_transfer_one()
422 return -EINVAL; in meson_spicc_transfer_one()
433 reinit_completion(&spicc->done); in meson_spicc_transfer_one()
435 /* For each byte we wait for 8 cycles of the SPI clock */ in meson_spicc_transfer_one()
436 timeout = 8LL * MSEC_PER_SEC * xfer->len; in meson_spicc_transfer_one()
437 do_div(timeout, xfer->speed_hz); in meson_spicc_transfer_one()
440 timeout += ((xfer->len >> 4) * 10) / MSEC_PER_SEC; in meson_spicc_transfer_one()
446 writel_bits_relaxed(SPICC_XCH, SPICC_XCH, spicc->base + SPICC_CONREG); in meson_spicc_transfer_one()
449 writel_relaxed(SPICC_TC_EN, spicc->base + SPICC_INTREG); in meson_spicc_transfer_one()
451 if (!wait_for_completion_timeout(&spicc->done, msecs_to_jiffies(timeout))) in meson_spicc_transfer_one()
452 return -ETIMEDOUT; in meson_spicc_transfer_one()
461 struct spi_device *spi = message->spi; in meson_spicc_prepare_message() local
462 u32 conf = readl_relaxed(spicc->base + SPICC_CONREG) & SPICC_DATARATE_MASK; in meson_spicc_prepare_message()
465 spicc->message = message; in meson_spicc_prepare_message()
474 if (spi->mode & SPI_CPOL) in meson_spicc_prepare_message()
479 if (!spicc->data->has_oen) { in meson_spicc_prepare_message()
480 if (spi->mode & SPI_CPOL) { in meson_spicc_prepare_message()
481 if (spicc->pins_idle_high) in meson_spicc_prepare_message()
482 pinctrl_select_state(spicc->pinctrl, spicc->pins_idle_high); in meson_spicc_prepare_message()
484 if (spicc->pins_idle_low) in meson_spicc_prepare_message()
485 pinctrl_select_state(spicc->pinctrl, spicc->pins_idle_low); in meson_spicc_prepare_message()
489 if (spi->mode & SPI_CPHA) in meson_spicc_prepare_message()
496 if (spi->mode & SPI_CS_HIGH) in meson_spicc_prepare_message()
501 if (spi->mode & SPI_READY) in meson_spicc_prepare_message()
507 conf |= FIELD_PREP(SPICC_CS_MASK, spi_get_chipselect(spi, 0)); in meson_spicc_prepare_message()
510 conf |= FIELD_PREP(SPICC_BITLENGTH_MASK, 8 - 1); in meson_spicc_prepare_message()
512 writel_relaxed(conf, spicc->base + SPICC_CONREG); in meson_spicc_prepare_message()
515 writel_relaxed(0, spicc->base + SPICC_PERIODREG); in meson_spicc_prepare_message()
518 spi->mode & SPI_LOOP ? SPICC_LBC_W1 : 0, in meson_spicc_prepare_message()
519 spicc->base + SPICC_TESTREG); in meson_spicc_prepare_message()
527 u32 conf = readl_relaxed(spicc->base + SPICC_CONREG) & SPICC_DATARATE_MASK; in meson_spicc_unprepare_transfer()
530 writel(0, spicc->base + SPICC_INTREG); in meson_spicc_unprepare_transfer()
532 device_reset_optional(&spicc->pdev->dev); in meson_spicc_unprepare_transfer()
535 writel_relaxed(conf, spicc->base + SPICC_CONREG); in meson_spicc_unprepare_transfer()
537 if (!spicc->data->has_oen) in meson_spicc_unprepare_transfer()
538 pinctrl_select_default_state(&spicc->pdev->dev); in meson_spicc_unprepare_transfer()
543 static int meson_spicc_setup(struct spi_device *spi) in meson_spicc_setup() argument
545 if (!spi->controller_state) in meson_spicc_setup()
546 spi->controller_state = spi_controller_get_devdata(spi->controller); in meson_spicc_setup()
551 static void meson_spicc_cleanup(struct spi_device *spi) in meson_spicc_cleanup() argument
553 spi->controller_state = NULL; in meson_spicc_cleanup()
557 * The Clock Mux
558 * x-----------------x x------------x x------\
559 * |---| pow2 fixed div |---| pow2 div |----| |
560 * | x-----------------x x------------x | |
561 * src ---| | mux |-- out
562 * | x-----------------x x------------x | |
563 * |---| enh fixed div |---| enh div |0---| |
564 * x-----------------x x------------x x------/
567 * src -> pow2 fixed div -> pow2 div -> out
570 * src -> pow2 fixed div -> pow2 div -> mux -> out
571 * src -> enh fixed div -> enh div -> mux -> out
574 * pclk -> pow2 fixed div -> pow2 div -> mux -> out
575 * pclk -> enh fixed div -> enh div -> mux -> out
590 if (!spicc->host->cur_msg) in meson_spicc_pow2_recalc_rate()
602 if (!spicc->host->cur_msg) in meson_spicc_pow2_determine_rate()
603 return -EINVAL; in meson_spicc_pow2_determine_rate()
614 if (!spicc->host->cur_msg) in meson_spicc_pow2_set_rate()
615 return -EINVAL; in meson_spicc_pow2_set_rate()
628 struct device *dev = &spicc->pdev->dev; in meson_spicc_pow2_clk_init()
644 return -ENOMEM; in meson_spicc_pow2_clk_init()
649 if (spicc->data->has_pclk) { in meson_spicc_pow2_clk_init()
651 parent_data[0].hw = __clk_get_hw(spicc->pclk); in meson_spicc_pow2_clk_init()
654 parent_data[0].hw = __clk_get_hw(spicc->core); in meson_spicc_pow2_clk_init()
658 pow2_fixed_div->mult = 1; in meson_spicc_pow2_clk_init()
659 pow2_fixed_div->div = 4; in meson_spicc_pow2_clk_init()
660 pow2_fixed_div->hw.init = &init; in meson_spicc_pow2_clk_init()
662 clk = devm_clk_register(dev, &pow2_fixed_div->hw); in meson_spicc_pow2_clk_init()
674 parent_data[0].hw = &pow2_fixed_div->hw; in meson_spicc_pow2_clk_init()
677 spicc->pow2_div.shift = 16; in meson_spicc_pow2_clk_init()
678 spicc->pow2_div.width = 3; in meson_spicc_pow2_clk_init()
679 spicc->pow2_div.flags = CLK_DIVIDER_POWER_OF_TWO; in meson_spicc_pow2_clk_init()
680 spicc->pow2_div.reg = spicc->base + SPICC_CONREG; in meson_spicc_pow2_clk_init()
681 spicc->pow2_div.hw.init = &init; in meson_spicc_pow2_clk_init()
683 spicc->clk = devm_clk_register(dev, &spicc->pow2_div.hw); in meson_spicc_pow2_clk_init()
684 if (WARN_ON(IS_ERR(spicc->clk))) in meson_spicc_pow2_clk_init()
685 return PTR_ERR(spicc->clk); in meson_spicc_pow2_clk_init()
692 struct device *dev = &spicc->pdev->dev; in meson_spicc_enh_clk_init()
695 struct clk_mux *mux; in meson_spicc_enh_clk_init() local
710 return -ENOMEM; in meson_spicc_enh_clk_init()
715 if (spicc->data->has_pclk) { in meson_spicc_enh_clk_init()
717 parent_data[0].hw = __clk_get_hw(spicc->pclk); in meson_spicc_enh_clk_init()
720 parent_data[0].hw = __clk_get_hw(spicc->core); in meson_spicc_enh_clk_init()
724 enh_fixed_div->mult = 1; in meson_spicc_enh_clk_init()
725 enh_fixed_div->div = 2; in meson_spicc_enh_clk_init()
726 enh_fixed_div->hw.init = &init; in meson_spicc_enh_clk_init()
728 clk = devm_clk_register(dev, &enh_fixed_div->hw); in meson_spicc_enh_clk_init()
734 return -ENOMEM; in meson_spicc_enh_clk_init()
740 parent_data[0].hw = &enh_fixed_div->hw; in meson_spicc_enh_clk_init()
743 enh_div->shift = 16; in meson_spicc_enh_clk_init()
744 enh_div->width = 8; in meson_spicc_enh_clk_init()
745 enh_div->reg = spicc->base + SPICC_ENH_CTL0; in meson_spicc_enh_clk_init()
746 enh_div->hw.init = &init; in meson_spicc_enh_clk_init()
748 clk = devm_clk_register(dev, &enh_div->hw); in meson_spicc_enh_clk_init()
752 mux = devm_kzalloc(dev, sizeof(*mux), GFP_KERNEL); in meson_spicc_enh_clk_init()
753 if (!mux) in meson_spicc_enh_clk_init()
754 return -ENOMEM; in meson_spicc_enh_clk_init()
759 parent_data[0].hw = &spicc->pow2_div.hw; in meson_spicc_enh_clk_init()
760 parent_data[1].hw = &enh_div->hw; in meson_spicc_enh_clk_init()
764 mux->mask = 0x1; in meson_spicc_enh_clk_init()
765 mux->shift = 24; in meson_spicc_enh_clk_init()
766 mux->reg = spicc->base + SPICC_ENH_CTL0; in meson_spicc_enh_clk_init()
767 mux->hw.init = &init; in meson_spicc_enh_clk_init()
769 spicc->clk = devm_clk_register(dev, &mux->hw); in meson_spicc_enh_clk_init()
770 if (WARN_ON(IS_ERR(spicc->clk))) in meson_spicc_enh_clk_init()
771 return PTR_ERR(spicc->clk); in meson_spicc_enh_clk_init()
782 host = spi_alloc_host(&pdev->dev, sizeof(*spicc)); in meson_spicc_probe()
784 dev_err(&pdev->dev, "host allocation failed\n"); in meson_spicc_probe()
785 return -ENOMEM; in meson_spicc_probe()
788 spicc->host = host; in meson_spicc_probe()
790 spicc->data = of_device_get_match_data(&pdev->dev); in meson_spicc_probe()
791 if (!spicc->data) { in meson_spicc_probe()
792 dev_err(&pdev->dev, "failed to get match data\n"); in meson_spicc_probe()
793 ret = -EINVAL; in meson_spicc_probe()
797 spicc->pdev = pdev; in meson_spicc_probe()
800 init_completion(&spicc->done); in meson_spicc_probe()
802 spicc->base = devm_platform_ioremap_resource(pdev, 0); in meson_spicc_probe()
803 if (IS_ERR(spicc->base)) { in meson_spicc_probe()
804 dev_err(&pdev->dev, "io resource mapping failed\n"); in meson_spicc_probe()
805 ret = PTR_ERR(spicc->base); in meson_spicc_probe()
811 spicc->base + SPICC_CONREG); in meson_spicc_probe()
814 writel_relaxed(0, spicc->base + SPICC_INTREG); in meson_spicc_probe()
822 ret = devm_request_irq(&pdev->dev, irq, meson_spicc_irq, in meson_spicc_probe()
825 dev_err(&pdev->dev, "irq request failed\n"); in meson_spicc_probe()
829 spicc->core = devm_clk_get_enabled(&pdev->dev, "core"); in meson_spicc_probe()
830 if (IS_ERR(spicc->core)) { in meson_spicc_probe()
831 dev_err(&pdev->dev, "core clock request failed\n"); in meson_spicc_probe()
832 ret = PTR_ERR(spicc->core); in meson_spicc_probe()
836 if (spicc->data->has_pclk) { in meson_spicc_probe()
837 spicc->pclk = devm_clk_get_enabled(&pdev->dev, "pclk"); in meson_spicc_probe()
838 if (IS_ERR(spicc->pclk)) { in meson_spicc_probe()
839 dev_err(&pdev->dev, "pclk clock request failed\n"); in meson_spicc_probe()
840 ret = PTR_ERR(spicc->pclk); in meson_spicc_probe()
845 spicc->pinctrl = devm_pinctrl_get(&pdev->dev); in meson_spicc_probe()
846 if (IS_ERR(spicc->pinctrl)) { in meson_spicc_probe()
847 ret = PTR_ERR(spicc->pinctrl); in meson_spicc_probe()
851 device_reset_optional(&pdev->dev); in meson_spicc_probe()
853 host->num_chipselect = 4; in meson_spicc_probe()
854 host->dev.of_node = pdev->dev.of_node; in meson_spicc_probe()
855 host->mode_bits = SPI_CPHA | SPI_CPOL | SPI_CS_HIGH | SPI_LOOP; in meson_spicc_probe()
856 host->bits_per_word_mask = SPI_BPW_MASK(32) | in meson_spicc_probe()
860 host->flags = (SPI_CONTROLLER_MUST_RX | SPI_CONTROLLER_MUST_TX); in meson_spicc_probe()
861 host->min_speed_hz = spicc->data->min_speed_hz; in meson_spicc_probe()
862 host->max_speed_hz = spicc->data->max_speed_hz; in meson_spicc_probe()
863 host->setup = meson_spicc_setup; in meson_spicc_probe()
864 host->cleanup = meson_spicc_cleanup; in meson_spicc_probe()
865 host->prepare_message = meson_spicc_prepare_message; in meson_spicc_probe()
866 host->unprepare_transfer_hardware = meson_spicc_unprepare_transfer; in meson_spicc_probe()
867 host->transfer_one = meson_spicc_transfer_one; in meson_spicc_probe()
868 host->use_gpio_descriptors = true; in meson_spicc_probe()
874 dev_err(&pdev->dev, "pow2 clock registration failed\n"); in meson_spicc_probe()
878 if (spicc->data->has_enhance_clk_div) { in meson_spicc_probe()
881 dev_err(&pdev->dev, "clock registration failed\n"); in meson_spicc_probe()
886 ret = devm_spi_register_controller(&pdev->dev, host); in meson_spicc_probe()
888 dev_err(&pdev->dev, "spi registration failed\n"); in meson_spicc_probe()
904 /* Disable SPI */ in meson_spicc_remove()
905 writel(0, spicc->base + SPICC_CONREG); in meson_spicc_remove()
907 spi_controller_put(spicc->host); in meson_spicc_remove()
935 .compatible = "amlogic,meson-gx-spicc",
939 .compatible = "amlogic,meson-axg-spicc",
943 .compatible = "amlogic,meson-g12a-spicc",
954 .name = "meson-spicc",
961 MODULE_DESCRIPTION("Meson SPI Communication Controller driver");