Commit a56f4895 authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'spi-v4.7' of git://git.kernel.org/pub/scm/linux/kernel/git/broonie/spi

Pull spi updates from Mark Brown:
 "Another quiet release for SPI, almost entirely driver specific changes
  with the diffstat dominated by two new drivers which are about two
  thirds of it in terms of lines of code:

   - new drivers for PIC32 standard and SQI controllers
   - the Cadence driver has had runtime PM support added and quite a few
     fixes and cleanups
   - flash-specific accelerated path support now has a feature query
     interface
   - the pxa2xx driver has been moved to use the core DMA mapping support"

* tag 'spi-v4.7' of git://git.kernel.org/pub/scm/linux/kernel/git/broonie/spi: (48 commits)
  spi: pic32-sqi: Fix linker error, undefined reference to `bad_dma_ops'
  spi: dw-pci: Spelling s/paltforms/platforms/g
  spi: pic32-sqi: Remove pic32_sqi_setup and pic32_sqi_cleanup
  spi: Fix simple typo s/impelment/implement
  spi: rockchip: potential NULL dereference on error
  spi: zynqmp: disable clocks in error paths
  spi: Drop unnecessary dependencies on relaxed I/O accessors
  spi: qup: Add spi_master_put in remove function
  spi: qup: Handle clocks in pm_runtime suspend and resume
  spi: st-ssc4: Fix missing spi_master_put in spi_st_probe error paths
  spi: st-ssc4: Allow compile test build
  spi: omap2-mcspi: Use dma_request_chan() for requesting DMA channel
  spi: davinci: Use dma_request_chan() for requesting DMA channel
  spi: pic32: Fix checking return value of devm_ioremap_resource
  spi: spi-fsl-dspi: Update DT binding documentation
  spi: Drop duplicate code to set master->dev.parent
  spi: pic32: Set proper bits_per_word_mask
  spi: return error if kmap'd buffers passed to spi_map_buf()
  spi: core: add hook flash_read_supported to spi_master
  spi: pic32-sqi: silence array overflow warning
  ...
parents 8bc4d5f3 c4e85b7e
Microchip PIC32 SPI Master controller
Required properties:
- compatible: Should be "microchip,pic32mzda-spi".
- reg: Address and length of register space for the device.
- interrupts: Should contain all three spi interrupts in sequence
of <fault-irq>, <receive-irq>, <transmit-irq>.
- interrupt-names: Should be "fault", "rx", "tx" in order.
- clocks: Phandle of the clock generating SPI clock on the bus.
- clock-names: Should be "mck0".
- cs-gpios: Specifies the gpio pins to be used for chipselects.
See: Documentation/devicetree/bindings/spi/spi-bus.txt
Optional properties:
- dmas: Two or more DMA channel specifiers following the convention outlined
in Documentation/devicetree/bindings/dma/dma.txt
- dma-names: Names for the dma channels. There must be at least one channel
named "spi-tx" for transmit and named "spi-rx" for receive.
Example:
spi1: spi@1f821000 {
compatible = "microchip,pic32mzda-spi";
reg = <0x1f821000 0x200>;
interrupts = <109 IRQ_TYPE_LEVEL_HIGH>,
<110 IRQ_TYPE_LEVEL_HIGH>,
<111 IRQ_TYPE_LEVEL_HIGH>;
interrupt-names = "fault", "rx", "tx";
clocks = <&PBCLK2>;
clock-names = "mck0";
cs-gpios = <&gpio3 4 GPIO_ACTIVE_LOW>;
dmas = <&dma 134>, <&dma 135>;
dma-names = "spi-rx", "spi-tx";
};
...@@ -16,8 +16,7 @@ Required properties: ...@@ -16,8 +16,7 @@ Required properties:
Optional property: Optional property:
- big-endian: If present the dspi device's registers are implemented - big-endian: If present the dspi device's registers are implemented
in big endian mode, otherwise in native mode(same with CPU), for more in big endian mode.
detail please see: Documentation/devicetree/bindings/regmap/regmap.txt.
Optional SPI slave node properties: Optional SPI slave node properties:
- fsl,spi-cs-sck-delay: a delay in nanoseconds between activating chip - fsl,spi-cs-sck-delay: a delay in nanoseconds between activating chip
......
Microchip PIC32 Quad SPI controller
-----------------------------------
Required properties:
- compatible: Should be "microchip,pic32mzda-sqi".
- reg: Address and length of SQI controller register space.
- interrupts: Should contain SQI interrupt.
- clocks: Should contain phandle of two clocks in sequence, one that drives
clock on SPI bus and other that drives SQI controller.
- clock-names: Should be "spi_ck" and "reg_ck" in order.
Example:
sqi1: spi@1f8e2000 {
compatible = "microchip,pic32mzda-sqi";
reg = <0x1f8e2000 0x200>;
clocks = <&rootclk REF2CLK>, <&rootclk PB5CLK>;
clock-names = "spi_ck", "reg_ck";
interrupts = <169 IRQ_TYPE_LEVEL_HIGH>;
};
...@@ -410,7 +410,6 @@ config SPI_OMAP_UWIRE ...@@ -410,7 +410,6 @@ config SPI_OMAP_UWIRE
config SPI_OMAP24XX config SPI_OMAP24XX
tristate "McSPI driver for OMAP" tristate "McSPI driver for OMAP"
depends on HAS_DMA depends on HAS_DMA
depends on ARM || ARM64 || AVR32 || HEXAGON || MIPS || SUPERH
depends on ARCH_OMAP2PLUS || COMPILE_TEST depends on ARCH_OMAP2PLUS || COMPILE_TEST
help help
SPI master controller for OMAP24XX and later Multichannel SPI SPI master controller for OMAP24XX and later Multichannel SPI
...@@ -432,10 +431,23 @@ config SPI_OMAP_100K ...@@ -432,10 +431,23 @@ config SPI_OMAP_100K
config SPI_ORION config SPI_ORION
tristate "Orion SPI master" tristate "Orion SPI master"
depends on PLAT_ORION || COMPILE_TEST depends on PLAT_ORION || ARCH_MVEBU || COMPILE_TEST
help help
This enables using the SPI master controller on the Orion chips. This enables using the SPI master controller on the Orion chips.
config SPI_PIC32
tristate "Microchip PIC32 series SPI"
depends on MACH_PIC32 || COMPILE_TEST
help
SPI driver for Microchip PIC32 SPI master controller.
config SPI_PIC32_SQI
tristate "Microchip PIC32 Quad SPI driver"
depends on MACH_PIC32 || COMPILE_TEST
depends on HAS_DMA
help
SPI driver for PIC32 Quad SPI controller.
config SPI_PL022 config SPI_PL022
tristate "ARM AMBA PL022 SSP controller" tristate "ARM AMBA PL022 SSP controller"
depends on ARM_AMBA depends on ARM_AMBA
...@@ -469,7 +481,6 @@ config SPI_PXA2XX_PCI ...@@ -469,7 +481,6 @@ config SPI_PXA2XX_PCI
config SPI_ROCKCHIP config SPI_ROCKCHIP
tristate "Rockchip SPI controller driver" tristate "Rockchip SPI controller driver"
depends on ARM || ARM64 || AVR32 || HEXAGON || MIPS || SUPERH
help help
This selects a driver for Rockchip SPI controller. This selects a driver for Rockchip SPI controller.
...@@ -569,7 +580,7 @@ config SPI_SIRF ...@@ -569,7 +580,7 @@ config SPI_SIRF
config SPI_ST_SSC4 config SPI_ST_SSC4
tristate "STMicroelectronics SPI SSC-based driver" tristate "STMicroelectronics SPI SSC-based driver"
depends on ARCH_STI depends on ARCH_STI || COMPILE_TEST
help help
STMicroelectronics SoCs support for SPI. If you say yes to STMicroelectronics SoCs support for SPI. If you say yes to
this option, support will be included for the SSC driven SPI. this option, support will be included for the SSC driven SPI.
...@@ -656,7 +667,7 @@ config SPI_XILINX ...@@ -656,7 +667,7 @@ config SPI_XILINX
config SPI_XLP config SPI_XLP
tristate "Netlogic XLP SPI controller driver" tristate "Netlogic XLP SPI controller driver"
depends on CPU_XLP || COMPILE_TEST depends on CPU_XLP || ARCH_VULCAN || COMPILE_TEST
help help
Enable support for the SPI controller on the Netlogic XLP SoCs. Enable support for the SPI controller on the Netlogic XLP SoCs.
Currently supported XLP variants are XLP8XX, XLP3XX, XLP2XX, XLP9XX Currently supported XLP variants are XLP8XX, XLP3XX, XLP2XX, XLP9XX
......
...@@ -62,6 +62,8 @@ obj-$(CONFIG_SPI_OMAP_100K) += spi-omap-100k.o ...@@ -62,6 +62,8 @@ obj-$(CONFIG_SPI_OMAP_100K) += spi-omap-100k.o
obj-$(CONFIG_SPI_OMAP24XX) += spi-omap2-mcspi.o obj-$(CONFIG_SPI_OMAP24XX) += spi-omap2-mcspi.o
obj-$(CONFIG_SPI_TI_QSPI) += spi-ti-qspi.o obj-$(CONFIG_SPI_TI_QSPI) += spi-ti-qspi.o
obj-$(CONFIG_SPI_ORION) += spi-orion.o obj-$(CONFIG_SPI_ORION) += spi-orion.o
obj-$(CONFIG_SPI_PIC32) += spi-pic32.o
obj-$(CONFIG_SPI_PIC32_SQI) += spi-pic32-sqi.o
obj-$(CONFIG_SPI_PL022) += spi-pl022.o obj-$(CONFIG_SPI_PL022) += spi-pl022.o
obj-$(CONFIG_SPI_PPC4xx) += spi-ppc4xx.o obj-$(CONFIG_SPI_PPC4xx) += spi-ppc4xx.o
spi-pxa2xx-platform-objs := spi-pxa2xx.o spi-pxa2xx-dma.o spi-pxa2xx-platform-objs := spi-pxa2xx.o spi-pxa2xx-dma.o
......
...@@ -525,7 +525,6 @@ static int spi_engine_probe(struct platform_device *pdev) ...@@ -525,7 +525,6 @@ static int spi_engine_probe(struct platform_device *pdev)
if (ret) if (ret)
goto err_ref_clk_disable; goto err_ref_clk_disable;
master->dev.parent = &pdev->dev;
master->dev.of_node = pdev->dev.of_node; master->dev.of_node = pdev->dev.of_node;
master->mode_bits = SPI_CPOL | SPI_CPHA | SPI_3WIRE; master->mode_bits = SPI_CPOL | SPI_CPHA | SPI_3WIRE;
master->bits_per_word_mask = SPI_BPW_MASK(8); master->bits_per_word_mask = SPI_BPW_MASK(8);
......
...@@ -10,6 +10,7 @@ ...@@ -10,6 +10,7 @@
#include "spi-bcm53xx.h" #include "spi-bcm53xx.h"
#define BCM53XXSPI_MAX_SPI_BAUD 13500000 /* 216 MHz? */ #define BCM53XXSPI_MAX_SPI_BAUD 13500000 /* 216 MHz? */
#define BCM53XXSPI_FLASH_WINDOW SZ_32M
/* The longest observed required wait was 19 ms */ /* The longest observed required wait was 19 ms */
#define BCM53XXSPI_SPE_TIMEOUT_MS 80 #define BCM53XXSPI_SPE_TIMEOUT_MS 80
...@@ -17,8 +18,10 @@ ...@@ -17,8 +18,10 @@
struct bcm53xxspi { struct bcm53xxspi {
struct bcma_device *core; struct bcma_device *core;
struct spi_master *master; struct spi_master *master;
void __iomem *mmio_base;
size_t read_offset; size_t read_offset;
bool bspi; /* Boot SPI mode with memory mapping */
}; };
static inline u32 bcm53xxspi_read(struct bcm53xxspi *b53spi, u16 offset) static inline u32 bcm53xxspi_read(struct bcm53xxspi *b53spi, u16 offset)
...@@ -32,6 +35,50 @@ static inline void bcm53xxspi_write(struct bcm53xxspi *b53spi, u16 offset, ...@@ -32,6 +35,50 @@ static inline void bcm53xxspi_write(struct bcm53xxspi *b53spi, u16 offset,
bcma_write32(b53spi->core, offset, value); bcma_write32(b53spi->core, offset, value);
} }
static void bcm53xxspi_disable_bspi(struct bcm53xxspi *b53spi)
{
struct device *dev = &b53spi->core->dev;
unsigned long deadline;
u32 tmp;
if (!b53spi->bspi)
return;
tmp = bcm53xxspi_read(b53spi, B53SPI_BSPI_MAST_N_BOOT_CTRL);
if (tmp & 0x1)
return;
deadline = jiffies + usecs_to_jiffies(200);
do {
tmp = bcm53xxspi_read(b53spi, B53SPI_BSPI_BUSY_STATUS);
if (!(tmp & 0x1)) {
bcm53xxspi_write(b53spi, B53SPI_BSPI_MAST_N_BOOT_CTRL,
0x1);
ndelay(200);
b53spi->bspi = false;
return;
}
udelay(1);
} while (!time_after_eq(jiffies, deadline));
dev_warn(dev, "Timeout disabling BSPI\n");
}
static void bcm53xxspi_enable_bspi(struct bcm53xxspi *b53spi)
{
u32 tmp;
if (b53spi->bspi)
return;
tmp = bcm53xxspi_read(b53spi, B53SPI_BSPI_MAST_N_BOOT_CTRL);
if (!(tmp & 0x1))
return;
bcm53xxspi_write(b53spi, B53SPI_BSPI_MAST_N_BOOT_CTRL, 0x0);
b53spi->bspi = true;
}
static inline unsigned int bcm53xxspi_calc_timeout(size_t len) static inline unsigned int bcm53xxspi_calc_timeout(size_t len)
{ {
/* Do some magic calculation based on length and buad. Add 10% and 1. */ /* Do some magic calculation based on length and buad. Add 10% and 1. */
...@@ -176,6 +223,8 @@ static int bcm53xxspi_transfer_one(struct spi_master *master, ...@@ -176,6 +223,8 @@ static int bcm53xxspi_transfer_one(struct spi_master *master,
u8 *buf; u8 *buf;
size_t left; size_t left;
bcm53xxspi_disable_bspi(b53spi);
if (t->tx_buf) { if (t->tx_buf) {
buf = (u8 *)t->tx_buf; buf = (u8 *)t->tx_buf;
left = t->len; left = t->len;
...@@ -206,6 +255,22 @@ static int bcm53xxspi_transfer_one(struct spi_master *master, ...@@ -206,6 +255,22 @@ static int bcm53xxspi_transfer_one(struct spi_master *master,
return 0; return 0;
} }
static int bcm53xxspi_flash_read(struct spi_device *spi,
struct spi_flash_read_message *msg)
{
struct bcm53xxspi *b53spi = spi_master_get_devdata(spi->master);
int ret = 0;
if (msg->from + msg->len > BCM53XXSPI_FLASH_WINDOW)
return -EINVAL;
bcm53xxspi_enable_bspi(b53spi);
memcpy_fromio(msg->buf, b53spi->mmio_base + msg->from, msg->len);
msg->retlen = msg->len;
return ret;
}
/************************************************** /**************************************************
* BCMA * BCMA
**************************************************/ **************************************************/
...@@ -222,6 +287,7 @@ MODULE_DEVICE_TABLE(bcma, bcm53xxspi_bcma_tbl); ...@@ -222,6 +287,7 @@ MODULE_DEVICE_TABLE(bcma, bcm53xxspi_bcma_tbl);
static int bcm53xxspi_bcma_probe(struct bcma_device *core) static int bcm53xxspi_bcma_probe(struct bcma_device *core)
{ {
struct device *dev = &core->dev;
struct bcm53xxspi *b53spi; struct bcm53xxspi *b53spi;
struct spi_master *master; struct spi_master *master;
int err; int err;
...@@ -231,7 +297,7 @@ static int bcm53xxspi_bcma_probe(struct bcma_device *core) ...@@ -231,7 +297,7 @@ static int bcm53xxspi_bcma_probe(struct bcma_device *core)
return -ENOTSUPP; return -ENOTSUPP;
} }
master = spi_alloc_master(&core->dev, sizeof(*b53spi)); master = spi_alloc_master(dev, sizeof(*b53spi));
if (!master) if (!master)
return -ENOMEM; return -ENOMEM;
...@@ -239,11 +305,19 @@ static int bcm53xxspi_bcma_probe(struct bcma_device *core) ...@@ -239,11 +305,19 @@ static int bcm53xxspi_bcma_probe(struct bcma_device *core)
b53spi->master = master; b53spi->master = master;
b53spi->core = core; b53spi->core = core;
if (core->addr_s[0])
b53spi->mmio_base = devm_ioremap(dev, core->addr_s[0],
BCM53XXSPI_FLASH_WINDOW);
b53spi->bspi = true;
bcm53xxspi_disable_bspi(b53spi);
master->transfer_one = bcm53xxspi_transfer_one; master->transfer_one = bcm53xxspi_transfer_one;
if (b53spi->mmio_base)
master->spi_flash_read = bcm53xxspi_flash_read;
bcma_set_drvdata(core, b53spi); bcma_set_drvdata(core, b53spi);
err = devm_spi_register_master(&core->dev, master); err = devm_spi_register_master(dev, master);
if (err) { if (err) {
spi_master_put(master); spi_master_put(master);
bcma_set_drvdata(core, NULL); bcma_set_drvdata(core, NULL);
......
This diff is collapsed.
...@@ -23,7 +23,6 @@ ...@@ -23,7 +23,6 @@
#include <linux/clk.h> #include <linux/clk.h>
#include <linux/dmaengine.h> #include <linux/dmaengine.h>
#include <linux/dma-mapping.h> #include <linux/dma-mapping.h>
#include <linux/edma.h>
#include <linux/of.h> #include <linux/of.h>
#include <linux/of_device.h> #include <linux/of_device.h>
#include <linux/of_gpio.h> #include <linux/of_gpio.h>
...@@ -33,8 +32,6 @@ ...@@ -33,8 +32,6 @@
#include <linux/platform_data/spi-davinci.h> #include <linux/platform_data/spi-davinci.h>
#define SPI_NO_RESOURCE ((resource_size_t)-1)
#define CS_DEFAULT 0xFF #define CS_DEFAULT 0xFF
#define SPIFMT_PHASE_MASK BIT(16) #define SPIFMT_PHASE_MASK BIT(16)
...@@ -130,8 +127,6 @@ struct davinci_spi { ...@@ -130,8 +127,6 @@ struct davinci_spi {
struct dma_chan *dma_rx; struct dma_chan *dma_rx;
struct dma_chan *dma_tx; struct dma_chan *dma_tx;
int dma_rx_chnum;
int dma_tx_chnum;
struct davinci_spi_platform_data pdata; struct davinci_spi_platform_data pdata;
...@@ -797,35 +792,19 @@ static irqreturn_t davinci_spi_irq(s32 irq, void *data) ...@@ -797,35 +792,19 @@ static irqreturn_t davinci_spi_irq(s32 irq, void *data)
static int davinci_spi_request_dma(struct davinci_spi *dspi) static int davinci_spi_request_dma(struct davinci_spi *dspi)
{ {
dma_cap_mask_t mask;
struct device *sdev = dspi->bitbang.master->dev.parent; struct device *sdev = dspi->bitbang.master->dev.parent;
int r;
dma_cap_zero(mask);
dma_cap_set(DMA_SLAVE, mask);
dspi->dma_rx = dma_request_channel(mask, edma_filter_fn, dspi->dma_rx = dma_request_chan(sdev, "rx");
&dspi->dma_rx_chnum); if (IS_ERR(dspi->dma_rx))
if (!dspi->dma_rx) { return PTR_ERR(dspi->dma_rx);
dev_err(sdev, "request RX DMA channel failed\n");
r = -ENODEV;
goto rx_dma_failed;
}
dspi->dma_tx = dma_request_channel(mask, edma_filter_fn, dspi->dma_tx = dma_request_chan(sdev, "tx");
&dspi->dma_tx_chnum); if (IS_ERR(dspi->dma_tx)) {
if (!dspi->dma_tx) { dma_release_channel(dspi->dma_rx);
dev_err(sdev, "request TX DMA channel failed\n"); return PTR_ERR(dspi->dma_tx);
r = -ENODEV;
goto tx_dma_failed;
} }
return 0; return 0;
tx_dma_failed:
dma_release_channel(dspi->dma_rx);
rx_dma_failed:
return r;
} }
#if defined(CONFIG_OF) #if defined(CONFIG_OF)
...@@ -936,8 +915,6 @@ static int davinci_spi_probe(struct platform_device *pdev) ...@@ -936,8 +915,6 @@ static int davinci_spi_probe(struct platform_device *pdev)
struct davinci_spi *dspi; struct davinci_spi *dspi;
struct davinci_spi_platform_data *pdata; struct davinci_spi_platform_data *pdata;
struct resource *r; struct resource *r;
resource_size_t dma_rx_chan = SPI_NO_RESOURCE;
resource_size_t dma_tx_chan = SPI_NO_RESOURCE;
int ret = 0; int ret = 0;
u32 spipc0; u32 spipc0;
...@@ -1044,27 +1021,15 @@ static int davinci_spi_probe(struct platform_device *pdev) ...@@ -1044,27 +1021,15 @@ static int davinci_spi_probe(struct platform_device *pdev)
} }
} }
r = platform_get_resource(pdev, IORESOURCE_DMA, 0);
if (r)
dma_rx_chan = r->start;
r = platform_get_resource(pdev, IORESOURCE_DMA, 1);
if (r)
dma_tx_chan = r->start;
dspi->bitbang.txrx_bufs = davinci_spi_bufs; dspi->bitbang.txrx_bufs = davinci_spi_bufs;
if (dma_rx_chan != SPI_NO_RESOURCE &&
dma_tx_chan != SPI_NO_RESOURCE) { ret = davinci_spi_request_dma(dspi);
dspi->dma_rx_chnum = dma_rx_chan; if (ret == -EPROBE_DEFER) {
dspi->dma_tx_chnum = dma_tx_chan; goto free_clk;
} else if (ret) {
ret = davinci_spi_request_dma(dspi); dev_info(&pdev->dev, "DMA is not supported (%d)\n", ret);
if (ret) dspi->dma_rx = NULL;
goto free_clk; dspi->dma_tx = NULL;
dev_info(&pdev->dev, "DMA: supported\n");
dev_info(&pdev->dev, "DMA: RX channel: %pa, TX channel: %pa, event queue: %d\n",
&dma_rx_chan, &dma_tx_chan,
pdata->dma_event_q);
} }
dspi->get_rx = davinci_spi_rx_buf_u8; dspi->get_rx = davinci_spi_rx_buf_u8;
...@@ -1102,8 +1067,10 @@ static int davinci_spi_probe(struct platform_device *pdev) ...@@ -1102,8 +1067,10 @@ static int davinci_spi_probe(struct platform_device *pdev)
return ret; return ret;
free_dma: free_dma:
dma_release_channel(dspi->dma_rx); if (dspi->dma_rx) {
dma_release_channel(dspi->dma_tx); dma_release_channel(dspi->dma_rx);
dma_release_channel(dspi->dma_tx);
}
free_clk: free_clk:
clk_disable_unprepare(dspi->clk); clk_disable_unprepare(dspi->clk);
free_master: free_master:
...@@ -1134,6 +1101,11 @@ static int davinci_spi_remove(struct platform_device *pdev) ...@@ -1134,6 +1101,11 @@ static int davinci_spi_remove(struct platform_device *pdev)
clk_disable_unprepare(dspi->clk); clk_disable_unprepare(dspi->clk);
spi_master_put(master); spi_master_put(master);
if (dspi->dma_rx) {
dma_release_channel(dspi->dma_rx);
dma_release_channel(dspi->dma_tx);
}
return 0; return 0;
} }
......
...@@ -683,6 +683,7 @@ static int dln2_spi_probe(struct platform_device *pdev) ...@@ -683,6 +683,7 @@ static int dln2_spi_probe(struct platform_device *pdev)
struct spi_master *master; struct spi_master *master;
struct dln2_spi *dln2; struct dln2_spi *dln2;
struct dln2_platform_data *pdata = dev_get_platdata(&pdev->dev); struct dln2_platform_data *pdata = dev_get_platdata(&pdev->dev);
struct device *dev = &pdev->dev;
int ret; int ret;
master = spi_alloc_master(&pdev->dev, sizeof(*dln2)); master = spi_alloc_master(&pdev->dev, sizeof(*dln2));
...@@ -700,6 +701,7 @@ static int dln2_spi_probe(struct platform_device *pdev) ...@@ -700,6 +701,7 @@ static int dln2_spi_probe(struct platform_device *pdev)
} }
dln2->master = master; dln2->master = master;
dln2->master->dev.of_node = dev->of_node;
dln2->pdev = pdev; dln2->pdev = pdev;
dln2->port = pdata->port; dln2->port = pdata->port;
/* cs/mode can never be 0xff, so the first transfer will set them */ /* cs/mode can never be 0xff, so the first transfer will set them */
......
...@@ -67,7 +67,7 @@ static int spi_pci_probe(struct pci_dev *pdev, const struct pci_device_id *ent) ...@@ -67,7 +67,7 @@ static int spi_pci_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
dws->irq = pdev->irq; dws->irq = pdev->irq;
/* /*
* Specific handling for paltforms, like dma setup, * Specific handling for platforms, like dma setup,
* clock rate, FIFO depth. * clock rate, FIFO depth.
*/ */
if (desc) { if (desc) {
......
...@@ -121,18 +121,22 @@ enum dspi_trans_mode { ...@@ -121,18 +121,22 @@ enum dspi_trans_mode {
struct fsl_dspi_devtype_data { struct fsl_dspi_devtype_data {
enum dspi_trans_mode trans_mode; enum dspi_trans_mode trans_mode;
u8 max_clock_factor;
}; };
static const struct fsl_dspi_devtype_data vf610_data = { static const struct fsl_dspi_devtype_data vf610_data = {
.trans_mode = DSPI_EOQ_MODE, .trans_mode = DSPI_EOQ_MODE,
.max_clock_factor = 2,
}; };
static const struct fsl_dspi_devtype_data ls1021a_v1_data = { static const struct fsl_dspi_devtype_data ls1021a_v1_data = {
.trans_mode = DSPI_TCFQ_MODE, .trans_mode = DSPI_TCFQ_MODE,
.max_clock_factor = 8,
}; };
static const struct fsl_dspi_devtype_data ls2085a_data = { static const struct fsl_dspi_devtype_data ls2085a_data = {
.trans_mode = DSPI_TCFQ_MODE, .trans_mode = DSPI_TCFQ_MODE,
.max_clock_factor = 8,
}; };
struct fsl_dspi { struct fsl_dspi {
...@@ -726,6 +730,9 @@ static int dspi_probe(struct platform_device *pdev) ...@@ -726,6 +730,9 @@ static int dspi_probe(struct platform_device *pdev)
} }
clk_prepare_enable(dspi->clk); clk_prepare_enable(dspi->clk);
master->max_speed_hz =
clk_get_rate(dspi->clk) / dspi->devtype_data->max_clock_factor;
init_waitqueue_head(&dspi->waitq); init_waitqueue_head(&dspi->waitq);
platform_set_drvdata(pdev, master); platform_set_drvdata(pdev, master);
......
...@@ -245,7 +245,12 @@ static int fsl_espi_bufs(struct spi_device *spi, struct spi_transfer *t) ...@@ -245,7 +245,12 @@ static int fsl_espi_bufs(struct spi_device *spi, struct spi_transfer *t)
if (ret) if (ret)
return ret; return ret;
wait_for_completion(&mpc8xxx_spi->done); /* Won't hang up forever, SPI bus sometimes got lost interrupts... */
ret = wait_for_completion_timeout(&mpc8xxx_spi->done, 2 * HZ);
if (ret == 0)
dev_err(mpc8xxx_spi->dev,
"Transaction hanging up (left %d bytes)\n",
mpc8xxx_spi->count);
/* disable rx ints */ /* disable rx ints */
mpc8xxx_spi_write_reg(&reg_base->mask, 0); mpc8xxx_spi_write_reg(&reg_base->mask, 0);
...@@ -539,16 +544,31 @@ void fsl_espi_cpu_irq(struct mpc8xxx_spi *mspi, u32 events) ...@@ -539,16 +544,31 @@ void fsl_espi_cpu_irq(struct mpc8xxx_spi *mspi, u32 events)
if (events & SPIE_NE) { if (events & SPIE_NE) {
u32 rx_data, tmp; u32 rx_data, tmp;
u8 rx_data_8; u8 rx_data_8;
int rx_nr_bytes = 4;
int ret;
/* Spin until RX is done */ /* Spin until RX is done */
while (SPIE_RXCNT(events) < min(4, mspi->len)) { if (SPIE_RXCNT(events) < min(4, mspi->len)) {
cpu_relax(); ret = spin_event_timeout(
events = mpc8xxx_spi_read_reg(&reg_base->event); !(SPIE_RXCNT(events =
mpc8xxx_spi_read_reg(&reg_base->event)) <
min(4, mspi->len)),
10000, 0); /* 10 msec */
if (!ret)
dev_err(mspi->dev,
"tired waiting for SPIE_RXCNT\n");
} }
if (mspi->len >= 4) { if (mspi->len >= 4) {
rx_data = mpc8xxx_spi_read_reg(&reg_base->receive); rx_data = mpc8xxx_spi_read_reg(&reg_base->receive);
} else if (mspi->len <= 0) {
dev_err(mspi->dev,
"unexpected RX(SPIE_NE) interrupt occurred,\n"
"(local rxlen %d bytes, reg rxlen %d bytes)\n",
min(4, mspi->len), SPIE_RXCNT(events));
rx_nr_bytes = 0;
} else { } else {
rx_nr_bytes = mspi->len;
tmp = mspi->len; tmp = mspi->len;
rx_data = 0; rx_data = 0;
while (tmp--) { while (tmp--) {
...@@ -559,7 +579,7 @@ void fsl_espi_cpu_irq(struct mpc8xxx_spi *mspi, u32 events) ...@@ -559,7 +579,7 @@ void fsl_espi_cpu_irq(struct mpc8xxx_spi *mspi, u32 events)
rx_data <<= (4 - mspi->len) * 8; rx_data <<= (4 - mspi->len) * 8;
} }
mspi->len -= 4; mspi->len -= rx_nr_bytes;
if (mspi->rx) if (mspi->rx)
mspi->get_rx(rx_data, mspi); mspi->get_rx(rx_data, mspi);
......
...@@ -175,6 +175,7 @@ static int octeon_spi_transfer_one_message(struct spi_master *master, ...@@ -175,6 +175,7 @@ static int octeon_spi_transfer_one_message(struct spi_master *master,
static int octeon_spi_probe(struct platform_device *pdev) static int octeon_spi_probe(struct platform_device *pdev)
{ {
struct resource *res_mem; struct resource *res_mem;
void __iomem *reg_base;
struct spi_master *master; struct spi_master *master;
struct octeon_spi *p; struct octeon_spi *p;
int err = -ENOENT; int err = -ENOENT;
...@@ -186,19 +187,13 @@ static int octeon_spi_probe(struct platform_device *pdev) ...@@ -186,19 +187,13 @@ static int octeon_spi_probe(struct platform_device *pdev)
platform_set_drvdata(pdev, master); platform_set_drvdata(pdev, master);
res_mem = platform_get_resource(pdev, IORESOURCE_MEM, 0); res_mem = platform_get_resource(pdev, IORESOURCE_MEM, 0);
reg_base = devm_ioremap_resource(&pdev->dev, res_mem);
if (res_mem == NULL) { if (IS_ERR(reg_base)) {
dev_err(&pdev->dev, "found no memory resource\n"); err = PTR_ERR(reg_base);
err = -ENXIO;
goto fail;
}
if (!devm_request_mem_region(&pdev->dev, res_mem->start,
resource_size(res_mem), res_mem->name)) {
dev_err(&pdev->dev, "request_mem_region failed\n");
goto fail; goto fail;
} }
p->register_base = (u64)devm_ioremap(&pdev->dev, res_mem->start,
resource_size(res_mem)); p->register_base = (u64)reg_base;
master->num_chipselect = 4; master->num_chipselect = 4;
master->mode_bits = SPI_CPHA | master->mode_bits = SPI_CPHA |
......
...@@ -23,7 +23,6 @@ ...@@ -23,7 +23,6 @@
#include <linux/delay.h> #include <linux/delay.h>
#include <linux/dma-mapping.h> #include <linux/dma-mapping.h>
#include <linux/dmaengine.h> #include <linux/dmaengine.h>
#include <linux/omap-dma.h>
#include <linux/pinctrl/consumer.h> #include <linux/pinctrl/consumer.h>
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include <linux/err.h> #include <linux/err.h>
...@@ -103,9 +102,6 @@ struct omap2_mcspi_dma { ...@@ -103,9 +102,6 @@ struct omap2_mcspi_dma {
struct dma_chan *dma_tx; struct dma_chan *dma_tx;
struct dma_chan *dma_rx; struct dma_chan *dma_rx;
int dma_tx_sync_dev;
int dma_rx_sync_dev;
struct completion dma_tx_completion; struct completion dma_tx_completion;
struct completion dma_rx_completion; struct completion dma_rx_completion;
...@@ -964,8 +960,7 @@ static int omap2_mcspi_request_dma(struct spi_device *spi) ...@@ -964,8 +960,7 @@ static int omap2_mcspi_request_dma(struct spi_device *spi)
struct spi_master *master = spi->master; struct spi_master *master = spi->master;
struct omap2_mcspi *mcspi; struct omap2_mcspi *mcspi;
struct omap2_mcspi_dma *mcspi_dma; struct omap2_mcspi_dma *mcspi_dma;
dma_cap_mask_t mask; int ret = 0;
unsigned sig;
mcspi = spi_master_get_devdata(master); mcspi = spi_master_get_devdata(master);
mcspi_dma = mcspi->dma_channels + spi->chip_select; mcspi_dma = mcspi->dma_channels + spi->chip_select;
...@@ -973,34 +968,25 @@ static int omap2_mcspi_request_dma(struct spi_device *spi) ...@@ -973,34 +968,25 @@ static int omap2_mcspi_request_dma(struct spi_device *spi)
init_completion(&mcspi_dma->dma_rx_completion); init_completion(&mcspi_dma->dma_rx_completion);
init_completion(&mcspi_dma->dma_tx_completion); init_completion(&mcspi_dma->dma_tx_completion);
dma_cap_zero(mask); mcspi_dma->dma_rx = dma_request_chan(&master->dev,
dma_cap_set(DMA_SLAVE, mask); mcspi_dma->dma_rx_ch_name);
sig = mcspi_dma->dma_rx_sync_dev; if (IS_ERR(mcspi_dma->dma_rx)) {
ret = PTR_ERR(mcspi_dma->dma_rx);
mcspi_dma->dma_rx = mcspi_dma->dma_rx = NULL;
dma_request_slave_channel_compat(mask, omap_dma_filter_fn,
&sig, &master->dev,
mcspi_dma->dma_rx_ch_name);
if (!mcspi_dma->dma_rx)
goto no_dma; goto no_dma;
}
sig = mcspi_dma->dma_tx_sync_dev; mcspi_dma->dma_tx = dma_request_chan(&master->dev,
mcspi_dma->dma_tx = mcspi_dma->dma_tx_ch_name);
dma_request_slave_channel_compat(mask, omap_dma_filter_fn, if (IS_ERR(mcspi_dma->dma_tx)) {
&sig, &master->dev, ret = PTR_ERR(mcspi_dma->dma_tx);
mcspi_dma->dma_tx_ch_name); mcspi_dma->dma_tx = NULL;
if (!mcspi_dma->dma_tx) {
dma_release_channel(mcspi_dma->dma_rx); dma_release_channel(mcspi_dma->dma_rx);
mcspi_dma->dma_rx = NULL; mcspi_dma->dma_rx = NULL;
goto no_dma;
} }
return 0;
no_dma: no_dma:
dev_warn(&spi->dev, "not using DMA for McSPI\n"); return ret;
return -EAGAIN;
} }
static int omap2_mcspi_setup(struct spi_device *spi) static int omap2_mcspi_setup(struct spi_device *spi)
...@@ -1039,8 +1025,9 @@ static int omap2_mcspi_setup(struct spi_device *spi) ...@@ -1039,8 +1025,9 @@ static int omap2_mcspi_setup(struct spi_device *spi)
if (!mcspi_dma->dma_rx || !mcspi_dma->dma_tx) { if (!mcspi_dma->dma_rx || !mcspi_dma->dma_tx) {
ret = omap2_mcspi_request_dma(spi); ret = omap2_mcspi_request_dma(spi);
if (ret < 0 && ret != -EAGAIN) if (ret)
return ret; dev_warn(&spi->dev, "not using DMA for McSPI (%d)\n",
ret);
} }
ret = pm_runtime_get_sync(mcspi->dev); ret = pm_runtime_get_sync(mcspi->dev);
...@@ -1434,42 +1421,8 @@ static int omap2_mcspi_probe(struct platform_device *pdev) ...@@ -1434,42 +1421,8 @@ static int omap2_mcspi_probe(struct platform_device *pdev)
} }
for (i = 0; i < master->num_chipselect; i++) { for (i = 0; i < master->num_chipselect; i++) {
char *dma_rx_ch_name = mcspi->dma_channels[i].dma_rx_ch_name; sprintf(mcspi->dma_channels[i].dma_rx_ch_name, "rx%d", i);
char *dma_tx_ch_name = mcspi->dma_channels[i].dma_tx_ch_name; sprintf(mcspi->dma_channels[i].dma_tx_ch_name, "tx%d", i);
struct resource *dma_res;
sprintf(dma_rx_ch_name, "rx%d", i);
if (!pdev->dev.of_node) {
dma_res =
platform_get_resource_byname(pdev,
IORESOURCE_DMA,
dma_rx_ch_name);
if (!dma_res) {
dev_dbg(&pdev->dev,
"cannot get DMA RX channel\n");
status = -ENODEV;
break;
}
mcspi->dma_channels[i].dma_rx_sync_dev =
dma_res->start;
}
sprintf(dma_tx_ch_name, "tx%d", i);
if (!pdev->dev.of_node) {
dma_res =
platform_get_resource_byname(pdev,
IORESOURCE_DMA,
dma_tx_ch_name);
if (!dma_res) {
dev_dbg(&pdev->dev,
"cannot get DMA TX channel\n");
status = -ENODEV;
break;
}
mcspi->dma_channels[i].dma_tx_sync_dev =
dma_res->start;
}
} }
if (status < 0) if (status < 0)
......
This diff is collapsed.
This diff is collapsed.
...@@ -33,12 +33,10 @@ static int pxa2xx_spi_map_dma_buffer(struct driver_data *drv_data, ...@@ -33,12 +33,10 @@ static int pxa2xx_spi_map_dma_buffer(struct driver_data *drv_data,
dmadev = drv_data->tx_chan->device->dev; dmadev = drv_data->tx_chan->device->dev;
sgt = &drv_data->tx_sgt; sgt = &drv_data->tx_sgt;
buf = drv_data->tx; buf = drv_data->tx;
drv_data->tx_map_len = len;
} else { } else {
dmadev = drv_data->rx_chan->device->dev; dmadev = drv_data->rx_chan->device->dev;
sgt = &drv_data->rx_sgt; sgt = &drv_data->rx_sgt;
buf = drv_data->rx; buf = drv_data->rx;
drv_data->rx_map_len = len;
} }
nents = DIV_ROUND_UP(len, SZ_2K); nents = DIV_ROUND_UP(len, SZ_2K);
...@@ -55,11 +53,7 @@ static int pxa2xx_spi_map_dma_buffer(struct driver_data *drv_data, ...@@ -55,11 +53,7 @@ static int pxa2xx_spi_map_dma_buffer(struct driver_data *drv_data,
for_each_sg(sgt->sgl, sg, sgt->nents, i) { for_each_sg(sgt->sgl, sg, sgt->nents, i) {
size_t bytes = min_t(size_t, len, SZ_2K); size_t bytes = min_t(size_t, len, SZ_2K);
if (buf) sg_set_buf(sg, pbuf, bytes);
sg_set_buf(sg, pbuf, bytes);
else
sg_set_buf(sg, drv_data->dummy, bytes);
pbuf += bytes; pbuf += bytes;
len -= bytes; len -= bytes;
} }
...@@ -133,9 +127,6 @@ static void pxa2xx_spi_dma_transfer_complete(struct driver_data *drv_data, ...@@ -133,9 +127,6 @@ static void pxa2xx_spi_dma_transfer_complete(struct driver_data *drv_data,
if (!error) { if (!error) {
pxa2xx_spi_unmap_dma_buffers(drv_data); pxa2xx_spi_unmap_dma_buffers(drv_data);
drv_data->tx += drv_data->tx_map_len;
drv_data->rx += drv_data->rx_map_len;
msg->actual_length += drv_data->len; msg->actual_length += drv_data->len;
msg->state = pxa2xx_spi_next_transfer(drv_data); msg->state = pxa2xx_spi_next_transfer(drv_data);
} else { } else {
...@@ -267,19 +258,22 @@ irqreturn_t pxa2xx_spi_dma_transfer(struct driver_data *drv_data) ...@@ -267,19 +258,22 @@ irqreturn_t pxa2xx_spi_dma_transfer(struct driver_data *drv_data)
int pxa2xx_spi_dma_prepare(struct driver_data *drv_data, u32 dma_burst) int pxa2xx_spi_dma_prepare(struct driver_data *drv_data, u32 dma_burst)
{ {
struct dma_async_tx_descriptor *tx_desc, *rx_desc; struct dma_async_tx_descriptor *tx_desc, *rx_desc;
int err = 0;
tx_desc = pxa2xx_spi_dma_prepare_one(drv_data, DMA_MEM_TO_DEV); tx_desc = pxa2xx_spi_dma_prepare_one(drv_data, DMA_MEM_TO_DEV);
if (!tx_desc) { if (!tx_desc) {
dev_err(&drv_data->pdev->dev, dev_err(&drv_data->pdev->dev,
"failed to get DMA TX descriptor\n"); "failed to get DMA TX descriptor\n");
return -EBUSY; err = -EBUSY;
goto err_tx;
} }
rx_desc = pxa2xx_spi_dma_prepare_one(drv_data, DMA_DEV_TO_MEM); rx_desc = pxa2xx_spi_dma_prepare_one(drv_data, DMA_DEV_TO_MEM);
if (!rx_desc) { if (!rx_desc) {
dev_err(&drv_data->pdev->dev, dev_err(&drv_data->pdev->dev,
"failed to get DMA RX descriptor\n"); "failed to get DMA RX descriptor\n");
return -EBUSY; err = -EBUSY;
goto err_rx;
} }
/* We are ready when RX completes */ /* We are ready when RX completes */
...@@ -289,6 +283,12 @@ int pxa2xx_spi_dma_prepare(struct driver_data *drv_data, u32 dma_burst) ...@@ -289,6 +283,12 @@ int pxa2xx_spi_dma_prepare(struct driver_data *drv_data, u32 dma_burst)
dmaengine_submit(rx_desc); dmaengine_submit(rx_desc);
dmaengine_submit(tx_desc); dmaengine_submit(tx_desc);
return 0; return 0;
err_rx:
dmaengine_terminate_async(drv_data->tx_chan);
err_tx:
pxa2xx_spi_unmap_dma_buffers(drv_data);
return err;
} }
void pxa2xx_spi_dma_start(struct driver_data *drv_data) void pxa2xx_spi_dma_start(struct driver_data *drv_data)
...@@ -308,10 +308,6 @@ int pxa2xx_spi_dma_setup(struct driver_data *drv_data) ...@@ -308,10 +308,6 @@ int pxa2xx_spi_dma_setup(struct driver_data *drv_data)
dma_cap_zero(mask); dma_cap_zero(mask);
dma_cap_set(DMA_SLAVE, mask); dma_cap_set(DMA_SLAVE, mask);
drv_data->dummy = devm_kzalloc(dev, SZ_2K, GFP_KERNEL);
if (!drv_data->dummy)
return -ENOMEM;
drv_data->tx_chan = dma_request_slave_channel_compat(mask, drv_data->tx_chan = dma_request_slave_channel_compat(mask,
pdata->dma_filter, pdata->tx_param, dev, "tx"); pdata->dma_filter, pdata->tx_param, dev, "tx");
if (!drv_data->tx_chan) if (!drv_data->tx_chan)
......
...@@ -173,8 +173,8 @@ static int pxa2xx_spi_pci_probe(struct pci_dev *dev, ...@@ -173,8 +173,8 @@ static int pxa2xx_spi_pci_probe(struct pci_dev *dev,
ssp->type = c->type; ssp->type = c->type;
snprintf(buf, sizeof(buf), "pxa2xx-spi.%d", ssp->port_id); snprintf(buf, sizeof(buf), "pxa2xx-spi.%d", ssp->port_id);
ssp->clk = clk_register_fixed_rate(&dev->dev, buf , NULL, ssp->clk = clk_register_fixed_rate(&dev->dev, buf , NULL, 0,
CLK_IS_ROOT, c->max_clk_rate); c->max_clk_rate);
if (IS_ERR(ssp->clk)) if (IS_ERR(ssp->clk))
return PTR_ERR(ssp->clk); return PTR_ERR(ssp->clk);
......
...@@ -570,9 +570,8 @@ static void giveback(struct driver_data *drv_data) ...@@ -570,9 +570,8 @@ static void giveback(struct driver_data *drv_data)
/* see if the next and current messages point /* see if the next and current messages point
* to the same chip * to the same chip
*/ */
if (next_msg && next_msg->spi != msg->spi) if ((next_msg && next_msg->spi != msg->spi) ||
next_msg = NULL; msg->state == ERROR_STATE)
if (!next_msg || msg->state == ERROR_STATE)
cs_deassert(drv_data); cs_deassert(drv_data);
} }
...@@ -928,6 +927,7 @@ static void pump_transfers(unsigned long data) ...@@ -928,6 +927,7 @@ static void pump_transfers(unsigned long data)
u32 dma_thresh = drv_data->cur_chip->dma_threshold; u32 dma_thresh = drv_data->cur_chip->dma_threshold;
u32 dma_burst = drv_data->cur_chip->dma_burst_size; u32 dma_burst = drv_data->cur_chip->dma_burst_size;
u32 change_mask = pxa2xx_spi_get_ssrc1_change_mask(drv_data); u32 change_mask = pxa2xx_spi_get_ssrc1_change_mask(drv_data);
int err;
/* Get current state information */ /* Get current state information */
message = drv_data->cur_msg; message = drv_data->cur_msg;
...@@ -1047,7 +1047,12 @@ static void pump_transfers(unsigned long data) ...@@ -1047,7 +1047,12 @@ static void pump_transfers(unsigned long data)
/* Ensure we have the correct interrupt handler */ /* Ensure we have the correct interrupt handler */
drv_data->transfer_handler = pxa2xx_spi_dma_transfer; drv_data->transfer_handler = pxa2xx_spi_dma_transfer;
pxa2xx_spi_dma_prepare(drv_data, dma_burst); err = pxa2xx_spi_dma_prepare(drv_data, dma_burst);
if (err) {
message->status = err;
giveback(drv_data);
return;
}
/* Clear status and start DMA engine */ /* Clear status and start DMA engine */
cr1 = chip->cr1 | dma_thresh | drv_data->dma_cr1; cr1 = chip->cr1 | dma_thresh | drv_data->dma_cr1;
...@@ -1543,7 +1548,6 @@ static int pxa2xx_spi_probe(struct platform_device *pdev) ...@@ -1543,7 +1548,6 @@ static int pxa2xx_spi_probe(struct platform_device *pdev)
drv_data->pdev = pdev; drv_data->pdev = pdev;
drv_data->ssp = ssp; drv_data->ssp = ssp;
master->dev.parent = &pdev->dev;
master->dev.of_node = pdev->dev.of_node; master->dev.of_node = pdev->dev.of_node;
/* the spi->mode bits understood by this driver: */ /* the spi->mode bits understood by this driver: */
master->mode_bits = SPI_CPOL | SPI_CPHA | SPI_CS_HIGH | SPI_LOOP; master->mode_bits = SPI_CPOL | SPI_CPHA | SPI_CS_HIGH | SPI_LOOP;
...@@ -1556,6 +1560,7 @@ static int pxa2xx_spi_probe(struct platform_device *pdev) ...@@ -1556,6 +1560,7 @@ static int pxa2xx_spi_probe(struct platform_device *pdev)
master->unprepare_transfer_hardware = pxa2xx_spi_unprepare_transfer; master->unprepare_transfer_hardware = pxa2xx_spi_unprepare_transfer;
master->fw_translate_cs = pxa2xx_spi_fw_translate_cs; master->fw_translate_cs = pxa2xx_spi_fw_translate_cs;
master->auto_runtime_pm = true; master->auto_runtime_pm = true;
master->flags = SPI_MASTER_MUST_RX | SPI_MASTER_MUST_TX;
drv_data->ssp_type = ssp->type; drv_data->ssp_type = ssp->type;
......
...@@ -56,7 +56,6 @@ struct driver_data { ...@@ -56,7 +56,6 @@ struct driver_data {
struct sg_table tx_sgt; struct sg_table tx_sgt;
int rx_nents; int rx_nents;
int tx_nents; int tx_nents;
void *dummy;
atomic_t dma_running; atomic_t dma_running;
/* Current message transfer state info */ /* Current message transfer state info */
...@@ -69,8 +68,6 @@ struct driver_data { ...@@ -69,8 +68,6 @@ struct driver_data {
void *rx; void *rx;
void *rx_end; void *rx_end;
int dma_mapped; int dma_mapped;
size_t rx_map_len;
size_t tx_map_len;
u8 n_bytes; u8 n_bytes;
int (*write)(struct driver_data *drv_data); int (*write)(struct driver_data *drv_data);
int (*read)(struct driver_data *drv_data); int (*read)(struct driver_data *drv_data);
......
...@@ -937,6 +937,10 @@ static int spi_qup_pm_suspend_runtime(struct device *device) ...@@ -937,6 +937,10 @@ static int spi_qup_pm_suspend_runtime(struct device *device)
config = readl(controller->base + QUP_CONFIG); config = readl(controller->base + QUP_CONFIG);
config |= QUP_CONFIG_CLOCK_AUTO_GATE; config |= QUP_CONFIG_CLOCK_AUTO_GATE;
writel_relaxed(config, controller->base + QUP_CONFIG); writel_relaxed(config, controller->base + QUP_CONFIG);
clk_disable_unprepare(controller->cclk);
clk_disable_unprepare(controller->iclk);
return 0; return 0;
} }
...@@ -945,6 +949,15 @@ static int spi_qup_pm_resume_runtime(struct device *device) ...@@ -945,6 +949,15 @@ static int spi_qup_pm_resume_runtime(struct device *device)
struct spi_master *master = dev_get_drvdata(device); struct spi_master *master = dev_get_drvdata(device);
struct spi_qup *controller = spi_master_get_devdata(master); struct spi_qup *controller = spi_master_get_devdata(master);
u32 config; u32 config;
int ret;
ret = clk_prepare_enable(controller->iclk);
if (ret)
return ret;
ret = clk_prepare_enable(controller->cclk);
if (ret)
return ret;
/* Disable clocks auto gaiting */ /* Disable clocks auto gaiting */
config = readl_relaxed(controller->base + QUP_CONFIG); config = readl_relaxed(controller->base + QUP_CONFIG);
...@@ -1017,6 +1030,8 @@ static int spi_qup_remove(struct platform_device *pdev) ...@@ -1017,6 +1030,8 @@ static int spi_qup_remove(struct platform_device *pdev)
pm_runtime_put_noidle(&pdev->dev); pm_runtime_put_noidle(&pdev->dev);
pm_runtime_disable(&pdev->dev); pm_runtime_disable(&pdev->dev);
spi_master_put(master);
return 0; return 0;
} }
......
...@@ -744,10 +744,8 @@ static int rockchip_spi_probe(struct platform_device *pdev) ...@@ -744,10 +744,8 @@ static int rockchip_spi_probe(struct platform_device *pdev)
rs->dma_rx.ch = dma_request_chan(rs->dev, "rx"); rs->dma_rx.ch = dma_request_chan(rs->dev, "rx");
if (IS_ERR(rs->dma_rx.ch)) { if (IS_ERR(rs->dma_rx.ch)) {
if (PTR_ERR(rs->dma_rx.ch) == -EPROBE_DEFER) { if (PTR_ERR(rs->dma_rx.ch) == -EPROBE_DEFER) {
dma_release_channel(rs->dma_tx.ch);
rs->dma_tx.ch = NULL;
ret = -EPROBE_DEFER; ret = -EPROBE_DEFER;
goto err_get_fifo_len; goto err_free_dma_tx;
} }
dev_warn(rs->dev, "Failed to request RX DMA channel\n"); dev_warn(rs->dev, "Failed to request RX DMA channel\n");
rs->dma_rx.ch = NULL; rs->dma_rx.ch = NULL;
...@@ -775,10 +773,11 @@ static int rockchip_spi_probe(struct platform_device *pdev) ...@@ -775,10 +773,11 @@ static int rockchip_spi_probe(struct platform_device *pdev)
err_register_master: err_register_master:
pm_runtime_disable(&pdev->dev); pm_runtime_disable(&pdev->dev);
if (rs->dma_tx.ch)
dma_release_channel(rs->dma_tx.ch);
if (rs->dma_rx.ch) if (rs->dma_rx.ch)
dma_release_channel(rs->dma_rx.ch); dma_release_channel(rs->dma_rx.ch);
err_free_dma_tx:
if (rs->dma_tx.ch)
dma_release_channel(rs->dma_tx.ch);
err_get_fifo_len: err_get_fifo_len:
clk_disable_unprepare(rs->spiclk); clk_disable_unprepare(rs->spiclk);
err_spiclk_enable: err_spiclk_enable:
......
...@@ -345,12 +345,13 @@ static int spi_st_probe(struct platform_device *pdev) ...@@ -345,12 +345,13 @@ static int spi_st_probe(struct platform_device *pdev)
spi_st->clk = devm_clk_get(&pdev->dev, "ssc"); spi_st->clk = devm_clk_get(&pdev->dev, "ssc");
if (IS_ERR(spi_st->clk)) { if (IS_ERR(spi_st->clk)) {
dev_err(&pdev->dev, "Unable to request clock\n"); dev_err(&pdev->dev, "Unable to request clock\n");
return PTR_ERR(spi_st->clk); ret = PTR_ERR(spi_st->clk);
goto put_master;
} }
ret = spi_st_clk_enable(spi_st); ret = spi_st_clk_enable(spi_st);
if (ret) if (ret)
return ret; goto put_master;
init_completion(&spi_st->done); init_completion(&spi_st->done);
...@@ -408,7 +409,8 @@ static int spi_st_probe(struct platform_device *pdev) ...@@ -408,7 +409,8 @@ static int spi_st_probe(struct platform_device *pdev)
clk_disable: clk_disable:
spi_st_clk_disable(spi_st); spi_st_clk_disable(spi_st);
put_master:
spi_master_put(master);
return ret; return ret;
} }
......
...@@ -360,7 +360,7 @@ static int zynqmp_prepare_transfer_hardware(struct spi_master *master) ...@@ -360,7 +360,7 @@ static int zynqmp_prepare_transfer_hardware(struct spi_master *master)
ret = clk_enable(xqspi->refclk); ret = clk_enable(xqspi->refclk);
if (ret) if (ret)
goto clk_err; return ret;
ret = clk_enable(xqspi->pclk); ret = clk_enable(xqspi->pclk);
if (ret) if (ret)
...@@ -369,6 +369,7 @@ static int zynqmp_prepare_transfer_hardware(struct spi_master *master) ...@@ -369,6 +369,7 @@ static int zynqmp_prepare_transfer_hardware(struct spi_master *master)
zynqmp_gqspi_write(xqspi, GQSPI_EN_OFST, GQSPI_EN_MASK); zynqmp_gqspi_write(xqspi, GQSPI_EN_OFST, GQSPI_EN_MASK);
return 0; return 0;
clk_err: clk_err:
clk_disable(xqspi->refclk);
return ret; return ret;
} }
......
...@@ -717,9 +717,11 @@ static int spi_map_buf(struct spi_master *master, struct device *dev, ...@@ -717,9 +717,11 @@ static int spi_map_buf(struct spi_master *master, struct device *dev,
if (vmalloced_buf) { if (vmalloced_buf) {
desc_len = min_t(int, max_seg_size, PAGE_SIZE); desc_len = min_t(int, max_seg_size, PAGE_SIZE);
sgs = DIV_ROUND_UP(len + offset_in_page(buf), desc_len); sgs = DIV_ROUND_UP(len + offset_in_page(buf), desc_len);
} else { } else if (virt_addr_valid(buf)) {
desc_len = min_t(int, max_seg_size, master->max_dma_len); desc_len = min_t(int, max_seg_size, master->max_dma_len);
sgs = DIV_ROUND_UP(len, desc_len); sgs = DIV_ROUND_UP(len, desc_len);
} else {
return -EINVAL;
} }
ret = sg_alloc_table(sgt, sgs, GFP_KERNEL); ret = sg_alloc_table(sgt, sgs, GFP_KERNEL);
...@@ -933,7 +935,7 @@ static int spi_map_msg(struct spi_master *master, struct spi_message *msg) ...@@ -933,7 +935,7 @@ static int spi_map_msg(struct spi_master *master, struct spi_message *msg)
* spi_transfer_one_message - Default implementation of transfer_one_message() * spi_transfer_one_message - Default implementation of transfer_one_message()
* *
* This is a standard implementation of transfer_one_message() for * This is a standard implementation of transfer_one_message() for
* drivers which impelment a transfer_one() operation. It provides * drivers which implement a transfer_one() operation. It provides
* standard handling of delays and chip select management. * standard handling of delays and chip select management.
*/ */
static int spi_transfer_one_message(struct spi_master *master, static int spi_transfer_one_message(struct spi_master *master,
...@@ -1764,6 +1766,7 @@ struct spi_master *spi_alloc_master(struct device *dev, unsigned size) ...@@ -1764,6 +1766,7 @@ struct spi_master *spi_alloc_master(struct device *dev, unsigned size)
master->num_chipselect = 1; master->num_chipselect = 1;
master->dev.class = &spi_master_class; master->dev.class = &spi_master_class;
master->dev.parent = dev; master->dev.parent = dev;
pm_suspend_ignore_children(&master->dev, true);
spi_master_set_devdata(master, &master[1]); spi_master_set_devdata(master, &master[1]);
return master; return master;
......
...@@ -372,6 +372,7 @@ static inline void spi_unregister_driver(struct spi_driver *sdrv) ...@@ -372,6 +372,7 @@ static inline void spi_unregister_driver(struct spi_driver *sdrv)
* @unprepare_message: undo any work done by prepare_message(). * @unprepare_message: undo any work done by prepare_message().
* @spi_flash_read: to support spi-controller hardwares that provide * @spi_flash_read: to support spi-controller hardwares that provide
* accelerated interface to read from flash devices. * accelerated interface to read from flash devices.
* @flash_read_supported: spi device supports flash read
* @cs_gpios: Array of GPIOs to use as chip select lines; one per CS * @cs_gpios: Array of GPIOs to use as chip select lines; one per CS
* number. Any individual value may be -ENOENT for CS lines that * number. Any individual value may be -ENOENT for CS lines that
* are not GPIOs (driven by the SPI controller itself). * are not GPIOs (driven by the SPI controller itself).
...@@ -529,6 +530,7 @@ struct spi_master { ...@@ -529,6 +530,7 @@ struct spi_master {
struct spi_message *message); struct spi_message *message);
int (*spi_flash_read)(struct spi_device *spi, int (*spi_flash_read)(struct spi_device *spi,
struct spi_flash_read_message *msg); struct spi_flash_read_message *msg);
bool (*flash_read_supported)(struct spi_device *spi);
/* /*
* These hooks are for drivers that use a generic implementation * These hooks are for drivers that use a generic implementation
...@@ -1158,7 +1160,9 @@ struct spi_flash_read_message { ...@@ -1158,7 +1160,9 @@ struct spi_flash_read_message {
/* SPI core interface for flash read support */ /* SPI core interface for flash read support */
static inline bool spi_flash_read_supported(struct spi_device *spi) static inline bool spi_flash_read_supported(struct spi_device *spi)
{ {
return spi->master->spi_flash_read ? true : false; return spi->master->spi_flash_read &&
(!spi->master->flash_read_supported ||
spi->master->flash_read_supported(spi));
} }
int spi_flash_read(struct spi_device *spi, int spi_flash_read(struct spi_device *spi,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment