Merge branch 'spi-5.1' into spi-linus
This commit is contained in:
commit
58b860ed4a
@ -570,7 +570,8 @@ static int atmel_qspi_remove(struct platform_device *pdev)
|
||||
|
||||
static int __maybe_unused atmel_qspi_suspend(struct device *dev)
|
||||
{
|
||||
struct atmel_qspi *aq = dev_get_drvdata(dev);
|
||||
struct spi_controller *ctrl = dev_get_drvdata(dev);
|
||||
struct atmel_qspi *aq = spi_controller_get_devdata(ctrl);
|
||||
|
||||
clk_disable_unprepare(aq->qspick);
|
||||
clk_disable_unprepare(aq->pclk);
|
||||
@ -580,7 +581,8 @@ static int __maybe_unused atmel_qspi_suspend(struct device *dev)
|
||||
|
||||
static int __maybe_unused atmel_qspi_resume(struct device *dev)
|
||||
{
|
||||
struct atmel_qspi *aq = dev_get_drvdata(dev);
|
||||
struct spi_controller *ctrl = dev_get_drvdata(dev);
|
||||
struct atmel_qspi *aq = spi_controller_get_devdata(ctrl);
|
||||
|
||||
clk_prepare_enable(aq->pclk);
|
||||
clk_prepare_enable(aq->qspick);
|
||||
|
@ -62,7 +62,7 @@
|
||||
#define CFGR1_PCSPOL BIT(8)
|
||||
#define CFGR1_NOSTALL BIT(3)
|
||||
#define CFGR1_MASTER BIT(0)
|
||||
#define FSR_RXCOUNT (BIT(16)|BIT(17)|BIT(18))
|
||||
#define FSR_TXCOUNT (0xFF)
|
||||
#define RSR_RXEMPTY BIT(1)
|
||||
#define TCR_CPOL BIT(31)
|
||||
#define TCR_CPHA BIT(30)
|
||||
@ -452,7 +452,7 @@ static irqreturn_t fsl_lpspi_isr(int irq, void *dev_id)
|
||||
}
|
||||
|
||||
if (temp_SR & SR_MBF ||
|
||||
readl(fsl_lpspi->base + IMX7ULP_FSR) & FSR_RXCOUNT) {
|
||||
readl(fsl_lpspi->base + IMX7ULP_FSR) & FSR_TXCOUNT) {
|
||||
writel(SR_FCF, fsl_lpspi->base + IMX7ULP_SR);
|
||||
fsl_lpspi_intctrl(fsl_lpspi, IER_FCIE);
|
||||
return IRQ_HANDLED;
|
||||
|
@ -35,7 +35,6 @@
|
||||
* platform_device->driver_data ... points to spi_gpio
|
||||
*
|
||||
* spi->controller_state ... reserved for bitbang framework code
|
||||
* spi->controller_data ... holds chipselect GPIO
|
||||
*
|
||||
* spi->master->dev.driver_data ... points to spi_gpio->bitbang
|
||||
*/
|
||||
|
@ -1494,7 +1494,7 @@ static int spi_imx_transfer(struct spi_device *spi,
|
||||
|
||||
/* flush rxfifo before transfer */
|
||||
while (spi_imx->devtype_data->rx_available(spi_imx))
|
||||
spi_imx->rx(spi_imx);
|
||||
readl(spi_imx->base + MXC_CSPIRXDATA);
|
||||
|
||||
if (spi_imx->slave_mode)
|
||||
return spi_imx_pio_transfer_slave(spi, transfer);
|
||||
|
@ -559,7 +559,7 @@ static int pic32_spi_one_transfer(struct spi_master *master,
|
||||
dev_err(&spi->dev, "wait error/timedout\n");
|
||||
if (dma_issued) {
|
||||
dmaengine_terminate_all(master->dma_rx);
|
||||
dmaengine_terminate_all(master->dma_rx);
|
||||
dmaengine_terminate_all(master->dma_tx);
|
||||
}
|
||||
ret = -ETIMEDOUT;
|
||||
} else {
|
||||
|
@ -884,10 +884,14 @@ static unsigned int ssp_get_clk_div(struct driver_data *drv_data, int rate)
|
||||
|
||||
rate = min_t(int, ssp_clk, rate);
|
||||
|
||||
/*
|
||||
* Calculate the divisor for the SCR (Serial Clock Rate), avoiding
|
||||
* that the SSP transmission rate can be greater than the device rate
|
||||
*/
|
||||
if (ssp->type == PXA25x_SSP || ssp->type == CE4100_SSP)
|
||||
return (ssp_clk / (2 * rate) - 1) & 0xff;
|
||||
return (DIV_ROUND_UP(ssp_clk, 2 * rate) - 1) & 0xff;
|
||||
else
|
||||
return (ssp_clk / rate - 1) & 0xfff;
|
||||
return (DIV_ROUND_UP(ssp_clk, rate) - 1) & 0xfff;
|
||||
}
|
||||
|
||||
static unsigned int pxa2xx_ssp_get_clk_div(struct driver_data *drv_data,
|
||||
|
@ -271,7 +271,8 @@ static int rspi_set_config_register(struct rspi_data *rspi, int access_size)
|
||||
/* Sets parity, interrupt mask */
|
||||
rspi_write8(rspi, 0x00, RSPI_SPCR2);
|
||||
|
||||
/* Sets SPCMD */
|
||||
/* Resets sequencer */
|
||||
rspi_write8(rspi, 0, RSPI_SPSCR);
|
||||
rspi->spcmd |= SPCMD_SPB_8_TO_16(access_size);
|
||||
rspi_write16(rspi, rspi->spcmd, RSPI_SPCMD0);
|
||||
|
||||
@ -315,7 +316,8 @@ static int rspi_rz_set_config_register(struct rspi_data *rspi, int access_size)
|
||||
rspi_write8(rspi, 0x00, RSPI_SSLND);
|
||||
rspi_write8(rspi, 0x00, RSPI_SPND);
|
||||
|
||||
/* Sets SPCMD */
|
||||
/* Resets sequencer */
|
||||
rspi_write8(rspi, 0, RSPI_SPSCR);
|
||||
rspi->spcmd |= SPCMD_SPB_8_TO_16(access_size);
|
||||
rspi_write16(rspi, rspi->spcmd, RSPI_SPCMD0);
|
||||
|
||||
@ -366,7 +368,8 @@ static int qspi_set_config_register(struct rspi_data *rspi, int access_size)
|
||||
/* Sets buffer to allow normal operation */
|
||||
rspi_write8(rspi, 0x00, QSPI_SPBFCR);
|
||||
|
||||
/* Sets SPCMD */
|
||||
/* Resets sequencer */
|
||||
rspi_write8(rspi, 0, RSPI_SPSCR);
|
||||
rspi_write16(rspi, rspi->spcmd, RSPI_SPCMD0);
|
||||
|
||||
/* Sets RSPI mode */
|
||||
@ -736,27 +739,22 @@ static int qspi_trigger_transfer_out_in(struct rspi_data *rspi, const u8 *tx,
|
||||
while (len > 0) {
|
||||
n = qspi_set_send_trigger(rspi, len);
|
||||
qspi_set_receive_trigger(rspi, len);
|
||||
if (n == QSPI_BUFFER_SIZE) {
|
||||
ret = rspi_wait_for_tx_empty(rspi);
|
||||
if (ret < 0) {
|
||||
dev_err(&rspi->ctlr->dev, "transmit timeout\n");
|
||||
return ret;
|
||||
}
|
||||
for (i = 0; i < n; i++)
|
||||
rspi_write_data(rspi, *tx++);
|
||||
|
||||
ret = rspi_wait_for_rx_full(rspi);
|
||||
if (ret < 0) {
|
||||
dev_err(&rspi->ctlr->dev, "receive timeout\n");
|
||||
return ret;
|
||||
}
|
||||
for (i = 0; i < n; i++)
|
||||
*rx++ = rspi_read_data(rspi);
|
||||
} else {
|
||||
ret = rspi_pio_transfer(rspi, tx, rx, n);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
ret = rspi_wait_for_tx_empty(rspi);
|
||||
if (ret < 0) {
|
||||
dev_err(&rspi->ctlr->dev, "transmit timeout\n");
|
||||
return ret;
|
||||
}
|
||||
for (i = 0; i < n; i++)
|
||||
rspi_write_data(rspi, *tx++);
|
||||
|
||||
ret = rspi_wait_for_rx_full(rspi);
|
||||
if (ret < 0) {
|
||||
dev_err(&rspi->ctlr->dev, "receive timeout\n");
|
||||
return ret;
|
||||
}
|
||||
for (i = 0; i < n; i++)
|
||||
*rx++ = rspi_read_data(rspi);
|
||||
|
||||
len -= n;
|
||||
}
|
||||
|
||||
@ -793,19 +791,14 @@ static int qspi_transfer_out(struct rspi_data *rspi, struct spi_transfer *xfer)
|
||||
|
||||
while (n > 0) {
|
||||
len = qspi_set_send_trigger(rspi, n);
|
||||
if (len == QSPI_BUFFER_SIZE) {
|
||||
ret = rspi_wait_for_tx_empty(rspi);
|
||||
if (ret < 0) {
|
||||
dev_err(&rspi->ctlr->dev, "transmit timeout\n");
|
||||
return ret;
|
||||
}
|
||||
for (i = 0; i < len; i++)
|
||||
rspi_write_data(rspi, *tx++);
|
||||
} else {
|
||||
ret = rspi_pio_transfer(rspi, tx, NULL, len);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
ret = rspi_wait_for_tx_empty(rspi);
|
||||
if (ret < 0) {
|
||||
dev_err(&rspi->ctlr->dev, "transmit timeout\n");
|
||||
return ret;
|
||||
}
|
||||
for (i = 0; i < len; i++)
|
||||
rspi_write_data(rspi, *tx++);
|
||||
|
||||
n -= len;
|
||||
}
|
||||
|
||||
@ -830,19 +823,14 @@ static int qspi_transfer_in(struct rspi_data *rspi, struct spi_transfer *xfer)
|
||||
|
||||
while (n > 0) {
|
||||
len = qspi_set_receive_trigger(rspi, n);
|
||||
if (len == QSPI_BUFFER_SIZE) {
|
||||
ret = rspi_wait_for_rx_full(rspi);
|
||||
if (ret < 0) {
|
||||
dev_err(&rspi->ctlr->dev, "receive timeout\n");
|
||||
return ret;
|
||||
}
|
||||
for (i = 0; i < len; i++)
|
||||
*rx++ = rspi_read_data(rspi);
|
||||
} else {
|
||||
ret = rspi_pio_transfer(rspi, NULL, rx, len);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
ret = rspi_wait_for_rx_full(rspi);
|
||||
if (ret < 0) {
|
||||
dev_err(&rspi->ctlr->dev, "receive timeout\n");
|
||||
return ret;
|
||||
}
|
||||
for (i = 0; i < len; i++)
|
||||
*rx++ = rspi_read_data(rspi);
|
||||
|
||||
n -= len;
|
||||
}
|
||||
|
||||
@ -868,28 +856,6 @@ static int qspi_transfer_one(struct spi_controller *ctlr,
|
||||
}
|
||||
}
|
||||
|
||||
static int rspi_setup(struct spi_device *spi)
|
||||
{
|
||||
struct rspi_data *rspi = spi_controller_get_devdata(spi->controller);
|
||||
|
||||
rspi->max_speed_hz = spi->max_speed_hz;
|
||||
|
||||
rspi->spcmd = SPCMD_SSLKP;
|
||||
if (spi->mode & SPI_CPOL)
|
||||
rspi->spcmd |= SPCMD_CPOL;
|
||||
if (spi->mode & SPI_CPHA)
|
||||
rspi->spcmd |= SPCMD_CPHA;
|
||||
|
||||
/* CMOS output mode and MOSI signal from previous transfer */
|
||||
rspi->sppcr = 0;
|
||||
if (spi->mode & SPI_LOOP)
|
||||
rspi->sppcr |= SPPCR_SPLP;
|
||||
|
||||
set_config_register(rspi, 8);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static u16 qspi_transfer_mode(const struct spi_transfer *xfer)
|
||||
{
|
||||
if (xfer->tx_buf)
|
||||
@ -959,8 +925,24 @@ static int rspi_prepare_message(struct spi_controller *ctlr,
|
||||
struct spi_message *msg)
|
||||
{
|
||||
struct rspi_data *rspi = spi_controller_get_devdata(ctlr);
|
||||
struct spi_device *spi = msg->spi;
|
||||
int ret;
|
||||
|
||||
rspi->max_speed_hz = spi->max_speed_hz;
|
||||
|
||||
rspi->spcmd = SPCMD_SSLKP;
|
||||
if (spi->mode & SPI_CPOL)
|
||||
rspi->spcmd |= SPCMD_CPOL;
|
||||
if (spi->mode & SPI_CPHA)
|
||||
rspi->spcmd |= SPCMD_CPHA;
|
||||
|
||||
/* CMOS output mode and MOSI signal from previous transfer */
|
||||
rspi->sppcr = 0;
|
||||
if (spi->mode & SPI_LOOP)
|
||||
rspi->sppcr |= SPPCR_SPLP;
|
||||
|
||||
set_config_register(rspi, 8);
|
||||
|
||||
if (msg->spi->mode &
|
||||
(SPI_TX_DUAL | SPI_TX_QUAD | SPI_RX_DUAL | SPI_RX_QUAD)) {
|
||||
/* Setup sequencer for messages with multiple transfer modes */
|
||||
@ -1267,7 +1249,6 @@ static int rspi_probe(struct platform_device *pdev)
|
||||
init_waitqueue_head(&rspi->wait);
|
||||
|
||||
ctlr->bus_num = pdev->id;
|
||||
ctlr->setup = rspi_setup;
|
||||
ctlr->auto_runtime_pm = true;
|
||||
ctlr->transfer_one = ops->transfer_one;
|
||||
ctlr->prepare_message = rspi_prepare_message;
|
||||
|
@ -76,7 +76,6 @@
|
||||
#define QSPI_PSMAR 0x28
|
||||
#define QSPI_PIR 0x2c
|
||||
#define QSPI_LPTR 0x30
|
||||
#define LPTR_DFT_TIMEOUT 0x10
|
||||
|
||||
#define STM32_QSPI_MAX_MMAP_SZ SZ_256M
|
||||
#define STM32_QSPI_MAX_NORCHIP 2
|
||||
@ -372,8 +371,7 @@ static int stm32_qspi_setup(struct spi_device *spi)
|
||||
flash->presc = presc;
|
||||
|
||||
mutex_lock(&qspi->lock);
|
||||
writel_relaxed(LPTR_DFT_TIMEOUT, qspi->io_base + QSPI_LPTR);
|
||||
cr = FIELD_PREP(CR_FTHRES_MASK, 3) | CR_TCEN | CR_SSHIFT | CR_EN;
|
||||
cr = FIELD_PREP(CR_FTHRES_MASK, 3) | CR_SSHIFT | CR_EN;
|
||||
writel_relaxed(cr, qspi->io_base + QSPI_CR);
|
||||
|
||||
/* set dcr fsize to max address */
|
||||
|
@ -1839,8 +1839,9 @@ static int stm32_spi_probe(struct platform_device *pdev)
|
||||
|
||||
spi->irq = platform_get_irq(pdev, 0);
|
||||
if (spi->irq <= 0) {
|
||||
dev_err(&pdev->dev, "no irq: %d\n", spi->irq);
|
||||
ret = -ENOENT;
|
||||
ret = spi->irq;
|
||||
if (ret != -EPROBE_DEFER)
|
||||
dev_err(&pdev->dev, "failed to get irq: %d\n", ret);
|
||||
goto err_master_put;
|
||||
}
|
||||
ret = devm_request_threaded_irq(&pdev->dev, spi->irq,
|
||||
|
@ -307,10 +307,16 @@ static unsigned tegra_spi_fill_tx_fifo_from_client_txbuf(
|
||||
x |= (u32)(*tx_buf++) << (i * 8);
|
||||
tegra_spi_writel(tspi, x, SPI_TX_FIFO);
|
||||
}
|
||||
|
||||
tspi->cur_tx_pos += written_words * tspi->bytes_per_word;
|
||||
} else {
|
||||
unsigned int write_bytes;
|
||||
max_n_32bit = min(tspi->curr_dma_words, tx_empty_count);
|
||||
written_words = max_n_32bit;
|
||||
nbytes = written_words * tspi->bytes_per_word;
|
||||
if (nbytes > t->len - tspi->cur_pos)
|
||||
nbytes = t->len - tspi->cur_pos;
|
||||
write_bytes = nbytes;
|
||||
for (count = 0; count < max_n_32bit; count++) {
|
||||
u32 x = 0;
|
||||
|
||||
@ -319,8 +325,10 @@ static unsigned tegra_spi_fill_tx_fifo_from_client_txbuf(
|
||||
x |= (u32)(*tx_buf++) << (i * 8);
|
||||
tegra_spi_writel(tspi, x, SPI_TX_FIFO);
|
||||
}
|
||||
|
||||
tspi->cur_tx_pos += write_bytes;
|
||||
}
|
||||
tspi->cur_tx_pos += written_words * tspi->bytes_per_word;
|
||||
|
||||
return written_words;
|
||||
}
|
||||
|
||||
@ -344,20 +352,27 @@ static unsigned int tegra_spi_read_rx_fifo_to_client_rxbuf(
|
||||
for (i = 0; len && (i < 4); i++, len--)
|
||||
*rx_buf++ = (x >> i*8) & 0xFF;
|
||||
}
|
||||
tspi->cur_rx_pos += tspi->curr_dma_words * tspi->bytes_per_word;
|
||||
read_words += tspi->curr_dma_words;
|
||||
tspi->cur_rx_pos += tspi->curr_dma_words * tspi->bytes_per_word;
|
||||
} else {
|
||||
u32 rx_mask = ((u32)1 << t->bits_per_word) - 1;
|
||||
u8 bytes_per_word = tspi->bytes_per_word;
|
||||
unsigned int read_bytes;
|
||||
|
||||
len = rx_full_count * bytes_per_word;
|
||||
if (len > t->len - tspi->cur_pos)
|
||||
len = t->len - tspi->cur_pos;
|
||||
read_bytes = len;
|
||||
for (count = 0; count < rx_full_count; count++) {
|
||||
u32 x = tegra_spi_readl(tspi, SPI_RX_FIFO) & rx_mask;
|
||||
|
||||
for (i = 0; (i < tspi->bytes_per_word); i++)
|
||||
for (i = 0; len && (i < bytes_per_word); i++, len--)
|
||||
*rx_buf++ = (x >> (i*8)) & 0xFF;
|
||||
}
|
||||
tspi->cur_rx_pos += rx_full_count * tspi->bytes_per_word;
|
||||
read_words += rx_full_count;
|
||||
tspi->cur_rx_pos += read_bytes;
|
||||
}
|
||||
|
||||
return read_words;
|
||||
}
|
||||
|
||||
@ -372,12 +387,17 @@ static void tegra_spi_copy_client_txbuf_to_spi_txbuf(
|
||||
unsigned len = tspi->curr_dma_words * tspi->bytes_per_word;
|
||||
|
||||
memcpy(tspi->tx_dma_buf, t->tx_buf + tspi->cur_pos, len);
|
||||
tspi->cur_tx_pos += tspi->curr_dma_words * tspi->bytes_per_word;
|
||||
} else {
|
||||
unsigned int i;
|
||||
unsigned int count;
|
||||
u8 *tx_buf = (u8 *)t->tx_buf + tspi->cur_tx_pos;
|
||||
unsigned consume = tspi->curr_dma_words * tspi->bytes_per_word;
|
||||
unsigned int write_bytes;
|
||||
|
||||
if (consume > t->len - tspi->cur_pos)
|
||||
consume = t->len - tspi->cur_pos;
|
||||
write_bytes = consume;
|
||||
for (count = 0; count < tspi->curr_dma_words; count++) {
|
||||
u32 x = 0;
|
||||
|
||||
@ -386,8 +406,9 @@ static void tegra_spi_copy_client_txbuf_to_spi_txbuf(
|
||||
x |= (u32)(*tx_buf++) << (i * 8);
|
||||
tspi->tx_dma_buf[count] = x;
|
||||
}
|
||||
|
||||
tspi->cur_tx_pos += write_bytes;
|
||||
}
|
||||
tspi->cur_tx_pos += tspi->curr_dma_words * tspi->bytes_per_word;
|
||||
|
||||
/* Make the dma buffer to read by dma */
|
||||
dma_sync_single_for_device(tspi->dev, tspi->tx_dma_phys,
|
||||
@ -405,20 +426,28 @@ static void tegra_spi_copy_spi_rxbuf_to_client_rxbuf(
|
||||
unsigned len = tspi->curr_dma_words * tspi->bytes_per_word;
|
||||
|
||||
memcpy(t->rx_buf + tspi->cur_rx_pos, tspi->rx_dma_buf, len);
|
||||
tspi->cur_rx_pos += tspi->curr_dma_words * tspi->bytes_per_word;
|
||||
} else {
|
||||
unsigned int i;
|
||||
unsigned int count;
|
||||
unsigned char *rx_buf = t->rx_buf + tspi->cur_rx_pos;
|
||||
u32 rx_mask = ((u32)1 << t->bits_per_word) - 1;
|
||||
unsigned consume = tspi->curr_dma_words * tspi->bytes_per_word;
|
||||
unsigned int read_bytes;
|
||||
|
||||
if (consume > t->len - tspi->cur_pos)
|
||||
consume = t->len - tspi->cur_pos;
|
||||
read_bytes = consume;
|
||||
for (count = 0; count < tspi->curr_dma_words; count++) {
|
||||
u32 x = tspi->rx_dma_buf[count] & rx_mask;
|
||||
|
||||
for (i = 0; (i < tspi->bytes_per_word); i++)
|
||||
for (i = 0; consume && (i < tspi->bytes_per_word);
|
||||
i++, consume--)
|
||||
*rx_buf++ = (x >> (i*8)) & 0xFF;
|
||||
}
|
||||
|
||||
tspi->cur_rx_pos += read_bytes;
|
||||
}
|
||||
tspi->cur_rx_pos += tspi->curr_dma_words * tspi->bytes_per_word;
|
||||
|
||||
/* Make the dma buffer to read by dma */
|
||||
dma_sync_single_for_device(tspi->dev, tspi->rx_dma_phys,
|
||||
@ -470,21 +499,38 @@ static int tegra_spi_start_rx_dma(struct tegra_spi_data *tspi, int len)
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int tegra_spi_flush_fifos(struct tegra_spi_data *tspi)
|
||||
{
|
||||
unsigned long timeout = jiffies + HZ;
|
||||
u32 status;
|
||||
|
||||
status = tegra_spi_readl(tspi, SPI_FIFO_STATUS);
|
||||
if ((status & SPI_FIFO_EMPTY) != SPI_FIFO_EMPTY) {
|
||||
status |= SPI_RX_FIFO_FLUSH | SPI_TX_FIFO_FLUSH;
|
||||
tegra_spi_writel(tspi, status, SPI_FIFO_STATUS);
|
||||
while ((status & SPI_FIFO_EMPTY) != SPI_FIFO_EMPTY) {
|
||||
status = tegra_spi_readl(tspi, SPI_FIFO_STATUS);
|
||||
if (time_after(jiffies, timeout)) {
|
||||
dev_err(tspi->dev,
|
||||
"timeout waiting for fifo flush\n");
|
||||
return -EIO;
|
||||
}
|
||||
|
||||
udelay(1);
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int tegra_spi_start_dma_based_transfer(
|
||||
struct tegra_spi_data *tspi, struct spi_transfer *t)
|
||||
{
|
||||
u32 val;
|
||||
unsigned int len;
|
||||
int ret = 0;
|
||||
u32 status;
|
||||
|
||||
/* Make sure that Rx and Tx fifo are empty */
|
||||
status = tegra_spi_readl(tspi, SPI_FIFO_STATUS);
|
||||
if ((status & SPI_FIFO_EMPTY) != SPI_FIFO_EMPTY) {
|
||||
dev_err(tspi->dev, "Rx/Tx fifo are not empty status 0x%08x\n",
|
||||
(unsigned)status);
|
||||
return -EIO;
|
||||
}
|
||||
u8 dma_burst;
|
||||
struct dma_slave_config dma_sconfig = {0};
|
||||
|
||||
val = SPI_DMA_BLK_SET(tspi->curr_dma_words - 1);
|
||||
tegra_spi_writel(tspi, val, SPI_DMA_BLK);
|
||||
@ -496,12 +542,16 @@ static int tegra_spi_start_dma_based_transfer(
|
||||
len = tspi->curr_dma_words * 4;
|
||||
|
||||
/* Set attention level based on length of transfer */
|
||||
if (len & 0xF)
|
||||
if (len & 0xF) {
|
||||
val |= SPI_TX_TRIG_1 | SPI_RX_TRIG_1;
|
||||
else if (((len) >> 4) & 0x1)
|
||||
dma_burst = 1;
|
||||
} else if (((len) >> 4) & 0x1) {
|
||||
val |= SPI_TX_TRIG_4 | SPI_RX_TRIG_4;
|
||||
else
|
||||
dma_burst = 4;
|
||||
} else {
|
||||
val |= SPI_TX_TRIG_8 | SPI_RX_TRIG_8;
|
||||
dma_burst = 8;
|
||||
}
|
||||
|
||||
if (tspi->cur_direction & DATA_DIR_TX)
|
||||
val |= SPI_IE_TX;
|
||||
@ -512,7 +562,18 @@ static int tegra_spi_start_dma_based_transfer(
|
||||
tegra_spi_writel(tspi, val, SPI_DMA_CTL);
|
||||
tspi->dma_control_reg = val;
|
||||
|
||||
dma_sconfig.device_fc = true;
|
||||
if (tspi->cur_direction & DATA_DIR_TX) {
|
||||
dma_sconfig.dst_addr = tspi->phys + SPI_TX_FIFO;
|
||||
dma_sconfig.dst_addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES;
|
||||
dma_sconfig.dst_maxburst = dma_burst;
|
||||
ret = dmaengine_slave_config(tspi->tx_dma_chan, &dma_sconfig);
|
||||
if (ret < 0) {
|
||||
dev_err(tspi->dev,
|
||||
"DMA slave config failed: %d\n", ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
tegra_spi_copy_client_txbuf_to_spi_txbuf(tspi, t);
|
||||
ret = tegra_spi_start_tx_dma(tspi, len);
|
||||
if (ret < 0) {
|
||||
@ -523,6 +584,16 @@ static int tegra_spi_start_dma_based_transfer(
|
||||
}
|
||||
|
||||
if (tspi->cur_direction & DATA_DIR_RX) {
|
||||
dma_sconfig.src_addr = tspi->phys + SPI_RX_FIFO;
|
||||
dma_sconfig.src_addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES;
|
||||
dma_sconfig.src_maxburst = dma_burst;
|
||||
ret = dmaengine_slave_config(tspi->rx_dma_chan, &dma_sconfig);
|
||||
if (ret < 0) {
|
||||
dev_err(tspi->dev,
|
||||
"DMA slave config failed: %d\n", ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
/* Make the dma buffer to read by dma */
|
||||
dma_sync_single_for_device(tspi->dev, tspi->rx_dma_phys,
|
||||
tspi->dma_buf_size, DMA_FROM_DEVICE);
|
||||
@ -570,8 +641,9 @@ static int tegra_spi_start_cpu_based_transfer(
|
||||
|
||||
tspi->is_curr_dma_xfer = false;
|
||||
|
||||
val |= SPI_DMA_EN;
|
||||
tegra_spi_writel(tspi, val, SPI_DMA_CTL);
|
||||
val = tspi->command1_reg;
|
||||
val |= SPI_PIO;
|
||||
tegra_spi_writel(tspi, val, SPI_COMMAND1);
|
||||
return 0;
|
||||
}
|
||||
|
||||
@ -582,7 +654,6 @@ static int tegra_spi_init_dma_param(struct tegra_spi_data *tspi,
|
||||
u32 *dma_buf;
|
||||
dma_addr_t dma_phys;
|
||||
int ret;
|
||||
struct dma_slave_config dma_sconfig;
|
||||
|
||||
dma_chan = dma_request_slave_channel_reason(tspi->dev,
|
||||
dma_to_memory ? "rx" : "tx");
|
||||
@ -602,19 +673,6 @@ static int tegra_spi_init_dma_param(struct tegra_spi_data *tspi,
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
if (dma_to_memory) {
|
||||
dma_sconfig.src_addr = tspi->phys + SPI_RX_FIFO;
|
||||
dma_sconfig.src_addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES;
|
||||
dma_sconfig.src_maxburst = 0;
|
||||
} else {
|
||||
dma_sconfig.dst_addr = tspi->phys + SPI_TX_FIFO;
|
||||
dma_sconfig.dst_addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES;
|
||||
dma_sconfig.dst_maxburst = 0;
|
||||
}
|
||||
|
||||
ret = dmaengine_slave_config(dma_chan, &dma_sconfig);
|
||||
if (ret)
|
||||
goto scrub;
|
||||
if (dma_to_memory) {
|
||||
tspi->rx_dma_chan = dma_chan;
|
||||
tspi->rx_dma_buf = dma_buf;
|
||||
@ -625,11 +683,6 @@ static int tegra_spi_init_dma_param(struct tegra_spi_data *tspi,
|
||||
tspi->tx_dma_phys = dma_phys;
|
||||
}
|
||||
return 0;
|
||||
|
||||
scrub:
|
||||
dma_free_coherent(tspi->dev, tspi->dma_buf_size, dma_buf, dma_phys);
|
||||
dma_release_channel(dma_chan);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void tegra_spi_deinit_dma_param(struct tegra_spi_data *tspi,
|
||||
@ -730,6 +783,8 @@ static int tegra_spi_start_transfer_one(struct spi_device *spi,
|
||||
|
||||
if (tspi->is_packed)
|
||||
command1 |= SPI_PACKED;
|
||||
else
|
||||
command1 &= ~SPI_PACKED;
|
||||
|
||||
command1 &= ~(SPI_CS_SEL_MASK | SPI_TX_EN | SPI_RX_EN);
|
||||
tspi->cur_direction = 0;
|
||||
@ -748,6 +803,9 @@ static int tegra_spi_start_transfer_one(struct spi_device *spi,
|
||||
dev_dbg(tspi->dev, "The def 0x%x and written 0x%x\n",
|
||||
tspi->def_command1_reg, (unsigned)command1);
|
||||
|
||||
ret = tegra_spi_flush_fifos(tspi);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
if (total_fifo_words > SPI_FIFO_DEPTH)
|
||||
ret = tegra_spi_start_dma_based_transfer(tspi, t);
|
||||
else
|
||||
@ -799,6 +857,19 @@ static void tegra_spi_transfer_delay(int delay)
|
||||
udelay(delay % 1000);
|
||||
}
|
||||
|
||||
static void tegra_spi_transfer_end(struct spi_device *spi)
|
||||
{
|
||||
struct tegra_spi_data *tspi = spi_master_get_devdata(spi->master);
|
||||
int cs_val = (spi->mode & SPI_CS_HIGH) ? 0 : 1;
|
||||
|
||||
if (cs_val)
|
||||
tspi->command1_reg |= SPI_CS_SW_VAL;
|
||||
else
|
||||
tspi->command1_reg &= ~SPI_CS_SW_VAL;
|
||||
tegra_spi_writel(tspi, tspi->command1_reg, SPI_COMMAND1);
|
||||
tegra_spi_writel(tspi, tspi->def_command1_reg, SPI_COMMAND1);
|
||||
}
|
||||
|
||||
static int tegra_spi_transfer_one_message(struct spi_master *master,
|
||||
struct spi_message *msg)
|
||||
{
|
||||
@ -838,7 +909,17 @@ static int tegra_spi_transfer_one_message(struct spi_master *master,
|
||||
if (WARN_ON(ret == 0)) {
|
||||
dev_err(tspi->dev,
|
||||
"spi transfer timeout, err %d\n", ret);
|
||||
if (tspi->is_curr_dma_xfer &&
|
||||
(tspi->cur_direction & DATA_DIR_TX))
|
||||
dmaengine_terminate_all(tspi->tx_dma_chan);
|
||||
if (tspi->is_curr_dma_xfer &&
|
||||
(tspi->cur_direction & DATA_DIR_RX))
|
||||
dmaengine_terminate_all(tspi->rx_dma_chan);
|
||||
ret = -EIO;
|
||||
tegra_spi_flush_fifos(tspi);
|
||||
reset_control_assert(tspi->rst);
|
||||
udelay(2);
|
||||
reset_control_deassert(tspi->rst);
|
||||
goto complete_xfer;
|
||||
}
|
||||
|
||||
@ -851,8 +932,7 @@ static int tegra_spi_transfer_one_message(struct spi_master *master,
|
||||
|
||||
complete_xfer:
|
||||
if (ret < 0 || skip) {
|
||||
tegra_spi_writel(tspi, tspi->def_command1_reg,
|
||||
SPI_COMMAND1);
|
||||
tegra_spi_transfer_end(spi);
|
||||
tegra_spi_transfer_delay(xfer->delay_usecs);
|
||||
goto exit;
|
||||
} else if (list_is_last(&xfer->transfer_list,
|
||||
@ -860,13 +940,11 @@ complete_xfer:
|
||||
if (xfer->cs_change)
|
||||
tspi->cs_control = spi;
|
||||
else {
|
||||
tegra_spi_writel(tspi, tspi->def_command1_reg,
|
||||
SPI_COMMAND1);
|
||||
tegra_spi_transfer_end(spi);
|
||||
tegra_spi_transfer_delay(xfer->delay_usecs);
|
||||
}
|
||||
} else if (xfer->cs_change) {
|
||||
tegra_spi_writel(tspi, tspi->def_command1_reg,
|
||||
SPI_COMMAND1);
|
||||
tegra_spi_transfer_end(spi);
|
||||
tegra_spi_transfer_delay(xfer->delay_usecs);
|
||||
}
|
||||
|
||||
@ -889,11 +967,13 @@ static irqreturn_t handle_cpu_based_xfer(struct tegra_spi_data *tspi)
|
||||
tspi->status_reg);
|
||||
dev_err(tspi->dev, "CpuXfer 0x%08x:0x%08x\n",
|
||||
tspi->command1_reg, tspi->dma_control_reg);
|
||||
tegra_spi_flush_fifos(tspi);
|
||||
complete(&tspi->xfer_completion);
|
||||
spin_unlock_irqrestore(&tspi->lock, flags);
|
||||
reset_control_assert(tspi->rst);
|
||||
udelay(2);
|
||||
reset_control_deassert(tspi->rst);
|
||||
complete(&tspi->xfer_completion);
|
||||
goto exit;
|
||||
return IRQ_HANDLED;
|
||||
}
|
||||
|
||||
if (tspi->cur_direction & DATA_DIR_RX)
|
||||
@ -961,11 +1041,12 @@ static irqreturn_t handle_dma_based_xfer(struct tegra_spi_data *tspi)
|
||||
tspi->status_reg);
|
||||
dev_err(tspi->dev, "DmaXfer 0x%08x:0x%08x\n",
|
||||
tspi->command1_reg, tspi->dma_control_reg);
|
||||
tegra_spi_flush_fifos(tspi);
|
||||
complete(&tspi->xfer_completion);
|
||||
spin_unlock_irqrestore(&tspi->lock, flags);
|
||||
reset_control_assert(tspi->rst);
|
||||
udelay(2);
|
||||
reset_control_deassert(tspi->rst);
|
||||
complete(&tspi->xfer_completion);
|
||||
spin_unlock_irqrestore(&tspi->lock, flags);
|
||||
return IRQ_HANDLED;
|
||||
}
|
||||
|
||||
@ -1067,27 +1148,19 @@ static int tegra_spi_probe(struct platform_device *pdev)
|
||||
|
||||
spi_irq = platform_get_irq(pdev, 0);
|
||||
tspi->irq = spi_irq;
|
||||
ret = request_threaded_irq(tspi->irq, tegra_spi_isr,
|
||||
tegra_spi_isr_thread, IRQF_ONESHOT,
|
||||
dev_name(&pdev->dev), tspi);
|
||||
if (ret < 0) {
|
||||
dev_err(&pdev->dev, "Failed to register ISR for IRQ %d\n",
|
||||
tspi->irq);
|
||||
goto exit_free_master;
|
||||
}
|
||||
|
||||
tspi->clk = devm_clk_get(&pdev->dev, "spi");
|
||||
if (IS_ERR(tspi->clk)) {
|
||||
dev_err(&pdev->dev, "can not get clock\n");
|
||||
ret = PTR_ERR(tspi->clk);
|
||||
goto exit_free_irq;
|
||||
goto exit_free_master;
|
||||
}
|
||||
|
||||
tspi->rst = devm_reset_control_get_exclusive(&pdev->dev, "spi");
|
||||
if (IS_ERR(tspi->rst)) {
|
||||
dev_err(&pdev->dev, "can not get reset\n");
|
||||
ret = PTR_ERR(tspi->rst);
|
||||
goto exit_free_irq;
|
||||
goto exit_free_master;
|
||||
}
|
||||
|
||||
tspi->max_buf_size = SPI_FIFO_DEPTH << 2;
|
||||
@ -1095,7 +1168,7 @@ static int tegra_spi_probe(struct platform_device *pdev)
|
||||
|
||||
ret = tegra_spi_init_dma_param(tspi, true);
|
||||
if (ret < 0)
|
||||
goto exit_free_irq;
|
||||
goto exit_free_master;
|
||||
ret = tegra_spi_init_dma_param(tspi, false);
|
||||
if (ret < 0)
|
||||
goto exit_rx_dma_free;
|
||||
@ -1117,18 +1190,32 @@ static int tegra_spi_probe(struct platform_device *pdev)
|
||||
dev_err(&pdev->dev, "pm runtime get failed, e = %d\n", ret);
|
||||
goto exit_pm_disable;
|
||||
}
|
||||
|
||||
reset_control_assert(tspi->rst);
|
||||
udelay(2);
|
||||
reset_control_deassert(tspi->rst);
|
||||
tspi->def_command1_reg = SPI_M_S;
|
||||
tegra_spi_writel(tspi, tspi->def_command1_reg, SPI_COMMAND1);
|
||||
pm_runtime_put(&pdev->dev);
|
||||
ret = request_threaded_irq(tspi->irq, tegra_spi_isr,
|
||||
tegra_spi_isr_thread, IRQF_ONESHOT,
|
||||
dev_name(&pdev->dev), tspi);
|
||||
if (ret < 0) {
|
||||
dev_err(&pdev->dev, "Failed to register ISR for IRQ %d\n",
|
||||
tspi->irq);
|
||||
goto exit_pm_disable;
|
||||
}
|
||||
|
||||
master->dev.of_node = pdev->dev.of_node;
|
||||
ret = devm_spi_register_master(&pdev->dev, master);
|
||||
if (ret < 0) {
|
||||
dev_err(&pdev->dev, "can not register to master err %d\n", ret);
|
||||
goto exit_pm_disable;
|
||||
goto exit_free_irq;
|
||||
}
|
||||
return ret;
|
||||
|
||||
exit_free_irq:
|
||||
free_irq(spi_irq, tspi);
|
||||
exit_pm_disable:
|
||||
pm_runtime_disable(&pdev->dev);
|
||||
if (!pm_runtime_status_suspended(&pdev->dev))
|
||||
@ -1136,8 +1223,6 @@ exit_pm_disable:
|
||||
tegra_spi_deinit_dma_param(tspi, false);
|
||||
exit_rx_dma_free:
|
||||
tegra_spi_deinit_dma_param(tspi, true);
|
||||
exit_free_irq:
|
||||
free_irq(spi_irq, tspi);
|
||||
exit_free_master:
|
||||
spi_master_put(master);
|
||||
return ret;
|
||||
|
@ -717,9 +717,6 @@ static int tegra_slink_start_transfer_one(struct spi_device *spi,
|
||||
command2 = tspi->command2_reg;
|
||||
command2 &= ~(SLINK_RXEN | SLINK_TXEN);
|
||||
|
||||
tegra_slink_writel(tspi, command, SLINK_COMMAND);
|
||||
tspi->command_reg = command;
|
||||
|
||||
tspi->cur_direction = 0;
|
||||
if (t->rx_buf) {
|
||||
command2 |= SLINK_RXEN;
|
||||
@ -729,9 +726,18 @@ static int tegra_slink_start_transfer_one(struct spi_device *spi,
|
||||
command2 |= SLINK_TXEN;
|
||||
tspi->cur_direction |= DATA_DIR_TX;
|
||||
}
|
||||
|
||||
/*
|
||||
* Writing to the command2 register bevore the command register prevents
|
||||
* a spike in chip_select line 0. This selects the chip_select line
|
||||
* before changing the chip_select value.
|
||||
*/
|
||||
tegra_slink_writel(tspi, command2, SLINK_COMMAND2);
|
||||
tspi->command2_reg = command2;
|
||||
|
||||
tegra_slink_writel(tspi, command, SLINK_COMMAND);
|
||||
tspi->command_reg = command;
|
||||
|
||||
if (total_fifo_words > SLINK_FIFO_DEPTH)
|
||||
ret = tegra_slink_start_dma_based_transfer(tspi, t);
|
||||
else
|
||||
|
@ -1299,18 +1299,27 @@ static void pch_free_dma_buf(struct pch_spi_board_data *board_dat,
|
||||
dma->rx_buf_virt, dma->rx_buf_dma);
|
||||
}
|
||||
|
||||
static void pch_alloc_dma_buf(struct pch_spi_board_data *board_dat,
|
||||
static int pch_alloc_dma_buf(struct pch_spi_board_data *board_dat,
|
||||
struct pch_spi_data *data)
|
||||
{
|
||||
struct pch_spi_dma_ctrl *dma;
|
||||
int ret;
|
||||
|
||||
dma = &data->dma;
|
||||
ret = 0;
|
||||
/* Get Consistent memory for Tx DMA */
|
||||
dma->tx_buf_virt = dma_alloc_coherent(&board_dat->pdev->dev,
|
||||
PCH_BUF_SIZE, &dma->tx_buf_dma, GFP_KERNEL);
|
||||
if (!dma->tx_buf_virt)
|
||||
ret = -ENOMEM;
|
||||
|
||||
/* Get Consistent memory for Rx DMA */
|
||||
dma->rx_buf_virt = dma_alloc_coherent(&board_dat->pdev->dev,
|
||||
PCH_BUF_SIZE, &dma->rx_buf_dma, GFP_KERNEL);
|
||||
if (!dma->rx_buf_virt)
|
||||
ret = -ENOMEM;
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int pch_spi_pd_probe(struct platform_device *plat_dev)
|
||||
@ -1387,7 +1396,9 @@ static int pch_spi_pd_probe(struct platform_device *plat_dev)
|
||||
|
||||
if (use_dma) {
|
||||
dev_info(&plat_dev->dev, "Use DMA for data transfers\n");
|
||||
pch_alloc_dma_buf(board_dat, data);
|
||||
ret = pch_alloc_dma_buf(board_dat, data);
|
||||
if (ret)
|
||||
goto err_spi_register_master;
|
||||
}
|
||||
|
||||
ret = spi_register_master(master);
|
||||
|
@ -1039,6 +1039,8 @@ static int spi_map_msg(struct spi_controller *ctlr, struct spi_message *msg)
|
||||
if (max_tx || max_rx) {
|
||||
list_for_each_entry(xfer, &msg->transfers,
|
||||
transfer_list) {
|
||||
if (!xfer->len)
|
||||
continue;
|
||||
if (!xfer->tx_buf)
|
||||
xfer->tx_buf = ctlr->dummy_tx;
|
||||
if (!xfer->rx_buf)
|
||||
@ -2195,6 +2197,8 @@ static int spi_get_gpio_descs(struct spi_controller *ctlr)
|
||||
*/
|
||||
cs[i] = devm_gpiod_get_index_optional(dev, "cs", i,
|
||||
GPIOD_OUT_LOW);
|
||||
if (IS_ERR(cs[i]))
|
||||
return PTR_ERR(cs[i]);
|
||||
|
||||
if (cs[i]) {
|
||||
/*
|
||||
|
Loading…
Reference in New Issue
Block a user