Quellcodebibliothek Statistik Leitseite products/Sources/formale Sprachen/C/Linux/drivers/spi/   (Open Source Betriebssystem Version 6.17.9©)  Datei vom 24.10.2025 mit Größe 14 kB image not shown  

Quelle  spi-apple.c   Sprache: C

 
// SPDX-License-Identifier: GPL-2.0
//
// Apple SoC SPI device driver
//
// Copyright The Asahi Linux Contributors
//
// Based on spi-sifive.c, Copyright 2018 SiFive, Inc.

#include <linux/bitfield.h>
#include <linux/bits.h>
#include <linux/clk.h>
#include <linux/interrupt.h>
#include <linux/io.h>
#include <linux/module.h>
#include <linux/of.h>
#include <linux/platform_device.h>
#include <linux/pm_runtime.h>
#include <linux/spi/spi.h>

#define APPLE_SPI_CTRL   0x000
#define APPLE_SPI_CTRL_RUN  BIT(0)
#define APPLE_SPI_CTRL_TX_RESET  BIT(2)
#define APPLE_SPI_CTRL_RX_RESET  BIT(3)

#define APPLE_SPI_CFG   0x004
#define APPLE_SPI_CFG_CPHA  BIT(1)
#define APPLE_SPI_CFG_CPOL  BIT(2)
#define APPLE_SPI_CFG_MODE  GENMASK(6, 5)
#define APPLE_SPI_CFG_MODE_POLLED 0
#define APPLE_SPI_CFG_MODE_IRQ  1
#define APPLE_SPI_CFG_MODE_DMA  2
#define APPLE_SPI_CFG_IE_RXCOMPLETE BIT(7)
#define APPLE_SPI_CFG_IE_TXRXTHRESH BIT(8)
#define APPLE_SPI_CFG_LSB_FIRST  BIT(13)
#define APPLE_SPI_CFG_WORD_SIZE  GENMASK(16, 15)
#define APPLE_SPI_CFG_WORD_SIZE_8B 0
#define APPLE_SPI_CFG_WORD_SIZE_16B 1
#define APPLE_SPI_CFG_WORD_SIZE_32B 2
#define APPLE_SPI_CFG_FIFO_THRESH GENMASK(18, 17)
#define APPLE_SPI_CFG_FIFO_THRESH_8B 0
#define APPLE_SPI_CFG_FIFO_THRESH_4B 1
#define APPLE_SPI_CFG_FIFO_THRESH_1B 2
#define APPLE_SPI_CFG_IE_TXCOMPLETE BIT(21)

#define APPLE_SPI_STATUS  0x008
#define APPLE_SPI_STATUS_RXCOMPLETE BIT(0)
#define APPLE_SPI_STATUS_TXRXTHRESH BIT(1)
#define APPLE_SPI_STATUS_TXCOMPLETE BIT(2)

#define APPLE_SPI_PIN   0x00c
#define APPLE_SPI_PIN_KEEP_MOSI  BIT(0)
#define APPLE_SPI_PIN_CS  BIT(1)

#define APPLE_SPI_TXDATA  0x010
#define APPLE_SPI_RXDATA  0x020
#define APPLE_SPI_CLKDIV  0x030
#define APPLE_SPI_CLKDIV_MAX  0x7ff
#define APPLE_SPI_RXCNT   0x034
#define APPLE_SPI_WORD_DELAY  0x038
#define APPLE_SPI_TXCNT   0x04c

#define APPLE_SPI_FIFOSTAT  0x10c
#define APPLE_SPI_FIFOSTAT_TXFULL BIT(4)
#define APPLE_SPI_FIFOSTAT_LEVEL_TX GENMASK(15, 8)
#define APPLE_SPI_FIFOSTAT_RXEMPTY BIT(20)
#define APPLE_SPI_FIFOSTAT_LEVEL_RX GENMASK(31, 24)

#define APPLE_SPI_IE_XFER  0x130
#define APPLE_SPI_IF_XFER  0x134
#define APPLE_SPI_XFER_RXCOMPLETE BIT(0)
#define APPLE_SPI_XFER_TXCOMPLETE BIT(1)

#define APPLE_SPI_IE_FIFO  0x138
#define APPLE_SPI_IF_FIFO  0x13c
#define APPLE_SPI_FIFO_RXTHRESH  BIT(4)
#define APPLE_SPI_FIFO_TXTHRESH  BIT(5)
#define APPLE_SPI_FIFO_RXFULL  BIT(8)
#define APPLE_SPI_FIFO_TXEMPTY  BIT(9)
#define APPLE_SPI_FIFO_RXUNDERRUN BIT(16)
#define APPLE_SPI_FIFO_TXOVERFLOW BIT(17)

#define APPLE_SPI_SHIFTCFG  0x150
#define APPLE_SPI_SHIFTCFG_CLK_ENABLE BIT(0)
#define APPLE_SPI_SHIFTCFG_CS_ENABLE BIT(1)
#define APPLE_SPI_SHIFTCFG_AND_CLK_DATA BIT(8)
#define APPLE_SPI_SHIFTCFG_CS_AS_DATA BIT(9)
#define APPLE_SPI_SHIFTCFG_TX_ENABLE BIT(10)
#define APPLE_SPI_SHIFTCFG_RX_ENABLE BIT(11)
#define APPLE_SPI_SHIFTCFG_BITS  GENMASK(21, 16)
#define APPLE_SPI_SHIFTCFG_OVERRIDE_CS BIT(24)

#define APPLE_SPI_PINCFG  0x154
#define APPLE_SPI_PINCFG_KEEP_CLK BIT(0)
#define APPLE_SPI_PINCFG_KEEP_CS BIT(1)
#define APPLE_SPI_PINCFG_KEEP_MOSI BIT(2)
#define APPLE_SPI_PINCFG_CLK_IDLE_VAL BIT(8)
#define APPLE_SPI_PINCFG_CS_IDLE_VAL BIT(9)
#define APPLE_SPI_PINCFG_MOSI_IDLE_VAL BIT(10)

#define APPLE_SPI_DELAY_PRE  0x160
#define APPLE_SPI_DELAY_POST  0x168
#define APPLE_SPI_DELAY_ENABLE  BIT(0)
#define APPLE_SPI_DELAY_NO_INTERBYTE BIT(1)
#define APPLE_SPI_DELAY_SET_SCK  BIT(4)
#define APPLE_SPI_DELAY_SET_MOSI BIT(6)
#define APPLE_SPI_DELAY_SCK_VAL  BIT(8)
#define APPLE_SPI_DELAY_MOSI_VAL BIT(12)

#define APPLE_SPI_FIFO_DEPTH  16

/*
 * The slowest refclock available is 24MHz, the highest divider is 0x7ff,
 * the largest word size is 32 bits, the FIFO depth is 16, the maximum
 * intra-word delay is 0xffff refclocks. So the maximum time a transfer
 * cycle can take is:
 *
 * (0x7ff * 32 + 0xffff) * 16 / 24e6 Hz ~= 87ms
 *
 * Double it and round it up to 200ms for good measure.
 */

#define APPLE_SPI_TIMEOUT_MS  200

struct apple_spi {
 void __iomem      *regs;        /* MMIO register address */
 struct clk        *clk;         /* bus clock */
 struct completion done;         /* wake-up from interrupt */
};

static inline void reg_write(struct apple_spi *spi, int offset, u32 value)
{
 writel_relaxed(value, spi->regs + offset);
}

static inline u32 reg_read(struct apple_spi *spi, int offset)
{
 return readl_relaxed(spi->regs + offset);
}

static inline void reg_mask(struct apple_spi *spi, int offset, u32 clear, u32 set)
{
 u32 val = reg_read(spi, offset);

 val &= ~clear;
 val |= set;
 reg_write(spi, offset, val);
}

static void apple_spi_init(struct apple_spi *spi)
{
 /* Set CS high (inactive) and disable override and auto-CS */
 reg_write(spi, APPLE_SPI_PIN, APPLE_SPI_PIN_CS);
 reg_mask(spi, APPLE_SPI_SHIFTCFG, APPLE_SPI_SHIFTCFG_OVERRIDE_CS, 0);
 reg_mask(spi, APPLE_SPI_PINCFG, APPLE_SPI_PINCFG_CS_IDLE_VAL, APPLE_SPI_PINCFG_KEEP_CS);

 /* Reset FIFOs */
 reg_write(spi, APPLE_SPI_CTRL, APPLE_SPI_CTRL_RX_RESET | APPLE_SPI_CTRL_TX_RESET);

 /* Configure defaults */
 reg_write(spi, APPLE_SPI_CFG,
    FIELD_PREP(APPLE_SPI_CFG_FIFO_THRESH, APPLE_SPI_CFG_FIFO_THRESH_8B) |
    FIELD_PREP(APPLE_SPI_CFG_MODE, APPLE_SPI_CFG_MODE_IRQ) |
    FIELD_PREP(APPLE_SPI_CFG_WORD_SIZE, APPLE_SPI_CFG_WORD_SIZE_8B));

 /* Disable IRQs */
 reg_write(spi, APPLE_SPI_IE_FIFO, 0);
 reg_write(spi, APPLE_SPI_IE_XFER, 0);

 /* Disable delays */
 reg_write(spi, APPLE_SPI_DELAY_PRE, 0);
 reg_write(spi, APPLE_SPI_DELAY_POST, 0);
}

static int apple_spi_prepare_message(struct spi_controller *ctlr, struct spi_message *msg)
{
 struct apple_spi *spi = spi_controller_get_devdata(ctlr);
 struct spi_device *device = msg->spi;

 u32 cfg = ((device->mode & SPI_CPHA ? APPLE_SPI_CFG_CPHA : 0) |
     (device->mode & SPI_CPOL ? APPLE_SPI_CFG_CPOL : 0) |
     (device->mode & SPI_LSB_FIRST ? APPLE_SPI_CFG_LSB_FIRST : 0));

 /* Update core config */
 reg_mask(spi, APPLE_SPI_CFG,
   APPLE_SPI_CFG_CPHA | APPLE_SPI_CFG_CPOL | APPLE_SPI_CFG_LSB_FIRST, cfg);

 return 0;
}

static void apple_spi_set_cs(struct spi_device *device, bool is_high)
{
 struct apple_spi *spi = spi_controller_get_devdata(device->controller);

 reg_mask(spi, APPLE_SPI_PIN, APPLE_SPI_PIN_CS, is_high ? APPLE_SPI_PIN_CS : 0);
}

static bool apple_spi_prep_transfer(struct apple_spi *spi, struct spi_transfer *t)
{
 u32 cr, fifo_threshold;

 /* Calculate and program the clock rate */
 cr = DIV_ROUND_UP(clk_get_rate(spi->clk), t->speed_hz);
 reg_write(spi, APPLE_SPI_CLKDIV, min_t(u32, cr, APPLE_SPI_CLKDIV_MAX));

 /* Update bits per word */
 reg_mask(spi, APPLE_SPI_SHIFTCFG, APPLE_SPI_SHIFTCFG_BITS,
   FIELD_PREP(APPLE_SPI_SHIFTCFG_BITS, t->bits_per_word));

 /* We will want to poll if the time we need to wait is
 * less than the context switching time.
 * Let's call that threshold 5us. The operation will take:
 *    bits_per_word * fifo_threshold / hz <= 5 * 10^-6
 *    200000 * bits_per_word * fifo_threshold <= hz
 */

 fifo_threshold = APPLE_SPI_FIFO_DEPTH / 2;
 return (200000 * t->bits_per_word * fifo_threshold) <= t->speed_hz;
}

static irqreturn_t apple_spi_irq(int irq, void *dev_id)
{
 struct apple_spi *spi = dev_id;
 u32 fifo = reg_read(spi, APPLE_SPI_IF_FIFO) & reg_read(spi, APPLE_SPI_IE_FIFO);
 u32 xfer = reg_read(spi, APPLE_SPI_IF_XFER) & reg_read(spi, APPLE_SPI_IE_XFER);

 if (fifo || xfer) {
  /* Disable interrupts until next transfer */
  reg_write(spi, APPLE_SPI_IE_XFER, 0);
  reg_write(spi, APPLE_SPI_IE_FIFO, 0);
  complete(&spi->done);
  return IRQ_HANDLED;
 }

 return IRQ_NONE;
}

static int apple_spi_wait(struct apple_spi *spi, u32 fifo_bit, u32 xfer_bit, int poll)
{
 int ret = 0;

 if (poll) {
  u32 fifo, xfer;
  unsigned long timeout = jiffies + APPLE_SPI_TIMEOUT_MS * HZ / 1000;

  do {
   fifo = reg_read(spi, APPLE_SPI_IF_FIFO);
   xfer = reg_read(spi, APPLE_SPI_IF_XFER);
   if (time_after(jiffies, timeout)) {
    ret = -ETIMEDOUT;
    break;
   }
  } while (!((fifo & fifo_bit) || (xfer & xfer_bit)));
 } else {
  reinit_completion(&spi->done);
  reg_write(spi, APPLE_SPI_IE_XFER, xfer_bit);
  reg_write(spi, APPLE_SPI_IE_FIFO, fifo_bit);

  if (!wait_for_completion_timeout(&spi->done,
       msecs_to_jiffies(APPLE_SPI_TIMEOUT_MS)))
   ret = -ETIMEDOUT;

  reg_write(spi, APPLE_SPI_IE_XFER, 0);
  reg_write(spi, APPLE_SPI_IE_FIFO, 0);
 }

 return ret;
}

static void apple_spi_tx(struct apple_spi *spi, const void **tx_ptr, u32 *left,
    unsigned int bytes_per_word)
{
 u32 inuse, words, wrote;

 if (!*tx_ptr)
  return;

 inuse = FIELD_GET(APPLE_SPI_FIFOSTAT_LEVEL_TX, reg_read(spi, APPLE_SPI_FIFOSTAT));
 words = wrote = min_t(u32, *left, APPLE_SPI_FIFO_DEPTH - inuse);

 if (!words)
  return;

 *left -= words;

 switch (bytes_per_word) {
 case 1: {
  const u8 *p = *tx_ptr;

  while (words--)
   reg_write(spi, APPLE_SPI_TXDATA, *p++);
  break;
 }
 case 2: {
  const u16 *p = *tx_ptr;

  while (words--)
   reg_write(spi, APPLE_SPI_TXDATA, *p++);
  break;
 }
 case 4: {
  const u32 *p = *tx_ptr;

  while (words--)
   reg_write(spi, APPLE_SPI_TXDATA, *p++);
  break;
 }
 default:
  WARN_ON(1);
 }

 *tx_ptr = ((u8 *)*tx_ptr) + bytes_per_word * wrote;
}

static void apple_spi_rx(struct apple_spi *spi, void **rx_ptr, u32 *left,
    unsigned int bytes_per_word)
{
 u32 words, read;

 if (!*rx_ptr)
  return;

 words = read = FIELD_GET(APPLE_SPI_FIFOSTAT_LEVEL_RX, reg_read(spi, APPLE_SPI_FIFOSTAT));
 WARN_ON(words > *left);

 if (!words)
  return;

 *left -= min_t(u32, *left, words);

 switch (bytes_per_word) {
 case 1: {
  u8 *p = *rx_ptr;

  while (words--)
   *p++ = reg_read(spi, APPLE_SPI_RXDATA);
  break;
 }
 case 2: {
  u16 *p = *rx_ptr;

  while (words--)
   *p++ = reg_read(spi, APPLE_SPI_RXDATA);
  break;
 }
 case 4: {
  u32 *p = *rx_ptr;

  while (words--)
   *p++ = reg_read(spi, APPLE_SPI_RXDATA);
  break;
 }
 default:
  WARN_ON(1);
 }

 *rx_ptr = ((u8 *)*rx_ptr) + bytes_per_word * read;
}

static int apple_spi_transfer_one(struct spi_controller *ctlr, struct spi_device *device,
      struct spi_transfer *t)
{
 struct apple_spi *spi = spi_controller_get_devdata(ctlr);
 bool poll = apple_spi_prep_transfer(spi, t);
 const void *tx_ptr = t->tx_buf;
 void *rx_ptr = t->rx_buf;
 unsigned int bytes_per_word;
 u32 words, remaining_tx, remaining_rx;
 u32 xfer_flags = 0;
 u32 fifo_flags;
 int retries = 100;
 int ret = 0;

 if (t->bits_per_word > 16)
  bytes_per_word = 4;
 else if (t->bits_per_word > 8)
  bytes_per_word = 2;
 else
  bytes_per_word = 1;

 words = t->len / bytes_per_word;
 remaining_tx = tx_ptr ? words : 0;
 remaining_rx = rx_ptr ? words : 0;

 /* Reset FIFOs */
 reg_write(spi, APPLE_SPI_CTRL, APPLE_SPI_CTRL_RX_RESET | APPLE_SPI_CTRL_TX_RESET);

 /* Clear IRQ flags */
 reg_write(spi, APPLE_SPI_IF_XFER, ~0);
 reg_write(spi, APPLE_SPI_IF_FIFO, ~0);

 /* Determine transfer completion flags we wait for */
 if (tx_ptr)
  xfer_flags |= APPLE_SPI_XFER_TXCOMPLETE;
 if (rx_ptr)
  xfer_flags |= APPLE_SPI_XFER_RXCOMPLETE;

 /* Set transfer length */
 reg_write(spi, APPLE_SPI_TXCNT, remaining_tx);
 reg_write(spi, APPLE_SPI_RXCNT, remaining_rx);

 /* Prime transmit FIFO */
 apple_spi_tx(spi, &tx_ptr, &remaining_tx, bytes_per_word);

 /* Start transfer */
 reg_write(spi, APPLE_SPI_CTRL, APPLE_SPI_CTRL_RUN);

 /* TX again since a few words get popped off immediately */
 apple_spi_tx(spi, &tx_ptr, &remaining_tx, bytes_per_word);

 while (xfer_flags) {
  fifo_flags = 0;

  if (remaining_tx)
   fifo_flags |= APPLE_SPI_FIFO_TXTHRESH;
  if (remaining_rx)
   fifo_flags |= APPLE_SPI_FIFO_RXTHRESH;

  /* Wait for anything to happen */
  ret = apple_spi_wait(spi, fifo_flags, xfer_flags, poll);
  if (ret) {
   dev_err(&ctlr->dev, "transfer timed out (remaining %d tx, %d rx)\n",
    remaining_tx, remaining_rx);
   goto err;
  }

  /* Stop waiting on transfer halves once they complete */
  xfer_flags &= ~reg_read(spi, APPLE_SPI_IF_XFER);

  /* Transmit and receive everything we can */
  apple_spi_tx(spi, &tx_ptr, &remaining_tx, bytes_per_word);
  apple_spi_rx(spi, &rx_ptr, &remaining_rx, bytes_per_word);
 }

 /*
 * Sometimes the transfer completes before the last word is in the RX FIFO.
 * Normally one retry is all it takes to get the last word out.
 */

 while (remaining_rx && retries--)
  apple_spi_rx(spi, &rx_ptr, &remaining_rx, bytes_per_word);

 if (remaining_tx)
  dev_err(&ctlr->dev, "transfer completed with %d words left to transmit\n",
   remaining_tx);
 if (remaining_rx)
  dev_err(&ctlr->dev, "transfer completed with %d words left to receive\n",
   remaining_rx);

err:
 fifo_flags = reg_read(spi, APPLE_SPI_IF_FIFO);
 WARN_ON(fifo_flags & APPLE_SPI_FIFO_TXOVERFLOW);
 WARN_ON(fifo_flags & APPLE_SPI_FIFO_RXUNDERRUN);

 /* Stop transfer */
 reg_write(spi, APPLE_SPI_CTRL, 0);

 return ret;
}

static int apple_spi_probe(struct platform_device *pdev)
{
 struct apple_spi *spi;
 int ret, irq;
 struct spi_controller *ctlr;

 ctlr = devm_spi_alloc_host(&pdev->dev, sizeof(struct apple_spi));
 if (!ctlr)
  return -ENOMEM;

 spi = spi_controller_get_devdata(ctlr);
 init_completion(&spi->done);

 spi->regs = devm_platform_ioremap_resource(pdev, 0);
 if (IS_ERR(spi->regs))
  return PTR_ERR(spi->regs);

 spi->clk = devm_clk_get_enabled(&pdev->dev, NULL);
 if (IS_ERR(spi->clk))
  return dev_err_probe(&pdev->dev, PTR_ERR(spi->clk),
         "Unable to find or enable bus clock\n");

 irq = platform_get_irq(pdev, 0);
 if (irq < 0)
  return irq;

 ret = devm_request_irq(&pdev->dev, irq, apple_spi_irq, 0,
          dev_name(&pdev->dev), spi);
 if (ret)
  return dev_err_probe(&pdev->dev, ret, "Unable to bind to interrupt\n");

 ctlr->dev.of_node = pdev->dev.of_node;
 ctlr->bus_num = pdev->id;
 ctlr->num_chipselect = 1;
 ctlr->mode_bits = SPI_CPHA | SPI_CPOL | SPI_LSB_FIRST;
 ctlr->bits_per_word_mask = SPI_BPW_RANGE_MASK(1, 32);
 ctlr->prepare_message = apple_spi_prepare_message;
 ctlr->set_cs = apple_spi_set_cs;
 ctlr->transfer_one = apple_spi_transfer_one;
 ctlr->use_gpio_descriptors = true;
 ctlr->auto_runtime_pm = true;

 pm_runtime_set_active(&pdev->dev);
 ret = devm_pm_runtime_enable(&pdev->dev);
 if (ret < 0)
  return ret;

 apple_spi_init(spi);

 ret = devm_spi_register_controller(&pdev->dev, ctlr);
 if (ret < 0)
  return dev_err_probe(&pdev->dev, ret, "devm_spi_register_controller failed\n");

 return 0;
}

static const struct of_device_id apple_spi_of_match[] = {
 { .compatible = "apple,spi", },
 {}
};
MODULE_DEVICE_TABLE(of, apple_spi_of_match);

static struct platform_driver apple_spi_driver = {
 .probe = apple_spi_probe,
 .driver = {
  .name = "apple-spi",
  .of_match_table = apple_spi_of_match,
 },
};
module_platform_driver(apple_spi_driver);

MODULE_AUTHOR("Hector Martin ");
MODULE_DESCRIPTION("Apple SoC SPI driver");
MODULE_LICENSE("GPL");

Messung V0.5
C=97 H=96 G=96

¤ Dauer der Verarbeitung: 0.5 Sekunden  ¤

*© Formatika GbR, Deutschland






Wurzel

Suchen

Beweissystem der NASA

Beweissystem Isabelle

NIST Cobol Testsuite

Cephes Mathematical Library

Wiener Entwicklungsmethode

Haftungshinweis

Die Informationen auf dieser Webseite wurden nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit, noch Qualität der bereit gestellten Informationen zugesichert.

Bemerkung:

Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.