spi: Updates for v5.19
This is quite a quiet release but some new drivers mean that the diffstat is fairly large, the new drivers include the aspeed driver which is migrated from MTD as part of the ongoing move of controllers with specialised support for SPI flashes into the SPI subsystem. - Support for devices which flip CPHA during recieve only transfers (eg, if MOSI and MISO have inverted polarity). - Overhaul of the i.MX driver, including the addition of PIO support for better performance on small transfers. - Migration of the Aspeed driver from MTD. - Support for Aspeed AST2400, Ingenic JZ4775 and X1/2000 and MediaTek IPM and SFI. -----BEGIN PGP SIGNATURE----- iQEzBAABCgAdFiEEreZoqmdXGLWf4p/qJNaLcl1Uh9AFAmKLh9wACgkQJNaLcl1U h9ApgAf/SvcHRyM06hw+00D2PAco7mxSiAs7rebxj4f0zPRNvUrYbUDWjakeKjoB ocmpY6AW/vRBUM9xZsiQiLp/9NPltnVghZIBgp5gcS8zaXdxiZjZ68Z+uYp9SP9v zj8F6bBQzk3lvuY+Cr1f68iXICA62Aa1yX28UFVLZkV0d+0AazpECZkb3hjOiL6P 0qSV6IfoFlJXYsyvAS4/MqYrHklSCD/0Ek09V9jPpZJPHn1ldbFs5zkstfs0PMc/ YeDlpStkjJE4OQ6+z6Ou/wGbcw11nWvKCSqF4bGRXLrgR/uIjOKomwOzaBUf/A6Q AxQcKMtsBTx0VzYmwJwopj0qDvOq/A== =6yj3 -----END PGP SIGNATURE----- Merge tag 'spi-v5.19' of git://git.kernel.org/pub/scm/linux/kernel/git/broonie/spi Pull spi updates from Mark Brown: "This is quite a quiet release but some new drivers mean that the diffstat is fairly large. The new drivers include the aspeed driver which is migrated from MTD as part of the ongoing move of controllers with specialised support for SPI flashes into the SPI subsystem. - Support for devices which flip CPHA during recieve only transfers (eg, if MOSI and MISO have inverted polarity). - Overhaul of the i.MX driver, including the addition of PIO support for better performance on small transfers. - Migration of the Aspeed driver from MTD. - Support for Aspeed AST2400, Ingenic JZ4775 and X1/2000 and MediaTek IPM and SFI" * tag 'spi-v5.19' of git://git.kernel.org/pub/scm/linux/kernel/git/broonie/spi: (84 commits) spi: spi-au1550: replace ternary operator with min() mtd: spi-nor: aspeed: set the decoding size to at least 2MB for AST2600 spi: aspeed: Calibrate read timings spi: aspeed: Add support for the AST2400 SPI controller spi: aspeed: Workaround AST2500 limitations spi: aspeed: Adjust direct mapping to device size spi: aspeed: Add support for direct mapping spi: spi-mem: Convert Aspeed SMC driver to spi-mem spi: Convert the Aspeed SMC controllers device tree binding spi: spi-cadence: Update ISR status variable type to irqreturn_t spi: Doc fix - Describe add_lock and dma_map_dev in spi_controller spi: cadence-quadspi: Handle spi_unregister_master() in remove() spi: stm32-qspi: Remove SR_BUSY bit check before sending command spi: stm32-qspi: Always check SR_TCF flags in stm32_qspi_wait_cmd() spi: stm32-qspi: Fix wait_cmd timeout in APM mode spi: cadence-quadspi: remove unnecessary (void *) casts spi: cadence-quadspi: Add missing blank line in cqspi_request_mmap_dma() spi: spi-imx: mx51_ecspi_prepare_message(): skip writing MX51_ECSPI_CONFIG register if unchanged spi: spi-imx: add PIO polling support spi: spi-imx: replace struct spi_imx_data::bitbang by pointer to struct spi_controller ...
This commit is contained in:
commit
d8e0f976f1
@ -1,51 +0,0 @@
|
||||
* Aspeed Firmware Memory controller
|
||||
* Aspeed SPI Flash Memory Controller
|
||||
|
||||
The Firmware Memory Controller in the Aspeed AST2500 SoC supports
|
||||
three chip selects, two of which are always of SPI type and the third
|
||||
can be SPI or NOR type flash. These bindings only describe SPI.
|
||||
|
||||
The two SPI flash memory controllers in the AST2500 each support two
|
||||
chip selects.
|
||||
|
||||
Required properties:
|
||||
- compatible : Should be one of
|
||||
"aspeed,ast2400-fmc" for the AST2400 Firmware Memory Controller
|
||||
"aspeed,ast2400-spi" for the AST2400 SPI Flash memory Controller
|
||||
"aspeed,ast2500-fmc" for the AST2500 Firmware Memory Controller
|
||||
"aspeed,ast2500-spi" for the AST2500 SPI flash memory controllers
|
||||
|
||||
- reg : the first contains the control register location and length,
|
||||
the second contains the memory window mapping address and length
|
||||
- #address-cells : must be 1 corresponding to chip select child binding
|
||||
- #size-cells : must be 0 corresponding to chip select child binding
|
||||
|
||||
Optional properties:
|
||||
- interrupts : Should contain the interrupt for the dma device if an
|
||||
FMC
|
||||
|
||||
The child nodes are the SPI flash modules which must have a compatible
|
||||
property as specified in bindings/mtd/jedec,spi-nor.txt
|
||||
|
||||
Optionally, the child node can contain properties for SPI mode (may be
|
||||
ignored):
|
||||
- spi-max-frequency - max frequency of spi bus
|
||||
|
||||
|
||||
Example:
|
||||
fmc: fmc@1e620000 {
|
||||
compatible = "aspeed,ast2500-fmc";
|
||||
reg = < 0x1e620000 0x94
|
||||
0x20000000 0x02000000 >;
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
interrupts = <19>;
|
||||
flash@0 {
|
||||
reg = < 0 >;
|
||||
compatible = "jedec,spi-nor";
|
||||
/* spi-max-frequency = <>; */
|
||||
/* m25p,fast-read; */
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
};
|
||||
};
|
@ -0,0 +1,82 @@
|
||||
# SPDX-License-Identifier: (GPL-2.0 OR BSD-2-Clause)
|
||||
%YAML 1.2
|
||||
---
|
||||
$id: http://devicetree.org/schemas/spi/aspeed,ast2600-fmc.yaml#
|
||||
$schema: http://devicetree.org/meta-schemas/core.yaml#
|
||||
|
||||
title: Aspeed SMC controllers bindings
|
||||
|
||||
maintainers:
|
||||
- Chin-Ting Kuo <chin-ting_kuo@aspeedtech.com>
|
||||
- Cédric Le Goater <clg@kaod.org>
|
||||
|
||||
description: |
|
||||
This binding describes the Aspeed Static Memory Controllers (FMC and
|
||||
SPI) of the AST2400, AST2500 and AST2600 SOCs.
|
||||
|
||||
allOf:
|
||||
- $ref: "spi-controller.yaml#"
|
||||
|
||||
properties:
|
||||
compatible:
|
||||
enum:
|
||||
- aspeed,ast2600-fmc
|
||||
- aspeed,ast2600-spi
|
||||
- aspeed,ast2500-fmc
|
||||
- aspeed,ast2500-spi
|
||||
- aspeed,ast2400-fmc
|
||||
- aspeed,ast2400-spi
|
||||
|
||||
reg:
|
||||
items:
|
||||
- description: registers
|
||||
- description: memory mapping
|
||||
|
||||
clocks:
|
||||
maxItems: 1
|
||||
|
||||
interrupts:
|
||||
maxItems: 1
|
||||
|
||||
required:
|
||||
- compatible
|
||||
- reg
|
||||
- clocks
|
||||
|
||||
unevaluatedProperties: false
|
||||
|
||||
examples:
|
||||
- |
|
||||
#include <dt-bindings/interrupt-controller/arm-gic.h>
|
||||
#include <dt-bindings/interrupt-controller/aspeed-scu-ic.h>
|
||||
#include <dt-bindings/clock/ast2600-clock.h>
|
||||
|
||||
spi@1e620000 {
|
||||
reg = <0x1e620000 0xc4>, <0x20000000 0x10000000>;
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
compatible = "aspeed,ast2600-fmc";
|
||||
clocks = <&syscon ASPEED_CLK_AHB>;
|
||||
interrupts = <GIC_SPI 39 IRQ_TYPE_LEVEL_HIGH>;
|
||||
|
||||
flash@0 {
|
||||
reg = < 0 >;
|
||||
compatible = "jedec,spi-nor";
|
||||
spi-max-frequency = <50000000>;
|
||||
spi-rx-bus-width = <2>;
|
||||
};
|
||||
|
||||
flash@1 {
|
||||
reg = < 1 >;
|
||||
compatible = "jedec,spi-nor";
|
||||
spi-max-frequency = <50000000>;
|
||||
spi-rx-bus-width = <2>;
|
||||
};
|
||||
|
||||
flash@2 {
|
||||
reg = < 2 >;
|
||||
compatible = "jedec,spi-nor";
|
||||
spi-max-frequency = <50000000>;
|
||||
spi-rx-bus-width = <2>;
|
||||
};
|
||||
};
|
@ -18,7 +18,10 @@ properties:
|
||||
oneOf:
|
||||
- enum:
|
||||
- ingenic,jz4750-spi
|
||||
- ingenic,jz4775-spi
|
||||
- ingenic,jz4780-spi
|
||||
- ingenic,x1000-spi
|
||||
- ingenic,x2000-spi
|
||||
- items:
|
||||
- enum:
|
||||
- ingenic,jz4760-spi
|
||||
|
@ -53,16 +53,20 @@ properties:
|
||||
maxItems: 1
|
||||
|
||||
clocks:
|
||||
minItems: 3
|
||||
items:
|
||||
- description: clock used for the parent clock
|
||||
- description: clock used for the muxes clock
|
||||
- description: clock used for the clock gate
|
||||
- description: clock used for the AHB bus, this clock is optional
|
||||
|
||||
clock-names:
|
||||
minItems: 3
|
||||
items:
|
||||
- const: parent-clk
|
||||
- const: sel-clk
|
||||
- const: spi-clk
|
||||
- const: hclk
|
||||
|
||||
mediatek,pad-select:
|
||||
$ref: /schemas/types.yaml#/definitions/uint32-array
|
||||
|
@ -0,0 +1,88 @@
|
||||
# SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause)
|
||||
%YAML 1.2
|
||||
---
|
||||
$id: http://devicetree.org/schemas/spi/mediatek,spi-mtk-snfi.yaml#
|
||||
$schema: http://devicetree.org/meta-schemas/core.yaml#
|
||||
|
||||
title: SPI-NAND flash controller for MediaTek ARM SoCs
|
||||
|
||||
maintainers:
|
||||
- Chuanhong Guo <gch981213@gmail.com>
|
||||
|
||||
description: |
|
||||
The Mediatek SPI-NAND flash controller is an extended version of
|
||||
the Mediatek NAND flash controller. It can perform standard SPI
|
||||
instructions with one continuous write and one read for up-to 0xa0
|
||||
bytes. It also supports typical SPI-NAND page cache operations
|
||||
in single, dual or quad IO mode with pipelined ECC encoding/decoding
|
||||
using the accompanying ECC engine. There should be only one spi
|
||||
slave device following generic spi bindings.
|
||||
|
||||
allOf:
|
||||
- $ref: /schemas/spi/spi-controller.yaml#
|
||||
|
||||
properties:
|
||||
compatible:
|
||||
enum:
|
||||
- mediatek,mt7622-snand
|
||||
- mediatek,mt7629-snand
|
||||
|
||||
reg:
|
||||
items:
|
||||
- description: core registers
|
||||
|
||||
interrupts:
|
||||
items:
|
||||
- description: NFI interrupt
|
||||
|
||||
clocks:
|
||||
items:
|
||||
- description: clock used for the controller
|
||||
- description: clock used for the SPI bus
|
||||
|
||||
clock-names:
|
||||
items:
|
||||
- const: nfi_clk
|
||||
- const: pad_clk
|
||||
|
||||
nand-ecc-engine:
|
||||
description: device-tree node of the accompanying ECC engine.
|
||||
$ref: /schemas/types.yaml#/definitions/phandle
|
||||
|
||||
required:
|
||||
- compatible
|
||||
- reg
|
||||
- interrupts
|
||||
- clocks
|
||||
- clock-names
|
||||
- nand-ecc-engine
|
||||
|
||||
unevaluatedProperties: false
|
||||
|
||||
examples:
|
||||
- |
|
||||
#include <dt-bindings/interrupt-controller/irq.h>
|
||||
#include <dt-bindings/interrupt-controller/arm-gic.h>
|
||||
#include <dt-bindings/clock/mt7622-clk.h>
|
||||
soc {
|
||||
#address-cells = <2>;
|
||||
#size-cells = <2>;
|
||||
snfi: spi@1100d000 {
|
||||
compatible = "mediatek,mt7622-snand";
|
||||
reg = <0 0x1100d000 0 0x1000>;
|
||||
interrupts = <GIC_SPI 96 IRQ_TYPE_LEVEL_LOW>;
|
||||
clocks = <&pericfg CLK_PERI_NFI_PD>, <&pericfg CLK_PERI_SNFI_PD>;
|
||||
clock-names = "nfi_clk", "pad_clk";
|
||||
nand-ecc-engine = <&bch>;
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
|
||||
flash@0 {
|
||||
compatible = "spi-nand";
|
||||
reg = <0>;
|
||||
spi-tx-bus-width = <4>;
|
||||
spi-rx-bus-width = <4>;
|
||||
nand-ecc-engine = <&snfi>;
|
||||
};
|
||||
};
|
||||
};
|
@ -49,6 +49,7 @@ properties:
|
||||
maxItems: 2
|
||||
|
||||
interconnect-names:
|
||||
minItems: 1
|
||||
items:
|
||||
- const: qspi-config
|
||||
- const: qspi-memory
|
||||
|
@ -21,6 +21,7 @@ properties:
|
||||
- enum:
|
||||
- renesas,rspi-r7s72100 # RZ/A1H
|
||||
- renesas,rspi-r7s9210 # RZ/A2
|
||||
- renesas,r9a07g043-rspi # RZ/G2UL
|
||||
- renesas,r9a07g044-rspi # RZ/G2{L,LC}
|
||||
- renesas,r9a07g054-rspi # RZ/V2L
|
||||
- const: renesas,rspi-rz
|
||||
@ -124,6 +125,7 @@ allOf:
|
||||
contains:
|
||||
enum:
|
||||
- renesas,qspi
|
||||
- renesas,r9a07g043-rspi
|
||||
- renesas,r9a07g044-rspi
|
||||
- renesas,r9a07g054-rspi
|
||||
then:
|
||||
|
10
MAINTAINERS
10
MAINTAINERS
@ -3102,6 +3102,16 @@ S: Maintained
|
||||
F: Documentation/devicetree/bindings/mmc/aspeed,sdhci.yaml
|
||||
F: drivers/mmc/host/sdhci-of-aspeed*
|
||||
|
||||
ASPEED SMC SPI DRIVER
|
||||
M: Chin-Ting Kuo <chin-ting_kuo@aspeedtech.com>
|
||||
M: Cédric Le Goater <clg@kaod.org>
|
||||
L: linux-aspeed@lists.ozlabs.org (moderated for non-subscribers)
|
||||
L: openbmc@lists.ozlabs.org (moderated for non-subscribers)
|
||||
L: linux-spi@vger.kernel.org
|
||||
S: Maintained
|
||||
F: Documentation/devicetree/bindings/spi/aspeed,ast2600-fmc.yaml
|
||||
F: drivers/spi/spi-aspeed-smc.c
|
||||
|
||||
ASPEED VIDEO ENGINE DRIVER
|
||||
M: Eddie James <eajames@linux.ibm.com>
|
||||
L: linux-media@vger.kernel.org
|
||||
|
@ -1,14 +1,4 @@
|
||||
# SPDX-License-Identifier: GPL-2.0-only
|
||||
config SPI_ASPEED_SMC
|
||||
tristate "Aspeed flash controllers in SPI mode"
|
||||
depends on ARCH_ASPEED || COMPILE_TEST
|
||||
depends on HAS_IOMEM && OF
|
||||
help
|
||||
This enables support for the Firmware Memory controller (FMC)
|
||||
in the Aspeed AST2500/AST2400 SoCs when attached to SPI NOR chips,
|
||||
and support for the SPI flash memory controller (SPI) for
|
||||
the host firmware. The implementation only supports SPI NOR.
|
||||
|
||||
config SPI_HISI_SFC
|
||||
tristate "Hisilicon FMC SPI NOR Flash Controller(SFC)"
|
||||
depends on ARCH_HISI || COMPILE_TEST
|
||||
|
@ -1,4 +1,3 @@
|
||||
# SPDX-License-Identifier: GPL-2.0
|
||||
obj-$(CONFIG_SPI_ASPEED_SMC) += aspeed-smc.o
|
||||
obj-$(CONFIG_SPI_HISI_SFC) += hisi-sfc.o
|
||||
obj-$(CONFIG_SPI_NXP_SPIFI) += nxp-spifi.o
|
||||
|
@ -1,921 +0,0 @@
|
||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
/*
|
||||
* ASPEED Static Memory Controller driver
|
||||
*
|
||||
* Copyright (c) 2015-2016, IBM Corporation.
|
||||
*/
|
||||
|
||||
#include <linux/bug.h>
|
||||
#include <linux/device.h>
|
||||
#include <linux/io.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/mutex.h>
|
||||
#include <linux/mtd/mtd.h>
|
||||
#include <linux/mtd/partitions.h>
|
||||
#include <linux/mtd/spi-nor.h>
|
||||
#include <linux/of.h>
|
||||
#include <linux/of_platform.h>
|
||||
#include <linux/sizes.h>
|
||||
#include <linux/sysfs.h>
|
||||
|
||||
#define DEVICE_NAME "aspeed-smc"
|
||||
|
||||
/*
|
||||
* The driver only support SPI flash
|
||||
*/
|
||||
enum aspeed_smc_flash_type {
|
||||
smc_type_nor = 0,
|
||||
smc_type_nand = 1,
|
||||
smc_type_spi = 2,
|
||||
};
|
||||
|
||||
struct aspeed_smc_chip;
|
||||
|
||||
struct aspeed_smc_info {
|
||||
u32 maxsize; /* maximum size of chip window */
|
||||
u8 nce; /* number of chip enables */
|
||||
bool hastype; /* flash type field exists in config reg */
|
||||
u8 we0; /* shift for write enable bit for CE0 */
|
||||
u8 ctl0; /* offset in regs of ctl for CE0 */
|
||||
|
||||
void (*set_4b)(struct aspeed_smc_chip *chip);
|
||||
};
|
||||
|
||||
static void aspeed_smc_chip_set_4b_spi_2400(struct aspeed_smc_chip *chip);
|
||||
static void aspeed_smc_chip_set_4b(struct aspeed_smc_chip *chip);
|
||||
|
||||
static const struct aspeed_smc_info fmc_2400_info = {
|
||||
.maxsize = 64 * 1024 * 1024,
|
||||
.nce = 5,
|
||||
.hastype = true,
|
||||
.we0 = 16,
|
||||
.ctl0 = 0x10,
|
||||
.set_4b = aspeed_smc_chip_set_4b,
|
||||
};
|
||||
|
||||
static const struct aspeed_smc_info spi_2400_info = {
|
||||
.maxsize = 64 * 1024 * 1024,
|
||||
.nce = 1,
|
||||
.hastype = false,
|
||||
.we0 = 0,
|
||||
.ctl0 = 0x04,
|
||||
.set_4b = aspeed_smc_chip_set_4b_spi_2400,
|
||||
};
|
||||
|
||||
static const struct aspeed_smc_info fmc_2500_info = {
|
||||
.maxsize = 256 * 1024 * 1024,
|
||||
.nce = 3,
|
||||
.hastype = true,
|
||||
.we0 = 16,
|
||||
.ctl0 = 0x10,
|
||||
.set_4b = aspeed_smc_chip_set_4b,
|
||||
};
|
||||
|
||||
static const struct aspeed_smc_info spi_2500_info = {
|
||||
.maxsize = 128 * 1024 * 1024,
|
||||
.nce = 2,
|
||||
.hastype = false,
|
||||
.we0 = 16,
|
||||
.ctl0 = 0x10,
|
||||
.set_4b = aspeed_smc_chip_set_4b,
|
||||
};
|
||||
|
||||
enum aspeed_smc_ctl_reg_value {
|
||||
smc_base, /* base value without mode for other commands */
|
||||
smc_read, /* command reg for (maybe fast) reads */
|
||||
smc_write, /* command reg for writes */
|
||||
smc_max,
|
||||
};
|
||||
|
||||
struct aspeed_smc_controller;
|
||||
|
||||
struct aspeed_smc_chip {
|
||||
int cs;
|
||||
struct aspeed_smc_controller *controller;
|
||||
void __iomem *ctl; /* control register */
|
||||
void __iomem *ahb_base; /* base of chip window */
|
||||
u32 ahb_window_size; /* chip mapping window size */
|
||||
u32 ctl_val[smc_max]; /* control settings */
|
||||
enum aspeed_smc_flash_type type; /* what type of flash */
|
||||
struct spi_nor nor;
|
||||
};
|
||||
|
||||
struct aspeed_smc_controller {
|
||||
struct device *dev;
|
||||
|
||||
struct mutex mutex; /* controller access mutex */
|
||||
const struct aspeed_smc_info *info; /* type info of controller */
|
||||
void __iomem *regs; /* controller registers */
|
||||
void __iomem *ahb_base; /* per-chip windows resource */
|
||||
u32 ahb_window_size; /* full mapping window size */
|
||||
|
||||
struct aspeed_smc_chip *chips[]; /* pointers to attached chips */
|
||||
};
|
||||
|
||||
/*
|
||||
* SPI Flash Configuration Register (AST2500 SPI)
|
||||
* or
|
||||
* Type setting Register (AST2500 FMC).
|
||||
* CE0 and CE1 can only be of type SPI. CE2 can be of type NOR but the
|
||||
* driver does not support it.
|
||||
*/
|
||||
#define CONFIG_REG 0x0
|
||||
#define CONFIG_DISABLE_LEGACY BIT(31) /* 1 */
|
||||
|
||||
#define CONFIG_CE2_WRITE BIT(18)
|
||||
#define CONFIG_CE1_WRITE BIT(17)
|
||||
#define CONFIG_CE0_WRITE BIT(16)
|
||||
|
||||
#define CONFIG_CE2_TYPE BIT(4) /* AST2500 FMC only */
|
||||
#define CONFIG_CE1_TYPE BIT(2) /* AST2500 FMC only */
|
||||
#define CONFIG_CE0_TYPE BIT(0) /* AST2500 FMC only */
|
||||
|
||||
/*
|
||||
* CE Control Register
|
||||
*/
|
||||
#define CE_CONTROL_REG 0x4
|
||||
|
||||
/*
|
||||
* CEx Control Register
|
||||
*/
|
||||
#define CONTROL_AAF_MODE BIT(31)
|
||||
#define CONTROL_IO_MODE_MASK GENMASK(30, 28)
|
||||
#define CONTROL_IO_DUAL_DATA BIT(29)
|
||||
#define CONTROL_IO_DUAL_ADDR_DATA (BIT(29) | BIT(28))
|
||||
#define CONTROL_IO_QUAD_DATA BIT(30)
|
||||
#define CONTROL_IO_QUAD_ADDR_DATA (BIT(30) | BIT(28))
|
||||
#define CONTROL_CE_INACTIVE_SHIFT 24
|
||||
#define CONTROL_CE_INACTIVE_MASK GENMASK(27, \
|
||||
CONTROL_CE_INACTIVE_SHIFT)
|
||||
/* 0 = 16T ... 15 = 1T T=HCLK */
|
||||
#define CONTROL_COMMAND_SHIFT 16
|
||||
#define CONTROL_DUMMY_COMMAND_OUT BIT(15)
|
||||
#define CONTROL_IO_DUMMY_HI BIT(14)
|
||||
#define CONTROL_IO_DUMMY_HI_SHIFT 14
|
||||
#define CONTROL_CLK_DIV4 BIT(13) /* others */
|
||||
#define CONTROL_IO_ADDRESS_4B BIT(13) /* AST2400 SPI */
|
||||
#define CONTROL_RW_MERGE BIT(12)
|
||||
#define CONTROL_IO_DUMMY_LO_SHIFT 6
|
||||
#define CONTROL_IO_DUMMY_LO GENMASK(7, \
|
||||
CONTROL_IO_DUMMY_LO_SHIFT)
|
||||
#define CONTROL_IO_DUMMY_MASK (CONTROL_IO_DUMMY_HI | \
|
||||
CONTROL_IO_DUMMY_LO)
|
||||
#define CONTROL_IO_DUMMY_SET(dummy) \
|
||||
(((((dummy) >> 2) & 0x1) << CONTROL_IO_DUMMY_HI_SHIFT) | \
|
||||
(((dummy) & 0x3) << CONTROL_IO_DUMMY_LO_SHIFT))
|
||||
|
||||
#define CONTROL_CLOCK_FREQ_SEL_SHIFT 8
|
||||
#define CONTROL_CLOCK_FREQ_SEL_MASK GENMASK(11, \
|
||||
CONTROL_CLOCK_FREQ_SEL_SHIFT)
|
||||
#define CONTROL_LSB_FIRST BIT(5)
|
||||
#define CONTROL_CLOCK_MODE_3 BIT(4)
|
||||
#define CONTROL_IN_DUAL_DATA BIT(3)
|
||||
#define CONTROL_CE_STOP_ACTIVE_CONTROL BIT(2)
|
||||
#define CONTROL_COMMAND_MODE_MASK GENMASK(1, 0)
|
||||
#define CONTROL_COMMAND_MODE_NORMAL 0
|
||||
#define CONTROL_COMMAND_MODE_FREAD 1
|
||||
#define CONTROL_COMMAND_MODE_WRITE 2
|
||||
#define CONTROL_COMMAND_MODE_USER 3
|
||||
|
||||
#define CONTROL_KEEP_MASK \
|
||||
(CONTROL_AAF_MODE | CONTROL_CE_INACTIVE_MASK | CONTROL_CLK_DIV4 | \
|
||||
CONTROL_CLOCK_FREQ_SEL_MASK | CONTROL_LSB_FIRST | CONTROL_CLOCK_MODE_3)
|
||||
|
||||
/*
|
||||
* The Segment Register uses a 8MB unit to encode the start address
|
||||
* and the end address of the mapping window of a flash SPI slave :
|
||||
*
|
||||
* | byte 1 | byte 2 | byte 3 | byte 4 |
|
||||
* +--------+--------+--------+--------+
|
||||
* | end | start | 0 | 0 |
|
||||
*/
|
||||
#define SEGMENT_ADDR_REG0 0x30
|
||||
#define SEGMENT_ADDR_START(_r) ((((_r) >> 16) & 0xFF) << 23)
|
||||
#define SEGMENT_ADDR_END(_r) ((((_r) >> 24) & 0xFF) << 23)
|
||||
#define SEGMENT_ADDR_VALUE(start, end) \
|
||||
(((((start) >> 23) & 0xFF) << 16) | ((((end) >> 23) & 0xFF) << 24))
|
||||
#define SEGMENT_ADDR_REG(controller, cs) \
|
||||
((controller)->regs + SEGMENT_ADDR_REG0 + (cs) * 4)
|
||||
|
||||
/*
|
||||
* In user mode all data bytes read or written to the chip decode address
|
||||
* range are transferred to or from the SPI bus. The range is treated as a
|
||||
* fifo of arbitratry 1, 2, or 4 byte width but each write has to be aligned
|
||||
* to its size. The address within the multiple 8kB range is ignored when
|
||||
* sending bytes to the SPI bus.
|
||||
*
|
||||
* On the arm architecture, as of Linux version 4.3, memcpy_fromio and
|
||||
* memcpy_toio on little endian targets use the optimized memcpy routines
|
||||
* that were designed for well behavied memory storage. These routines
|
||||
* have a stutter if the source and destination are not both word aligned,
|
||||
* once with a duplicate access to the source after aligning to the
|
||||
* destination to a word boundary, and again with a duplicate access to
|
||||
* the source when the final byte count is not word aligned.
|
||||
*
|
||||
* When writing or reading the fifo this stutter discards data or sends
|
||||
* too much data to the fifo and can not be used by this driver.
|
||||
*
|
||||
* While the low level io string routines that implement the insl family do
|
||||
* the desired accesses and memory increments, the cross architecture io
|
||||
* macros make them essentially impossible to use on a memory mapped address
|
||||
* instead of a a token from the call to iomap of an io port.
|
||||
*
|
||||
* These fifo routines use readl and friends to a constant io port and update
|
||||
* the memory buffer pointer and count via explicit code. The final updates
|
||||
* to len are optimistically suppressed.
|
||||
*/
|
||||
static int aspeed_smc_read_from_ahb(void *buf, void __iomem *src, size_t len)
|
||||
{
|
||||
size_t offset = 0;
|
||||
|
||||
if (IS_ALIGNED((uintptr_t)src, sizeof(uintptr_t)) &&
|
||||
IS_ALIGNED((uintptr_t)buf, sizeof(uintptr_t))) {
|
||||
ioread32_rep(src, buf, len >> 2);
|
||||
offset = len & ~0x3;
|
||||
len -= offset;
|
||||
}
|
||||
ioread8_rep(src, (u8 *)buf + offset, len);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int aspeed_smc_write_to_ahb(void __iomem *dst, const void *buf,
|
||||
size_t len)
|
||||
{
|
||||
size_t offset = 0;
|
||||
|
||||
if (IS_ALIGNED((uintptr_t)dst, sizeof(uintptr_t)) &&
|
||||
IS_ALIGNED((uintptr_t)buf, sizeof(uintptr_t))) {
|
||||
iowrite32_rep(dst, buf, len >> 2);
|
||||
offset = len & ~0x3;
|
||||
len -= offset;
|
||||
}
|
||||
iowrite8_rep(dst, (const u8 *)buf + offset, len);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static inline u32 aspeed_smc_chip_write_bit(struct aspeed_smc_chip *chip)
|
||||
{
|
||||
return BIT(chip->controller->info->we0 + chip->cs);
|
||||
}
|
||||
|
||||
static void aspeed_smc_chip_check_config(struct aspeed_smc_chip *chip)
|
||||
{
|
||||
struct aspeed_smc_controller *controller = chip->controller;
|
||||
u32 reg;
|
||||
|
||||
reg = readl(controller->regs + CONFIG_REG);
|
||||
|
||||
if (reg & aspeed_smc_chip_write_bit(chip))
|
||||
return;
|
||||
|
||||
dev_dbg(controller->dev, "config write is not set ! @%p: 0x%08x\n",
|
||||
controller->regs + CONFIG_REG, reg);
|
||||
reg |= aspeed_smc_chip_write_bit(chip);
|
||||
writel(reg, controller->regs + CONFIG_REG);
|
||||
}
|
||||
|
||||
static void aspeed_smc_start_user(struct spi_nor *nor)
|
||||
{
|
||||
struct aspeed_smc_chip *chip = nor->priv;
|
||||
u32 ctl = chip->ctl_val[smc_base];
|
||||
|
||||
/*
|
||||
* When the chip is controlled in user mode, we need write
|
||||
* access to send the opcodes to it. So check the config.
|
||||
*/
|
||||
aspeed_smc_chip_check_config(chip);
|
||||
|
||||
ctl |= CONTROL_COMMAND_MODE_USER |
|
||||
CONTROL_CE_STOP_ACTIVE_CONTROL;
|
||||
writel(ctl, chip->ctl);
|
||||
|
||||
ctl &= ~CONTROL_CE_STOP_ACTIVE_CONTROL;
|
||||
writel(ctl, chip->ctl);
|
||||
}
|
||||
|
||||
static void aspeed_smc_stop_user(struct spi_nor *nor)
|
||||
{
|
||||
struct aspeed_smc_chip *chip = nor->priv;
|
||||
|
||||
u32 ctl = chip->ctl_val[smc_read];
|
||||
u32 ctl2 = ctl | CONTROL_COMMAND_MODE_USER |
|
||||
CONTROL_CE_STOP_ACTIVE_CONTROL;
|
||||
|
||||
writel(ctl2, chip->ctl); /* stop user CE control */
|
||||
writel(ctl, chip->ctl); /* default to fread or read mode */
|
||||
}
|
||||
|
||||
static int aspeed_smc_prep(struct spi_nor *nor)
|
||||
{
|
||||
struct aspeed_smc_chip *chip = nor->priv;
|
||||
|
||||
mutex_lock(&chip->controller->mutex);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void aspeed_smc_unprep(struct spi_nor *nor)
|
||||
{
|
||||
struct aspeed_smc_chip *chip = nor->priv;
|
||||
|
||||
mutex_unlock(&chip->controller->mutex);
|
||||
}
|
||||
|
||||
static int aspeed_smc_read_reg(struct spi_nor *nor, u8 opcode, u8 *buf,
|
||||
size_t len)
|
||||
{
|
||||
struct aspeed_smc_chip *chip = nor->priv;
|
||||
|
||||
aspeed_smc_start_user(nor);
|
||||
aspeed_smc_write_to_ahb(chip->ahb_base, &opcode, 1);
|
||||
aspeed_smc_read_from_ahb(buf, chip->ahb_base, len);
|
||||
aspeed_smc_stop_user(nor);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int aspeed_smc_write_reg(struct spi_nor *nor, u8 opcode, const u8 *buf,
|
||||
size_t len)
|
||||
{
|
||||
struct aspeed_smc_chip *chip = nor->priv;
|
||||
|
||||
aspeed_smc_start_user(nor);
|
||||
aspeed_smc_write_to_ahb(chip->ahb_base, &opcode, 1);
|
||||
aspeed_smc_write_to_ahb(chip->ahb_base, buf, len);
|
||||
aspeed_smc_stop_user(nor);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void aspeed_smc_send_cmd_addr(struct spi_nor *nor, u8 cmd, u32 addr)
|
||||
{
|
||||
struct aspeed_smc_chip *chip = nor->priv;
|
||||
__be32 temp;
|
||||
u32 cmdaddr;
|
||||
|
||||
switch (nor->addr_width) {
|
||||
default:
|
||||
WARN_ONCE(1, "Unexpected address width %u, defaulting to 3\n",
|
||||
nor->addr_width);
|
||||
fallthrough;
|
||||
case 3:
|
||||
cmdaddr = addr & 0xFFFFFF;
|
||||
cmdaddr |= cmd << 24;
|
||||
|
||||
temp = cpu_to_be32(cmdaddr);
|
||||
aspeed_smc_write_to_ahb(chip->ahb_base, &temp, 4);
|
||||
break;
|
||||
case 4:
|
||||
temp = cpu_to_be32(addr);
|
||||
aspeed_smc_write_to_ahb(chip->ahb_base, &cmd, 1);
|
||||
aspeed_smc_write_to_ahb(chip->ahb_base, &temp, 4);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
static ssize_t aspeed_smc_read_user(struct spi_nor *nor, loff_t from,
|
||||
size_t len, u_char *read_buf)
|
||||
{
|
||||
struct aspeed_smc_chip *chip = nor->priv;
|
||||
int i;
|
||||
u8 dummy = 0xFF;
|
||||
|
||||
aspeed_smc_start_user(nor);
|
||||
aspeed_smc_send_cmd_addr(nor, nor->read_opcode, from);
|
||||
for (i = 0; i < chip->nor.read_dummy / 8; i++)
|
||||
aspeed_smc_write_to_ahb(chip->ahb_base, &dummy, sizeof(dummy));
|
||||
|
||||
aspeed_smc_read_from_ahb(read_buf, chip->ahb_base, len);
|
||||
aspeed_smc_stop_user(nor);
|
||||
return len;
|
||||
}
|
||||
|
||||
static ssize_t aspeed_smc_write_user(struct spi_nor *nor, loff_t to,
|
||||
size_t len, const u_char *write_buf)
|
||||
{
|
||||
struct aspeed_smc_chip *chip = nor->priv;
|
||||
|
||||
aspeed_smc_start_user(nor);
|
||||
aspeed_smc_send_cmd_addr(nor, nor->program_opcode, to);
|
||||
aspeed_smc_write_to_ahb(chip->ahb_base, write_buf, len);
|
||||
aspeed_smc_stop_user(nor);
|
||||
return len;
|
||||
}
|
||||
|
||||
static int aspeed_smc_unregister(struct aspeed_smc_controller *controller)
|
||||
{
|
||||
struct aspeed_smc_chip *chip;
|
||||
int n;
|
||||
|
||||
for (n = 0; n < controller->info->nce; n++) {
|
||||
chip = controller->chips[n];
|
||||
if (chip)
|
||||
mtd_device_unregister(&chip->nor.mtd);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int aspeed_smc_remove(struct platform_device *dev)
|
||||
{
|
||||
return aspeed_smc_unregister(platform_get_drvdata(dev));
|
||||
}
|
||||
|
||||
static const struct of_device_id aspeed_smc_matches[] = {
|
||||
{ .compatible = "aspeed,ast2400-fmc", .data = &fmc_2400_info },
|
||||
{ .compatible = "aspeed,ast2400-spi", .data = &spi_2400_info },
|
||||
{ .compatible = "aspeed,ast2500-fmc", .data = &fmc_2500_info },
|
||||
{ .compatible = "aspeed,ast2500-spi", .data = &spi_2500_info },
|
||||
{ }
|
||||
};
|
||||
MODULE_DEVICE_TABLE(of, aspeed_smc_matches);
|
||||
|
||||
/*
|
||||
* Each chip has a mapping window defined by a segment address
|
||||
* register defining a start and an end address on the AHB bus. These
|
||||
* addresses can be configured to fit the chip size and offer a
|
||||
* contiguous memory region across chips. For the moment, we only
|
||||
* check that each chip segment is valid.
|
||||
*/
|
||||
static void __iomem *aspeed_smc_chip_base(struct aspeed_smc_chip *chip,
|
||||
struct resource *res)
|
||||
{
|
||||
struct aspeed_smc_controller *controller = chip->controller;
|
||||
u32 offset = 0;
|
||||
u32 reg;
|
||||
|
||||
if (controller->info->nce > 1) {
|
||||
reg = readl(SEGMENT_ADDR_REG(controller, chip->cs));
|
||||
|
||||
if (SEGMENT_ADDR_START(reg) >= SEGMENT_ADDR_END(reg))
|
||||
return NULL;
|
||||
|
||||
offset = SEGMENT_ADDR_START(reg) - res->start;
|
||||
}
|
||||
|
||||
return controller->ahb_base + offset;
|
||||
}
|
||||
|
||||
static u32 aspeed_smc_ahb_base_phy(struct aspeed_smc_controller *controller)
|
||||
{
|
||||
u32 seg0_val = readl(SEGMENT_ADDR_REG(controller, 0));
|
||||
|
||||
return SEGMENT_ADDR_START(seg0_val);
|
||||
}
|
||||
|
||||
static u32 chip_set_segment(struct aspeed_smc_chip *chip, u32 cs, u32 start,
|
||||
u32 size)
|
||||
{
|
||||
struct aspeed_smc_controller *controller = chip->controller;
|
||||
void __iomem *seg_reg;
|
||||
u32 seg_oldval, seg_newval, ahb_base_phy, end;
|
||||
|
||||
ahb_base_phy = aspeed_smc_ahb_base_phy(controller);
|
||||
|
||||
seg_reg = SEGMENT_ADDR_REG(controller, cs);
|
||||
seg_oldval = readl(seg_reg);
|
||||
|
||||
/*
|
||||
* If the chip size is not specified, use the default segment
|
||||
* size, but take into account the possible overlap with the
|
||||
* previous segment
|
||||
*/
|
||||
if (!size)
|
||||
size = SEGMENT_ADDR_END(seg_oldval) - start;
|
||||
|
||||
/*
|
||||
* The segment cannot exceed the maximum window size of the
|
||||
* controller.
|
||||
*/
|
||||
if (start + size > ahb_base_phy + controller->ahb_window_size) {
|
||||
size = ahb_base_phy + controller->ahb_window_size - start;
|
||||
dev_warn(chip->nor.dev, "CE%d window resized to %dMB",
|
||||
cs, size >> 20);
|
||||
}
|
||||
|
||||
end = start + size;
|
||||
seg_newval = SEGMENT_ADDR_VALUE(start, end);
|
||||
writel(seg_newval, seg_reg);
|
||||
|
||||
/*
|
||||
* Restore default value if something goes wrong. The chip
|
||||
* might have set some bogus value and we would loose access
|
||||
* to the chip.
|
||||
*/
|
||||
if (seg_newval != readl(seg_reg)) {
|
||||
dev_err(chip->nor.dev, "CE%d window invalid", cs);
|
||||
writel(seg_oldval, seg_reg);
|
||||
start = SEGMENT_ADDR_START(seg_oldval);
|
||||
end = SEGMENT_ADDR_END(seg_oldval);
|
||||
size = end - start;
|
||||
}
|
||||
|
||||
dev_info(chip->nor.dev, "CE%d window [ 0x%.8x - 0x%.8x ] %dMB",
|
||||
cs, start, end, size >> 20);
|
||||
|
||||
return size;
|
||||
}
|
||||
|
||||
/*
|
||||
* The segment register defines the mapping window on the AHB bus and
|
||||
* it needs to be configured depending on the chip size. The segment
|
||||
* register of the following CE also needs to be tuned in order to
|
||||
* provide a contiguous window across multiple chips.
|
||||
*
|
||||
* This is expected to be called in increasing CE order
|
||||
*/
|
||||
static u32 aspeed_smc_chip_set_segment(struct aspeed_smc_chip *chip)
|
||||
{
|
||||
struct aspeed_smc_controller *controller = chip->controller;
|
||||
u32 ahb_base_phy, start;
|
||||
u32 size = chip->nor.mtd.size;
|
||||
|
||||
/*
|
||||
* Each controller has a chip size limit for direct memory
|
||||
* access
|
||||
*/
|
||||
if (size > controller->info->maxsize)
|
||||
size = controller->info->maxsize;
|
||||
|
||||
/*
|
||||
* The AST2400 SPI controller only handles one chip and does
|
||||
* not have segment registers. Let's use the chip size for the
|
||||
* AHB window.
|
||||
*/
|
||||
if (controller->info == &spi_2400_info)
|
||||
goto out;
|
||||
|
||||
/*
|
||||
* The AST2500 SPI controller has a HW bug when the CE0 chip
|
||||
* size reaches 128MB. Enforce a size limit of 120MB to
|
||||
* prevent the controller from using bogus settings in the
|
||||
* segment register.
|
||||
*/
|
||||
if (chip->cs == 0 && controller->info == &spi_2500_info &&
|
||||
size == SZ_128M) {
|
||||
size = 120 << 20;
|
||||
dev_info(chip->nor.dev,
|
||||
"CE%d window resized to %dMB (AST2500 HW quirk)",
|
||||
chip->cs, size >> 20);
|
||||
}
|
||||
|
||||
ahb_base_phy = aspeed_smc_ahb_base_phy(controller);
|
||||
|
||||
/*
|
||||
* As a start address for the current segment, use the default
|
||||
* start address if we are handling CE0 or use the previous
|
||||
* segment ending address
|
||||
*/
|
||||
if (chip->cs) {
|
||||
u32 prev = readl(SEGMENT_ADDR_REG(controller, chip->cs - 1));
|
||||
|
||||
start = SEGMENT_ADDR_END(prev);
|
||||
} else {
|
||||
start = ahb_base_phy;
|
||||
}
|
||||
|
||||
size = chip_set_segment(chip, chip->cs, start, size);
|
||||
|
||||
/* Update chip base address on the AHB bus */
|
||||
chip->ahb_base = controller->ahb_base + (start - ahb_base_phy);
|
||||
|
||||
/*
|
||||
* Now, make sure the next segment does not overlap with the
|
||||
* current one we just configured, even if there is no
|
||||
* available chip. That could break access in Command Mode.
|
||||
*/
|
||||
if (chip->cs < controller->info->nce - 1)
|
||||
chip_set_segment(chip, chip->cs + 1, start + size, 0);
|
||||
|
||||
out:
|
||||
if (size < chip->nor.mtd.size)
|
||||
dev_warn(chip->nor.dev,
|
||||
"CE%d window too small for chip %dMB",
|
||||
chip->cs, (u32)chip->nor.mtd.size >> 20);
|
||||
|
||||
return size;
|
||||
}
|
||||
|
||||
static void aspeed_smc_chip_enable_write(struct aspeed_smc_chip *chip)
|
||||
{
|
||||
struct aspeed_smc_controller *controller = chip->controller;
|
||||
u32 reg;
|
||||
|
||||
reg = readl(controller->regs + CONFIG_REG);
|
||||
|
||||
reg |= aspeed_smc_chip_write_bit(chip);
|
||||
writel(reg, controller->regs + CONFIG_REG);
|
||||
}
|
||||
|
||||
static void aspeed_smc_chip_set_type(struct aspeed_smc_chip *chip, int type)
|
||||
{
|
||||
struct aspeed_smc_controller *controller = chip->controller;
|
||||
u32 reg;
|
||||
|
||||
chip->type = type;
|
||||
|
||||
reg = readl(controller->regs + CONFIG_REG);
|
||||
reg &= ~(3 << (chip->cs * 2));
|
||||
reg |= chip->type << (chip->cs * 2);
|
||||
writel(reg, controller->regs + CONFIG_REG);
|
||||
}
|
||||
|
||||
/*
|
||||
* The first chip of the AST2500 FMC flash controller is strapped by
|
||||
* hardware, or autodetected, but other chips need to be set. Enforce
|
||||
* the 4B setting for all chips.
|
||||
*/
|
||||
static void aspeed_smc_chip_set_4b(struct aspeed_smc_chip *chip)
|
||||
{
|
||||
struct aspeed_smc_controller *controller = chip->controller;
|
||||
u32 reg;
|
||||
|
||||
reg = readl(controller->regs + CE_CONTROL_REG);
|
||||
reg |= 1 << chip->cs;
|
||||
writel(reg, controller->regs + CE_CONTROL_REG);
|
||||
}
|
||||
|
||||
/*
|
||||
* The AST2400 SPI flash controller does not have a CE Control
|
||||
* register. It uses the CE0 control register to set 4Byte mode at the
|
||||
* controller level.
|
||||
*/
|
||||
static void aspeed_smc_chip_set_4b_spi_2400(struct aspeed_smc_chip *chip)
|
||||
{
|
||||
chip->ctl_val[smc_base] |= CONTROL_IO_ADDRESS_4B;
|
||||
chip->ctl_val[smc_read] |= CONTROL_IO_ADDRESS_4B;
|
||||
}
|
||||
|
||||
static int aspeed_smc_chip_setup_init(struct aspeed_smc_chip *chip,
|
||||
struct resource *res)
|
||||
{
|
||||
struct aspeed_smc_controller *controller = chip->controller;
|
||||
const struct aspeed_smc_info *info = controller->info;
|
||||
u32 reg, base_reg;
|
||||
|
||||
/*
|
||||
* Always turn on the write enable bit to allow opcodes to be
|
||||
* sent in user mode.
|
||||
*/
|
||||
aspeed_smc_chip_enable_write(chip);
|
||||
|
||||
/* The driver only supports SPI type flash */
|
||||
if (info->hastype)
|
||||
aspeed_smc_chip_set_type(chip, smc_type_spi);
|
||||
|
||||
/*
|
||||
* Configure chip base address in memory
|
||||
*/
|
||||
chip->ahb_base = aspeed_smc_chip_base(chip, res);
|
||||
if (!chip->ahb_base) {
|
||||
dev_warn(chip->nor.dev, "CE%d window closed", chip->cs);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
/*
|
||||
* Get value of the inherited control register. U-Boot usually
|
||||
* does some timing calibration on the FMC chip, so it's good
|
||||
* to keep them. In the future, we should handle calibration
|
||||
* from Linux.
|
||||
*/
|
||||
reg = readl(chip->ctl);
|
||||
dev_dbg(controller->dev, "control register: %08x\n", reg);
|
||||
|
||||
base_reg = reg & CONTROL_KEEP_MASK;
|
||||
if (base_reg != reg) {
|
||||
dev_dbg(controller->dev,
|
||||
"control register changed to: %08x\n",
|
||||
base_reg);
|
||||
}
|
||||
chip->ctl_val[smc_base] = base_reg;
|
||||
|
||||
/*
|
||||
* Retain the prior value of the control register as the
|
||||
* default if it was normal access mode. Otherwise start with
|
||||
* the sanitized base value set to read mode.
|
||||
*/
|
||||
if ((reg & CONTROL_COMMAND_MODE_MASK) ==
|
||||
CONTROL_COMMAND_MODE_NORMAL)
|
||||
chip->ctl_val[smc_read] = reg;
|
||||
else
|
||||
chip->ctl_val[smc_read] = chip->ctl_val[smc_base] |
|
||||
CONTROL_COMMAND_MODE_NORMAL;
|
||||
|
||||
dev_dbg(controller->dev, "default control register: %08x\n",
|
||||
chip->ctl_val[smc_read]);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int aspeed_smc_chip_setup_finish(struct aspeed_smc_chip *chip)
|
||||
{
|
||||
struct aspeed_smc_controller *controller = chip->controller;
|
||||
const struct aspeed_smc_info *info = controller->info;
|
||||
u32 cmd;
|
||||
|
||||
if (chip->nor.addr_width == 4 && info->set_4b)
|
||||
info->set_4b(chip);
|
||||
|
||||
/* This is for direct AHB access when using Command Mode. */
|
||||
chip->ahb_window_size = aspeed_smc_chip_set_segment(chip);
|
||||
|
||||
/*
|
||||
* base mode has not been optimized yet. use it for writes.
|
||||
*/
|
||||
chip->ctl_val[smc_write] = chip->ctl_val[smc_base] |
|
||||
chip->nor.program_opcode << CONTROL_COMMAND_SHIFT |
|
||||
CONTROL_COMMAND_MODE_WRITE;
|
||||
|
||||
dev_dbg(controller->dev, "write control register: %08x\n",
|
||||
chip->ctl_val[smc_write]);
|
||||
|
||||
/*
|
||||
* TODO: Adjust clocks if fast read is supported and interpret
|
||||
* SPI NOR flags to adjust controller settings.
|
||||
*/
|
||||
if (chip->nor.read_proto == SNOR_PROTO_1_1_1) {
|
||||
if (chip->nor.read_dummy == 0)
|
||||
cmd = CONTROL_COMMAND_MODE_NORMAL;
|
||||
else
|
||||
cmd = CONTROL_COMMAND_MODE_FREAD;
|
||||
} else {
|
||||
dev_err(chip->nor.dev, "unsupported SPI read mode\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
chip->ctl_val[smc_read] |= cmd |
|
||||
CONTROL_IO_DUMMY_SET(chip->nor.read_dummy / 8);
|
||||
|
||||
dev_dbg(controller->dev, "base control register: %08x\n",
|
||||
chip->ctl_val[smc_read]);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const struct spi_nor_controller_ops aspeed_smc_controller_ops = {
|
||||
.prepare = aspeed_smc_prep,
|
||||
.unprepare = aspeed_smc_unprep,
|
||||
.read_reg = aspeed_smc_read_reg,
|
||||
.write_reg = aspeed_smc_write_reg,
|
||||
.read = aspeed_smc_read_user,
|
||||
.write = aspeed_smc_write_user,
|
||||
};
|
||||
|
||||
static int aspeed_smc_setup_flash(struct aspeed_smc_controller *controller,
|
||||
struct device_node *np, struct resource *r)
|
||||
{
|
||||
const struct spi_nor_hwcaps hwcaps = {
|
||||
.mask = SNOR_HWCAPS_READ |
|
||||
SNOR_HWCAPS_READ_FAST |
|
||||
SNOR_HWCAPS_PP,
|
||||
};
|
||||
const struct aspeed_smc_info *info = controller->info;
|
||||
struct device *dev = controller->dev;
|
||||
struct device_node *child;
|
||||
unsigned int cs;
|
||||
int ret = -ENODEV;
|
||||
bool found_one = false;
|
||||
|
||||
for_each_available_child_of_node(np, child) {
|
||||
struct aspeed_smc_chip *chip;
|
||||
struct spi_nor *nor;
|
||||
struct mtd_info *mtd;
|
||||
|
||||
/* This driver does not support NAND or NOR flash devices. */
|
||||
if (!of_device_is_compatible(child, "jedec,spi-nor"))
|
||||
continue;
|
||||
|
||||
ret = of_property_read_u32(child, "reg", &cs);
|
||||
if (ret) {
|
||||
dev_err(dev, "Couldn't not read chip select.\n");
|
||||
break;
|
||||
}
|
||||
|
||||
if (cs >= info->nce) {
|
||||
dev_err(dev, "Chip select %d out of range.\n",
|
||||
cs);
|
||||
ret = -ERANGE;
|
||||
break;
|
||||
}
|
||||
|
||||
if (controller->chips[cs]) {
|
||||
dev_err(dev, "Chip select %d already in use by %s\n",
|
||||
cs, dev_name(controller->chips[cs]->nor.dev));
|
||||
ret = -EBUSY;
|
||||
break;
|
||||
}
|
||||
|
||||
chip = devm_kzalloc(controller->dev, sizeof(*chip), GFP_KERNEL);
|
||||
if (!chip) {
|
||||
ret = -ENOMEM;
|
||||
break;
|
||||
}
|
||||
|
||||
chip->controller = controller;
|
||||
chip->ctl = controller->regs + info->ctl0 + cs * 4;
|
||||
chip->cs = cs;
|
||||
|
||||
nor = &chip->nor;
|
||||
mtd = &nor->mtd;
|
||||
|
||||
nor->dev = dev;
|
||||
nor->priv = chip;
|
||||
spi_nor_set_flash_node(nor, child);
|
||||
nor->controller_ops = &aspeed_smc_controller_ops;
|
||||
|
||||
ret = aspeed_smc_chip_setup_init(chip, r);
|
||||
if (ret)
|
||||
break;
|
||||
|
||||
/*
|
||||
* TODO: Add support for Dual and Quad SPI protocols
|
||||
* attach when board support is present as determined
|
||||
* by of property.
|
||||
*/
|
||||
ret = spi_nor_scan(nor, NULL, &hwcaps);
|
||||
/*
|
||||
* If we fail to scan the device it might not be present or
|
||||
* broken. Don't fail the whole controller if others work.
|
||||
*/
|
||||
if (ret) {
|
||||
if (found_one)
|
||||
ret = 0;
|
||||
|
||||
devm_kfree(controller->dev, chip);
|
||||
continue;
|
||||
}
|
||||
|
||||
ret = aspeed_smc_chip_setup_finish(chip);
|
||||
if (ret)
|
||||
break;
|
||||
|
||||
ret = mtd_device_register(mtd, NULL, 0);
|
||||
if (ret)
|
||||
break;
|
||||
|
||||
controller->chips[cs] = chip;
|
||||
found_one = true;
|
||||
}
|
||||
|
||||
if (ret) {
|
||||
of_node_put(child);
|
||||
aspeed_smc_unregister(controller);
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int aspeed_smc_probe(struct platform_device *pdev)
|
||||
{
|
||||
struct device_node *np = pdev->dev.of_node;
|
||||
struct device *dev = &pdev->dev;
|
||||
struct aspeed_smc_controller *controller;
|
||||
const struct of_device_id *match;
|
||||
const struct aspeed_smc_info *info;
|
||||
struct resource *res;
|
||||
int ret;
|
||||
|
||||
match = of_match_device(aspeed_smc_matches, &pdev->dev);
|
||||
if (!match || !match->data)
|
||||
return -ENODEV;
|
||||
info = match->data;
|
||||
|
||||
controller = devm_kzalloc(&pdev->dev,
|
||||
struct_size(controller, chips, info->nce),
|
||||
GFP_KERNEL);
|
||||
if (!controller)
|
||||
return -ENOMEM;
|
||||
controller->info = info;
|
||||
controller->dev = dev;
|
||||
|
||||
mutex_init(&controller->mutex);
|
||||
platform_set_drvdata(pdev, controller);
|
||||
|
||||
res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
||||
controller->regs = devm_ioremap_resource(dev, res);
|
||||
if (IS_ERR(controller->regs))
|
||||
return PTR_ERR(controller->regs);
|
||||
|
||||
res = platform_get_resource(pdev, IORESOURCE_MEM, 1);
|
||||
controller->ahb_base = devm_ioremap_resource(dev, res);
|
||||
if (IS_ERR(controller->ahb_base))
|
||||
return PTR_ERR(controller->ahb_base);
|
||||
|
||||
controller->ahb_window_size = resource_size(res);
|
||||
|
||||
ret = aspeed_smc_setup_flash(controller, np, res);
|
||||
if (ret)
|
||||
dev_err(dev, "Aspeed SMC probe failed %d\n", ret);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static struct platform_driver aspeed_smc_driver = {
|
||||
.probe = aspeed_smc_probe,
|
||||
.remove = aspeed_smc_remove,
|
||||
.driver = {
|
||||
.name = DEVICE_NAME,
|
||||
.of_match_table = aspeed_smc_matches,
|
||||
}
|
||||
};
|
||||
|
||||
module_platform_driver(aspeed_smc_driver);
|
||||
|
||||
MODULE_DESCRIPTION("ASPEED Static Memory Controller Driver");
|
||||
MODULE_AUTHOR("Cedric Le Goater <clg@kaod.org>");
|
||||
MODULE_LICENSE("GPL v2");
|
@ -101,6 +101,17 @@ config SPI_ARMADA_3700
|
||||
This enables support for the SPI controller present on the
|
||||
Marvell Armada 3700 SoCs.
|
||||
|
||||
config SPI_ASPEED_SMC
|
||||
tristate "Aspeed flash controllers in SPI mode"
|
||||
depends on ARCH_ASPEED || COMPILE_TEST
|
||||
depends on OF
|
||||
help
|
||||
This enables support for the Firmware Memory controller (FMC)
|
||||
in the Aspeed AST2600, AST2500 and AST2400 SoCs when attached
|
||||
to SPI NOR chips, and support for the SPI flash memory
|
||||
controller (SPI) for the host firmware. The implementation
|
||||
only supports SPI NOR.
|
||||
|
||||
config SPI_ATMEL
|
||||
tristate "Atmel SPI Controller"
|
||||
depends on ARCH_AT91 || COMPILE_TEST
|
||||
@ -414,15 +425,14 @@ config SPI_IMG_SPFI
|
||||
config SPI_IMX
|
||||
tristate "Freescale i.MX SPI controllers"
|
||||
depends on ARCH_MXC || COMPILE_TEST
|
||||
select SPI_BITBANG
|
||||
help
|
||||
This enables support for the Freescale i.MX SPI controllers.
|
||||
|
||||
config SPI_INGENIC
|
||||
tristate "Ingenic JZ47xx SoCs SPI controller"
|
||||
tristate "Ingenic SoCs SPI controller"
|
||||
depends on MACH_INGENIC || COMPILE_TEST
|
||||
help
|
||||
This enables support for the Ingenic JZ47xx SoCs SPI controller.
|
||||
This enables support for the Ingenic SoCs SPI controller.
|
||||
|
||||
To compile this driver as a module, choose M here: the module
|
||||
will be called spi-ingenic.
|
||||
@ -590,6 +600,16 @@ config SPI_MTK_NOR
|
||||
SPI interface as well as several SPI NOR specific instructions
|
||||
via SPI MEM interface.
|
||||
|
||||
config SPI_MTK_SNFI
|
||||
tristate "MediaTek SPI NAND Flash Interface"
|
||||
depends on ARCH_MEDIATEK || COMPILE_TEST
|
||||
depends on MTD_NAND_ECC_MEDIATEK
|
||||
help
|
||||
This enables support for SPI-NAND mode on the MediaTek NAND
|
||||
Flash Interface found on MediaTek ARM SoCs. This controller
|
||||
is implemented as a SPI-MEM controller with pipelined ECC
|
||||
capcability.
|
||||
|
||||
config SPI_NPCM_FIU
|
||||
tristate "Nuvoton NPCM FLASH Interface Unit"
|
||||
depends on ARCH_NPCM || COMPILE_TEST
|
||||
|
@ -19,6 +19,7 @@ obj-$(CONFIG_SPI_ALTERA_CORE) += spi-altera-core.o
|
||||
obj-$(CONFIG_SPI_ALTERA_DFL) += spi-altera-dfl.o
|
||||
obj-$(CONFIG_SPI_AR934X) += spi-ar934x.o
|
||||
obj-$(CONFIG_SPI_ARMADA_3700) += spi-armada-3700.o
|
||||
obj-$(CONFIG_SPI_ASPEED_SMC) += spi-aspeed-smc.o
|
||||
obj-$(CONFIG_SPI_ATMEL) += spi-atmel.o
|
||||
obj-$(CONFIG_SPI_ATMEL_QUADSPI) += atmel-quadspi.o
|
||||
obj-$(CONFIG_SPI_AT91_USART) += spi-at91-usart.o
|
||||
@ -76,6 +77,7 @@ obj-$(CONFIG_SPI_MPC52xx) += spi-mpc52xx.o
|
||||
obj-$(CONFIG_SPI_MT65XX) += spi-mt65xx.o
|
||||
obj-$(CONFIG_SPI_MT7621) += spi-mt7621.o
|
||||
obj-$(CONFIG_SPI_MTK_NOR) += spi-mtk-nor.o
|
||||
obj-$(CONFIG_SPI_MTK_SNFI) += spi-mtk-snfi.o
|
||||
obj-$(CONFIG_SPI_MXIC) += spi-mxic.o
|
||||
obj-$(CONFIG_SPI_MXS) += spi-mxs.o
|
||||
obj-$(CONFIG_SPI_NPCM_FIU) += spi-npcm-fiu.o
|
||||
|
@ -288,12 +288,6 @@ static bool atmel_qspi_supports_op(struct spi_mem *mem,
|
||||
op->dummy.nbytes == 0)
|
||||
return false;
|
||||
|
||||
/* DTR ops not supported. */
|
||||
if (op->cmd.dtr || op->addr.dtr || op->dummy.dtr || op->data.dtr)
|
||||
return false;
|
||||
if (op->cmd.nbytes != 1)
|
||||
return false;
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
|
1210
drivers/spi/spi-aspeed-smc.c
Normal file
1210
drivers/spi/spi-aspeed-smc.c
Normal file
File diff suppressed because it is too large
Load Diff
@ -405,7 +405,7 @@ static int au1550_spi_dma_txrxb(struct spi_device *spi, struct spi_transfer *t)
|
||||
dma_unmap_single(hw->dev, dma_tx_addr, t->len,
|
||||
DMA_TO_DEVICE);
|
||||
|
||||
return hw->rx_count < hw->tx_count ? hw->rx_count : hw->tx_count;
|
||||
return min(hw->rx_count, hw->tx_count);
|
||||
}
|
||||
|
||||
static irqreturn_t au1550_spi_dma_irq_callback(struct au1550_spi *hw)
|
||||
@ -539,7 +539,7 @@ static int au1550_spi_pio_txrxb(struct spi_device *spi, struct spi_transfer *t)
|
||||
|
||||
wait_for_completion(&hw->master_done);
|
||||
|
||||
return hw->rx_count < hw->tx_count ? hw->rx_count : hw->tx_count;
|
||||
return min(hw->rx_count, hw->tx_count);
|
||||
}
|
||||
|
||||
static irqreturn_t au1550_spi_pio_irq_callback(struct au1550_spi *hw)
|
||||
|
@ -43,6 +43,8 @@
|
||||
/* Capabilities */
|
||||
#define CQSPI_SUPPORTS_OCTAL BIT(0)
|
||||
|
||||
#define CQSPI_OP_WIDTH(part) ((part).nbytes ? ilog2((part).buswidth) : 0)
|
||||
|
||||
struct cqspi_st;
|
||||
|
||||
struct cqspi_flash_pdata {
|
||||
@ -53,16 +55,12 @@ struct cqspi_flash_pdata {
|
||||
u32 tsd2d_ns;
|
||||
u32 tchsh_ns;
|
||||
u32 tslch_ns;
|
||||
u8 inst_width;
|
||||
u8 addr_width;
|
||||
u8 data_width;
|
||||
bool dtr;
|
||||
u8 cs;
|
||||
};
|
||||
|
||||
struct cqspi_st {
|
||||
struct platform_device *pdev;
|
||||
|
||||
struct spi_master *master;
|
||||
struct clk *clk;
|
||||
unsigned int sclk;
|
||||
|
||||
@ -343,18 +341,18 @@ static irqreturn_t cqspi_irq_handler(int this_irq, void *dev)
|
||||
return IRQ_HANDLED;
|
||||
}
|
||||
|
||||
static unsigned int cqspi_calc_rdreg(struct cqspi_flash_pdata *f_pdata)
|
||||
static unsigned int cqspi_calc_rdreg(const struct spi_mem_op *op)
|
||||
{
|
||||
u32 rdreg = 0;
|
||||
|
||||
rdreg |= f_pdata->inst_width << CQSPI_REG_RD_INSTR_TYPE_INSTR_LSB;
|
||||
rdreg |= f_pdata->addr_width << CQSPI_REG_RD_INSTR_TYPE_ADDR_LSB;
|
||||
rdreg |= f_pdata->data_width << CQSPI_REG_RD_INSTR_TYPE_DATA_LSB;
|
||||
rdreg |= CQSPI_OP_WIDTH(op->cmd) << CQSPI_REG_RD_INSTR_TYPE_INSTR_LSB;
|
||||
rdreg |= CQSPI_OP_WIDTH(op->addr) << CQSPI_REG_RD_INSTR_TYPE_ADDR_LSB;
|
||||
rdreg |= CQSPI_OP_WIDTH(op->data) << CQSPI_REG_RD_INSTR_TYPE_DATA_LSB;
|
||||
|
||||
return rdreg;
|
||||
}
|
||||
|
||||
static unsigned int cqspi_calc_dummy(const struct spi_mem_op *op, bool dtr)
|
||||
static unsigned int cqspi_calc_dummy(const struct spi_mem_op *op)
|
||||
{
|
||||
unsigned int dummy_clk;
|
||||
|
||||
@ -362,66 +360,12 @@ static unsigned int cqspi_calc_dummy(const struct spi_mem_op *op, bool dtr)
|
||||
return 0;
|
||||
|
||||
dummy_clk = op->dummy.nbytes * (8 / op->dummy.buswidth);
|
||||
if (dtr)
|
||||
if (op->cmd.dtr)
|
||||
dummy_clk /= 2;
|
||||
|
||||
return dummy_clk;
|
||||
}
|
||||
|
||||
static int cqspi_set_protocol(struct cqspi_flash_pdata *f_pdata,
|
||||
const struct spi_mem_op *op)
|
||||
{
|
||||
/*
|
||||
* For an op to be DTR, cmd phase along with every other non-empty
|
||||
* phase should have dtr field set to 1. If an op phase has zero
|
||||
* nbytes, ignore its dtr field; otherwise, check its dtr field.
|
||||
*/
|
||||
f_pdata->dtr = op->cmd.dtr &&
|
||||
(!op->addr.nbytes || op->addr.dtr) &&
|
||||
(!op->data.nbytes || op->data.dtr);
|
||||
|
||||
f_pdata->inst_width = 0;
|
||||
if (op->cmd.buswidth)
|
||||
f_pdata->inst_width = ilog2(op->cmd.buswidth);
|
||||
|
||||
f_pdata->addr_width = 0;
|
||||
if (op->addr.buswidth)
|
||||
f_pdata->addr_width = ilog2(op->addr.buswidth);
|
||||
|
||||
f_pdata->data_width = 0;
|
||||
if (op->data.buswidth)
|
||||
f_pdata->data_width = ilog2(op->data.buswidth);
|
||||
|
||||
/* Right now we only support 8-8-8 DTR mode. */
|
||||
if (f_pdata->dtr) {
|
||||
switch (op->cmd.buswidth) {
|
||||
case 0:
|
||||
case 8:
|
||||
break;
|
||||
default:
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
switch (op->addr.buswidth) {
|
||||
case 0:
|
||||
case 8:
|
||||
break;
|
||||
default:
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
switch (op->data.buswidth) {
|
||||
case 0:
|
||||
case 8:
|
||||
break;
|
||||
default:
|
||||
return -EINVAL;
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int cqspi_wait_idle(struct cqspi_st *cqspi)
|
||||
{
|
||||
const unsigned int poll_idle_retry = 3;
|
||||
@ -503,8 +447,7 @@ static int cqspi_setup_opcode_ext(struct cqspi_flash_pdata *f_pdata,
|
||||
}
|
||||
|
||||
static int cqspi_enable_dtr(struct cqspi_flash_pdata *f_pdata,
|
||||
const struct spi_mem_op *op, unsigned int shift,
|
||||
bool enable)
|
||||
const struct spi_mem_op *op, unsigned int shift)
|
||||
{
|
||||
struct cqspi_st *cqspi = f_pdata->cqspi;
|
||||
void __iomem *reg_base = cqspi->iobase;
|
||||
@ -517,7 +460,7 @@ static int cqspi_enable_dtr(struct cqspi_flash_pdata *f_pdata,
|
||||
* We enable dual byte opcode here. The callers have to set up the
|
||||
* extension opcode based on which type of operation it is.
|
||||
*/
|
||||
if (enable) {
|
||||
if (op->cmd.dtr) {
|
||||
reg |= CQSPI_REG_CONFIG_DTR_PROTO;
|
||||
reg |= CQSPI_REG_CONFIG_DUAL_OPCODE;
|
||||
|
||||
@ -549,12 +492,7 @@ static int cqspi_command_read(struct cqspi_flash_pdata *f_pdata,
|
||||
size_t read_len;
|
||||
int status;
|
||||
|
||||
status = cqspi_set_protocol(f_pdata, op);
|
||||
if (status)
|
||||
return status;
|
||||
|
||||
status = cqspi_enable_dtr(f_pdata, op, CQSPI_REG_OP_EXT_STIG_LSB,
|
||||
f_pdata->dtr);
|
||||
status = cqspi_enable_dtr(f_pdata, op, CQSPI_REG_OP_EXT_STIG_LSB);
|
||||
if (status)
|
||||
return status;
|
||||
|
||||
@ -565,17 +503,17 @@ static int cqspi_command_read(struct cqspi_flash_pdata *f_pdata,
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
if (f_pdata->dtr)
|
||||
if (op->cmd.dtr)
|
||||
opcode = op->cmd.opcode >> 8;
|
||||
else
|
||||
opcode = op->cmd.opcode;
|
||||
|
||||
reg = opcode << CQSPI_REG_CMDCTRL_OPCODE_LSB;
|
||||
|
||||
rdreg = cqspi_calc_rdreg(f_pdata);
|
||||
rdreg = cqspi_calc_rdreg(op);
|
||||
writel(rdreg, reg_base + CQSPI_REG_RD_INSTR);
|
||||
|
||||
dummy_clk = cqspi_calc_dummy(op, f_pdata->dtr);
|
||||
dummy_clk = cqspi_calc_dummy(op);
|
||||
if (dummy_clk > CQSPI_DUMMY_CLKS_MAX)
|
||||
return -EOPNOTSUPP;
|
||||
|
||||
@ -622,12 +560,7 @@ static int cqspi_command_write(struct cqspi_flash_pdata *f_pdata,
|
||||
size_t write_len;
|
||||
int ret;
|
||||
|
||||
ret = cqspi_set_protocol(f_pdata, op);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
ret = cqspi_enable_dtr(f_pdata, op, CQSPI_REG_OP_EXT_STIG_LSB,
|
||||
f_pdata->dtr);
|
||||
ret = cqspi_enable_dtr(f_pdata, op, CQSPI_REG_OP_EXT_STIG_LSB);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
@ -638,10 +571,10 @@ static int cqspi_command_write(struct cqspi_flash_pdata *f_pdata,
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
reg = cqspi_calc_rdreg(f_pdata);
|
||||
reg = cqspi_calc_rdreg(op);
|
||||
writel(reg, reg_base + CQSPI_REG_RD_INSTR);
|
||||
|
||||
if (f_pdata->dtr)
|
||||
if (op->cmd.dtr)
|
||||
opcode = op->cmd.opcode >> 8;
|
||||
else
|
||||
opcode = op->cmd.opcode;
|
||||
@ -688,21 +621,20 @@ static int cqspi_read_setup(struct cqspi_flash_pdata *f_pdata,
|
||||
int ret;
|
||||
u8 opcode;
|
||||
|
||||
ret = cqspi_enable_dtr(f_pdata, op, CQSPI_REG_OP_EXT_READ_LSB,
|
||||
f_pdata->dtr);
|
||||
ret = cqspi_enable_dtr(f_pdata, op, CQSPI_REG_OP_EXT_READ_LSB);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
if (f_pdata->dtr)
|
||||
if (op->cmd.dtr)
|
||||
opcode = op->cmd.opcode >> 8;
|
||||
else
|
||||
opcode = op->cmd.opcode;
|
||||
|
||||
reg = opcode << CQSPI_REG_RD_INSTR_OPCODE_LSB;
|
||||
reg |= cqspi_calc_rdreg(f_pdata);
|
||||
reg |= cqspi_calc_rdreg(op);
|
||||
|
||||
/* Setup dummy clock cycles */
|
||||
dummy_clk = cqspi_calc_dummy(op, f_pdata->dtr);
|
||||
dummy_clk = cqspi_calc_dummy(op);
|
||||
|
||||
if (dummy_clk > CQSPI_DUMMY_CLKS_MAX)
|
||||
return -EOPNOTSUPP;
|
||||
@ -947,22 +879,21 @@ static int cqspi_write_setup(struct cqspi_flash_pdata *f_pdata,
|
||||
void __iomem *reg_base = cqspi->iobase;
|
||||
u8 opcode;
|
||||
|
||||
ret = cqspi_enable_dtr(f_pdata, op, CQSPI_REG_OP_EXT_WRITE_LSB,
|
||||
f_pdata->dtr);
|
||||
ret = cqspi_enable_dtr(f_pdata, op, CQSPI_REG_OP_EXT_WRITE_LSB);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
if (f_pdata->dtr)
|
||||
if (op->cmd.dtr)
|
||||
opcode = op->cmd.opcode >> 8;
|
||||
else
|
||||
opcode = op->cmd.opcode;
|
||||
|
||||
/* Set opcode. */
|
||||
reg = opcode << CQSPI_REG_WR_INSTR_OPCODE_LSB;
|
||||
reg |= f_pdata->data_width << CQSPI_REG_WR_INSTR_TYPE_DATA_LSB;
|
||||
reg |= f_pdata->addr_width << CQSPI_REG_WR_INSTR_TYPE_ADDR_LSB;
|
||||
reg |= CQSPI_OP_WIDTH(op->data) << CQSPI_REG_WR_INSTR_TYPE_DATA_LSB;
|
||||
reg |= CQSPI_OP_WIDTH(op->addr) << CQSPI_REG_WR_INSTR_TYPE_ADDR_LSB;
|
||||
writel(reg, reg_base + CQSPI_REG_WR_INSTR);
|
||||
reg = cqspi_calc_rdreg(f_pdata);
|
||||
reg = cqspi_calc_rdreg(op);
|
||||
writel(reg, reg_base + CQSPI_REG_RD_INSTR);
|
||||
|
||||
/*
|
||||
@ -1244,10 +1175,6 @@ static ssize_t cqspi_write(struct cqspi_flash_pdata *f_pdata,
|
||||
const u_char *buf = op->data.buf.out;
|
||||
int ret;
|
||||
|
||||
ret = cqspi_set_protocol(f_pdata, op);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
ret = cqspi_write_setup(f_pdata, op);
|
||||
if (ret)
|
||||
return ret;
|
||||
@ -1260,7 +1187,7 @@ static ssize_t cqspi_write(struct cqspi_flash_pdata *f_pdata,
|
||||
* mode. So, we can not use direct mode when in DTR mode for writing
|
||||
* data.
|
||||
*/
|
||||
if (!f_pdata->dtr && cqspi->use_direct_mode &&
|
||||
if (!op->cmd.dtr && cqspi->use_direct_mode &&
|
||||
((to + len) <= cqspi->ahb_size)) {
|
||||
memcpy_toio(cqspi->ahb_base + to, buf, len);
|
||||
return cqspi_wait_idle(cqspi);
|
||||
@ -1348,9 +1275,6 @@ static ssize_t cqspi_read(struct cqspi_flash_pdata *f_pdata,
|
||||
int ret;
|
||||
|
||||
ddata = of_device_get_match_data(dev);
|
||||
ret = cqspi_set_protocol(f_pdata, op);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
ret = cqspi_read_setup(f_pdata, op);
|
||||
if (ret)
|
||||
@ -1423,13 +1347,7 @@ static bool cqspi_supports_mem_op(struct spi_mem *mem,
|
||||
return false;
|
||||
if (op->data.nbytes && op->data.buswidth != 8)
|
||||
return false;
|
||||
} else if (all_false) {
|
||||
/* Only 1-1-X ops are supported without DTR */
|
||||
if (op->cmd.nbytes && op->cmd.buswidth > 1)
|
||||
return false;
|
||||
if (op->addr.nbytes && op->addr.buswidth > 1)
|
||||
return false;
|
||||
} else {
|
||||
} else if (!all_false) {
|
||||
/* Mixed DTR modes are not supported. */
|
||||
return false;
|
||||
}
|
||||
@ -1563,6 +1481,7 @@ static int cqspi_request_mmap_dma(struct cqspi_st *cqspi)
|
||||
cqspi->rx_chan = dma_request_chan_by_mask(&mask);
|
||||
if (IS_ERR(cqspi->rx_chan)) {
|
||||
int ret = PTR_ERR(cqspi->rx_chan);
|
||||
|
||||
cqspi->rx_chan = NULL;
|
||||
return dev_err_probe(&cqspi->pdev->dev, ret, "No Rx DMA available\n");
|
||||
}
|
||||
@ -1639,7 +1558,7 @@ static int cqspi_probe(struct platform_device *pdev)
|
||||
int ret;
|
||||
int irq;
|
||||
|
||||
master = spi_alloc_master(&pdev->dev, sizeof(*cqspi));
|
||||
master = devm_spi_alloc_master(&pdev->dev, sizeof(*cqspi));
|
||||
if (!master) {
|
||||
dev_err(&pdev->dev, "spi_alloc_master failed\n");
|
||||
return -ENOMEM;
|
||||
@ -1652,6 +1571,7 @@ static int cqspi_probe(struct platform_device *pdev)
|
||||
cqspi = spi_master_get_devdata(master);
|
||||
|
||||
cqspi->pdev = pdev;
|
||||
cqspi->master = master;
|
||||
platform_set_drvdata(pdev, cqspi);
|
||||
|
||||
/* Obtain configuration from OF. */
|
||||
@ -1700,11 +1620,9 @@ static int cqspi_probe(struct platform_device *pdev)
|
||||
}
|
||||
|
||||
pm_runtime_enable(dev);
|
||||
ret = pm_runtime_get_sync(dev);
|
||||
if (ret < 0) {
|
||||
pm_runtime_put_noidle(dev);
|
||||
ret = pm_runtime_resume_and_get(dev);
|
||||
if (ret < 0)
|
||||
goto probe_master_put;
|
||||
}
|
||||
|
||||
ret = clk_prepare_enable(cqspi->clk);
|
||||
if (ret) {
|
||||
@ -1784,7 +1702,7 @@ static int cqspi_probe(struct platform_device *pdev)
|
||||
goto probe_setup_failed;
|
||||
}
|
||||
|
||||
ret = devm_spi_register_master(dev, master);
|
||||
ret = spi_register_master(master);
|
||||
if (ret) {
|
||||
dev_err(&pdev->dev, "failed to register SPI ctlr %d\n", ret);
|
||||
goto probe_setup_failed;
|
||||
@ -1807,6 +1725,7 @@ static int cqspi_remove(struct platform_device *pdev)
|
||||
{
|
||||
struct cqspi_st *cqspi = platform_get_drvdata(pdev);
|
||||
|
||||
spi_unregister_master(cqspi->master);
|
||||
cqspi_controller_enable(cqspi, 0);
|
||||
|
||||
if (cqspi->rx_chan)
|
||||
@ -1865,7 +1784,7 @@ static const struct cqspi_driver_platdata intel_lgm_qspi = {
|
||||
};
|
||||
|
||||
static const struct cqspi_driver_platdata socfpga_qspi = {
|
||||
.quirks = CQSPI_NO_SUPPORT_WR_COMPLETION,
|
||||
.quirks = CQSPI_DISABLE_DAC_MODE | CQSPI_NO_SUPPORT_WR_COMPLETION,
|
||||
};
|
||||
|
||||
static const struct cqspi_driver_platdata versal_ospi = {
|
||||
@ -1894,11 +1813,11 @@ static const struct of_device_id cqspi_dt_ids[] = {
|
||||
},
|
||||
{
|
||||
.compatible = "xlnx,versal-ospi-1.0",
|
||||
.data = (void *)&versal_ospi,
|
||||
.data = &versal_ospi,
|
||||
},
|
||||
{
|
||||
.compatible = "intel,socfpga-qspi",
|
||||
.data = (void *)&socfpga_qspi,
|
||||
.data = &socfpga_qspi,
|
||||
},
|
||||
{ /* end of table */ }
|
||||
};
|
||||
|
@ -342,7 +342,8 @@ static irqreturn_t cdns_spi_irq(int irq, void *dev_id)
|
||||
{
|
||||
struct spi_master *master = dev_id;
|
||||
struct cdns_spi *xspi = spi_master_get_devdata(master);
|
||||
u32 intr_status, status;
|
||||
irqreturn_t status;
|
||||
u32 intr_status;
|
||||
|
||||
status = IRQ_NONE;
|
||||
intr_status = cdns_spi_read(xspi, CDNS_SPI_ISR);
|
||||
@ -657,7 +658,7 @@ static int __maybe_unused cdns_spi_resume(struct device *dev)
|
||||
*
|
||||
* Return: 0 on success and error value on error
|
||||
*/
|
||||
static int __maybe_unused cnds_runtime_resume(struct device *dev)
|
||||
static int __maybe_unused cdns_spi_runtime_resume(struct device *dev)
|
||||
{
|
||||
struct spi_master *master = dev_get_drvdata(dev);
|
||||
struct cdns_spi *xspi = spi_master_get_devdata(master);
|
||||
@ -686,7 +687,7 @@ static int __maybe_unused cnds_runtime_resume(struct device *dev)
|
||||
*
|
||||
* Return: Always 0
|
||||
*/
|
||||
static int __maybe_unused cnds_runtime_suspend(struct device *dev)
|
||||
static int __maybe_unused cdns_spi_runtime_suspend(struct device *dev)
|
||||
{
|
||||
struct spi_master *master = dev_get_drvdata(dev);
|
||||
struct cdns_spi *xspi = spi_master_get_devdata(master);
|
||||
@ -698,8 +699,8 @@ static int __maybe_unused cnds_runtime_suspend(struct device *dev)
|
||||
}
|
||||
|
||||
static const struct dev_pm_ops cdns_spi_dev_pm_ops = {
|
||||
SET_RUNTIME_PM_OPS(cnds_runtime_suspend,
|
||||
cnds_runtime_resume, NULL)
|
||||
SET_RUNTIME_PM_OPS(cdns_spi_runtime_suspend,
|
||||
cdns_spi_runtime_resume, NULL)
|
||||
SET_SYSTEM_SLEEP_PM_OPS(cdns_spi_suspend, cdns_spi_resume)
|
||||
};
|
||||
|
||||
|
@ -9,6 +9,7 @@
|
||||
#include <linux/clk.h>
|
||||
#include <linux/gpio/consumer.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/of.h>
|
||||
#include <linux/interrupt.h>
|
||||
#include <linux/platform_device.h>
|
||||
#include <linux/regmap.h>
|
||||
@ -89,6 +90,7 @@ static irqreturn_t spi_clps711x_isr(int irq, void *dev_id)
|
||||
|
||||
static int spi_clps711x_probe(struct platform_device *pdev)
|
||||
{
|
||||
struct device_node *np = pdev->dev.of_node;
|
||||
struct spi_clps711x_data *hw;
|
||||
struct spi_master *master;
|
||||
int irq, ret;
|
||||
@ -117,8 +119,7 @@ static int spi_clps711x_probe(struct platform_device *pdev)
|
||||
goto err_out;
|
||||
}
|
||||
|
||||
hw->syscon =
|
||||
syscon_regmap_lookup_by_compatible("cirrus,ep7209-syscon3");
|
||||
hw->syscon = syscon_regmap_lookup_by_phandle(np, "syscon");
|
||||
if (IS_ERR(hw->syscon)) {
|
||||
ret = PTR_ERR(hw->syscon);
|
||||
goto err_out;
|
||||
|
@ -876,6 +876,10 @@ static int fsl_qspi_probe(struct platform_device *pdev)
|
||||
|
||||
res = platform_get_resource_byname(pdev, IORESOURCE_MEM,
|
||||
"QuadSPI-memory");
|
||||
if (!res) {
|
||||
ret = -EINVAL;
|
||||
goto err_put_ctrl;
|
||||
}
|
||||
q->memmap_phy = res->start;
|
||||
/* Since there are 4 cs, map size required is 4 times ahb_buf_size */
|
||||
q->ahb_addr = devm_ioremap(dev, q->memmap_phy,
|
||||
|
@ -731,7 +731,7 @@ static int img_spfi_resume(struct device *dev)
|
||||
int ret;
|
||||
|
||||
ret = pm_runtime_get_sync(dev);
|
||||
if (ret) {
|
||||
if (ret < 0) {
|
||||
pm_runtime_put_noidle(dev);
|
||||
return ret;
|
||||
}
|
||||
|
@ -18,7 +18,6 @@
|
||||
#include <linux/pm_runtime.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/spi/spi.h>
|
||||
#include <linux/spi/spi_bitbang.h>
|
||||
#include <linux/types.h>
|
||||
#include <linux/of.h>
|
||||
#include <linux/of_device.h>
|
||||
@ -32,6 +31,12 @@ static bool use_dma = true;
|
||||
module_param(use_dma, bool, 0644);
|
||||
MODULE_PARM_DESC(use_dma, "Enable usage of DMA when available (default)");
|
||||
|
||||
/* define polling limits */
|
||||
static unsigned int polling_limit_us = 30;
|
||||
module_param(polling_limit_us, uint, 0664);
|
||||
MODULE_PARM_DESC(polling_limit_us,
|
||||
"time in us to run a transfer in polling mode\n");
|
||||
|
||||
#define MXC_RPM_TIMEOUT 2000 /* 2000ms */
|
||||
|
||||
#define MXC_CSPIRXDATA 0x00
|
||||
@ -64,15 +69,15 @@ enum spi_imx_devtype {
|
||||
struct spi_imx_data;
|
||||
|
||||
struct spi_imx_devtype_data {
|
||||
void (*intctrl)(struct spi_imx_data *, int);
|
||||
int (*prepare_message)(struct spi_imx_data *, struct spi_message *);
|
||||
int (*prepare_transfer)(struct spi_imx_data *, struct spi_device *);
|
||||
void (*trigger)(struct spi_imx_data *);
|
||||
int (*rx_available)(struct spi_imx_data *);
|
||||
void (*reset)(struct spi_imx_data *);
|
||||
void (*setup_wml)(struct spi_imx_data *);
|
||||
void (*disable)(struct spi_imx_data *);
|
||||
void (*disable_dma)(struct spi_imx_data *);
|
||||
void (*intctrl)(struct spi_imx_data *spi_imx, int enable);
|
||||
int (*prepare_message)(struct spi_imx_data *spi_imx, struct spi_message *msg);
|
||||
int (*prepare_transfer)(struct spi_imx_data *spi_imx, struct spi_device *spi);
|
||||
void (*trigger)(struct spi_imx_data *spi_imx);
|
||||
int (*rx_available)(struct spi_imx_data *spi_imx);
|
||||
void (*reset)(struct spi_imx_data *spi_imx);
|
||||
void (*setup_wml)(struct spi_imx_data *spi_imx);
|
||||
void (*disable)(struct spi_imx_data *spi_imx);
|
||||
void (*disable_dma)(struct spi_imx_data *spi_imx);
|
||||
bool has_dmamode;
|
||||
bool has_slavemode;
|
||||
unsigned int fifo_size;
|
||||
@ -86,7 +91,7 @@ struct spi_imx_devtype_data {
|
||||
};
|
||||
|
||||
struct spi_imx_data {
|
||||
struct spi_bitbang bitbang;
|
||||
struct spi_controller *controller;
|
||||
struct device *dev;
|
||||
|
||||
struct completion xfer_done;
|
||||
@ -102,12 +107,13 @@ struct spi_imx_data {
|
||||
unsigned int spi_drctl;
|
||||
|
||||
unsigned int count, remainder;
|
||||
void (*tx)(struct spi_imx_data *);
|
||||
void (*rx)(struct spi_imx_data *);
|
||||
void (*tx)(struct spi_imx_data *spi_imx);
|
||||
void (*rx)(struct spi_imx_data *spi_imx);
|
||||
void *rx_buf;
|
||||
const void *tx_buf;
|
||||
unsigned int txfifo; /* number of words pushed in tx FIFO */
|
||||
unsigned int dynamic_burst;
|
||||
bool rx_only;
|
||||
|
||||
/* Slave mode */
|
||||
bool slave_mode;
|
||||
@ -225,15 +231,15 @@ static int spi_imx_bytes_per_word(const int bits_per_word)
|
||||
return 4;
|
||||
}
|
||||
|
||||
static bool spi_imx_can_dma(struct spi_master *master, struct spi_device *spi,
|
||||
static bool spi_imx_can_dma(struct spi_controller *controller, struct spi_device *spi,
|
||||
struct spi_transfer *transfer)
|
||||
{
|
||||
struct spi_imx_data *spi_imx = spi_master_get_devdata(master);
|
||||
struct spi_imx_data *spi_imx = spi_controller_get_devdata(controller);
|
||||
|
||||
if (!use_dma || master->fallback)
|
||||
if (!use_dma || controller->fallback)
|
||||
return false;
|
||||
|
||||
if (!master->dma_rx)
|
||||
if (!controller->dma_rx)
|
||||
return false;
|
||||
|
||||
if (spi_imx->slave_mode)
|
||||
@ -289,17 +295,16 @@ static bool spi_imx_can_dma(struct spi_master *master, struct spi_device *spi,
|
||||
static void spi_imx_buf_rx_swap_u32(struct spi_imx_data *spi_imx)
|
||||
{
|
||||
unsigned int val = readl(spi_imx->base + MXC_CSPIRXDATA);
|
||||
#ifdef __LITTLE_ENDIAN
|
||||
unsigned int bytes_per_word;
|
||||
#endif
|
||||
|
||||
if (spi_imx->rx_buf) {
|
||||
#ifdef __LITTLE_ENDIAN
|
||||
unsigned int bytes_per_word;
|
||||
|
||||
bytes_per_word = spi_imx_bytes_per_word(spi_imx->bits_per_word);
|
||||
if (bytes_per_word == 1)
|
||||
val = cpu_to_be32(val);
|
||||
swab32s(&val);
|
||||
else if (bytes_per_word == 2)
|
||||
val = (val << 16) | (val >> 16);
|
||||
swahw32s(&val);
|
||||
#endif
|
||||
*(u32 *)spi_imx->rx_buf = val;
|
||||
spi_imx->rx_buf += sizeof(u32);
|
||||
@ -353,9 +358,9 @@ static void spi_imx_buf_tx_swap_u32(struct spi_imx_data *spi_imx)
|
||||
bytes_per_word = spi_imx_bytes_per_word(spi_imx->bits_per_word);
|
||||
|
||||
if (bytes_per_word == 1)
|
||||
val = cpu_to_be32(val);
|
||||
swab32s(&val);
|
||||
else if (bytes_per_word == 2)
|
||||
val = (val << 16) | (val >> 16);
|
||||
swahw32s(&val);
|
||||
#endif
|
||||
writel(val, spi_imx->base + MXC_CSPITXDATA);
|
||||
}
|
||||
@ -469,7 +474,7 @@ static unsigned int mx51_ecspi_clkdiv(struct spi_imx_data *spi_imx,
|
||||
|
||||
static void mx51_ecspi_intctrl(struct spi_imx_data *spi_imx, int enable)
|
||||
{
|
||||
unsigned val = 0;
|
||||
unsigned int val = 0;
|
||||
|
||||
if (enable & MXC_INT_TE)
|
||||
val |= MX51_ECSPI_INT_TEEN;
|
||||
@ -515,6 +520,7 @@ static int mx51_ecspi_prepare_message(struct spi_imx_data *spi_imx,
|
||||
u32 min_speed_hz = ~0U;
|
||||
u32 testreg, delay;
|
||||
u32 cfg = readl(spi_imx->base + MX51_ECSPI_CONFIG);
|
||||
u32 current_cfg = cfg;
|
||||
|
||||
/* set Master or Slave mode */
|
||||
if (spi_imx->slave_mode)
|
||||
@ -554,11 +560,6 @@ static int mx51_ecspi_prepare_message(struct spi_imx_data *spi_imx,
|
||||
else
|
||||
cfg |= MX51_ECSPI_CONFIG_SBBCTRL(spi->chip_select);
|
||||
|
||||
if (spi->mode & SPI_CPHA)
|
||||
cfg |= MX51_ECSPI_CONFIG_SCLKPHA(spi->chip_select);
|
||||
else
|
||||
cfg &= ~MX51_ECSPI_CONFIG_SCLKPHA(spi->chip_select);
|
||||
|
||||
if (spi->mode & SPI_CPOL) {
|
||||
cfg |= MX51_ECSPI_CONFIG_SCLKPOL(spi->chip_select);
|
||||
cfg |= MX51_ECSPI_CONFIG_SCLKCTL(spi->chip_select);
|
||||
@ -572,6 +573,9 @@ static int mx51_ecspi_prepare_message(struct spi_imx_data *spi_imx,
|
||||
else
|
||||
cfg &= ~MX51_ECSPI_CONFIG_SSBPOL(spi->chip_select);
|
||||
|
||||
if (cfg == current_cfg)
|
||||
return 0;
|
||||
|
||||
writel(cfg, spi_imx->base + MX51_ECSPI_CONFIG);
|
||||
|
||||
/*
|
||||
@ -585,7 +589,7 @@ static int mx51_ecspi_prepare_message(struct spi_imx_data *spi_imx,
|
||||
* the SPI communication as the device on the other end would consider
|
||||
* the change of SCLK polarity as a clock tick already.
|
||||
*
|
||||
* Because spi_imx->spi_bus_clk is only set in bitbang prepare_message
|
||||
* Because spi_imx->spi_bus_clk is only set in prepare_message
|
||||
* callback, iterate over all the transfers in spi_message, find the
|
||||
* one with lowest bus frequency, and use that bus frequency for the
|
||||
* delay calculation. In case all transfers have speed_hz == 0, then
|
||||
@ -606,6 +610,24 @@ static int mx51_ecspi_prepare_message(struct spi_imx_data *spi_imx,
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void mx51_configure_cpha(struct spi_imx_data *spi_imx,
|
||||
struct spi_device *spi)
|
||||
{
|
||||
bool cpha = (spi->mode & SPI_CPHA);
|
||||
bool flip_cpha = (spi->mode & SPI_RX_CPHA_FLIP) && spi_imx->rx_only;
|
||||
u32 cfg = readl(spi_imx->base + MX51_ECSPI_CONFIG);
|
||||
|
||||
/* Flip cpha logical value iff flip_cpha */
|
||||
cpha ^= flip_cpha;
|
||||
|
||||
if (cpha)
|
||||
cfg |= MX51_ECSPI_CONFIG_SCLKPHA(spi->chip_select);
|
||||
else
|
||||
cfg &= ~MX51_ECSPI_CONFIG_SCLKPHA(spi->chip_select);
|
||||
|
||||
writel(cfg, spi_imx->base + MX51_ECSPI_CONFIG);
|
||||
}
|
||||
|
||||
static int mx51_ecspi_prepare_transfer(struct spi_imx_data *spi_imx,
|
||||
struct spi_device *spi)
|
||||
{
|
||||
@ -627,6 +649,8 @@ static int mx51_ecspi_prepare_transfer(struct spi_imx_data *spi_imx,
|
||||
ctrl |= mx51_ecspi_clkdiv(spi_imx, spi_imx->spi_bus_clk, &clk);
|
||||
spi_imx->spi_bus_clk = clk;
|
||||
|
||||
mx51_configure_cpha(spi_imx, spi);
|
||||
|
||||
/*
|
||||
* ERR009165: work in XHC mode instead of SMC as PIO on the chips
|
||||
* before i.mx6ul.
|
||||
@ -1153,12 +1177,12 @@ static irqreturn_t spi_imx_isr(int irq, void *dev_id)
|
||||
return IRQ_HANDLED;
|
||||
}
|
||||
|
||||
static int spi_imx_dma_configure(struct spi_master *master)
|
||||
static int spi_imx_dma_configure(struct spi_controller *controller)
|
||||
{
|
||||
int ret;
|
||||
enum dma_slave_buswidth buswidth;
|
||||
struct dma_slave_config rx = {}, tx = {};
|
||||
struct spi_imx_data *spi_imx = spi_master_get_devdata(master);
|
||||
struct spi_imx_data *spi_imx = spi_controller_get_devdata(controller);
|
||||
|
||||
switch (spi_imx_bytes_per_word(spi_imx->bits_per_word)) {
|
||||
case 4:
|
||||
@ -1178,7 +1202,7 @@ static int spi_imx_dma_configure(struct spi_master *master)
|
||||
tx.dst_addr = spi_imx->base_phys + MXC_CSPITXDATA;
|
||||
tx.dst_addr_width = buswidth;
|
||||
tx.dst_maxburst = spi_imx->wml;
|
||||
ret = dmaengine_slave_config(master->dma_tx, &tx);
|
||||
ret = dmaengine_slave_config(controller->dma_tx, &tx);
|
||||
if (ret) {
|
||||
dev_err(spi_imx->dev, "TX dma configuration failed with %d\n", ret);
|
||||
return ret;
|
||||
@ -1188,7 +1212,7 @@ static int spi_imx_dma_configure(struct spi_master *master)
|
||||
rx.src_addr = spi_imx->base_phys + MXC_CSPIRXDATA;
|
||||
rx.src_addr_width = buswidth;
|
||||
rx.src_maxburst = spi_imx->wml;
|
||||
ret = dmaengine_slave_config(master->dma_rx, &rx);
|
||||
ret = dmaengine_slave_config(controller->dma_rx, &rx);
|
||||
if (ret) {
|
||||
dev_err(spi_imx->dev, "RX dma configuration failed with %d\n", ret);
|
||||
return ret;
|
||||
@ -1200,7 +1224,7 @@ static int spi_imx_dma_configure(struct spi_master *master)
|
||||
static int spi_imx_setupxfer(struct spi_device *spi,
|
||||
struct spi_transfer *t)
|
||||
{
|
||||
struct spi_imx_data *spi_imx = spi_master_get_devdata(spi->master);
|
||||
struct spi_imx_data *spi_imx = spi_controller_get_devdata(spi->controller);
|
||||
|
||||
if (!t)
|
||||
return 0;
|
||||
@ -1246,11 +1270,14 @@ static int spi_imx_setupxfer(struct spi_device *spi,
|
||||
spi_imx->dynamic_burst = 0;
|
||||
}
|
||||
|
||||
if (spi_imx_can_dma(spi_imx->bitbang.master, spi, t))
|
||||
if (spi_imx_can_dma(spi_imx->controller, spi, t))
|
||||
spi_imx->usedma = true;
|
||||
else
|
||||
spi_imx->usedma = false;
|
||||
|
||||
spi_imx->rx_only = ((t->tx_buf == NULL)
|
||||
|| (t->tx_buf == spi->controller->dummy_tx));
|
||||
|
||||
if (is_imx53_ecspi(spi_imx) && spi_imx->slave_mode) {
|
||||
spi_imx->rx = mx53_ecspi_rx_slave;
|
||||
spi_imx->tx = mx53_ecspi_tx_slave;
|
||||
@ -1264,50 +1291,50 @@ static int spi_imx_setupxfer(struct spi_device *spi,
|
||||
|
||||
static void spi_imx_sdma_exit(struct spi_imx_data *spi_imx)
|
||||
{
|
||||
struct spi_master *master = spi_imx->bitbang.master;
|
||||
struct spi_controller *controller = spi_imx->controller;
|
||||
|
||||
if (master->dma_rx) {
|
||||
dma_release_channel(master->dma_rx);
|
||||
master->dma_rx = NULL;
|
||||
if (controller->dma_rx) {
|
||||
dma_release_channel(controller->dma_rx);
|
||||
controller->dma_rx = NULL;
|
||||
}
|
||||
|
||||
if (master->dma_tx) {
|
||||
dma_release_channel(master->dma_tx);
|
||||
master->dma_tx = NULL;
|
||||
if (controller->dma_tx) {
|
||||
dma_release_channel(controller->dma_tx);
|
||||
controller->dma_tx = NULL;
|
||||
}
|
||||
}
|
||||
|
||||
static int spi_imx_sdma_init(struct device *dev, struct spi_imx_data *spi_imx,
|
||||
struct spi_master *master)
|
||||
struct spi_controller *controller)
|
||||
{
|
||||
int ret;
|
||||
|
||||
spi_imx->wml = spi_imx->devtype_data->fifo_size / 2;
|
||||
|
||||
/* Prepare for TX DMA: */
|
||||
master->dma_tx = dma_request_chan(dev, "tx");
|
||||
if (IS_ERR(master->dma_tx)) {
|
||||
ret = PTR_ERR(master->dma_tx);
|
||||
controller->dma_tx = dma_request_chan(dev, "tx");
|
||||
if (IS_ERR(controller->dma_tx)) {
|
||||
ret = PTR_ERR(controller->dma_tx);
|
||||
dev_dbg(dev, "can't get the TX DMA channel, error %d!\n", ret);
|
||||
master->dma_tx = NULL;
|
||||
controller->dma_tx = NULL;
|
||||
goto err;
|
||||
}
|
||||
|
||||
/* Prepare for RX : */
|
||||
master->dma_rx = dma_request_chan(dev, "rx");
|
||||
if (IS_ERR(master->dma_rx)) {
|
||||
ret = PTR_ERR(master->dma_rx);
|
||||
controller->dma_rx = dma_request_chan(dev, "rx");
|
||||
if (IS_ERR(controller->dma_rx)) {
|
||||
ret = PTR_ERR(controller->dma_rx);
|
||||
dev_dbg(dev, "can't get the RX DMA channel, error %d\n", ret);
|
||||
master->dma_rx = NULL;
|
||||
controller->dma_rx = NULL;
|
||||
goto err;
|
||||
}
|
||||
|
||||
init_completion(&spi_imx->dma_rx_completion);
|
||||
init_completion(&spi_imx->dma_tx_completion);
|
||||
master->can_dma = spi_imx_can_dma;
|
||||
master->max_dma_len = MAX_SDMA_BD_BYTES;
|
||||
spi_imx->bitbang.master->flags = SPI_MASTER_MUST_RX |
|
||||
SPI_MASTER_MUST_TX;
|
||||
controller->can_dma = spi_imx_can_dma;
|
||||
controller->max_dma_len = MAX_SDMA_BD_BYTES;
|
||||
spi_imx->controller->flags = SPI_CONTROLLER_MUST_RX |
|
||||
SPI_CONTROLLER_MUST_TX;
|
||||
|
||||
return 0;
|
||||
err:
|
||||
@ -1349,7 +1376,7 @@ static int spi_imx_dma_transfer(struct spi_imx_data *spi_imx,
|
||||
struct dma_async_tx_descriptor *desc_tx, *desc_rx;
|
||||
unsigned long transfer_timeout;
|
||||
unsigned long timeout;
|
||||
struct spi_master *master = spi_imx->bitbang.master;
|
||||
struct spi_controller *controller = spi_imx->controller;
|
||||
struct sg_table *tx = &transfer->tx_sg, *rx = &transfer->rx_sg;
|
||||
struct scatterlist *last_sg = sg_last(rx->sgl, rx->nents);
|
||||
unsigned int bytes_per_word, i;
|
||||
@ -1367,7 +1394,7 @@ static int spi_imx_dma_transfer(struct spi_imx_data *spi_imx,
|
||||
|
||||
spi_imx->wml = i;
|
||||
|
||||
ret = spi_imx_dma_configure(master);
|
||||
ret = spi_imx_dma_configure(controller);
|
||||
if (ret)
|
||||
goto dma_failure_no_start;
|
||||
|
||||
@ -1382,7 +1409,7 @@ static int spi_imx_dma_transfer(struct spi_imx_data *spi_imx,
|
||||
* The TX DMA setup starts the transfer, so make sure RX is configured
|
||||
* before TX.
|
||||
*/
|
||||
desc_rx = dmaengine_prep_slave_sg(master->dma_rx,
|
||||
desc_rx = dmaengine_prep_slave_sg(controller->dma_rx,
|
||||
rx->sgl, rx->nents, DMA_DEV_TO_MEM,
|
||||
DMA_PREP_INTERRUPT | DMA_CTRL_ACK);
|
||||
if (!desc_rx) {
|
||||
@ -1394,14 +1421,14 @@ static int spi_imx_dma_transfer(struct spi_imx_data *spi_imx,
|
||||
desc_rx->callback_param = (void *)spi_imx;
|
||||
dmaengine_submit(desc_rx);
|
||||
reinit_completion(&spi_imx->dma_rx_completion);
|
||||
dma_async_issue_pending(master->dma_rx);
|
||||
dma_async_issue_pending(controller->dma_rx);
|
||||
|
||||
desc_tx = dmaengine_prep_slave_sg(master->dma_tx,
|
||||
desc_tx = dmaengine_prep_slave_sg(controller->dma_tx,
|
||||
tx->sgl, tx->nents, DMA_MEM_TO_DEV,
|
||||
DMA_PREP_INTERRUPT | DMA_CTRL_ACK);
|
||||
if (!desc_tx) {
|
||||
dmaengine_terminate_all(master->dma_tx);
|
||||
dmaengine_terminate_all(master->dma_rx);
|
||||
dmaengine_terminate_all(controller->dma_tx);
|
||||
dmaengine_terminate_all(controller->dma_rx);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
@ -1409,7 +1436,7 @@ static int spi_imx_dma_transfer(struct spi_imx_data *spi_imx,
|
||||
desc_tx->callback_param = (void *)spi_imx;
|
||||
dmaengine_submit(desc_tx);
|
||||
reinit_completion(&spi_imx->dma_tx_completion);
|
||||
dma_async_issue_pending(master->dma_tx);
|
||||
dma_async_issue_pending(controller->dma_tx);
|
||||
|
||||
transfer_timeout = spi_imx_calculate_timeout(spi_imx, transfer->len);
|
||||
|
||||
@ -1418,21 +1445,21 @@ static int spi_imx_dma_transfer(struct spi_imx_data *spi_imx,
|
||||
transfer_timeout);
|
||||
if (!timeout) {
|
||||
dev_err(spi_imx->dev, "I/O Error in DMA TX\n");
|
||||
dmaengine_terminate_all(master->dma_tx);
|
||||
dmaengine_terminate_all(master->dma_rx);
|
||||
dmaengine_terminate_all(controller->dma_tx);
|
||||
dmaengine_terminate_all(controller->dma_rx);
|
||||
return -ETIMEDOUT;
|
||||
}
|
||||
|
||||
timeout = wait_for_completion_timeout(&spi_imx->dma_rx_completion,
|
||||
transfer_timeout);
|
||||
if (!timeout) {
|
||||
dev_err(&master->dev, "I/O Error in DMA RX\n");
|
||||
dev_err(&controller->dev, "I/O Error in DMA RX\n");
|
||||
spi_imx->devtype_data->reset(spi_imx);
|
||||
dmaengine_terminate_all(master->dma_rx);
|
||||
dmaengine_terminate_all(controller->dma_rx);
|
||||
return -ETIMEDOUT;
|
||||
}
|
||||
|
||||
return transfer->len;
|
||||
return 0;
|
||||
/* fallback to pio */
|
||||
dma_failure_no_start:
|
||||
transfer->error |= SPI_TRANS_FAIL_NO_START;
|
||||
@ -1442,7 +1469,7 @@ dma_failure_no_start:
|
||||
static int spi_imx_pio_transfer(struct spi_device *spi,
|
||||
struct spi_transfer *transfer)
|
||||
{
|
||||
struct spi_imx_data *spi_imx = spi_master_get_devdata(spi->master);
|
||||
struct spi_imx_data *spi_imx = spi_controller_get_devdata(spi->controller);
|
||||
unsigned long transfer_timeout;
|
||||
unsigned long timeout;
|
||||
|
||||
@ -1468,14 +1495,62 @@ static int spi_imx_pio_transfer(struct spi_device *spi,
|
||||
return -ETIMEDOUT;
|
||||
}
|
||||
|
||||
return transfer->len;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int spi_imx_poll_transfer(struct spi_device *spi,
|
||||
struct spi_transfer *transfer)
|
||||
{
|
||||
struct spi_imx_data *spi_imx = spi_controller_get_devdata(spi->controller);
|
||||
unsigned long timeout;
|
||||
|
||||
spi_imx->tx_buf = transfer->tx_buf;
|
||||
spi_imx->rx_buf = transfer->rx_buf;
|
||||
spi_imx->count = transfer->len;
|
||||
spi_imx->txfifo = 0;
|
||||
spi_imx->remainder = 0;
|
||||
|
||||
/* fill in the fifo before timeout calculations if we are
|
||||
* interrupted here, then the data is getting transferred by
|
||||
* the HW while we are interrupted
|
||||
*/
|
||||
spi_imx_push(spi_imx);
|
||||
|
||||
timeout = spi_imx_calculate_timeout(spi_imx, transfer->len) + jiffies;
|
||||
while (spi_imx->txfifo) {
|
||||
/* RX */
|
||||
while (spi_imx->txfifo &&
|
||||
spi_imx->devtype_data->rx_available(spi_imx)) {
|
||||
spi_imx->rx(spi_imx);
|
||||
spi_imx->txfifo--;
|
||||
}
|
||||
|
||||
/* TX */
|
||||
if (spi_imx->count) {
|
||||
spi_imx_push(spi_imx);
|
||||
continue;
|
||||
}
|
||||
|
||||
if (spi_imx->txfifo &&
|
||||
time_after(jiffies, timeout)) {
|
||||
|
||||
dev_err_ratelimited(&spi->dev,
|
||||
"timeout period reached: jiffies: %lu- falling back to interrupt mode\n",
|
||||
jiffies - timeout);
|
||||
|
||||
/* fall back to interrupt mode */
|
||||
return spi_imx_pio_transfer(spi, transfer);
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int spi_imx_pio_transfer_slave(struct spi_device *spi,
|
||||
struct spi_transfer *transfer)
|
||||
{
|
||||
struct spi_imx_data *spi_imx = spi_master_get_devdata(spi->master);
|
||||
int ret = transfer->len;
|
||||
struct spi_imx_data *spi_imx = spi_controller_get_devdata(spi->controller);
|
||||
int ret = 0;
|
||||
|
||||
if (is_imx53_ecspi(spi_imx) &&
|
||||
transfer->len > MX53_MAX_TRANSFER_BYTES) {
|
||||
@ -1515,11 +1590,14 @@ static int spi_imx_pio_transfer_slave(struct spi_device *spi,
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int spi_imx_transfer(struct spi_device *spi,
|
||||
static int spi_imx_transfer_one(struct spi_controller *controller,
|
||||
struct spi_device *spi,
|
||||
struct spi_transfer *transfer)
|
||||
{
|
||||
struct spi_imx_data *spi_imx = spi_master_get_devdata(spi->master);
|
||||
struct spi_imx_data *spi_imx = spi_controller_get_devdata(spi->controller);
|
||||
unsigned long hz_per_byte, byte_limit;
|
||||
|
||||
spi_imx_setupxfer(spi, transfer);
|
||||
transfer->effective_speed_hz = spi_imx->spi_bus_clk;
|
||||
|
||||
/* flush rxfifo before transfer */
|
||||
@ -1529,6 +1607,17 @@ static int spi_imx_transfer(struct spi_device *spi,
|
||||
if (spi_imx->slave_mode)
|
||||
return spi_imx_pio_transfer_slave(spi, transfer);
|
||||
|
||||
/*
|
||||
* Calculate the estimated time in us the transfer runs. Find
|
||||
* the number of Hz per byte per polling limit.
|
||||
*/
|
||||
hz_per_byte = polling_limit_us ? ((8 + 4) * USEC_PER_SEC) / polling_limit_us : 0;
|
||||
byte_limit = hz_per_byte ? transfer->effective_speed_hz / hz_per_byte : 1;
|
||||
|
||||
/* run in polling mode for short transfers */
|
||||
if (transfer->len < byte_limit)
|
||||
return spi_imx_poll_transfer(spi, transfer);
|
||||
|
||||
if (spi_imx->usedma)
|
||||
return spi_imx_dma_transfer(spi_imx, transfer);
|
||||
|
||||
@ -1548,14 +1637,13 @@ static void spi_imx_cleanup(struct spi_device *spi)
|
||||
}
|
||||
|
||||
static int
|
||||
spi_imx_prepare_message(struct spi_master *master, struct spi_message *msg)
|
||||
spi_imx_prepare_message(struct spi_controller *controller, struct spi_message *msg)
|
||||
{
|
||||
struct spi_imx_data *spi_imx = spi_master_get_devdata(master);
|
||||
struct spi_imx_data *spi_imx = spi_controller_get_devdata(controller);
|
||||
int ret;
|
||||
|
||||
ret = pm_runtime_get_sync(spi_imx->dev);
|
||||
ret = pm_runtime_resume_and_get(spi_imx->dev);
|
||||
if (ret < 0) {
|
||||
pm_runtime_put_noidle(spi_imx->dev);
|
||||
dev_err(spi_imx->dev, "failed to enable clock\n");
|
||||
return ret;
|
||||
}
|
||||
@ -1570,18 +1658,18 @@ spi_imx_prepare_message(struct spi_master *master, struct spi_message *msg)
|
||||
}
|
||||
|
||||
static int
|
||||
spi_imx_unprepare_message(struct spi_master *master, struct spi_message *msg)
|
||||
spi_imx_unprepare_message(struct spi_controller *controller, struct spi_message *msg)
|
||||
{
|
||||
struct spi_imx_data *spi_imx = spi_master_get_devdata(master);
|
||||
struct spi_imx_data *spi_imx = spi_controller_get_devdata(controller);
|
||||
|
||||
pm_runtime_mark_last_busy(spi_imx->dev);
|
||||
pm_runtime_put_autosuspend(spi_imx->dev);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int spi_imx_slave_abort(struct spi_master *master)
|
||||
static int spi_imx_slave_abort(struct spi_controller *controller)
|
||||
{
|
||||
struct spi_imx_data *spi_imx = spi_master_get_devdata(master);
|
||||
struct spi_imx_data *spi_imx = spi_controller_get_devdata(controller);
|
||||
|
||||
spi_imx->slave_aborted = true;
|
||||
complete(&spi_imx->xfer_done);
|
||||
@ -1592,7 +1680,7 @@ static int spi_imx_slave_abort(struct spi_master *master)
|
||||
static int spi_imx_probe(struct platform_device *pdev)
|
||||
{
|
||||
struct device_node *np = pdev->dev.of_node;
|
||||
struct spi_master *master;
|
||||
struct spi_controller *controller;
|
||||
struct spi_imx_data *spi_imx;
|
||||
struct resource *res;
|
||||
int ret, irq, spi_drctl;
|
||||
@ -1604,12 +1692,12 @@ static int spi_imx_probe(struct platform_device *pdev)
|
||||
slave_mode = devtype_data->has_slavemode &&
|
||||
of_property_read_bool(np, "spi-slave");
|
||||
if (slave_mode)
|
||||
master = spi_alloc_slave(&pdev->dev,
|
||||
sizeof(struct spi_imx_data));
|
||||
controller = spi_alloc_slave(&pdev->dev,
|
||||
sizeof(struct spi_imx_data));
|
||||
else
|
||||
master = spi_alloc_master(&pdev->dev,
|
||||
sizeof(struct spi_imx_data));
|
||||
if (!master)
|
||||
controller = spi_alloc_master(&pdev->dev,
|
||||
sizeof(struct spi_imx_data));
|
||||
if (!controller)
|
||||
return -ENOMEM;
|
||||
|
||||
ret = of_property_read_u32(np, "fsl,spi-rdy-drctl", &spi_drctl);
|
||||
@ -1618,14 +1706,14 @@ static int spi_imx_probe(struct platform_device *pdev)
|
||||
spi_drctl = 0;
|
||||
}
|
||||
|
||||
platform_set_drvdata(pdev, master);
|
||||
platform_set_drvdata(pdev, controller);
|
||||
|
||||
master->bits_per_word_mask = SPI_BPW_RANGE_MASK(1, 32);
|
||||
master->bus_num = np ? -1 : pdev->id;
|
||||
master->use_gpio_descriptors = true;
|
||||
controller->bits_per_word_mask = SPI_BPW_RANGE_MASK(1, 32);
|
||||
controller->bus_num = np ? -1 : pdev->id;
|
||||
controller->use_gpio_descriptors = true;
|
||||
|
||||
spi_imx = spi_master_get_devdata(master);
|
||||
spi_imx->bitbang.master = master;
|
||||
spi_imx = spi_controller_get_devdata(controller);
|
||||
spi_imx->controller = controller;
|
||||
spi_imx->dev = &pdev->dev;
|
||||
spi_imx->slave_mode = slave_mode;
|
||||
|
||||
@ -1638,22 +1726,24 @@ static int spi_imx_probe(struct platform_device *pdev)
|
||||
* board files have <= 3 chip selects.
|
||||
*/
|
||||
if (!device_property_read_u32(&pdev->dev, "num-cs", &val))
|
||||
master->num_chipselect = val;
|
||||
controller->num_chipselect = val;
|
||||
else
|
||||
master->num_chipselect = 3;
|
||||
controller->num_chipselect = 3;
|
||||
|
||||
spi_imx->controller->transfer_one = spi_imx_transfer_one;
|
||||
spi_imx->controller->setup = spi_imx_setup;
|
||||
spi_imx->controller->cleanup = spi_imx_cleanup;
|
||||
spi_imx->controller->prepare_message = spi_imx_prepare_message;
|
||||
spi_imx->controller->unprepare_message = spi_imx_unprepare_message;
|
||||
spi_imx->controller->slave_abort = spi_imx_slave_abort;
|
||||
spi_imx->controller->mode_bits = SPI_CPOL | SPI_CPHA | SPI_CS_HIGH | SPI_NO_CS;
|
||||
|
||||
spi_imx->bitbang.setup_transfer = spi_imx_setupxfer;
|
||||
spi_imx->bitbang.txrx_bufs = spi_imx_transfer;
|
||||
spi_imx->bitbang.master->setup = spi_imx_setup;
|
||||
spi_imx->bitbang.master->cleanup = spi_imx_cleanup;
|
||||
spi_imx->bitbang.master->prepare_message = spi_imx_prepare_message;
|
||||
spi_imx->bitbang.master->unprepare_message = spi_imx_unprepare_message;
|
||||
spi_imx->bitbang.master->slave_abort = spi_imx_slave_abort;
|
||||
spi_imx->bitbang.master->mode_bits = SPI_CPOL | SPI_CPHA | SPI_CS_HIGH \
|
||||
| SPI_NO_CS;
|
||||
if (is_imx35_cspi(spi_imx) || is_imx51_ecspi(spi_imx) ||
|
||||
is_imx53_ecspi(spi_imx))
|
||||
spi_imx->bitbang.master->mode_bits |= SPI_LOOP | SPI_READY;
|
||||
spi_imx->controller->mode_bits |= SPI_LOOP | SPI_READY;
|
||||
|
||||
if (is_imx51_ecspi(spi_imx) || is_imx53_ecspi(spi_imx))
|
||||
spi_imx->controller->mode_bits |= SPI_RX_CPHA_FLIP;
|
||||
|
||||
if (is_imx51_ecspi(spi_imx) &&
|
||||
device_property_read_u32(&pdev->dev, "cs-gpios", NULL))
|
||||
@ -1662,7 +1752,7 @@ static int spi_imx_probe(struct platform_device *pdev)
|
||||
* setting the burst length to the word size. This is
|
||||
* considerably faster than manually controlling the CS.
|
||||
*/
|
||||
spi_imx->bitbang.master->mode_bits |= SPI_CS_WORD;
|
||||
spi_imx->controller->mode_bits |= SPI_CS_WORD;
|
||||
|
||||
spi_imx->spi_drctl = spi_drctl;
|
||||
|
||||
@ -1672,38 +1762,38 @@ static int spi_imx_probe(struct platform_device *pdev)
|
||||
spi_imx->base = devm_ioremap_resource(&pdev->dev, res);
|
||||
if (IS_ERR(spi_imx->base)) {
|
||||
ret = PTR_ERR(spi_imx->base);
|
||||
goto out_master_put;
|
||||
goto out_controller_put;
|
||||
}
|
||||
spi_imx->base_phys = res->start;
|
||||
|
||||
irq = platform_get_irq(pdev, 0);
|
||||
if (irq < 0) {
|
||||
ret = irq;
|
||||
goto out_master_put;
|
||||
goto out_controller_put;
|
||||
}
|
||||
|
||||
ret = devm_request_irq(&pdev->dev, irq, spi_imx_isr, 0,
|
||||
dev_name(&pdev->dev), spi_imx);
|
||||
if (ret) {
|
||||
dev_err(&pdev->dev, "can't get irq%d: %d\n", irq, ret);
|
||||
goto out_master_put;
|
||||
goto out_controller_put;
|
||||
}
|
||||
|
||||
spi_imx->clk_ipg = devm_clk_get(&pdev->dev, "ipg");
|
||||
if (IS_ERR(spi_imx->clk_ipg)) {
|
||||
ret = PTR_ERR(spi_imx->clk_ipg);
|
||||
goto out_master_put;
|
||||
goto out_controller_put;
|
||||
}
|
||||
|
||||
spi_imx->clk_per = devm_clk_get(&pdev->dev, "per");
|
||||
if (IS_ERR(spi_imx->clk_per)) {
|
||||
ret = PTR_ERR(spi_imx->clk_per);
|
||||
goto out_master_put;
|
||||
goto out_controller_put;
|
||||
}
|
||||
|
||||
ret = clk_prepare_enable(spi_imx->clk_per);
|
||||
if (ret)
|
||||
goto out_master_put;
|
||||
goto out_controller_put;
|
||||
|
||||
ret = clk_prepare_enable(spi_imx->clk_ipg);
|
||||
if (ret)
|
||||
@ -1721,7 +1811,7 @@ static int spi_imx_probe(struct platform_device *pdev)
|
||||
* if validated on other chips.
|
||||
*/
|
||||
if (spi_imx->devtype_data->has_dmamode) {
|
||||
ret = spi_imx_sdma_init(&pdev->dev, spi_imx, master);
|
||||
ret = spi_imx_sdma_init(&pdev->dev, spi_imx, controller);
|
||||
if (ret == -EPROBE_DEFER)
|
||||
goto out_runtime_pm_put;
|
||||
|
||||
@ -1734,11 +1824,11 @@ static int spi_imx_probe(struct platform_device *pdev)
|
||||
|
||||
spi_imx->devtype_data->intctrl(spi_imx, 0);
|
||||
|
||||
master->dev.of_node = pdev->dev.of_node;
|
||||
ret = spi_bitbang_start(&spi_imx->bitbang);
|
||||
controller->dev.of_node = pdev->dev.of_node;
|
||||
ret = spi_register_controller(controller);
|
||||
if (ret) {
|
||||
dev_err_probe(&pdev->dev, ret, "bitbang start failed\n");
|
||||
goto out_bitbang_start;
|
||||
dev_err_probe(&pdev->dev, ret, "register controller failed\n");
|
||||
goto out_register_controller;
|
||||
}
|
||||
|
||||
pm_runtime_mark_last_busy(spi_imx->dev);
|
||||
@ -1746,7 +1836,7 @@ static int spi_imx_probe(struct platform_device *pdev)
|
||||
|
||||
return ret;
|
||||
|
||||
out_bitbang_start:
|
||||
out_register_controller:
|
||||
if (spi_imx->devtype_data->has_dmamode)
|
||||
spi_imx_sdma_exit(spi_imx);
|
||||
out_runtime_pm_put:
|
||||
@ -1757,23 +1847,22 @@ out_runtime_pm_put:
|
||||
clk_disable_unprepare(spi_imx->clk_ipg);
|
||||
out_put_per:
|
||||
clk_disable_unprepare(spi_imx->clk_per);
|
||||
out_master_put:
|
||||
spi_master_put(master);
|
||||
out_controller_put:
|
||||
spi_controller_put(controller);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int spi_imx_remove(struct platform_device *pdev)
|
||||
{
|
||||
struct spi_master *master = platform_get_drvdata(pdev);
|
||||
struct spi_imx_data *spi_imx = spi_master_get_devdata(master);
|
||||
struct spi_controller *controller = platform_get_drvdata(pdev);
|
||||
struct spi_imx_data *spi_imx = spi_controller_get_devdata(controller);
|
||||
int ret;
|
||||
|
||||
spi_bitbang_stop(&spi_imx->bitbang);
|
||||
spi_unregister_controller(controller);
|
||||
|
||||
ret = pm_runtime_get_sync(spi_imx->dev);
|
||||
ret = pm_runtime_resume_and_get(spi_imx->dev);
|
||||
if (ret < 0) {
|
||||
pm_runtime_put_noidle(spi_imx->dev);
|
||||
dev_err(spi_imx->dev, "failed to enable clock\n");
|
||||
return ret;
|
||||
}
|
||||
@ -1785,18 +1874,17 @@ static int spi_imx_remove(struct platform_device *pdev)
|
||||
pm_runtime_disable(spi_imx->dev);
|
||||
|
||||
spi_imx_sdma_exit(spi_imx);
|
||||
spi_master_put(master);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int __maybe_unused spi_imx_runtime_resume(struct device *dev)
|
||||
{
|
||||
struct spi_master *master = dev_get_drvdata(dev);
|
||||
struct spi_controller *controller = dev_get_drvdata(dev);
|
||||
struct spi_imx_data *spi_imx;
|
||||
int ret;
|
||||
|
||||
spi_imx = spi_master_get_devdata(master);
|
||||
spi_imx = spi_controller_get_devdata(controller);
|
||||
|
||||
ret = clk_prepare_enable(spi_imx->clk_per);
|
||||
if (ret)
|
||||
@ -1813,10 +1901,10 @@ static int __maybe_unused spi_imx_runtime_resume(struct device *dev)
|
||||
|
||||
static int __maybe_unused spi_imx_runtime_suspend(struct device *dev)
|
||||
{
|
||||
struct spi_master *master = dev_get_drvdata(dev);
|
||||
struct spi_controller *controller = dev_get_drvdata(dev);
|
||||
struct spi_imx_data *spi_imx;
|
||||
|
||||
spi_imx = spi_master_get_devdata(master);
|
||||
spi_imx = spi_controller_get_devdata(controller);
|
||||
|
||||
clk_disable_unprepare(spi_imx->clk_per);
|
||||
clk_disable_unprepare(spi_imx->clk_ipg);
|
||||
|
@ -1,8 +1,9 @@
|
||||
// SPDX-License-Identifier: GPL-2.0
|
||||
/*
|
||||
* SPI bus driver for the Ingenic JZ47xx SoCs
|
||||
* SPI bus driver for the Ingenic SoCs
|
||||
* Copyright (c) 2017-2021 Artur Rojek <contact@artur-rojek.eu>
|
||||
* Copyright (c) 2017-2021 Paul Cercueil <paul@crapouillou.net>
|
||||
* Copyright (c) 2022 周琰杰 (Zhou Yanjie) <zhouyanjie@wanyeetech.com>
|
||||
*/
|
||||
|
||||
#include <linux/clk.h>
|
||||
@ -52,6 +53,9 @@ struct jz_soc_info {
|
||||
u32 bits_per_word_mask;
|
||||
struct reg_field flen_field;
|
||||
bool has_trendian;
|
||||
|
||||
unsigned int max_speed_hz;
|
||||
unsigned int max_native_cs;
|
||||
};
|
||||
|
||||
struct ingenic_spi {
|
||||
@ -380,7 +384,7 @@ static int spi_ingenic_probe(struct platform_device *pdev)
|
||||
struct spi_controller *ctlr;
|
||||
struct ingenic_spi *priv;
|
||||
void __iomem *base;
|
||||
int ret;
|
||||
int num_cs, ret;
|
||||
|
||||
pdata = of_device_get_match_data(dev);
|
||||
if (!pdata) {
|
||||
@ -416,6 +420,9 @@ static int spi_ingenic_probe(struct platform_device *pdev)
|
||||
if (IS_ERR(priv->flen_field))
|
||||
return PTR_ERR(priv->flen_field);
|
||||
|
||||
if (device_property_read_u32(dev, "num-cs", &num_cs))
|
||||
num_cs = pdata->max_native_cs;
|
||||
|
||||
platform_set_drvdata(pdev, ctlr);
|
||||
|
||||
ctlr->prepare_transfer_hardware = spi_ingenic_prepare_hardware;
|
||||
@ -428,8 +435,10 @@ static int spi_ingenic_probe(struct platform_device *pdev)
|
||||
ctlr->max_dma_len = SPI_INGENIC_FIFO_SIZE;
|
||||
ctlr->bits_per_word_mask = pdata->bits_per_word_mask;
|
||||
ctlr->min_speed_hz = 7200;
|
||||
ctlr->max_speed_hz = 54000000;
|
||||
ctlr->num_chipselect = 2;
|
||||
ctlr->max_speed_hz = pdata->max_speed_hz;
|
||||
ctlr->use_gpio_descriptors = true;
|
||||
ctlr->max_native_cs = pdata->max_native_cs;
|
||||
ctlr->num_chipselect = num_cs;
|
||||
ctlr->dev.of_node = pdev->dev.of_node;
|
||||
|
||||
if (spi_ingenic_request_dma(ctlr, dev))
|
||||
@ -452,17 +461,44 @@ static const struct jz_soc_info jz4750_soc_info = {
|
||||
.bits_per_word_mask = SPI_BPW_RANGE_MASK(2, 17),
|
||||
.flen_field = REG_FIELD(REG_SSICR1, 4, 7),
|
||||
.has_trendian = false,
|
||||
|
||||
.max_speed_hz = 54000000,
|
||||
.max_native_cs = 2,
|
||||
};
|
||||
|
||||
static const struct jz_soc_info jz4780_soc_info = {
|
||||
.bits_per_word_mask = SPI_BPW_RANGE_MASK(2, 32),
|
||||
.flen_field = REG_FIELD(REG_SSICR1, 3, 7),
|
||||
.has_trendian = true,
|
||||
|
||||
.max_speed_hz = 54000000,
|
||||
.max_native_cs = 2,
|
||||
};
|
||||
|
||||
static const struct jz_soc_info x1000_soc_info = {
|
||||
.bits_per_word_mask = SPI_BPW_RANGE_MASK(2, 32),
|
||||
.flen_field = REG_FIELD(REG_SSICR1, 3, 7),
|
||||
.has_trendian = true,
|
||||
|
||||
.max_speed_hz = 50000000,
|
||||
.max_native_cs = 2,
|
||||
};
|
||||
|
||||
static const struct jz_soc_info x2000_soc_info = {
|
||||
.bits_per_word_mask = SPI_BPW_RANGE_MASK(2, 32),
|
||||
.flen_field = REG_FIELD(REG_SSICR1, 3, 7),
|
||||
.has_trendian = true,
|
||||
|
||||
.max_speed_hz = 50000000,
|
||||
.max_native_cs = 1,
|
||||
};
|
||||
|
||||
static const struct of_device_id spi_ingenic_of_match[] = {
|
||||
{ .compatible = "ingenic,jz4750-spi", .data = &jz4750_soc_info },
|
||||
{ .compatible = "ingenic,jz4775-spi", .data = &jz4780_soc_info },
|
||||
{ .compatible = "ingenic,jz4780-spi", .data = &jz4780_soc_info },
|
||||
{ .compatible = "ingenic,x1000-spi", .data = &x1000_soc_info },
|
||||
{ .compatible = "ingenic,x2000-spi", .data = &x2000_soc_info },
|
||||
{}
|
||||
};
|
||||
MODULE_DEVICE_TABLE(of, spi_ingenic_of_match);
|
||||
@ -476,7 +512,8 @@ static struct platform_driver spi_ingenic_driver = {
|
||||
};
|
||||
|
||||
module_platform_driver(spi_ingenic_driver);
|
||||
MODULE_DESCRIPTION("SPI bus driver for the Ingenic JZ47xx SoCs");
|
||||
MODULE_DESCRIPTION("SPI bus driver for the Ingenic SoCs");
|
||||
MODULE_AUTHOR("Artur Rojek <contact@artur-rojek.eu>");
|
||||
MODULE_AUTHOR("Paul Cercueil <paul@crapouillou.net>");
|
||||
MODULE_AUTHOR("周琰杰 (Zhou Yanjie) <zhouyanjie@wanyeetech.com>");
|
||||
MODULE_LICENSE("GPL");
|
||||
|
@ -779,10 +779,59 @@ static const char *intel_spi_get_name(struct spi_mem *mem)
|
||||
return dev_name(ispi->dev);
|
||||
}
|
||||
|
||||
static int intel_spi_dirmap_create(struct spi_mem_dirmap_desc *desc)
|
||||
{
|
||||
struct intel_spi *ispi = spi_master_get_devdata(desc->mem->spi->master);
|
||||
const struct intel_spi_mem_op *iop;
|
||||
|
||||
iop = intel_spi_match_mem_op(ispi, &desc->info.op_tmpl);
|
||||
if (!iop)
|
||||
return -EOPNOTSUPP;
|
||||
|
||||
desc->priv = (void *)iop;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static ssize_t intel_spi_dirmap_read(struct spi_mem_dirmap_desc *desc, u64 offs,
|
||||
size_t len, void *buf)
|
||||
{
|
||||
struct intel_spi *ispi = spi_master_get_devdata(desc->mem->spi->master);
|
||||
const struct intel_spi_mem_op *iop = desc->priv;
|
||||
struct spi_mem_op op = desc->info.op_tmpl;
|
||||
int ret;
|
||||
|
||||
/* Fill in the gaps */
|
||||
op.addr.val = offs;
|
||||
op.data.nbytes = len;
|
||||
op.data.buf.in = buf;
|
||||
|
||||
ret = iop->exec_op(ispi, iop, &op);
|
||||
return ret ? ret : len;
|
||||
}
|
||||
|
||||
static ssize_t intel_spi_dirmap_write(struct spi_mem_dirmap_desc *desc, u64 offs,
|
||||
size_t len, const void *buf)
|
||||
{
|
||||
struct intel_spi *ispi = spi_master_get_devdata(desc->mem->spi->master);
|
||||
const struct intel_spi_mem_op *iop = desc->priv;
|
||||
struct spi_mem_op op = desc->info.op_tmpl;
|
||||
int ret;
|
||||
|
||||
op.addr.val = offs;
|
||||
op.data.nbytes = len;
|
||||
op.data.buf.out = buf;
|
||||
|
||||
ret = iop->exec_op(ispi, iop, &op);
|
||||
return ret ? ret : len;
|
||||
}
|
||||
|
||||
static const struct spi_controller_mem_ops intel_spi_mem_ops = {
|
||||
.supports_op = intel_spi_supports_mem_op,
|
||||
.exec_op = intel_spi_exec_mem_op,
|
||||
.get_name = intel_spi_get_name,
|
||||
.dirmap_create = intel_spi_dirmap_create,
|
||||
.dirmap_read = intel_spi_dirmap_read,
|
||||
.dirmap_write = intel_spi_dirmap_write,
|
||||
};
|
||||
|
||||
#define INTEL_SPI_OP_ADDR(__nbytes) \
|
||||
@ -1205,7 +1254,7 @@ static int intel_spi_populate_chip(struct intel_spi *ispi)
|
||||
* intel_spi_probe() - Probe the Intel SPI flash controller
|
||||
* @dev: Pointer to the parent device
|
||||
* @mem: MMIO resource
|
||||
* @info: Platform spefific information
|
||||
* @info: Platform specific information
|
||||
*
|
||||
* Probes Intel SPI flash controller and creates the flash chip device.
|
||||
* Returns %0 on success and negative errno in case of failure.
|
||||
|
@ -10,6 +10,7 @@
|
||||
#include <linux/pm_runtime.h>
|
||||
#include <linux/spi/spi.h>
|
||||
#include <linux/spi/spi-mem.h>
|
||||
#include <linux/sched/task_stack.h>
|
||||
|
||||
#include "internals.h"
|
||||
|
||||
@ -211,6 +212,15 @@ static int spi_mem_check_op(const struct spi_mem_op *op)
|
||||
!spi_mem_buswidth_is_valid(op->data.buswidth))
|
||||
return -EINVAL;
|
||||
|
||||
/* Buffers must be DMA-able. */
|
||||
if (WARN_ON_ONCE(op->data.dir == SPI_MEM_DATA_IN &&
|
||||
object_is_on_stack(op->data.buf.in)))
|
||||
return -EINVAL;
|
||||
|
||||
if (WARN_ON_ONCE(op->data.dir == SPI_MEM_DATA_OUT &&
|
||||
object_is_on_stack(op->data.buf.out)))
|
||||
return -EINVAL;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
@ -262,9 +272,8 @@ static int spi_mem_access_start(struct spi_mem *mem)
|
||||
if (ctlr->auto_runtime_pm) {
|
||||
int ret;
|
||||
|
||||
ret = pm_runtime_get_sync(ctlr->dev.parent);
|
||||
ret = pm_runtime_resume_and_get(ctlr->dev.parent);
|
||||
if (ret < 0) {
|
||||
pm_runtime_put_noidle(ctlr->dev.parent);
|
||||
dev_err(&ctlr->dev, "Failed to power device: %d\n",
|
||||
ret);
|
||||
return ret;
|
||||
|
@ -20,6 +20,7 @@
|
||||
#include <linux/spi/spi.h>
|
||||
#include <linux/fsl_devices.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/of_irq.h>
|
||||
|
||||
#include <asm/mpc52xx.h>
|
||||
#include <asm/mpc52xx_psc.h>
|
||||
|
@ -19,6 +19,9 @@
|
||||
#include <linux/io.h>
|
||||
#include <linux/of_gpio.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/of_address.h>
|
||||
#include <linux/of_irq.h>
|
||||
|
||||
#include <asm/time.h>
|
||||
#include <asm/mpc52xx.h>
|
||||
|
||||
|
@ -17,105 +17,148 @@
|
||||
#include <linux/platform_data/spi-mt65xx.h>
|
||||
#include <linux/pm_runtime.h>
|
||||
#include <linux/spi/spi.h>
|
||||
#include <linux/spi/spi-mem.h>
|
||||
#include <linux/dma-mapping.h>
|
||||
|
||||
#define SPI_CFG0_REG 0x0000
|
||||
#define SPI_CFG1_REG 0x0004
|
||||
#define SPI_TX_SRC_REG 0x0008
|
||||
#define SPI_RX_DST_REG 0x000c
|
||||
#define SPI_TX_DATA_REG 0x0010
|
||||
#define SPI_RX_DATA_REG 0x0014
|
||||
#define SPI_CMD_REG 0x0018
|
||||
#define SPI_STATUS0_REG 0x001c
|
||||
#define SPI_PAD_SEL_REG 0x0024
|
||||
#define SPI_CFG2_REG 0x0028
|
||||
#define SPI_TX_SRC_REG_64 0x002c
|
||||
#define SPI_RX_DST_REG_64 0x0030
|
||||
#define SPI_CFG3_IPM_REG 0x0040
|
||||
#define SPI_CFG0_REG 0x0000
|
||||
#define SPI_CFG1_REG 0x0004
|
||||
#define SPI_TX_SRC_REG 0x0008
|
||||
#define SPI_RX_DST_REG 0x000c
|
||||
#define SPI_TX_DATA_REG 0x0010
|
||||
#define SPI_RX_DATA_REG 0x0014
|
||||
#define SPI_CMD_REG 0x0018
|
||||
#define SPI_STATUS0_REG 0x001c
|
||||
#define SPI_PAD_SEL_REG 0x0024
|
||||
#define SPI_CFG2_REG 0x0028
|
||||
#define SPI_TX_SRC_REG_64 0x002c
|
||||
#define SPI_RX_DST_REG_64 0x0030
|
||||
#define SPI_CFG3_IPM_REG 0x0040
|
||||
|
||||
#define SPI_CFG0_SCK_HIGH_OFFSET 0
|
||||
#define SPI_CFG0_SCK_LOW_OFFSET 8
|
||||
#define SPI_CFG0_CS_HOLD_OFFSET 16
|
||||
#define SPI_CFG0_CS_SETUP_OFFSET 24
|
||||
#define SPI_ADJUST_CFG0_CS_HOLD_OFFSET 0
|
||||
#define SPI_ADJUST_CFG0_CS_SETUP_OFFSET 16
|
||||
#define SPI_CFG0_SCK_HIGH_OFFSET 0
|
||||
#define SPI_CFG0_SCK_LOW_OFFSET 8
|
||||
#define SPI_CFG0_CS_HOLD_OFFSET 16
|
||||
#define SPI_CFG0_CS_SETUP_OFFSET 24
|
||||
#define SPI_ADJUST_CFG0_CS_HOLD_OFFSET 0
|
||||
#define SPI_ADJUST_CFG0_CS_SETUP_OFFSET 16
|
||||
|
||||
#define SPI_CFG1_CS_IDLE_OFFSET 0
|
||||
#define SPI_CFG1_PACKET_LOOP_OFFSET 8
|
||||
#define SPI_CFG1_PACKET_LENGTH_OFFSET 16
|
||||
#define SPI_CFG1_GET_TICK_DLY_OFFSET 29
|
||||
#define SPI_CFG1_GET_TICK_DLY_OFFSET_V1 30
|
||||
#define SPI_CFG1_CS_IDLE_OFFSET 0
|
||||
#define SPI_CFG1_PACKET_LOOP_OFFSET 8
|
||||
#define SPI_CFG1_PACKET_LENGTH_OFFSET 16
|
||||
#define SPI_CFG1_GET_TICK_DLY_OFFSET 29
|
||||
#define SPI_CFG1_GET_TICK_DLY_OFFSET_V1 30
|
||||
|
||||
#define SPI_CFG1_GET_TICK_DLY_MASK 0xe0000000
|
||||
#define SPI_CFG1_GET_TICK_DLY_MASK_V1 0xc0000000
|
||||
#define SPI_CFG1_GET_TICK_DLY_MASK 0xe0000000
|
||||
#define SPI_CFG1_GET_TICK_DLY_MASK_V1 0xc0000000
|
||||
|
||||
#define SPI_CFG1_CS_IDLE_MASK 0xff
|
||||
#define SPI_CFG1_PACKET_LOOP_MASK 0xff00
|
||||
#define SPI_CFG1_PACKET_LENGTH_MASK 0x3ff0000
|
||||
#define SPI_CFG1_IPM_PACKET_LENGTH_MASK GENMASK(31, 16)
|
||||
#define SPI_CFG2_SCK_HIGH_OFFSET 0
|
||||
#define SPI_CFG2_SCK_LOW_OFFSET 16
|
||||
#define SPI_CFG1_CS_IDLE_MASK 0xff
|
||||
#define SPI_CFG1_PACKET_LOOP_MASK 0xff00
|
||||
#define SPI_CFG1_PACKET_LENGTH_MASK 0x3ff0000
|
||||
#define SPI_CFG1_IPM_PACKET_LENGTH_MASK GENMASK(31, 16)
|
||||
#define SPI_CFG2_SCK_HIGH_OFFSET 0
|
||||
#define SPI_CFG2_SCK_LOW_OFFSET 16
|
||||
|
||||
#define SPI_CMD_ACT BIT(0)
|
||||
#define SPI_CMD_RESUME BIT(1)
|
||||
#define SPI_CMD_RST BIT(2)
|
||||
#define SPI_CMD_PAUSE_EN BIT(4)
|
||||
#define SPI_CMD_DEASSERT BIT(5)
|
||||
#define SPI_CMD_SAMPLE_SEL BIT(6)
|
||||
#define SPI_CMD_CS_POL BIT(7)
|
||||
#define SPI_CMD_CPHA BIT(8)
|
||||
#define SPI_CMD_CPOL BIT(9)
|
||||
#define SPI_CMD_RX_DMA BIT(10)
|
||||
#define SPI_CMD_TX_DMA BIT(11)
|
||||
#define SPI_CMD_TXMSBF BIT(12)
|
||||
#define SPI_CMD_RXMSBF BIT(13)
|
||||
#define SPI_CMD_RX_ENDIAN BIT(14)
|
||||
#define SPI_CMD_TX_ENDIAN BIT(15)
|
||||
#define SPI_CMD_FINISH_IE BIT(16)
|
||||
#define SPI_CMD_PAUSE_IE BIT(17)
|
||||
#define SPI_CMD_IPM_NONIDLE_MODE BIT(19)
|
||||
#define SPI_CMD_IPM_SPIM_LOOP BIT(21)
|
||||
#define SPI_CMD_IPM_GET_TICKDLY_OFFSET 22
|
||||
#define SPI_CMD_ACT BIT(0)
|
||||
#define SPI_CMD_RESUME BIT(1)
|
||||
#define SPI_CMD_RST BIT(2)
|
||||
#define SPI_CMD_PAUSE_EN BIT(4)
|
||||
#define SPI_CMD_DEASSERT BIT(5)
|
||||
#define SPI_CMD_SAMPLE_SEL BIT(6)
|
||||
#define SPI_CMD_CS_POL BIT(7)
|
||||
#define SPI_CMD_CPHA BIT(8)
|
||||
#define SPI_CMD_CPOL BIT(9)
|
||||
#define SPI_CMD_RX_DMA BIT(10)
|
||||
#define SPI_CMD_TX_DMA BIT(11)
|
||||
#define SPI_CMD_TXMSBF BIT(12)
|
||||
#define SPI_CMD_RXMSBF BIT(13)
|
||||
#define SPI_CMD_RX_ENDIAN BIT(14)
|
||||
#define SPI_CMD_TX_ENDIAN BIT(15)
|
||||
#define SPI_CMD_FINISH_IE BIT(16)
|
||||
#define SPI_CMD_PAUSE_IE BIT(17)
|
||||
#define SPI_CMD_IPM_NONIDLE_MODE BIT(19)
|
||||
#define SPI_CMD_IPM_SPIM_LOOP BIT(21)
|
||||
#define SPI_CMD_IPM_GET_TICKDLY_OFFSET 22
|
||||
|
||||
#define SPI_CMD_IPM_GET_TICKDLY_MASK GENMASK(24, 22)
|
||||
#define SPI_CFG3_IPM_HALF_DUPLEX_DIR BIT(2)
|
||||
#define SPI_CFG3_IPM_HALF_DUPLEX_EN BIT(3)
|
||||
#define MT8173_SPI_MAX_PAD_SEL 3
|
||||
|
||||
#define MTK_SPI_PAUSE_INT_STATUS 0x2
|
||||
#define PIN_MODE_CFG(x) ((x) / 2)
|
||||
|
||||
#define MTK_SPI_IDLE 0
|
||||
#define MTK_SPI_PAUSED 1
|
||||
#define SPI_CFG3_IPM_HALF_DUPLEX_DIR BIT(2)
|
||||
#define SPI_CFG3_IPM_HALF_DUPLEX_EN BIT(3)
|
||||
#define SPI_CFG3_IPM_XMODE_EN BIT(4)
|
||||
#define SPI_CFG3_IPM_NODATA_FLAG BIT(5)
|
||||
#define SPI_CFG3_IPM_CMD_BYTELEN_OFFSET 8
|
||||
#define SPI_CFG3_IPM_ADDR_BYTELEN_OFFSET 12
|
||||
|
||||
#define MTK_SPI_MAX_FIFO_SIZE 32U
|
||||
#define MTK_SPI_PACKET_SIZE 1024
|
||||
#define MTK_SPI_IPM_PACKET_SIZE SZ_64K
|
||||
#define MTK_SPI_32BITS_MASK (0xffffffff)
|
||||
#define SPI_CFG3_IPM_CMD_PIN_MODE_MASK GENMASK(1, 0)
|
||||
#define SPI_CFG3_IPM_CMD_BYTELEN_MASK GENMASK(11, 8)
|
||||
#define SPI_CFG3_IPM_ADDR_BYTELEN_MASK GENMASK(15, 12)
|
||||
|
||||
#define DMA_ADDR_EXT_BITS (36)
|
||||
#define DMA_ADDR_DEF_BITS (32)
|
||||
#define MT8173_SPI_MAX_PAD_SEL 3
|
||||
|
||||
#define MTK_SPI_PAUSE_INT_STATUS 0x2
|
||||
|
||||
#define MTK_SPI_MAX_FIFO_SIZE 32U
|
||||
#define MTK_SPI_PACKET_SIZE 1024
|
||||
#define MTK_SPI_IPM_PACKET_SIZE SZ_64K
|
||||
#define MTK_SPI_IPM_PACKET_LOOP SZ_256
|
||||
|
||||
#define MTK_SPI_IDLE 0
|
||||
#define MTK_SPI_PAUSED 1
|
||||
|
||||
#define MTK_SPI_32BITS_MASK (0xffffffff)
|
||||
|
||||
#define DMA_ADDR_EXT_BITS (36)
|
||||
#define DMA_ADDR_DEF_BITS (32)
|
||||
|
||||
/**
|
||||
* struct mtk_spi_compatible - device data structure
|
||||
* @need_pad_sel: Enable pad (pins) selection in SPI controller
|
||||
* @must_tx: Must explicitly send dummy TX bytes to do RX only transfer
|
||||
* @enhance_timing: Enable adjusting cfg register to enhance time accuracy
|
||||
* @dma_ext: DMA address extension supported
|
||||
* @no_need_unprepare: Don't unprepare the SPI clk during runtime
|
||||
* @ipm_design: Adjust/extend registers to support IPM design IP features
|
||||
*/
|
||||
struct mtk_spi_compatible {
|
||||
bool need_pad_sel;
|
||||
/* Must explicitly send dummy Tx bytes to do Rx only transfer */
|
||||
bool must_tx;
|
||||
/* some IC design adjust cfg register to enhance time accuracy */
|
||||
bool enhance_timing;
|
||||
/* some IC support DMA addr extension */
|
||||
bool dma_ext;
|
||||
/* some IC no need unprepare SPI clk */
|
||||
bool no_need_unprepare;
|
||||
/* IPM design adjust and extend register to support more features */
|
||||
bool ipm_design;
|
||||
|
||||
};
|
||||
|
||||
/**
|
||||
* struct mtk_spi - SPI driver instance
|
||||
* @base: Start address of the SPI controller registers
|
||||
* @state: SPI controller state
|
||||
* @pad_num: Number of pad_sel entries
|
||||
* @pad_sel: Groups of pins to select
|
||||
* @parent_clk: Parent of sel_clk
|
||||
* @sel_clk: SPI master mux clock
|
||||
* @spi_clk: Peripheral clock
|
||||
* @spi_hclk: AHB bus clock
|
||||
* @cur_transfer: Currently processed SPI transfer
|
||||
* @xfer_len: Number of bytes to transfer
|
||||
* @num_xfered: Number of transferred bytes
|
||||
* @tx_sgl: TX transfer scatterlist
|
||||
* @rx_sgl: RX transfer scatterlist
|
||||
* @tx_sgl_len: Size of TX DMA transfer
|
||||
* @rx_sgl_len: Size of RX DMA transfer
|
||||
* @dev_comp: Device data structure
|
||||
* @spi_clk_hz: Current SPI clock in Hz
|
||||
* @spimem_done: SPI-MEM operation completion
|
||||
* @use_spimem: Enables SPI-MEM
|
||||
* @dev: Device pointer
|
||||
* @tx_dma: DMA start for SPI-MEM TX
|
||||
* @rx_dma: DMA start for SPI-MEM RX
|
||||
*/
|
||||
struct mtk_spi {
|
||||
void __iomem *base;
|
||||
u32 state;
|
||||
int pad_num;
|
||||
u32 *pad_sel;
|
||||
struct clk *parent_clk, *sel_clk, *spi_clk;
|
||||
struct clk *parent_clk, *sel_clk, *spi_clk, *spi_hclk;
|
||||
struct spi_transfer *cur_transfer;
|
||||
u32 xfer_len;
|
||||
u32 num_xfered;
|
||||
@ -123,6 +166,11 @@ struct mtk_spi {
|
||||
u32 tx_sgl_len, rx_sgl_len;
|
||||
const struct mtk_spi_compatible *dev_comp;
|
||||
u32 spi_clk_hz;
|
||||
struct completion spimem_done;
|
||||
bool use_spimem;
|
||||
struct device *dev;
|
||||
dma_addr_t tx_dma;
|
||||
dma_addr_t rx_dma;
|
||||
};
|
||||
|
||||
static const struct mtk_spi_compatible mtk_common_compat;
|
||||
@ -704,6 +752,12 @@ static irqreturn_t mtk_spi_interrupt(int irq, void *dev_id)
|
||||
else
|
||||
mdata->state = MTK_SPI_IDLE;
|
||||
|
||||
/* SPI-MEM ops */
|
||||
if (mdata->use_spimem) {
|
||||
complete(&mdata->spimem_done);
|
||||
return IRQ_HANDLED;
|
||||
}
|
||||
|
||||
if (!master->can_dma(master, NULL, trans)) {
|
||||
if (trans->rx_buf) {
|
||||
cnt = mdata->xfer_len / 4;
|
||||
@ -787,21 +841,287 @@ static irqreturn_t mtk_spi_interrupt(int irq, void *dev_id)
|
||||
return IRQ_HANDLED;
|
||||
}
|
||||
|
||||
static int mtk_spi_probe(struct platform_device *pdev)
|
||||
static int mtk_spi_mem_adjust_op_size(struct spi_mem *mem,
|
||||
struct spi_mem_op *op)
|
||||
{
|
||||
struct spi_master *master;
|
||||
struct mtk_spi *mdata;
|
||||
const struct of_device_id *of_id;
|
||||
int i, irq, ret, addr_bits;
|
||||
int opcode_len;
|
||||
|
||||
master = spi_alloc_master(&pdev->dev, sizeof(*mdata));
|
||||
if (!master) {
|
||||
dev_err(&pdev->dev, "failed to alloc spi master\n");
|
||||
if (op->data.dir != SPI_MEM_NO_DATA) {
|
||||
opcode_len = 1 + op->addr.nbytes + op->dummy.nbytes;
|
||||
if (opcode_len + op->data.nbytes > MTK_SPI_IPM_PACKET_SIZE) {
|
||||
op->data.nbytes = MTK_SPI_IPM_PACKET_SIZE - opcode_len;
|
||||
/* force data buffer dma-aligned. */
|
||||
op->data.nbytes -= op->data.nbytes % 4;
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static bool mtk_spi_mem_supports_op(struct spi_mem *mem,
|
||||
const struct spi_mem_op *op)
|
||||
{
|
||||
if (!spi_mem_default_supports_op(mem, op))
|
||||
return false;
|
||||
|
||||
if (op->addr.nbytes && op->dummy.nbytes &&
|
||||
op->addr.buswidth != op->dummy.buswidth)
|
||||
return false;
|
||||
|
||||
if (op->addr.nbytes + op->dummy.nbytes > 16)
|
||||
return false;
|
||||
|
||||
if (op->data.nbytes > MTK_SPI_IPM_PACKET_SIZE) {
|
||||
if (op->data.nbytes / MTK_SPI_IPM_PACKET_SIZE >
|
||||
MTK_SPI_IPM_PACKET_LOOP ||
|
||||
op->data.nbytes % MTK_SPI_IPM_PACKET_SIZE != 0)
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
static void mtk_spi_mem_setup_dma_xfer(struct spi_master *master,
|
||||
const struct spi_mem_op *op)
|
||||
{
|
||||
struct mtk_spi *mdata = spi_master_get_devdata(master);
|
||||
|
||||
writel((u32)(mdata->tx_dma & MTK_SPI_32BITS_MASK),
|
||||
mdata->base + SPI_TX_SRC_REG);
|
||||
#ifdef CONFIG_ARCH_DMA_ADDR_T_64BIT
|
||||
if (mdata->dev_comp->dma_ext)
|
||||
writel((u32)(mdata->tx_dma >> 32),
|
||||
mdata->base + SPI_TX_SRC_REG_64);
|
||||
#endif
|
||||
|
||||
if (op->data.dir == SPI_MEM_DATA_IN) {
|
||||
writel((u32)(mdata->rx_dma & MTK_SPI_32BITS_MASK),
|
||||
mdata->base + SPI_RX_DST_REG);
|
||||
#ifdef CONFIG_ARCH_DMA_ADDR_T_64BIT
|
||||
if (mdata->dev_comp->dma_ext)
|
||||
writel((u32)(mdata->rx_dma >> 32),
|
||||
mdata->base + SPI_RX_DST_REG_64);
|
||||
#endif
|
||||
}
|
||||
}
|
||||
|
||||
static int mtk_spi_transfer_wait(struct spi_mem *mem,
|
||||
const struct spi_mem_op *op)
|
||||
{
|
||||
struct mtk_spi *mdata = spi_master_get_devdata(mem->spi->master);
|
||||
/*
|
||||
* For each byte we wait for 8 cycles of the SPI clock.
|
||||
* Since speed is defined in Hz and we want milliseconds,
|
||||
* so it should be 8 * 1000.
|
||||
*/
|
||||
u64 ms = 8000LL;
|
||||
|
||||
if (op->data.dir == SPI_MEM_NO_DATA)
|
||||
ms *= 32; /* prevent we may get 0 for short transfers. */
|
||||
else
|
||||
ms *= op->data.nbytes;
|
||||
ms = div_u64(ms, mem->spi->max_speed_hz);
|
||||
ms += ms + 1000; /* 1s tolerance */
|
||||
|
||||
if (ms > UINT_MAX)
|
||||
ms = UINT_MAX;
|
||||
|
||||
if (!wait_for_completion_timeout(&mdata->spimem_done,
|
||||
msecs_to_jiffies(ms))) {
|
||||
dev_err(mdata->dev, "spi-mem transfer timeout\n");
|
||||
return -ETIMEDOUT;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int mtk_spi_mem_exec_op(struct spi_mem *mem,
|
||||
const struct spi_mem_op *op)
|
||||
{
|
||||
struct mtk_spi *mdata = spi_master_get_devdata(mem->spi->master);
|
||||
u32 reg_val, nio, tx_size;
|
||||
char *tx_tmp_buf, *rx_tmp_buf;
|
||||
int ret = 0;
|
||||
|
||||
mdata->use_spimem = true;
|
||||
reinit_completion(&mdata->spimem_done);
|
||||
|
||||
mtk_spi_reset(mdata);
|
||||
mtk_spi_hw_init(mem->spi->master, mem->spi);
|
||||
mtk_spi_prepare_transfer(mem->spi->master, mem->spi->max_speed_hz);
|
||||
|
||||
reg_val = readl(mdata->base + SPI_CFG3_IPM_REG);
|
||||
/* opcode byte len */
|
||||
reg_val &= ~SPI_CFG3_IPM_CMD_BYTELEN_MASK;
|
||||
reg_val |= 1 << SPI_CFG3_IPM_CMD_BYTELEN_OFFSET;
|
||||
|
||||
/* addr & dummy byte len */
|
||||
reg_val &= ~SPI_CFG3_IPM_ADDR_BYTELEN_MASK;
|
||||
if (op->addr.nbytes || op->dummy.nbytes)
|
||||
reg_val |= (op->addr.nbytes + op->dummy.nbytes) <<
|
||||
SPI_CFG3_IPM_ADDR_BYTELEN_OFFSET;
|
||||
|
||||
/* data byte len */
|
||||
if (op->data.dir == SPI_MEM_NO_DATA) {
|
||||
reg_val |= SPI_CFG3_IPM_NODATA_FLAG;
|
||||
writel(0, mdata->base + SPI_CFG1_REG);
|
||||
} else {
|
||||
reg_val &= ~SPI_CFG3_IPM_NODATA_FLAG;
|
||||
mdata->xfer_len = op->data.nbytes;
|
||||
mtk_spi_setup_packet(mem->spi->master);
|
||||
}
|
||||
|
||||
if (op->addr.nbytes || op->dummy.nbytes) {
|
||||
if (op->addr.buswidth == 1 || op->dummy.buswidth == 1)
|
||||
reg_val |= SPI_CFG3_IPM_XMODE_EN;
|
||||
else
|
||||
reg_val &= ~SPI_CFG3_IPM_XMODE_EN;
|
||||
}
|
||||
|
||||
if (op->addr.buswidth == 2 ||
|
||||
op->dummy.buswidth == 2 ||
|
||||
op->data.buswidth == 2)
|
||||
nio = 2;
|
||||
else if (op->addr.buswidth == 4 ||
|
||||
op->dummy.buswidth == 4 ||
|
||||
op->data.buswidth == 4)
|
||||
nio = 4;
|
||||
else
|
||||
nio = 1;
|
||||
|
||||
reg_val &= ~SPI_CFG3_IPM_CMD_PIN_MODE_MASK;
|
||||
reg_val |= PIN_MODE_CFG(nio);
|
||||
|
||||
reg_val |= SPI_CFG3_IPM_HALF_DUPLEX_EN;
|
||||
if (op->data.dir == SPI_MEM_DATA_IN)
|
||||
reg_val |= SPI_CFG3_IPM_HALF_DUPLEX_DIR;
|
||||
else
|
||||
reg_val &= ~SPI_CFG3_IPM_HALF_DUPLEX_DIR;
|
||||
writel(reg_val, mdata->base + SPI_CFG3_IPM_REG);
|
||||
|
||||
tx_size = 1 + op->addr.nbytes + op->dummy.nbytes;
|
||||
if (op->data.dir == SPI_MEM_DATA_OUT)
|
||||
tx_size += op->data.nbytes;
|
||||
|
||||
tx_size = max_t(u32, tx_size, 32);
|
||||
|
||||
tx_tmp_buf = kzalloc(tx_size, GFP_KERNEL | GFP_DMA);
|
||||
if (!tx_tmp_buf) {
|
||||
mdata->use_spimem = false;
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
tx_tmp_buf[0] = op->cmd.opcode;
|
||||
|
||||
if (op->addr.nbytes) {
|
||||
int i;
|
||||
|
||||
for (i = 0; i < op->addr.nbytes; i++)
|
||||
tx_tmp_buf[i + 1] = op->addr.val >>
|
||||
(8 * (op->addr.nbytes - i - 1));
|
||||
}
|
||||
|
||||
if (op->dummy.nbytes)
|
||||
memset(tx_tmp_buf + op->addr.nbytes + 1,
|
||||
0xff,
|
||||
op->dummy.nbytes);
|
||||
|
||||
if (op->data.nbytes && op->data.dir == SPI_MEM_DATA_OUT)
|
||||
memcpy(tx_tmp_buf + op->dummy.nbytes + op->addr.nbytes + 1,
|
||||
op->data.buf.out,
|
||||
op->data.nbytes);
|
||||
|
||||
mdata->tx_dma = dma_map_single(mdata->dev, tx_tmp_buf,
|
||||
tx_size, DMA_TO_DEVICE);
|
||||
if (dma_mapping_error(mdata->dev, mdata->tx_dma)) {
|
||||
ret = -ENOMEM;
|
||||
goto err_exit;
|
||||
}
|
||||
|
||||
if (op->data.dir == SPI_MEM_DATA_IN) {
|
||||
if (!IS_ALIGNED((size_t)op->data.buf.in, 4)) {
|
||||
rx_tmp_buf = kzalloc(op->data.nbytes,
|
||||
GFP_KERNEL | GFP_DMA);
|
||||
if (!rx_tmp_buf) {
|
||||
ret = -ENOMEM;
|
||||
goto unmap_tx_dma;
|
||||
}
|
||||
} else {
|
||||
rx_tmp_buf = op->data.buf.in;
|
||||
}
|
||||
|
||||
mdata->rx_dma = dma_map_single(mdata->dev,
|
||||
rx_tmp_buf,
|
||||
op->data.nbytes,
|
||||
DMA_FROM_DEVICE);
|
||||
if (dma_mapping_error(mdata->dev, mdata->rx_dma)) {
|
||||
ret = -ENOMEM;
|
||||
goto kfree_rx_tmp_buf;
|
||||
}
|
||||
}
|
||||
|
||||
reg_val = readl(mdata->base + SPI_CMD_REG);
|
||||
reg_val |= SPI_CMD_TX_DMA;
|
||||
if (op->data.dir == SPI_MEM_DATA_IN)
|
||||
reg_val |= SPI_CMD_RX_DMA;
|
||||
writel(reg_val, mdata->base + SPI_CMD_REG);
|
||||
|
||||
mtk_spi_mem_setup_dma_xfer(mem->spi->master, op);
|
||||
|
||||
mtk_spi_enable_transfer(mem->spi->master);
|
||||
|
||||
/* Wait for the interrupt. */
|
||||
ret = mtk_spi_transfer_wait(mem, op);
|
||||
if (ret)
|
||||
goto unmap_rx_dma;
|
||||
|
||||
/* spi disable dma */
|
||||
reg_val = readl(mdata->base + SPI_CMD_REG);
|
||||
reg_val &= ~SPI_CMD_TX_DMA;
|
||||
if (op->data.dir == SPI_MEM_DATA_IN)
|
||||
reg_val &= ~SPI_CMD_RX_DMA;
|
||||
writel(reg_val, mdata->base + SPI_CMD_REG);
|
||||
|
||||
unmap_rx_dma:
|
||||
if (op->data.dir == SPI_MEM_DATA_IN) {
|
||||
dma_unmap_single(mdata->dev, mdata->rx_dma,
|
||||
op->data.nbytes, DMA_FROM_DEVICE);
|
||||
if (!IS_ALIGNED((size_t)op->data.buf.in, 4))
|
||||
memcpy(op->data.buf.in, rx_tmp_buf, op->data.nbytes);
|
||||
}
|
||||
kfree_rx_tmp_buf:
|
||||
if (op->data.dir == SPI_MEM_DATA_IN &&
|
||||
!IS_ALIGNED((size_t)op->data.buf.in, 4))
|
||||
kfree(rx_tmp_buf);
|
||||
unmap_tx_dma:
|
||||
dma_unmap_single(mdata->dev, mdata->tx_dma,
|
||||
tx_size, DMA_TO_DEVICE);
|
||||
err_exit:
|
||||
kfree(tx_tmp_buf);
|
||||
mdata->use_spimem = false;
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static const struct spi_controller_mem_ops mtk_spi_mem_ops = {
|
||||
.adjust_op_size = mtk_spi_mem_adjust_op_size,
|
||||
.supports_op = mtk_spi_mem_supports_op,
|
||||
.exec_op = mtk_spi_mem_exec_op,
|
||||
};
|
||||
|
||||
static int mtk_spi_probe(struct platform_device *pdev)
|
||||
{
|
||||
struct device *dev = &pdev->dev;
|
||||
struct spi_master *master;
|
||||
struct mtk_spi *mdata;
|
||||
int i, irq, ret, addr_bits;
|
||||
|
||||
master = devm_spi_alloc_master(dev, sizeof(*mdata));
|
||||
if (!master)
|
||||
return dev_err_probe(dev, -ENOMEM, "failed to alloc spi master\n");
|
||||
|
||||
master->auto_runtime_pm = true;
|
||||
master->dev.of_node = pdev->dev.of_node;
|
||||
master->dev.of_node = dev->of_node;
|
||||
master->mode_bits = SPI_CPOL | SPI_CPHA | SPI_LSB_FIRST;
|
||||
|
||||
master->set_cs = mtk_spi_set_cs;
|
||||
@ -812,15 +1132,8 @@ static int mtk_spi_probe(struct platform_device *pdev)
|
||||
master->set_cs_timing = mtk_spi_set_hw_cs_timing;
|
||||
master->use_gpio_descriptors = true;
|
||||
|
||||
of_id = of_match_node(mtk_spi_of_match, pdev->dev.of_node);
|
||||
if (!of_id) {
|
||||
dev_err(&pdev->dev, "failed to probe of_node\n");
|
||||
ret = -EINVAL;
|
||||
goto err_put_master;
|
||||
}
|
||||
|
||||
mdata = spi_master_get_devdata(master);
|
||||
mdata->dev_comp = of_id->data;
|
||||
mdata->dev_comp = device_get_match_data(dev);
|
||||
|
||||
if (mdata->dev_comp->enhance_timing)
|
||||
master->mode_bits |= SPI_CS_HIGH;
|
||||
@ -830,143 +1143,122 @@ static int mtk_spi_probe(struct platform_device *pdev)
|
||||
if (mdata->dev_comp->ipm_design)
|
||||
master->mode_bits |= SPI_LOOP;
|
||||
|
||||
if (mdata->dev_comp->need_pad_sel) {
|
||||
mdata->pad_num = of_property_count_u32_elems(
|
||||
pdev->dev.of_node,
|
||||
"mediatek,pad-select");
|
||||
if (mdata->pad_num < 0) {
|
||||
dev_err(&pdev->dev,
|
||||
"No 'mediatek,pad-select' property\n");
|
||||
ret = -EINVAL;
|
||||
goto err_put_master;
|
||||
}
|
||||
if (mdata->dev_comp->ipm_design) {
|
||||
mdata->dev = dev;
|
||||
master->mem_ops = &mtk_spi_mem_ops;
|
||||
init_completion(&mdata->spimem_done);
|
||||
}
|
||||
|
||||
mdata->pad_sel = devm_kmalloc_array(&pdev->dev, mdata->pad_num,
|
||||
if (mdata->dev_comp->need_pad_sel) {
|
||||
mdata->pad_num = of_property_count_u32_elems(dev->of_node,
|
||||
"mediatek,pad-select");
|
||||
if (mdata->pad_num < 0)
|
||||
return dev_err_probe(dev, -EINVAL,
|
||||
"No 'mediatek,pad-select' property\n");
|
||||
|
||||
mdata->pad_sel = devm_kmalloc_array(dev, mdata->pad_num,
|
||||
sizeof(u32), GFP_KERNEL);
|
||||
if (!mdata->pad_sel) {
|
||||
ret = -ENOMEM;
|
||||
goto err_put_master;
|
||||
}
|
||||
if (!mdata->pad_sel)
|
||||
return -ENOMEM;
|
||||
|
||||
for (i = 0; i < mdata->pad_num; i++) {
|
||||
of_property_read_u32_index(pdev->dev.of_node,
|
||||
of_property_read_u32_index(dev->of_node,
|
||||
"mediatek,pad-select",
|
||||
i, &mdata->pad_sel[i]);
|
||||
if (mdata->pad_sel[i] > MT8173_SPI_MAX_PAD_SEL) {
|
||||
dev_err(&pdev->dev, "wrong pad-sel[%d]: %u\n",
|
||||
i, mdata->pad_sel[i]);
|
||||
ret = -EINVAL;
|
||||
goto err_put_master;
|
||||
}
|
||||
if (mdata->pad_sel[i] > MT8173_SPI_MAX_PAD_SEL)
|
||||
return dev_err_probe(dev, -EINVAL,
|
||||
"wrong pad-sel[%d]: %u\n",
|
||||
i, mdata->pad_sel[i]);
|
||||
}
|
||||
}
|
||||
|
||||
platform_set_drvdata(pdev, master);
|
||||
mdata->base = devm_platform_ioremap_resource(pdev, 0);
|
||||
if (IS_ERR(mdata->base)) {
|
||||
ret = PTR_ERR(mdata->base);
|
||||
goto err_put_master;
|
||||
}
|
||||
if (IS_ERR(mdata->base))
|
||||
return PTR_ERR(mdata->base);
|
||||
|
||||
irq = platform_get_irq(pdev, 0);
|
||||
if (irq < 0) {
|
||||
ret = irq;
|
||||
goto err_put_master;
|
||||
}
|
||||
if (irq < 0)
|
||||
return irq;
|
||||
|
||||
if (!pdev->dev.dma_mask)
|
||||
pdev->dev.dma_mask = &pdev->dev.coherent_dma_mask;
|
||||
if (!dev->dma_mask)
|
||||
dev->dma_mask = &dev->coherent_dma_mask;
|
||||
|
||||
ret = devm_request_irq(&pdev->dev, irq, mtk_spi_interrupt,
|
||||
IRQF_TRIGGER_NONE, dev_name(&pdev->dev), master);
|
||||
if (ret) {
|
||||
dev_err(&pdev->dev, "failed to register irq (%d)\n", ret);
|
||||
goto err_put_master;
|
||||
}
|
||||
ret = devm_request_irq(dev, irq, mtk_spi_interrupt,
|
||||
IRQF_TRIGGER_NONE, dev_name(dev), master);
|
||||
if (ret)
|
||||
return dev_err_probe(dev, ret, "failed to register irq\n");
|
||||
|
||||
mdata->parent_clk = devm_clk_get(&pdev->dev, "parent-clk");
|
||||
if (IS_ERR(mdata->parent_clk)) {
|
||||
ret = PTR_ERR(mdata->parent_clk);
|
||||
dev_err(&pdev->dev, "failed to get parent-clk: %d\n", ret);
|
||||
goto err_put_master;
|
||||
}
|
||||
mdata->parent_clk = devm_clk_get(dev, "parent-clk");
|
||||
if (IS_ERR(mdata->parent_clk))
|
||||
return dev_err_probe(dev, PTR_ERR(mdata->parent_clk),
|
||||
"failed to get parent-clk\n");
|
||||
|
||||
mdata->sel_clk = devm_clk_get(&pdev->dev, "sel-clk");
|
||||
if (IS_ERR(mdata->sel_clk)) {
|
||||
ret = PTR_ERR(mdata->sel_clk);
|
||||
dev_err(&pdev->dev, "failed to get sel-clk: %d\n", ret);
|
||||
goto err_put_master;
|
||||
}
|
||||
mdata->sel_clk = devm_clk_get(dev, "sel-clk");
|
||||
if (IS_ERR(mdata->sel_clk))
|
||||
return dev_err_probe(dev, PTR_ERR(mdata->sel_clk), "failed to get sel-clk\n");
|
||||
|
||||
mdata->spi_clk = devm_clk_get(&pdev->dev, "spi-clk");
|
||||
if (IS_ERR(mdata->spi_clk)) {
|
||||
ret = PTR_ERR(mdata->spi_clk);
|
||||
dev_err(&pdev->dev, "failed to get spi-clk: %d\n", ret);
|
||||
goto err_put_master;
|
||||
}
|
||||
mdata->spi_clk = devm_clk_get(dev, "spi-clk");
|
||||
if (IS_ERR(mdata->spi_clk))
|
||||
return dev_err_probe(dev, PTR_ERR(mdata->spi_clk), "failed to get spi-clk\n");
|
||||
|
||||
mdata->spi_hclk = devm_clk_get_optional(dev, "hclk");
|
||||
if (IS_ERR(mdata->spi_hclk))
|
||||
return dev_err_probe(dev, PTR_ERR(mdata->spi_hclk), "failed to get hclk\n");
|
||||
|
||||
ret = clk_set_parent(mdata->sel_clk, mdata->parent_clk);
|
||||
if (ret < 0)
|
||||
return dev_err_probe(dev, ret, "failed to clk_set_parent\n");
|
||||
|
||||
ret = clk_prepare_enable(mdata->spi_hclk);
|
||||
if (ret < 0)
|
||||
return dev_err_probe(dev, ret, "failed to enable hclk\n");
|
||||
|
||||
ret = clk_prepare_enable(mdata->spi_clk);
|
||||
if (ret < 0) {
|
||||
dev_err(&pdev->dev, "failed to enable spi_clk (%d)\n", ret);
|
||||
goto err_put_master;
|
||||
}
|
||||
|
||||
ret = clk_set_parent(mdata->sel_clk, mdata->parent_clk);
|
||||
if (ret < 0) {
|
||||
dev_err(&pdev->dev, "failed to clk_set_parent (%d)\n", ret);
|
||||
clk_disable_unprepare(mdata->spi_clk);
|
||||
goto err_put_master;
|
||||
clk_disable_unprepare(mdata->spi_hclk);
|
||||
return dev_err_probe(dev, ret, "failed to enable spi_clk\n");
|
||||
}
|
||||
|
||||
mdata->spi_clk_hz = clk_get_rate(mdata->spi_clk);
|
||||
|
||||
if (mdata->dev_comp->no_need_unprepare)
|
||||
if (mdata->dev_comp->no_need_unprepare) {
|
||||
clk_disable(mdata->spi_clk);
|
||||
else
|
||||
clk_disable(mdata->spi_hclk);
|
||||
} else {
|
||||
clk_disable_unprepare(mdata->spi_clk);
|
||||
|
||||
pm_runtime_enable(&pdev->dev);
|
||||
clk_disable_unprepare(mdata->spi_hclk);
|
||||
}
|
||||
|
||||
if (mdata->dev_comp->need_pad_sel) {
|
||||
if (mdata->pad_num != master->num_chipselect) {
|
||||
dev_err(&pdev->dev,
|
||||
if (mdata->pad_num != master->num_chipselect)
|
||||
return dev_err_probe(dev, -EINVAL,
|
||||
"pad_num does not match num_chipselect(%d != %d)\n",
|
||||
mdata->pad_num, master->num_chipselect);
|
||||
ret = -EINVAL;
|
||||
goto err_disable_runtime_pm;
|
||||
}
|
||||
|
||||
if (!master->cs_gpiods && master->num_chipselect > 1) {
|
||||
dev_err(&pdev->dev,
|
||||
if (!master->cs_gpiods && master->num_chipselect > 1)
|
||||
return dev_err_probe(dev, -EINVAL,
|
||||
"cs_gpios not specified and num_chipselect > 1\n");
|
||||
ret = -EINVAL;
|
||||
goto err_disable_runtime_pm;
|
||||
}
|
||||
}
|
||||
|
||||
if (mdata->dev_comp->dma_ext)
|
||||
addr_bits = DMA_ADDR_EXT_BITS;
|
||||
else
|
||||
addr_bits = DMA_ADDR_DEF_BITS;
|
||||
ret = dma_set_mask(&pdev->dev, DMA_BIT_MASK(addr_bits));
|
||||
ret = dma_set_mask(dev, DMA_BIT_MASK(addr_bits));
|
||||
if (ret)
|
||||
dev_notice(&pdev->dev, "SPI dma_set_mask(%d) failed, ret:%d\n",
|
||||
dev_notice(dev, "SPI dma_set_mask(%d) failed, ret:%d\n",
|
||||
addr_bits, ret);
|
||||
|
||||
ret = devm_spi_register_master(&pdev->dev, master);
|
||||
pm_runtime_enable(dev);
|
||||
|
||||
ret = devm_spi_register_master(dev, master);
|
||||
if (ret) {
|
||||
dev_err(&pdev->dev, "failed to register master (%d)\n", ret);
|
||||
goto err_disable_runtime_pm;
|
||||
pm_runtime_disable(dev);
|
||||
return dev_err_probe(dev, ret, "failed to register master\n");
|
||||
}
|
||||
|
||||
return 0;
|
||||
|
||||
err_disable_runtime_pm:
|
||||
pm_runtime_disable(&pdev->dev);
|
||||
err_put_master:
|
||||
spi_master_put(master);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int mtk_spi_remove(struct platform_device *pdev)
|
||||
@ -978,8 +1270,10 @@ static int mtk_spi_remove(struct platform_device *pdev)
|
||||
|
||||
mtk_spi_reset(mdata);
|
||||
|
||||
if (mdata->dev_comp->no_need_unprepare)
|
||||
if (mdata->dev_comp->no_need_unprepare) {
|
||||
clk_unprepare(mdata->spi_clk);
|
||||
clk_unprepare(mdata->spi_hclk);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
@ -995,8 +1289,10 @@ static int mtk_spi_suspend(struct device *dev)
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
if (!pm_runtime_suspended(dev))
|
||||
if (!pm_runtime_suspended(dev)) {
|
||||
clk_disable_unprepare(mdata->spi_clk);
|
||||
clk_disable_unprepare(mdata->spi_hclk);
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
@ -1013,11 +1309,20 @@ static int mtk_spi_resume(struct device *dev)
|
||||
dev_err(dev, "failed to enable spi_clk (%d)\n", ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
ret = clk_prepare_enable(mdata->spi_hclk);
|
||||
if (ret < 0) {
|
||||
dev_err(dev, "failed to enable spi_hclk (%d)\n", ret);
|
||||
clk_disable_unprepare(mdata->spi_clk);
|
||||
return ret;
|
||||
}
|
||||
}
|
||||
|
||||
ret = spi_master_resume(master);
|
||||
if (ret < 0)
|
||||
if (ret < 0) {
|
||||
clk_disable_unprepare(mdata->spi_clk);
|
||||
clk_disable_unprepare(mdata->spi_hclk);
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
@ -1029,10 +1334,13 @@ static int mtk_spi_runtime_suspend(struct device *dev)
|
||||
struct spi_master *master = dev_get_drvdata(dev);
|
||||
struct mtk_spi *mdata = spi_master_get_devdata(master);
|
||||
|
||||
if (mdata->dev_comp->no_need_unprepare)
|
||||
if (mdata->dev_comp->no_need_unprepare) {
|
||||
clk_disable(mdata->spi_clk);
|
||||
else
|
||||
clk_disable(mdata->spi_hclk);
|
||||
} else {
|
||||
clk_disable_unprepare(mdata->spi_clk);
|
||||
clk_disable_unprepare(mdata->spi_hclk);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
@ -1043,13 +1351,31 @@ static int mtk_spi_runtime_resume(struct device *dev)
|
||||
struct mtk_spi *mdata = spi_master_get_devdata(master);
|
||||
int ret;
|
||||
|
||||
if (mdata->dev_comp->no_need_unprepare)
|
||||
if (mdata->dev_comp->no_need_unprepare) {
|
||||
ret = clk_enable(mdata->spi_clk);
|
||||
else
|
||||
if (ret < 0) {
|
||||
dev_err(dev, "failed to enable spi_clk (%d)\n", ret);
|
||||
return ret;
|
||||
}
|
||||
ret = clk_enable(mdata->spi_hclk);
|
||||
if (ret < 0) {
|
||||
dev_err(dev, "failed to enable spi_hclk (%d)\n", ret);
|
||||
clk_disable(mdata->spi_clk);
|
||||
return ret;
|
||||
}
|
||||
} else {
|
||||
ret = clk_prepare_enable(mdata->spi_clk);
|
||||
if (ret < 0) {
|
||||
dev_err(dev, "failed to enable spi_clk (%d)\n", ret);
|
||||
return ret;
|
||||
if (ret < 0) {
|
||||
dev_err(dev, "failed to prepare_enable spi_clk (%d)\n", ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
ret = clk_prepare_enable(mdata->spi_hclk);
|
||||
if (ret < 0) {
|
||||
dev_err(dev, "failed to prepare_enable spi_hclk (%d)\n", ret);
|
||||
clk_disable_unprepare(mdata->spi_clk);
|
||||
return ret;
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
|
1472
drivers/spi/spi-mtk-snfi.c
Normal file
1472
drivers/spi/spi-mtk-snfi.c
Normal file
File diff suppressed because it is too large
Load Diff
@ -605,9 +605,8 @@ static int mxs_spi_probe(struct platform_device *pdev)
|
||||
}
|
||||
}
|
||||
|
||||
ret = pm_runtime_get_sync(ssp->dev);
|
||||
ret = pm_runtime_resume_and_get(ssp->dev);
|
||||
if (ret < 0) {
|
||||
pm_runtime_put_noidle(ssp->dev);
|
||||
dev_err(ssp->dev, "runtime_get_sync failed\n");
|
||||
goto out_pm_runtime_disable;
|
||||
}
|
||||
|
@ -246,9 +246,8 @@ static void omap2_mcspi_set_cs(struct spi_device *spi, bool enable)
|
||||
enable = !enable;
|
||||
|
||||
if (spi->controller_state) {
|
||||
int err = pm_runtime_get_sync(mcspi->dev);
|
||||
int err = pm_runtime_resume_and_get(mcspi->dev);
|
||||
if (err < 0) {
|
||||
pm_runtime_put_noidle(mcspi->dev);
|
||||
dev_err(mcspi->dev, "failed to get sync: %d\n", err);
|
||||
return;
|
||||
}
|
||||
@ -758,6 +757,8 @@ omap2_mcspi_txrx_pio(struct spi_device *spi, struct spi_transfer *xfer)
|
||||
dev_vdbg(&spi->dev, "read-%d %02x\n",
|
||||
word_len, *(rx - 1));
|
||||
}
|
||||
/* Add word delay between each word */
|
||||
spi_delay_exec(&xfer->word_delay, xfer);
|
||||
} while (c);
|
||||
} else if (word_len <= 16) {
|
||||
u16 *rx;
|
||||
@ -805,6 +806,8 @@ omap2_mcspi_txrx_pio(struct spi_device *spi, struct spi_transfer *xfer)
|
||||
dev_vdbg(&spi->dev, "read-%d %04x\n",
|
||||
word_len, *(rx - 1));
|
||||
}
|
||||
/* Add word delay between each word */
|
||||
spi_delay_exec(&xfer->word_delay, xfer);
|
||||
} while (c >= 2);
|
||||
} else if (word_len <= 32) {
|
||||
u32 *rx;
|
||||
@ -852,6 +855,8 @@ omap2_mcspi_txrx_pio(struct spi_device *spi, struct spi_transfer *xfer)
|
||||
dev_vdbg(&spi->dev, "read-%d %08x\n",
|
||||
word_len, *(rx - 1));
|
||||
}
|
||||
/* Add word delay between each word */
|
||||
spi_delay_exec(&xfer->word_delay, xfer);
|
||||
} while (c >= 4);
|
||||
}
|
||||
|
||||
@ -1068,9 +1073,8 @@ static int omap2_mcspi_setup(struct spi_device *spi)
|
||||
initial_setup = true;
|
||||
}
|
||||
|
||||
ret = pm_runtime_get_sync(mcspi->dev);
|
||||
ret = pm_runtime_resume_and_get(mcspi->dev);
|
||||
if (ret < 0) {
|
||||
pm_runtime_put_noidle(mcspi->dev);
|
||||
if (initial_setup)
|
||||
omap2_mcspi_cleanup(spi);
|
||||
|
||||
@ -1317,12 +1321,9 @@ static int omap2_mcspi_controller_setup(struct omap2_mcspi *mcspi)
|
||||
struct omap2_mcspi_regs *ctx = &mcspi->ctx;
|
||||
int ret = 0;
|
||||
|
||||
ret = pm_runtime_get_sync(mcspi->dev);
|
||||
if (ret < 0) {
|
||||
pm_runtime_put_noidle(mcspi->dev);
|
||||
|
||||
ret = pm_runtime_resume_and_get(mcspi->dev);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
}
|
||||
|
||||
mcspi_write_reg(master, OMAP2_MCSPI_WAKEUPENABLE,
|
||||
OMAP2_MCSPI_WAKEUPENABLE_WKEN);
|
||||
|
@ -196,6 +196,8 @@ struct rockchip_spi {
|
||||
|
||||
bool slave_abort;
|
||||
bool cs_inactive; /* spi slave tansmition stop when cs inactive */
|
||||
bool cs_high_supported; /* native CS supports active-high polarity */
|
||||
|
||||
struct spi_transfer *xfer; /* Store xfer temporarily */
|
||||
};
|
||||
|
||||
@ -719,6 +721,11 @@ static int rockchip_spi_setup(struct spi_device *spi)
|
||||
struct rockchip_spi *rs = spi_controller_get_devdata(spi->controller);
|
||||
u32 cr0;
|
||||
|
||||
if (!spi->cs_gpiod && (spi->mode & SPI_CS_HIGH) && !rs->cs_high_supported) {
|
||||
dev_warn(&spi->dev, "setup: non GPIO CS can't be active-high\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
pm_runtime_get_sync(rs->dev);
|
||||
|
||||
cr0 = readl_relaxed(rs->regs + ROCKCHIP_SPI_CTRLR0);
|
||||
@ -899,6 +906,7 @@ static int rockchip_spi_probe(struct platform_device *pdev)
|
||||
|
||||
switch (readl_relaxed(rs->regs + ROCKCHIP_SPI_VERSION)) {
|
||||
case ROCKCHIP_SPI_VER2_TYPE2:
|
||||
rs->cs_high_supported = true;
|
||||
ctlr->mode_bits |= SPI_CS_HIGH;
|
||||
if (ctlr->can_dma && slave_mode)
|
||||
rs->cs_inactive = true;
|
||||
|
@ -1108,14 +1108,11 @@ static struct dma_chan *rspi_request_dma_chan(struct device *dev,
|
||||
}
|
||||
|
||||
memset(&cfg, 0, sizeof(cfg));
|
||||
cfg.dst_addr = port_addr + RSPI_SPDR;
|
||||
cfg.src_addr = port_addr + RSPI_SPDR;
|
||||
cfg.dst_addr_width = DMA_SLAVE_BUSWIDTH_1_BYTE;
|
||||
cfg.src_addr_width = DMA_SLAVE_BUSWIDTH_1_BYTE;
|
||||
cfg.direction = dir;
|
||||
if (dir == DMA_MEM_TO_DEV) {
|
||||
cfg.dst_addr = port_addr;
|
||||
cfg.dst_addr_width = DMA_SLAVE_BUSWIDTH_1_BYTE;
|
||||
} else {
|
||||
cfg.src_addr = port_addr;
|
||||
cfg.src_addr_width = DMA_SLAVE_BUSWIDTH_1_BYTE;
|
||||
}
|
||||
|
||||
ret = dmaengine_slave_config(chan, &cfg);
|
||||
if (ret) {
|
||||
@ -1146,12 +1143,12 @@ static int rspi_request_dma(struct device *dev, struct spi_controller *ctlr,
|
||||
}
|
||||
|
||||
ctlr->dma_tx = rspi_request_dma_chan(dev, DMA_MEM_TO_DEV, dma_tx_id,
|
||||
res->start + RSPI_SPDR);
|
||||
res->start);
|
||||
if (!ctlr->dma_tx)
|
||||
return -ENODEV;
|
||||
|
||||
ctlr->dma_rx = rspi_request_dma_chan(dev, DMA_DEV_TO_MEM, dma_rx_id,
|
||||
res->start + RSPI_SPDR);
|
||||
res->start);
|
||||
if (!ctlr->dma_rx) {
|
||||
dma_release_channel(ctlr->dma_tx);
|
||||
ctlr->dma_tx = NULL;
|
||||
|
@ -1008,9 +1008,8 @@ static int sprd_spi_remove(struct platform_device *pdev)
|
||||
struct sprd_spi *ss = spi_controller_get_devdata(sctlr);
|
||||
int ret;
|
||||
|
||||
ret = pm_runtime_get_sync(ss->dev);
|
||||
ret = pm_runtime_resume_and_get(ss->dev);
|
||||
if (ret < 0) {
|
||||
pm_runtime_put_noidle(ss->dev);
|
||||
dev_err(ss->dev, "failed to resume SPI controller\n");
|
||||
return ret;
|
||||
}
|
||||
|
@ -305,10 +305,8 @@ static int stm32_qspi_wait_cmd(struct stm32_qspi *qspi,
|
||||
u32 cr, sr;
|
||||
int err = 0;
|
||||
|
||||
if (!op->data.nbytes)
|
||||
goto wait_nobusy;
|
||||
|
||||
if (readl_relaxed(qspi->io_base + QSPI_SR) & SR_TCF)
|
||||
if ((readl_relaxed(qspi->io_base + QSPI_SR) & SR_TCF) ||
|
||||
qspi->fmode == CCR_FMODE_APM)
|
||||
goto out;
|
||||
|
||||
reinit_completion(&qspi->data_completion);
|
||||
@ -327,7 +325,6 @@ static int stm32_qspi_wait_cmd(struct stm32_qspi *qspi,
|
||||
out:
|
||||
/* clear flags */
|
||||
writel_relaxed(FCR_CTCF | FCR_CTEF, qspi->io_base + QSPI_FCR);
|
||||
wait_nobusy:
|
||||
if (!err)
|
||||
err = stm32_qspi_wait_nobusy(qspi);
|
||||
|
||||
@ -372,10 +369,6 @@ static int stm32_qspi_send(struct spi_mem *mem, const struct spi_mem_op *op)
|
||||
op->dummy.buswidth, op->data.buswidth,
|
||||
op->addr.val, op->data.nbytes);
|
||||
|
||||
err = stm32_qspi_wait_nobusy(qspi);
|
||||
if (err)
|
||||
goto abort;
|
||||
|
||||
cr = readl_relaxed(qspi->io_base + QSPI_CR);
|
||||
cr &= ~CR_PRESC_MASK & ~CR_FSEL;
|
||||
cr |= FIELD_PREP(CR_PRESC_MASK, flash->presc);
|
||||
@ -463,11 +456,9 @@ static int stm32_qspi_poll_status(struct spi_mem *mem, const struct spi_mem_op *
|
||||
if (!spi_mem_supports_op(mem, op))
|
||||
return -EOPNOTSUPP;
|
||||
|
||||
ret = pm_runtime_get_sync(qspi->dev);
|
||||
if (ret < 0) {
|
||||
pm_runtime_put_noidle(qspi->dev);
|
||||
ret = pm_runtime_resume_and_get(qspi->dev);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
}
|
||||
|
||||
mutex_lock(&qspi->lock);
|
||||
|
||||
@ -490,11 +481,9 @@ static int stm32_qspi_exec_op(struct spi_mem *mem, const struct spi_mem_op *op)
|
||||
struct stm32_qspi *qspi = spi_controller_get_devdata(mem->spi->master);
|
||||
int ret;
|
||||
|
||||
ret = pm_runtime_get_sync(qspi->dev);
|
||||
if (ret < 0) {
|
||||
pm_runtime_put_noidle(qspi->dev);
|
||||
ret = pm_runtime_resume_and_get(qspi->dev);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
}
|
||||
|
||||
mutex_lock(&qspi->lock);
|
||||
if (op->data.dir == SPI_MEM_DATA_IN && op->data.nbytes)
|
||||
@ -536,11 +525,9 @@ static ssize_t stm32_qspi_dirmap_read(struct spi_mem_dirmap_desc *desc,
|
||||
u32 addr_max;
|
||||
int ret;
|
||||
|
||||
ret = pm_runtime_get_sync(qspi->dev);
|
||||
if (ret < 0) {
|
||||
pm_runtime_put_noidle(qspi->dev);
|
||||
ret = pm_runtime_resume_and_get(qspi->dev);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
}
|
||||
|
||||
mutex_lock(&qspi->lock);
|
||||
/* make a local copy of desc op_tmpl and complete dirmap rdesc
|
||||
@ -583,11 +570,9 @@ static int stm32_qspi_setup(struct spi_device *spi)
|
||||
if (!spi->max_speed_hz)
|
||||
return -EINVAL;
|
||||
|
||||
ret = pm_runtime_get_sync(qspi->dev);
|
||||
if (ret < 0) {
|
||||
pm_runtime_put_noidle(qspi->dev);
|
||||
ret = pm_runtime_resume_and_get(qspi->dev);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
}
|
||||
|
||||
presc = DIV_ROUND_UP(qspi->clk_rate, spi->max_speed_hz) - 1;
|
||||
|
||||
@ -851,11 +836,9 @@ static int __maybe_unused stm32_qspi_resume(struct device *dev)
|
||||
|
||||
pinctrl_pm_select_default_state(dev);
|
||||
|
||||
ret = pm_runtime_get_sync(dev);
|
||||
if (ret < 0) {
|
||||
pm_runtime_put_noidle(dev);
|
||||
ret = pm_runtime_resume_and_get(dev);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
}
|
||||
|
||||
writel_relaxed(qspi->cr_reg, qspi->io_base + QSPI_CR);
|
||||
writel_relaxed(qspi->dcr_reg, qspi->io_base + QSPI_DCR);
|
||||
|
@ -2000,9 +2000,8 @@ static int __maybe_unused stm32_spi_resume(struct device *dev)
|
||||
return ret;
|
||||
}
|
||||
|
||||
ret = pm_runtime_get_sync(dev);
|
||||
ret = pm_runtime_resume_and_get(dev);
|
||||
if (ret < 0) {
|
||||
pm_runtime_put_noidle(dev);
|
||||
dev_err(dev, "Unable to power device:%d\n", ret);
|
||||
return ret;
|
||||
}
|
||||
|
@ -85,8 +85,6 @@ struct sp7021_spi_ctlr {
|
||||
int s_irq;
|
||||
struct clk *spi_clk;
|
||||
struct reset_control *rstc;
|
||||
// irq spin lock
|
||||
spinlock_t lock;
|
||||
// data xfer lock
|
||||
struct mutex buf_lock;
|
||||
struct completion isr_done;
|
||||
@ -199,8 +197,6 @@ static irqreturn_t sp7021_spi_master_irq(int irq, void *dev)
|
||||
if (tx_len == 0 && total_len == 0)
|
||||
return IRQ_NONE;
|
||||
|
||||
spin_lock_irq(&pspim->lock);
|
||||
|
||||
rx_cnt = FIELD_GET(SP7021_RX_CNT_MASK, fd_status);
|
||||
if (fd_status & SP7021_RX_FULL_FLAG)
|
||||
rx_cnt = pspim->data_unit;
|
||||
@ -239,7 +235,6 @@ static irqreturn_t sp7021_spi_master_irq(int irq, void *dev)
|
||||
|
||||
if (isrdone)
|
||||
complete(&pspim->isr_done);
|
||||
spin_unlock_irq(&pspim->lock);
|
||||
return IRQ_HANDLED;
|
||||
}
|
||||
|
||||
@ -446,7 +441,6 @@ static int sp7021_spi_controller_probe(struct platform_device *pdev)
|
||||
pspim->mode = mode;
|
||||
pspim->ctlr = ctlr;
|
||||
pspim->dev = dev;
|
||||
spin_lock_init(&pspim->lock);
|
||||
mutex_init(&pspim->buf_lock);
|
||||
init_completion(&pspim->isr_done);
|
||||
init_completion(&pspim->slave_isr);
|
||||
|
@ -964,9 +964,8 @@ static int tegra_spi_setup(struct spi_device *spi)
|
||||
spi->controller_data = cdata;
|
||||
}
|
||||
|
||||
ret = pm_runtime_get_sync(tspi->dev);
|
||||
ret = pm_runtime_resume_and_get(tspi->dev);
|
||||
if (ret < 0) {
|
||||
pm_runtime_put_noidle(tspi->dev);
|
||||
dev_err(tspi->dev, "pm runtime failed, e = %d\n", ret);
|
||||
if (cdata)
|
||||
tegra_spi_cleanup(spi);
|
||||
@ -1394,10 +1393,9 @@ static int tegra_spi_probe(struct platform_device *pdev)
|
||||
goto exit_pm_disable;
|
||||
}
|
||||
|
||||
ret = pm_runtime_get_sync(&pdev->dev);
|
||||
ret = pm_runtime_resume_and_get(&pdev->dev);
|
||||
if (ret < 0) {
|
||||
dev_err(&pdev->dev, "pm runtime get failed, e = %d\n", ret);
|
||||
pm_runtime_put_noidle(&pdev->dev);
|
||||
goto exit_pm_disable;
|
||||
}
|
||||
|
||||
@ -1476,9 +1474,8 @@ static int tegra_spi_resume(struct device *dev)
|
||||
struct tegra_spi_data *tspi = spi_master_get_devdata(master);
|
||||
int ret;
|
||||
|
||||
ret = pm_runtime_get_sync(dev);
|
||||
ret = pm_runtime_resume_and_get(dev);
|
||||
if (ret < 0) {
|
||||
pm_runtime_put_noidle(dev);
|
||||
dev_err(dev, "pm runtime failed, e = %d\n", ret);
|
||||
return ret;
|
||||
}
|
||||
|
@ -486,10 +486,9 @@ static int tegra_sflash_probe(struct platform_device *pdev)
|
||||
goto exit_pm_disable;
|
||||
}
|
||||
|
||||
ret = pm_runtime_get_sync(&pdev->dev);
|
||||
ret = pm_runtime_resume_and_get(&pdev->dev);
|
||||
if (ret < 0) {
|
||||
dev_err(&pdev->dev, "pm runtime get failed, e = %d\n", ret);
|
||||
pm_runtime_put_noidle(&pdev->dev);
|
||||
goto exit_pm_disable;
|
||||
}
|
||||
|
||||
@ -549,9 +548,8 @@ static int tegra_sflash_resume(struct device *dev)
|
||||
struct tegra_sflash_data *tsd = spi_master_get_devdata(master);
|
||||
int ret;
|
||||
|
||||
ret = pm_runtime_get_sync(dev);
|
||||
ret = pm_runtime_resume_and_get(dev);
|
||||
if (ret < 0) {
|
||||
pm_runtime_put_noidle(dev);
|
||||
dev_err(dev, "pm runtime failed, e = %d\n", ret);
|
||||
return ret;
|
||||
}
|
||||
|
@ -749,9 +749,8 @@ static int tegra_slink_setup(struct spi_device *spi)
|
||||
spi->mode & SPI_CPHA ? "" : "~",
|
||||
spi->max_speed_hz);
|
||||
|
||||
ret = pm_runtime_get_sync(tspi->dev);
|
||||
ret = pm_runtime_resume_and_get(tspi->dev);
|
||||
if (ret < 0) {
|
||||
pm_runtime_put_noidle(tspi->dev);
|
||||
dev_err(tspi->dev, "pm runtime failed, e = %d\n", ret);
|
||||
return ret;
|
||||
}
|
||||
@ -1169,9 +1168,8 @@ static int tegra_slink_resume(struct device *dev)
|
||||
struct tegra_slink_data *tspi = spi_master_get_devdata(master);
|
||||
int ret;
|
||||
|
||||
ret = pm_runtime_get_sync(dev);
|
||||
ret = pm_runtime_resume_and_get(dev);
|
||||
if (ret < 0) {
|
||||
pm_runtime_put_noidle(dev);
|
||||
dev_err(dev, "pm runtime failed, e = %d\n", ret);
|
||||
return ret;
|
||||
}
|
||||
|
@ -172,9 +172,8 @@ static int ti_qspi_setup(struct spi_device *spi)
|
||||
dev_dbg(qspi->dev, "hz: %d, clock divider %d\n",
|
||||
qspi->spi_max_frequency, clk_div);
|
||||
|
||||
ret = pm_runtime_get_sync(qspi->dev);
|
||||
ret = pm_runtime_resume_and_get(qspi->dev);
|
||||
if (ret < 0) {
|
||||
pm_runtime_put_noidle(qspi->dev);
|
||||
dev_err(qspi->dev, "pm_runtime_get_sync() failed\n");
|
||||
return ret;
|
||||
}
|
||||
@ -448,6 +447,7 @@ static int ti_qspi_dma_xfer(struct ti_qspi *qspi, dma_addr_t dma_dst,
|
||||
enum dma_ctrl_flags flags = DMA_CTRL_ACK | DMA_PREP_INTERRUPT;
|
||||
struct dma_async_tx_descriptor *tx;
|
||||
int ret;
|
||||
unsigned long time_left;
|
||||
|
||||
tx = dmaengine_prep_dma_memcpy(chan, dma_dst, dma_src, len, flags);
|
||||
if (!tx) {
|
||||
@ -467,9 +467,9 @@ static int ti_qspi_dma_xfer(struct ti_qspi *qspi, dma_addr_t dma_dst,
|
||||
}
|
||||
|
||||
dma_async_issue_pending(chan);
|
||||
ret = wait_for_completion_timeout(&qspi->transfer_complete,
|
||||
time_left = wait_for_completion_timeout(&qspi->transfer_complete,
|
||||
msecs_to_jiffies(len));
|
||||
if (ret <= 0) {
|
||||
if (time_left == 0) {
|
||||
dmaengine_terminate_sync(chan);
|
||||
dev_err(qspi->dev, "DMA wait_for_completion_timeout\n");
|
||||
return -ETIMEDOUT;
|
||||
|
@ -1611,9 +1611,8 @@ static void __spi_pump_messages(struct spi_controller *ctlr, bool in_kthread)
|
||||
mutex_lock(&ctlr->io_mutex);
|
||||
|
||||
if (!was_busy && ctlr->auto_runtime_pm) {
|
||||
ret = pm_runtime_get_sync(ctlr->dev.parent);
|
||||
ret = pm_runtime_resume_and_get(ctlr->dev.parent);
|
||||
if (ret < 0) {
|
||||
pm_runtime_put_noidle(ctlr->dev.parent);
|
||||
dev_err(&ctlr->dev, "Failed to power device: %d\n",
|
||||
ret);
|
||||
mutex_unlock(&ctlr->io_mutex);
|
||||
@ -3475,7 +3474,7 @@ static int __spi_validate_bits_per_word(struct spi_controller *ctlr,
|
||||
int spi_setup(struct spi_device *spi)
|
||||
{
|
||||
unsigned bad_bits, ugly_bits;
|
||||
int status;
|
||||
int status = 0;
|
||||
|
||||
/*
|
||||
* Check mode to prevent that any two of DUAL, QUAD and NO_MOSI/MISO
|
||||
@ -3518,13 +3517,18 @@ int spi_setup(struct spi_device *spi)
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
if (!spi->bits_per_word)
|
||||
if (!spi->bits_per_word) {
|
||||
spi->bits_per_word = 8;
|
||||
|
||||
status = __spi_validate_bits_per_word(spi->controller,
|
||||
spi->bits_per_word);
|
||||
if (status)
|
||||
return status;
|
||||
} else {
|
||||
/*
|
||||
* Some controllers may not support the default 8 bits-per-word
|
||||
* so only perform the check when this is explicitly provided.
|
||||
*/
|
||||
status = __spi_validate_bits_per_word(spi->controller,
|
||||
spi->bits_per_word);
|
||||
if (status)
|
||||
return status;
|
||||
}
|
||||
|
||||
if (spi->controller->max_speed_hz &&
|
||||
(!spi->max_speed_hz ||
|
||||
@ -3544,10 +3548,9 @@ int spi_setup(struct spi_device *spi)
|
||||
}
|
||||
|
||||
if (spi->controller->auto_runtime_pm && spi->controller->set_cs) {
|
||||
status = pm_runtime_get_sync(spi->controller->dev.parent);
|
||||
status = pm_runtime_resume_and_get(spi->controller->dev.parent);
|
||||
if (status < 0) {
|
||||
mutex_unlock(&spi->controller->io_mutex);
|
||||
pm_runtime_put_noidle(spi->controller->dev.parent);
|
||||
dev_err(&spi->controller->dev, "Failed to power device: %d\n",
|
||||
status);
|
||||
return status;
|
||||
|
@ -8,19 +8,18 @@
|
||||
*/
|
||||
|
||||
#include <linux/init.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/ioctl.h>
|
||||
#include <linux/fs.h>
|
||||
#include <linux/device.h>
|
||||
#include <linux/err.h>
|
||||
#include <linux/list.h>
|
||||
#include <linux/errno.h>
|
||||
#include <linux/mod_devicetable.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/mutex.h>
|
||||
#include <linux/property.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/compat.h>
|
||||
#include <linux/of.h>
|
||||
#include <linux/of_device.h>
|
||||
#include <linux/acpi.h>
|
||||
|
||||
#include <linux/spi/spi.h>
|
||||
#include <linux/spi/spidev.h>
|
||||
@ -46,6 +45,7 @@
|
||||
|
||||
static DECLARE_BITMAP(minors, N_SPI_MINORS);
|
||||
|
||||
static_assert(N_SPI_MINORS > 0 && N_SPI_MINORS <= 256);
|
||||
|
||||
/* Bit masks for spi_device.mode management. Note that incorrect
|
||||
* settings for some settings can cause *lots* of trouble for other
|
||||
@ -63,7 +63,8 @@ static DECLARE_BITMAP(minors, N_SPI_MINORS);
|
||||
| SPI_LSB_FIRST | SPI_3WIRE | SPI_LOOP \
|
||||
| SPI_NO_CS | SPI_READY | SPI_TX_DUAL \
|
||||
| SPI_TX_QUAD | SPI_TX_OCTAL | SPI_RX_DUAL \
|
||||
| SPI_RX_QUAD | SPI_RX_OCTAL)
|
||||
| SPI_RX_QUAD | SPI_RX_OCTAL \
|
||||
| SPI_RX_CPHA_FLIP)
|
||||
|
||||
struct spidev_data {
|
||||
dev_t devt;
|
||||
@ -568,19 +569,20 @@ spidev_compat_ioctl(struct file *filp, unsigned int cmd, unsigned long arg)
|
||||
|
||||
static int spidev_open(struct inode *inode, struct file *filp)
|
||||
{
|
||||
struct spidev_data *spidev;
|
||||
struct spidev_data *spidev = NULL, *iter;
|
||||
int status = -ENXIO;
|
||||
|
||||
mutex_lock(&device_list_lock);
|
||||
|
||||
list_for_each_entry(spidev, &device_list, device_entry) {
|
||||
if (spidev->devt == inode->i_rdev) {
|
||||
list_for_each_entry(iter, &device_list, device_entry) {
|
||||
if (iter->devt == inode->i_rdev) {
|
||||
status = 0;
|
||||
spidev = iter;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (status) {
|
||||
if (!spidev) {
|
||||
pr_debug("spidev: nothing for minor %d\n", iminor(inode));
|
||||
goto err_find_dev;
|
||||
}
|
||||
@ -693,25 +695,38 @@ static const struct spi_device_id spidev_spi_ids[] = {
|
||||
};
|
||||
MODULE_DEVICE_TABLE(spi, spidev_spi_ids);
|
||||
|
||||
#ifdef CONFIG_OF
|
||||
/*
|
||||
* spidev should never be referenced in DT without a specific compatible string,
|
||||
* it is a Linux implementation thing rather than a description of the hardware.
|
||||
*/
|
||||
static int spidev_of_check(struct device *dev)
|
||||
{
|
||||
if (device_property_match_string(dev, "compatible", "spidev") < 0)
|
||||
return 0;
|
||||
|
||||
dev_err(dev, "spidev listed directly in DT is not supported\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
static const struct of_device_id spidev_dt_ids[] = {
|
||||
{ .compatible = "rohm,dh2228fv" },
|
||||
{ .compatible = "lineartechnology,ltc2488" },
|
||||
{ .compatible = "semtech,sx1301" },
|
||||
{ .compatible = "lwn,bk4" },
|
||||
{ .compatible = "dh,dhcom-board" },
|
||||
{ .compatible = "menlo,m53cpld" },
|
||||
{ .compatible = "cisco,spi-petra" },
|
||||
{ .compatible = "micron,spi-authenta" },
|
||||
{ .compatible = "rohm,dh2228fv", .data = &spidev_of_check },
|
||||
{ .compatible = "lineartechnology,ltc2488", .data = &spidev_of_check },
|
||||
{ .compatible = "semtech,sx1301", .data = &spidev_of_check },
|
||||
{ .compatible = "lwn,bk4", .data = &spidev_of_check },
|
||||
{ .compatible = "dh,dhcom-board", .data = &spidev_of_check },
|
||||
{ .compatible = "menlo,m53cpld", .data = &spidev_of_check },
|
||||
{ .compatible = "cisco,spi-petra", .data = &spidev_of_check },
|
||||
{ .compatible = "micron,spi-authenta", .data = &spidev_of_check },
|
||||
{},
|
||||
};
|
||||
MODULE_DEVICE_TABLE(of, spidev_dt_ids);
|
||||
#endif
|
||||
|
||||
#ifdef CONFIG_ACPI
|
||||
|
||||
/* Dummy SPI devices not to be used in production systems */
|
||||
#define SPIDEV_ACPI_DUMMY 1
|
||||
static int spidev_acpi_check(struct device *dev)
|
||||
{
|
||||
dev_warn(dev, "do not use this driver in production systems!\n");
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const struct acpi_device_id spidev_acpi_ids[] = {
|
||||
/*
|
||||
@ -720,51 +735,29 @@ static const struct acpi_device_id spidev_acpi_ids[] = {
|
||||
* description of the connected peripheral and they should also use
|
||||
* a proper driver instead of poking directly to the SPI bus.
|
||||
*/
|
||||
{ "SPT0001", SPIDEV_ACPI_DUMMY },
|
||||
{ "SPT0002", SPIDEV_ACPI_DUMMY },
|
||||
{ "SPT0003", SPIDEV_ACPI_DUMMY },
|
||||
{ "SPT0001", (kernel_ulong_t)&spidev_acpi_check },
|
||||
{ "SPT0002", (kernel_ulong_t)&spidev_acpi_check },
|
||||
{ "SPT0003", (kernel_ulong_t)&spidev_acpi_check },
|
||||
{},
|
||||
};
|
||||
MODULE_DEVICE_TABLE(acpi, spidev_acpi_ids);
|
||||
|
||||
static void spidev_probe_acpi(struct spi_device *spi)
|
||||
{
|
||||
const struct acpi_device_id *id;
|
||||
|
||||
if (!has_acpi_companion(&spi->dev))
|
||||
return;
|
||||
|
||||
id = acpi_match_device(spidev_acpi_ids, &spi->dev);
|
||||
if (WARN_ON(!id))
|
||||
return;
|
||||
|
||||
if (id->driver_data == SPIDEV_ACPI_DUMMY)
|
||||
dev_warn(&spi->dev, "do not use this driver in production systems!\n");
|
||||
}
|
||||
#else
|
||||
static inline void spidev_probe_acpi(struct spi_device *spi) {}
|
||||
#endif
|
||||
|
||||
/*-------------------------------------------------------------------------*/
|
||||
|
||||
static int spidev_probe(struct spi_device *spi)
|
||||
{
|
||||
int (*match)(struct device *dev);
|
||||
struct spidev_data *spidev;
|
||||
int status;
|
||||
unsigned long minor;
|
||||
|
||||
/*
|
||||
* spidev should never be referenced in DT without a specific
|
||||
* compatible string, it is a Linux implementation thing
|
||||
* rather than a description of the hardware.
|
||||
*/
|
||||
if (spi->dev.of_node && of_device_is_compatible(spi->dev.of_node, "spidev")) {
|
||||
dev_err(&spi->dev, "spidev listed directly in DT is not supported\n");
|
||||
return -EINVAL;
|
||||
match = device_get_match_data(&spi->dev);
|
||||
if (match) {
|
||||
status = match(&spi->dev);
|
||||
if (status)
|
||||
return status;
|
||||
}
|
||||
|
||||
spidev_probe_acpi(spi);
|
||||
|
||||
/* Allocate driver data */
|
||||
spidev = kzalloc(sizeof(*spidev), GFP_KERNEL);
|
||||
if (!spidev)
|
||||
@ -832,8 +825,8 @@ static void spidev_remove(struct spi_device *spi)
|
||||
static struct spi_driver spidev_spi_driver = {
|
||||
.driver = {
|
||||
.name = "spidev",
|
||||
.of_match_table = of_match_ptr(spidev_dt_ids),
|
||||
.acpi_match_table = ACPI_PTR(spidev_acpi_ids),
|
||||
.of_match_table = spidev_dt_ids,
|
||||
.acpi_match_table = spidev_acpi_ids,
|
||||
},
|
||||
.probe = spidev_probe,
|
||||
.remove = spidev_remove,
|
||||
@ -856,7 +849,6 @@ static int __init spidev_init(void)
|
||||
* that will key udev/mdev to add/remove /dev nodes. Last, register
|
||||
* the driver which manages those device numbers.
|
||||
*/
|
||||
BUILD_BUG_ON(N_SPI_MINORS > 256);
|
||||
status = register_chrdev(SPIDEV_MAJOR, "spi", &spidev_fops);
|
||||
if (status < 0)
|
||||
return status;
|
||||
|
@ -347,6 +347,7 @@ extern struct spi_device *spi_new_ancillary_device(struct spi_device *spi, u8 ch
|
||||
* @max_message_size: function that returns the max message size for
|
||||
* a &spi_device; may be %NULL, so the default %SIZE_MAX will be used.
|
||||
* @io_mutex: mutex for physical bus access
|
||||
* @add_lock: mutex to avoid adding devices to the same chipselect
|
||||
* @bus_lock_spinlock: spinlock for SPI bus locking
|
||||
* @bus_lock_mutex: mutex for exclusion of multiple callers
|
||||
* @bus_lock_flag: indicates that the SPI bus is locked for exclusive use
|
||||
@ -361,6 +362,7 @@ extern struct spi_device *spi_new_ancillary_device(struct spi_device *spi, u8 ch
|
||||
* @transfer: adds a message to the controller's transfer queue.
|
||||
* @cleanup: frees controller-specific state
|
||||
* @can_dma: determine whether this controller supports DMA
|
||||
* @dma_map_dev: device which can be used for DMA mapping
|
||||
* @queued: whether this controller is providing an internal message queue
|
||||
* @kworker: pointer to thread struct for message pump
|
||||
* @pump_messages: work struct for scheduling work to the message pump
|
||||
|
@ -27,6 +27,7 @@
|
||||
#define SPI_TX_OCTAL _BITUL(13) /* transmit with 8 wires */
|
||||
#define SPI_RX_OCTAL _BITUL(14) /* receive with 8 wires */
|
||||
#define SPI_3WIRE_HIZ _BITUL(15) /* high impedance turnaround */
|
||||
#define SPI_RX_CPHA_FLIP _BITUL(16) /* flip CPHA on Rx only xfer */
|
||||
|
||||
/*
|
||||
* All the bits defined above should be covered by SPI_MODE_USER_MASK.
|
||||
@ -36,6 +37,6 @@
|
||||
* These bits must not overlap. A static assert check should make sure of that.
|
||||
* If adding extra bits, make sure to increase the bit index below as well.
|
||||
*/
|
||||
#define SPI_MODE_USER_MASK (_BITUL(16) - 1)
|
||||
#define SPI_MODE_USER_MASK (_BITUL(17) - 1)
|
||||
|
||||
#endif /* _UAPI_SPI_H */
|
||||
|
Loading…
Reference in New Issue
Block a user