mirror of https://gitee.com/openkylin/linux.git
MMC core:
- Add CMD13 polling for MMC IOCTLS with R1B response. - Add common DT properties for clk-phase-delays for various speed modes. - Fix size overflow for mmc gp-partitions. - Re-work HW reset for SDIO cards, which also includes a re-work for Marvell's WiFi mwifiex SDIO func driver. MMC host: - jz4740: Add support for X1000 and JZ4760. - jz4740: Add support for 8-bit bus and for low power mode. - mmci: Add support for HW busy timeout for the stm32_sdmmc variant. - owl-mmc: Add driver for Actions Semi Owl SoCs SD/MMC controller. - renesas_sdhi: Add support for r8a774b1. - sdhci_am654: Add support for Command Queuing Engine for J721E. - sdhci-milbeaut: Add driver for the Milbeaut SD controller. - sdhci-of-arasan: Add support for ZynqMP tap-delays. - sdhci-of-arasan: Add support for clk-phase-delays for SD cards. - sdhci-of-arasan: Add support for Intel LGM SDXC. - sdhci-of-aspeed: Allow inversion of the internal card detect signal. - sdhci-of-esdhc: Fixup workaround for erratum A-008171 for tunings. - sdhci-of-at91: Improve support for calibration. - sdhci-pci: Add support for Intel JSL. - sdhci-pci: Add quirk for AMD SDHC Device 0x7906. - tmio: Enable support for erase/discard/trim requests. MMC/OMAP/pandora/wl1251: The TI wl1251 WiFi driver for SDIO on the OMAP openpandora board has been broken since v4.7. To fix the problems, changes have been made cross subsystems, but also to OMAP2 machine code and to openpandora DTS files, as summarized below. Note that, relevant changes have been tagged for stable. - mmc/wl1251: Re-introduce lost SDIO quirks and vendor-id for wl1251 - omap/omap_hsmmc: Remove redundant platform config for openpandora - omap_hsmmc: Initialize non-std SDIO card for wl1251 for pandora - omap/dts/pandora: Specify wl1251 through a child node of mmc3 - wl1251: Add devicetree support for TI wl1251 SDIO -----BEGIN PGP SIGNATURE----- iQJLBAABCgA1FiEEugLDXPmKSktSkQsV/iaEJXNYjCkFAl3dRnAXHHVsZi5oYW5z c29uQGxpbmFyby5vcmcACgkQ/iaEJXNYjClQqxAA0QFBhpdXMXPUv74wduxup4u/ JFKUBX81CvZ0XoES9Q6N96ZM+umaWo7mxKE0PhBPX4ScdcGvQTL7f7kqWamtUtd1 bKwgdEau1hyAAsAUMEUONw5+7Vyg3i5WXogItg+RqAefUXwW/KxeNBfBBC3kY/az 4/aRBFW2N52LMqOWndAbg2EdaX5nrTl72NXwecPXEsAPid0yoz7YUpBysLDdOVAm 3PYjz2R56wUfgXRpcMUWmSSkgAT6yicPAXtpGUsxZG5oFlOeEjaqB6ggDz9bj50z aBNXBpx+vFKiXKqX3vru67tqKo8nS7JTS9qiTDIzQjsS47MML9dOEpKQlTBRklOg z7XqQ8X8mic/YvQrxZO8c/pvZls/QZrssLfdy0edS/r4tbNC/NfPEkdi6wiXojPL QZySR1Mt3De/3daZSqxYQcca82GVCTIpV/3wsRML/pAqsDaM/ZAHRbfYJ6ZdAHUN bnZcr0gJ0uKDdwfaKewt5t/3TDftE5kb0UndlAi5xwffeKELQ0xTGrEaFlf0D7vr QCIJBI6HIQ/AIuMDRg1LIV1aTmJ3Q0oGxLKNWf1NKjDZfpNkVqWSSkrx5GxQuq30 AdR4prTtAlzkCKrGLNDUer7hMCutv/UIXbmbuuoPXjR/qLCR41qLYYg/AC2KUOTW ZT8iZBdcoK6frMzmRXU= =WOaQ -----END PGP SIGNATURE----- Merge tag 'mmc-v5.5' of git://git.kernel.org/pub/scm/linux/kernel/git/ulfh/mmc Pull MMC updates from Ulf Hansson: "These are the updates for MMC and MEMSTICK for v5.5. Note that this also contains quite some additional changes reaching beyond both the MMC and MEMSTICK subsystems. This is primarily because of fixing an old regression for a WiFi driver based on the SDIO interface on an OMAP openpandora board MMC core: - Add CMD13 polling for MMC IOCTLS with R1B response. - Add common DT properties for clk-phase-delays for various speed modes. - Fix size overflow for mmc gp-partitions. - Re-work HW reset for SDIO cards, which also includes a re-work for Marvell's WiFi mwifiex SDIO func driver. MMC host: - jz4740: Add support for X1000 and JZ4760. - jz4740: Add support for 8-bit bus and for low power mode. - mmci: Add support for HW busy timeout for the stm32_sdmmc variant. - owl-mmc: Add driver for Actions Semi Owl SoCs SD/MMC controller. - renesas_sdhi: Add support for r8a774b1. - sdhci_am654: Add support for Command Queuing Engine for J721E. - sdhci-milbeaut: Add driver for the Milbeaut SD controller. - sdhci-of-arasan: Add support for ZynqMP tap-delays. - sdhci-of-arasan: Add support for clk-phase-delays for SD cards. - sdhci-of-arasan: Add support for Intel LGM SDXC. - sdhci-of-aspeed: Allow inversion of the internal card detect signal. - sdhci-of-esdhc: Fixup workaround for erratum A-008171 for tunings. - sdhci-of-at91: Improve support for calibration. - sdhci-pci: Add support for Intel JSL. - sdhci-pci: Add quirk for AMD SDHC Device 0x7906. - tmio: Enable support for erase/discard/trim requests. MMC/OMAP/pandora/wl1251: The TI wl1251 WiFi driver for SDIO on the OMAP openpandora board has been broken since v4.7. To fix the problems, changes have been made cross subsystems, but also to OMAP2 machine code and to openpandora DTS files, as summarized below. Relevant changes have been tagged for stable. - mmc/wl1251: Re-introduce lost SDIO quirks and vendor-id for wl1251 - omap/omap_hsmmc: Remove redundant platform config for openpandora - omap_hsmmc: Initialize non-std SDIO card for wl1251 for pandora - omap/dts/pandora: Specify wl1251 through a child node of mmc3 - wl1251: Add devicetree support for TI wl1251 SDIO" * tag 'mmc-v5.5' of git://git.kernel.org/pub/scm/linux/kernel/git/ulfh/mmc: (73 commits) dt-bindings: mmc: Correct the type of the clk phase properties Revert "mmc: tmio: remove workaround for NON_REMOVABLE" memstick: Fix Kconfig indentation mmc: sdhci-of-arasan: Add support for ZynqMP Platform Tap Delays Setup dt-bindings: mmc: arasan: Document 'xlnx,zynqmp-8.9a' controller firmware: xilinx: Add SDIO Tap Delay nodes mmc: sdhci-of-arasan: Add support to set clock phase delays for SD dt-bindings: mmc: Add optional generic properties for mmc mmc: sdhci-of-arasan: Add sampling clock for a phy to use dt-bindings: mmc: arasan: Update Documentation for the input clock mmc: sdhci-of-arasan: Separate out clk related data to another structure mmc: sdhci: Fix grammar in warning message mmc: sdhci-of-aspeed: add inversion signal presence mmc: sdhci-of-aspeed: enable CONFIG_MMC_SDHCI_IO_ACCESSORS mmc: sdhci_am654: Add Support for Command Queuing Engine to J721E mmc: core: Fix size overflow for mmc partitions mmc: tmio: Add MMC_CAP_ERASE to allow erase/discard/trim requests net: wireless: ti: remove local VENDOR_ID and DEVICE_ID definitions net: wireless: ti: wl1251 use new SDIO_VENDOR_ID_TI_WL1251 definition mmc: core: fix wl1251 sdio quirks ...
This commit is contained in:
commit
0e45384cec
|
@ -15,10 +15,15 @@ Required Properties:
|
|||
- "arasan,sdhci-5.1": generic Arasan SDHCI 5.1 PHY
|
||||
- "rockchip,rk3399-sdhci-5.1", "arasan,sdhci-5.1": rk3399 eMMC PHY
|
||||
For this device it is strongly suggested to include arasan,soc-ctl-syscon.
|
||||
- "xlnx,zynqmp-8.9a": ZynqMP SDHCI 8.9a PHY
|
||||
For this device it is strongly suggested to include clock-output-names and
|
||||
#clock-cells.
|
||||
- "ti,am654-sdhci-5.1", "arasan,sdhci-5.1": TI AM654 MMC PHY
|
||||
Note: This binding has been deprecated and moved to [5].
|
||||
- "intel,lgm-sdhci-5.1-emmc", "arasan,sdhci-5.1": Intel LGM eMMC PHY
|
||||
For this device it is strongly suggested to include arasan,soc-ctl-syscon.
|
||||
- "intel,lgm-sdhci-5.1-sdxc", "arasan,sdhci-5.1": Intel LGM SDXC PHY
|
||||
For this device it is strongly suggested to include arasan,soc-ctl-syscon.
|
||||
|
||||
[5] Documentation/devicetree/bindings/mmc/sdhci-am654.txt
|
||||
|
||||
|
@ -38,15 +43,19 @@ Optional Properties:
|
|||
- clock-output-names: If specified, this will be the name of the card clock
|
||||
which will be exposed by this device. Required if #clock-cells is
|
||||
specified.
|
||||
- #clock-cells: If specified this should be the value <0>. With this property
|
||||
in place we will export a clock representing the Card Clock. This clock
|
||||
is expected to be consumed by our PHY. You must also specify
|
||||
- #clock-cells: If specified this should be the value <0> or <1>. With this
|
||||
property in place we will export one or two clocks representing the Card
|
||||
Clock. These clocks are expected to be consumed by our PHY.
|
||||
- xlnx,fails-without-test-cd: when present, the controller doesn't work when
|
||||
the CD line is not connected properly, and the line is not connected
|
||||
properly. Test mode can be used to force the controller to function.
|
||||
- xlnx,int-clock-stable-broken: when present, the controller always reports
|
||||
that the internal clock is stable even when it is not.
|
||||
|
||||
- xlnx,mio-bank: When specified, this will indicate the MIO bank number in
|
||||
which the command and data lines are configured. If not specified, driver
|
||||
will assume this as 0.
|
||||
|
||||
Example:
|
||||
sdhci@e0100000 {
|
||||
compatible = "arasan,sdhci-8.9a";
|
||||
|
@ -83,6 +92,18 @@ Example:
|
|||
#clock-cells = <0>;
|
||||
};
|
||||
|
||||
sdhci: mmc@ff160000 {
|
||||
compatible = "xlnx,zynqmp-8.9a", "arasan,sdhci-8.9a";
|
||||
interrupt-parent = <&gic>;
|
||||
interrupts = <0 48 4>;
|
||||
reg = <0x0 0xff160000 0x0 0x1000>;
|
||||
clocks = <&clk200>, <&clk200>;
|
||||
clock-names = "clk_xin", "clk_ahb";
|
||||
clock-output-names = "clk_out_sd0", "clk_in_sd0";
|
||||
#clock-cells = <1>;
|
||||
clk-phase-sd-hs = <63>, <72>;
|
||||
};
|
||||
|
||||
emmc: sdhci@ec700000 {
|
||||
compatible = "intel,lgm-sdhci-5.1-emmc", "arasan,sdhci-5.1";
|
||||
reg = <0xec700000 0x300>;
|
||||
|
@ -97,3 +118,18 @@ Example:
|
|||
phy-names = "phy_arasan";
|
||||
arasan,soc-ctl-syscon = <&sysconf>;
|
||||
};
|
||||
|
||||
sdxc: sdhci@ec600000 {
|
||||
compatible = "arasan,sdhci-5.1", "intel,lgm-sdhci-5.1-sdxc";
|
||||
reg = <0xec600000 0x300>;
|
||||
interrupt-parent = <&ioapic1>;
|
||||
interrupts = <43 1>;
|
||||
clocks = <&cgu0 LGM_CLK_SDIO>, <&cgu0 LGM_CLK_NGI>,
|
||||
<&cgu0 LGM_GCLK_SDXC>;
|
||||
clock-names = "clk_xin", "clk_ahb", "gate";
|
||||
clock-output-names = "sdxc_cardclock";
|
||||
#clock-cells = <0>;
|
||||
phys = <&sdxc_phy>;
|
||||
phy-names = "phy_arasan";
|
||||
arasan,soc-ctl-syscon = <&sysconf>;
|
||||
};
|
||||
|
|
|
@ -18,6 +18,9 @@ Required properties:
|
|||
"fsl,imx6ull-usdhc"
|
||||
"fsl,imx7d-usdhc"
|
||||
"fsl,imx7ulp-usdhc"
|
||||
"fsl,imx8mq-usdhc"
|
||||
"fsl,imx8mm-usdhc"
|
||||
"fsl,imx8mn-usdhc"
|
||||
"fsl,imx8qxp-usdhc"
|
||||
|
||||
Optional properties:
|
||||
|
|
|
@ -1,14 +1,16 @@
|
|||
* Ingenic JZ47xx MMC controllers
|
||||
* Ingenic XBurst MMC controllers
|
||||
|
||||
This file documents the device tree properties used for the MMC controller in
|
||||
Ingenic JZ4740/JZ4780 SoCs. These are in addition to the core MMC properties
|
||||
described in mmc.txt.
|
||||
Ingenic JZ4740/JZ4760/JZ4780/X1000 SoCs. These are in addition to the core MMC
|
||||
properties described in mmc.txt.
|
||||
|
||||
Required properties:
|
||||
- compatible: Should be one of the following:
|
||||
- "ingenic,jz4740-mmc" for the JZ4740
|
||||
- "ingenic,jz4725b-mmc" for the JZ4725B
|
||||
- "ingenic,jz4760-mmc" for the JZ4760
|
||||
- "ingenic,jz4780-mmc" for the JZ4780
|
||||
- "ingenic,x1000-mmc" for the X1000
|
||||
- reg: Should contain the MMC controller registers location and length.
|
||||
- interrupts: Should contain the interrupt specifier of the MMC controller.
|
||||
- clocks: Clock for the MMC controller.
|
||||
|
|
|
@ -333,6 +333,19 @@ patternProperties:
|
|||
required:
|
||||
- reg
|
||||
|
||||
"^clk-phase-(legacy|sd-hs|mmc-(hs|hs[24]00|ddr52)|uhs-(sdr(12|25|50|104)|ddr50))$":
|
||||
allOf:
|
||||
- $ref: /schemas/types.yaml#/definitions/uint32-array
|
||||
minItems: 2
|
||||
maxItems: 2
|
||||
items:
|
||||
minimum: 0
|
||||
maximum: 359
|
||||
description:
|
||||
Set the clock (phase) delays which are to be configured in the
|
||||
controller while switching to particular speed mode. These values
|
||||
are in pair of degrees.
|
||||
|
||||
dependencies:
|
||||
cd-debounce-delay-ms: [ cd-gpios ]
|
||||
fixed-emmc-driver-type: [ non-removable ]
|
||||
|
@ -351,6 +364,7 @@ examples:
|
|||
keep-power-in-suspend;
|
||||
wakeup-source;
|
||||
mmc-pwrseq = <&sdhci0_pwrseq>;
|
||||
clk-phase-sd-hs = <63>, <72>;
|
||||
};
|
||||
|
||||
- |
|
||||
|
|
|
@ -0,0 +1,59 @@
|
|||
# SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause)
|
||||
%YAML 1.2
|
||||
---
|
||||
$id: http://devicetree.org/schemas/mmc/owl-mmc.yaml#
|
||||
$schema: http://devicetree.org/meta-schemas/core.yaml#
|
||||
|
||||
title: Actions Semi Owl SoCs SD/MMC/SDIO controller
|
||||
|
||||
allOf:
|
||||
- $ref: "mmc-controller.yaml"
|
||||
|
||||
maintainers:
|
||||
- Manivannan Sadhasivam <manivannan.sadhasivam@linaro.org>
|
||||
|
||||
properties:
|
||||
compatible:
|
||||
const: actions,owl-mmc
|
||||
|
||||
reg:
|
||||
maxItems: 1
|
||||
|
||||
interrupts:
|
||||
maxItems: 1
|
||||
|
||||
clocks:
|
||||
minItems: 1
|
||||
|
||||
resets:
|
||||
maxItems: 1
|
||||
|
||||
dmas:
|
||||
maxItems: 1
|
||||
|
||||
dma-names:
|
||||
const: mmc
|
||||
|
||||
required:
|
||||
- compatible
|
||||
- reg
|
||||
- interrupts
|
||||
- clocks
|
||||
- resets
|
||||
- dmas
|
||||
- dma-names
|
||||
|
||||
examples:
|
||||
- |
|
||||
mmc0: mmc@e0330000 {
|
||||
compatible = "actions,owl-mmc";
|
||||
reg = <0x0 0xe0330000 0x0 0x4000>;
|
||||
interrupts = <0 42 4>;
|
||||
clocks = <&cmu 56>;
|
||||
resets = <&cmu 23>;
|
||||
dmas = <&dma 2>;
|
||||
dma-names = "mmc";
|
||||
bus-width = <4>;
|
||||
};
|
||||
|
||||
...
|
|
@ -11,6 +11,7 @@ Required properties:
|
|||
"renesas,sdhi-r8a7744" - SDHI IP on R8A7744 SoC
|
||||
"renesas,sdhi-r8a7745" - SDHI IP on R8A7745 SoC
|
||||
"renesas,sdhi-r8a774a1" - SDHI IP on R8A774A1 SoC
|
||||
"renesas,sdhi-r8a774b1" - SDHI IP on R8A774B1 SoC
|
||||
"renesas,sdhi-r8a774c0" - SDHI IP on R8A774C0 SoC
|
||||
"renesas,sdhi-r8a77470" - SDHI IP on R8A77470 SoC
|
||||
"renesas,sdhi-mmc-r8a77470" - SDHI/MMC IP on R8A77470 SoC
|
||||
|
|
|
@ -9,6 +9,11 @@ Required properties:
|
|||
- clocks: Phandlers to the clocks.
|
||||
- clock-names: Must be "hclock", "multclk", "baseclk";
|
||||
|
||||
Optional properties:
|
||||
- microchip,sdcal-inverted: when present, polarity on the SDCAL SoC pin is
|
||||
inverted. The default polarity for this signal is described in the datasheet.
|
||||
For instance on SAMA5D2, the pin is usually tied to the GND with a resistor
|
||||
and a capacitor (see "SDMMC I/O Calibration" chapter).
|
||||
|
||||
Example:
|
||||
|
||||
|
|
|
@ -0,0 +1,30 @@
|
|||
* SOCIONEXT Milbeaut SDHCI controller
|
||||
|
||||
This file documents differences between the core properties in mmc.txt
|
||||
and the properties used by the sdhci_milbeaut driver.
|
||||
|
||||
Required properties:
|
||||
- compatible: "socionext,milbeaut-m10v-sdhci-3.0"
|
||||
- clocks: Must contain an entry for each entry in clock-names. It is a
|
||||
list of phandles and clock-specifier pairs.
|
||||
See ../clocks/clock-bindings.txt for details.
|
||||
- clock-names: Should contain the following two entries:
|
||||
"iface" - clock used for sdhci interface
|
||||
"core" - core clock for sdhci controller
|
||||
|
||||
Optional properties:
|
||||
- fujitsu,cmd-dat-delay-select: boolean property indicating that this host
|
||||
requires the CMD_DAT_DELAY control to be enabled.
|
||||
|
||||
Example:
|
||||
sdhci3: mmc@1b010000 {
|
||||
compatible = "socionext,milbeaut-m10v-sdhci-3.0";
|
||||
reg = <0x1b010000 0x10000>;
|
||||
interrupts = <0 265 0x4>;
|
||||
voltage-ranges = <3300 3300>;
|
||||
bus-width = <4>;
|
||||
clocks = <&clk 7>, <&ahb_clk>;
|
||||
clock-names = "core", "iface";
|
||||
cap-sdio-irq;
|
||||
fujitsu,cmd-dat-delay-select;
|
||||
};
|
|
@ -35,3 +35,29 @@ Examples:
|
|||
ti,power-gpio = <&gpio3 23 GPIO_ACTIVE_HIGH>; /* 87 */
|
||||
};
|
||||
};
|
||||
|
||||
&mmc3 {
|
||||
vmmc-supply = <&wlan_en>;
|
||||
|
||||
bus-width = <4>;
|
||||
non-removable;
|
||||
ti,non-removable;
|
||||
cap-power-off-card;
|
||||
|
||||
pinctrl-names = "default";
|
||||
pinctrl-0 = <&mmc3_pins>;
|
||||
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
|
||||
wlan: wifi@1 {
|
||||
compatible = "ti,wl1251";
|
||||
|
||||
reg = <1>;
|
||||
|
||||
interrupt-parent = <&gpio1>;
|
||||
interrupts = <21 IRQ_TYPE_LEVEL_HIGH>; /* GPIO_21 */
|
||||
|
||||
ti,wl1251-has-eeprom;
|
||||
};
|
||||
};
|
||||
|
|
|
@ -1399,6 +1399,7 @@ F: drivers/clk/actions/
|
|||
F: drivers/clocksource/timer-owl*
|
||||
F: drivers/dma/owl-dma.c
|
||||
F: drivers/i2c/busses/i2c-owl.c
|
||||
F: drivers/mmc/host/owl-mmc.c
|
||||
F: drivers/pinctrl/actions/*
|
||||
F: drivers/soc/actions/
|
||||
F: include/dt-bindings/power/owl-*
|
||||
|
@ -1407,6 +1408,7 @@ F: Documentation/devicetree/bindings/arm/actions.yaml
|
|||
F: Documentation/devicetree/bindings/clock/actions,owl-cmu.txt
|
||||
F: Documentation/devicetree/bindings/dma/owl-dma.txt
|
||||
F: Documentation/devicetree/bindings/i2c/i2c-owl.txt
|
||||
F: Documentation/devicetree/bindings/mmc/owl-mmc.yaml
|
||||
F: Documentation/devicetree/bindings/pinctrl/actions,s900-pinctrl.txt
|
||||
F: Documentation/devicetree/bindings/power/actions,owl-sps.txt
|
||||
F: Documentation/devicetree/bindings/timer/actions,owl-timer.txt
|
||||
|
@ -17612,10 +17614,8 @@ S: Maintained
|
|||
F: drivers/hwmon/vt8231.c
|
||||
|
||||
VUB300 USB to SDIO/SD/MMC bridge chip
|
||||
M: Tony Olech <tony.olech@elandigitalsystems.com>
|
||||
L: linux-mmc@vger.kernel.org
|
||||
L: linux-usb@vger.kernel.org
|
||||
S: Supported
|
||||
S: Orphan
|
||||
F: drivers/mmc/host/vub300.c
|
||||
|
||||
W1 DALLAS'S 1-WIRE BUS
|
||||
|
|
|
@ -226,6 +226,17 @@ usb_host_5v: fixed-regulator-usb_host_5v {
|
|||
gpio = <&gpio6 4 GPIO_ACTIVE_HIGH>; /* GPIO_164 */
|
||||
};
|
||||
|
||||
/* wl1251 wifi+bt module */
|
||||
wlan_en: fixed-regulator-wg7210_en {
|
||||
compatible = "regulator-fixed";
|
||||
regulator-name = "vwlan";
|
||||
regulator-min-microvolt = <1800000>;
|
||||
regulator-max-microvolt = <1800000>;
|
||||
startup-delay-us = <50000>;
|
||||
enable-active-high;
|
||||
gpio = <&gpio1 23 GPIO_ACTIVE_HIGH>;
|
||||
};
|
||||
|
||||
/* wg7210 (wifi+bt module) 32k clock buffer */
|
||||
wg7210_32k: fixed-regulator-wg7210_32k {
|
||||
compatible = "regulator-fixed";
|
||||
|
@ -522,9 +533,30 @@ &mmc2 {
|
|||
/*wp-gpios = <&gpio4 31 GPIO_ACTIVE_HIGH>;*/ /* GPIO_127 */
|
||||
};
|
||||
|
||||
/* mmc3 is probed using pdata-quirks to pass wl1251 card data */
|
||||
&mmc3 {
|
||||
status = "disabled";
|
||||
vmmc-supply = <&wlan_en>;
|
||||
|
||||
bus-width = <4>;
|
||||
non-removable;
|
||||
ti,non-removable;
|
||||
cap-power-off-card;
|
||||
|
||||
pinctrl-names = "default";
|
||||
pinctrl-0 = <&mmc3_pins>;
|
||||
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
|
||||
wlan: wifi@1 {
|
||||
compatible = "ti,wl1251";
|
||||
|
||||
reg = <1>;
|
||||
|
||||
interrupt-parent = <&gpio1>;
|
||||
interrupts = <21 IRQ_TYPE_LEVEL_HIGH>; /* GPIO_21 */
|
||||
|
||||
ti,wl1251-has-eeprom;
|
||||
};
|
||||
};
|
||||
|
||||
/* bluetooth*/
|
||||
|
|
|
@ -216,9 +216,6 @@ obj-$(CONFIG_MACH_NOKIA_N8X0) += board-n8x0.o
|
|||
|
||||
# Platform specific device init code
|
||||
|
||||
omap-hsmmc-$(CONFIG_MMC_OMAP_HS) := hsmmc.o
|
||||
obj-y += $(omap-hsmmc-m) $(omap-hsmmc-y)
|
||||
|
||||
obj-y += omap_phy_internal.o
|
||||
|
||||
obj-$(CONFIG_MACH_OMAP2_TUSB6010) += usb-tusb6010.o
|
||||
|
|
|
@ -352,7 +352,6 @@ void omap_pcs_legacy_init(int irq, void (*rearm)(void));
|
|||
struct omap_sdrc_params;
|
||||
extern void omap_sdrc_init(struct omap_sdrc_params *sdrc_cs0,
|
||||
struct omap_sdrc_params *sdrc_cs1);
|
||||
struct omap2_hsmmc_info;
|
||||
extern void omap_reserve(void);
|
||||
|
||||
struct omap_hwmod;
|
||||
|
|
|
@ -1,171 +0,0 @@
|
|||
// SPDX-License-Identifier: GPL-2.0-only
|
||||
/*
|
||||
* linux/arch/arm/mach-omap2/hsmmc.c
|
||||
*
|
||||
* Copyright (C) 2007-2008 Texas Instruments
|
||||
* Copyright (C) 2008 Nokia Corporation
|
||||
* Author: Texas Instruments
|
||||
*/
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/string.h>
|
||||
#include <linux/delay.h>
|
||||
#include <linux/mmc/host.h>
|
||||
#include <linux/platform_data/hsmmc-omap.h>
|
||||
|
||||
#include "soc.h"
|
||||
#include "omap_device.h"
|
||||
|
||||
#include "hsmmc.h"
|
||||
#include "control.h"
|
||||
|
||||
#if IS_ENABLED(CONFIG_MMC_OMAP_HS)
|
||||
|
||||
static u16 control_pbias_offset;
|
||||
static u16 control_devconf1_offset;
|
||||
|
||||
#define HSMMC_NAME_LEN 9
|
||||
|
||||
static int __init omap_hsmmc_pdata_init(struct omap2_hsmmc_info *c,
|
||||
struct omap_hsmmc_platform_data *mmc)
|
||||
{
|
||||
char *hc_name;
|
||||
|
||||
hc_name = kzalloc(HSMMC_NAME_LEN + 1, GFP_KERNEL);
|
||||
if (!hc_name)
|
||||
return -ENOMEM;
|
||||
|
||||
snprintf(hc_name, (HSMMC_NAME_LEN + 1), "mmc%islot%i", c->mmc, 1);
|
||||
mmc->name = hc_name;
|
||||
mmc->caps = c->caps;
|
||||
mmc->reg_offset = 0;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int omap_hsmmc_done;
|
||||
|
||||
void omap_hsmmc_late_init(struct omap2_hsmmc_info *c)
|
||||
{
|
||||
struct platform_device *pdev;
|
||||
int res;
|
||||
|
||||
if (omap_hsmmc_done)
|
||||
return;
|
||||
|
||||
omap_hsmmc_done = 1;
|
||||
|
||||
for (; c->mmc; c++) {
|
||||
pdev = c->pdev;
|
||||
if (!pdev)
|
||||
continue;
|
||||
res = omap_device_register(pdev);
|
||||
if (res)
|
||||
pr_err("Could not late init MMC\n");
|
||||
}
|
||||
}
|
||||
|
||||
#define MAX_OMAP_MMC_HWMOD_NAME_LEN 16
|
||||
|
||||
static void __init omap_hsmmc_init_one(struct omap2_hsmmc_info *hsmmcinfo,
|
||||
int ctrl_nr)
|
||||
{
|
||||
struct omap_hwmod *oh;
|
||||
struct omap_hwmod *ohs[1];
|
||||
struct omap_device *od;
|
||||
struct platform_device *pdev;
|
||||
char oh_name[MAX_OMAP_MMC_HWMOD_NAME_LEN];
|
||||
struct omap_hsmmc_platform_data *mmc_data;
|
||||
struct omap_hsmmc_dev_attr *mmc_dev_attr;
|
||||
char *name;
|
||||
int res;
|
||||
|
||||
mmc_data = kzalloc(sizeof(*mmc_data), GFP_KERNEL);
|
||||
if (!mmc_data)
|
||||
return;
|
||||
|
||||
res = omap_hsmmc_pdata_init(hsmmcinfo, mmc_data);
|
||||
if (res < 0)
|
||||
goto free_mmc;
|
||||
|
||||
name = "omap_hsmmc";
|
||||
res = snprintf(oh_name, MAX_OMAP_MMC_HWMOD_NAME_LEN,
|
||||
"mmc%d", ctrl_nr);
|
||||
WARN(res >= MAX_OMAP_MMC_HWMOD_NAME_LEN,
|
||||
"String buffer overflow in MMC%d device setup\n", ctrl_nr);
|
||||
|
||||
oh = omap_hwmod_lookup(oh_name);
|
||||
if (!oh) {
|
||||
pr_err("Could not look up %s\n", oh_name);
|
||||
goto free_name;
|
||||
}
|
||||
ohs[0] = oh;
|
||||
if (oh->dev_attr != NULL) {
|
||||
mmc_dev_attr = oh->dev_attr;
|
||||
mmc_data->controller_flags = mmc_dev_attr->flags;
|
||||
}
|
||||
|
||||
pdev = platform_device_alloc(name, ctrl_nr - 1);
|
||||
if (!pdev) {
|
||||
pr_err("Could not allocate pdev for %s\n", name);
|
||||
goto free_name;
|
||||
}
|
||||
dev_set_name(&pdev->dev, "%s.%d", pdev->name, pdev->id);
|
||||
|
||||
od = omap_device_alloc(pdev, ohs, 1);
|
||||
if (IS_ERR(od)) {
|
||||
pr_err("Could not allocate od for %s\n", name);
|
||||
goto put_pdev;
|
||||
}
|
||||
|
||||
res = platform_device_add_data(pdev, mmc_data,
|
||||
sizeof(struct omap_hsmmc_platform_data));
|
||||
if (res) {
|
||||
pr_err("Could not add pdata for %s\n", name);
|
||||
goto put_pdev;
|
||||
}
|
||||
|
||||
hsmmcinfo->pdev = pdev;
|
||||
|
||||
res = omap_device_register(pdev);
|
||||
if (res) {
|
||||
pr_err("Could not register od for %s\n", name);
|
||||
goto free_od;
|
||||
}
|
||||
|
||||
goto free_mmc;
|
||||
|
||||
free_od:
|
||||
omap_device_delete(od);
|
||||
|
||||
put_pdev:
|
||||
platform_device_put(pdev);
|
||||
|
||||
free_name:
|
||||
kfree(mmc_data->name);
|
||||
|
||||
free_mmc:
|
||||
kfree(mmc_data);
|
||||
}
|
||||
|
||||
void __init omap_hsmmc_init(struct omap2_hsmmc_info *controllers)
|
||||
{
|
||||
if (omap_hsmmc_done)
|
||||
return;
|
||||
|
||||
omap_hsmmc_done = 1;
|
||||
|
||||
if (cpu_is_omap2430()) {
|
||||
control_pbias_offset = OMAP243X_CONTROL_PBIAS_LITE;
|
||||
control_devconf1_offset = OMAP243X_CONTROL_DEVCONF1;
|
||||
} else {
|
||||
control_pbias_offset = OMAP343X_CONTROL_PBIAS_LITE;
|
||||
control_devconf1_offset = OMAP343X_CONTROL_DEVCONF1;
|
||||
}
|
||||
|
||||
for (; controllers->mmc; controllers++)
|
||||
omap_hsmmc_init_one(controllers, controllers->mmc);
|
||||
|
||||
}
|
||||
|
||||
#endif
|
|
@ -1,32 +0,0 @@
|
|||
/* SPDX-License-Identifier: GPL-2.0-only */
|
||||
/*
|
||||
* MMC definitions for OMAP2
|
||||
*/
|
||||
|
||||
struct mmc_card;
|
||||
|
||||
struct omap2_hsmmc_info {
|
||||
u8 mmc; /* controller 1/2/3 */
|
||||
u32 caps; /* 4/8 wires and any additional host
|
||||
* capabilities OR'd (ref. linux/mmc/host.h) */
|
||||
struct platform_device *pdev; /* mmc controller instance */
|
||||
/* init some special card */
|
||||
void (*init_card)(struct mmc_card *card);
|
||||
};
|
||||
|
||||
#if IS_ENABLED(CONFIG_MMC_OMAP_HS)
|
||||
|
||||
void omap_hsmmc_init(struct omap2_hsmmc_info *);
|
||||
void omap_hsmmc_late_init(struct omap2_hsmmc_info *);
|
||||
|
||||
#else
|
||||
|
||||
static inline void omap_hsmmc_init(struct omap2_hsmmc_info *info)
|
||||
{
|
||||
}
|
||||
|
||||
static inline void omap_hsmmc_late_init(struct omap2_hsmmc_info *info)
|
||||
{
|
||||
}
|
||||
|
||||
#endif
|
|
@ -7,7 +7,6 @@
|
|||
#include <linux/clk.h>
|
||||
#include <linux/davinci_emac.h>
|
||||
#include <linux/gpio.h>
|
||||
#include <linux/gpio/machine.h>
|
||||
#include <linux/init.h>
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/of_platform.h>
|
||||
|
@ -33,7 +32,6 @@
|
|||
#include "omap_device.h"
|
||||
#include "omap-secure.h"
|
||||
#include "soc.h"
|
||||
#include "hsmmc.h"
|
||||
|
||||
static struct omap_hsmmc_platform_data __maybe_unused mmc_pdata[2];
|
||||
|
||||
|
@ -300,118 +298,15 @@ static void __init omap3_logicpd_torpedo_init(void)
|
|||
}
|
||||
|
||||
/* omap3pandora legacy devices */
|
||||
#define PANDORA_WIFI_IRQ_GPIO 21
|
||||
#define PANDORA_WIFI_NRESET_GPIO 23
|
||||
|
||||
static struct platform_device pandora_backlight = {
|
||||
.name = "pandora-backlight",
|
||||
.id = -1,
|
||||
};
|
||||
|
||||
static struct regulator_consumer_supply pandora_vmmc3_supply[] = {
|
||||
REGULATOR_SUPPLY("vmmc", "omap_hsmmc.2"),
|
||||
};
|
||||
|
||||
static struct regulator_init_data pandora_vmmc3 = {
|
||||
.constraints = {
|
||||
.valid_ops_mask = REGULATOR_CHANGE_STATUS,
|
||||
},
|
||||
.num_consumer_supplies = ARRAY_SIZE(pandora_vmmc3_supply),
|
||||
.consumer_supplies = pandora_vmmc3_supply,
|
||||
};
|
||||
|
||||
static struct fixed_voltage_config pandora_vwlan = {
|
||||
.supply_name = "vwlan",
|
||||
.microvolts = 1800000, /* 1.8V */
|
||||
.startup_delay = 50000, /* 50ms */
|
||||
.init_data = &pandora_vmmc3,
|
||||
};
|
||||
|
||||
static struct platform_device pandora_vwlan_device = {
|
||||
.name = "reg-fixed-voltage",
|
||||
.id = 1,
|
||||
.dev = {
|
||||
.platform_data = &pandora_vwlan,
|
||||
},
|
||||
};
|
||||
|
||||
static struct gpiod_lookup_table pandora_vwlan_gpiod_table = {
|
||||
.dev_id = "reg-fixed-voltage.1",
|
||||
.table = {
|
||||
/*
|
||||
* As this is a low GPIO number it should be at the first
|
||||
* GPIO bank.
|
||||
*/
|
||||
GPIO_LOOKUP("gpio-0-31", PANDORA_WIFI_NRESET_GPIO,
|
||||
NULL, GPIO_ACTIVE_HIGH),
|
||||
{ },
|
||||
},
|
||||
};
|
||||
|
||||
static void pandora_wl1251_init_card(struct mmc_card *card)
|
||||
{
|
||||
/*
|
||||
* We have TI wl1251 attached to MMC3. Pass this information to
|
||||
* SDIO core because it can't be probed by normal methods.
|
||||
*/
|
||||
if (card->type == MMC_TYPE_SDIO || card->type == MMC_TYPE_SD_COMBO) {
|
||||
card->quirks |= MMC_QUIRK_NONSTD_SDIO;
|
||||
card->cccr.wide_bus = 1;
|
||||
card->cis.vendor = 0x104c;
|
||||
card->cis.device = 0x9066;
|
||||
card->cis.blksize = 512;
|
||||
card->cis.max_dtr = 24000000;
|
||||
card->ocr = 0x80;
|
||||
}
|
||||
}
|
||||
|
||||
static struct omap2_hsmmc_info pandora_mmc3[] = {
|
||||
{
|
||||
.mmc = 3,
|
||||
.caps = MMC_CAP_4_BIT_DATA | MMC_CAP_POWER_OFF_CARD,
|
||||
.init_card = pandora_wl1251_init_card,
|
||||
},
|
||||
{} /* Terminator */
|
||||
};
|
||||
|
||||
static void __init pandora_wl1251_init(void)
|
||||
{
|
||||
struct wl1251_platform_data pandora_wl1251_pdata;
|
||||
int ret;
|
||||
|
||||
memset(&pandora_wl1251_pdata, 0, sizeof(pandora_wl1251_pdata));
|
||||
|
||||
pandora_wl1251_pdata.power_gpio = -1;
|
||||
|
||||
ret = gpio_request_one(PANDORA_WIFI_IRQ_GPIO, GPIOF_IN, "wl1251 irq");
|
||||
if (ret < 0)
|
||||
goto fail;
|
||||
|
||||
pandora_wl1251_pdata.irq = gpio_to_irq(PANDORA_WIFI_IRQ_GPIO);
|
||||
if (pandora_wl1251_pdata.irq < 0)
|
||||
goto fail_irq;
|
||||
|
||||
pandora_wl1251_pdata.use_eeprom = true;
|
||||
ret = wl1251_set_platform_data(&pandora_wl1251_pdata);
|
||||
if (ret < 0)
|
||||
goto fail_irq;
|
||||
|
||||
return;
|
||||
|
||||
fail_irq:
|
||||
gpio_free(PANDORA_WIFI_IRQ_GPIO);
|
||||
fail:
|
||||
pr_err("wl1251 board initialisation failed\n");
|
||||
}
|
||||
|
||||
static void __init omap3_pandora_legacy_init(void)
|
||||
{
|
||||
platform_device_register(&pandora_backlight);
|
||||
gpiod_add_lookup_table(&pandora_vwlan_gpiod_table);
|
||||
platform_device_register(&pandora_vwlan_device);
|
||||
omap_hsmmc_init(pandora_mmc3);
|
||||
omap_hsmmc_late_init(pandora_mmc3);
|
||||
pandora_wl1251_init();
|
||||
}
|
||||
#endif /* CONFIG_ARCH_OMAP3 */
|
||||
|
||||
|
|
|
@ -6,16 +6,16 @@
|
|||
comment "MemoryStick drivers"
|
||||
|
||||
config MEMSTICK_UNSAFE_RESUME
|
||||
bool "Allow unsafe resume (DANGEROUS)"
|
||||
help
|
||||
If you say Y here, the MemoryStick layer will assume that all
|
||||
cards stayed in their respective slots during the suspend. The
|
||||
normal behaviour is to remove them at suspend and
|
||||
redetecting them at resume. Breaking this assumption will
|
||||
in most cases result in data corruption.
|
||||
bool "Allow unsafe resume (DANGEROUS)"
|
||||
help
|
||||
If you say Y here, the MemoryStick layer will assume that all
|
||||
cards stayed in their respective slots during the suspend. The
|
||||
normal behaviour is to remove them at suspend and
|
||||
redetecting them at resume. Breaking this assumption will
|
||||
in most cases result in data corruption.
|
||||
|
||||
This option is usually just for embedded systems which use
|
||||
a MemoryStick card for rootfs. Most people should say N here.
|
||||
This option is usually just for embedded systems which use
|
||||
a MemoryStick card for rootfs. Most people should say N here.
|
||||
|
||||
config MSPRO_BLOCK
|
||||
tristate "MemoryStick Pro block device driver"
|
||||
|
|
|
@ -18,7 +18,7 @@ config MEMSTICK_TIFM_MS
|
|||
'Misc devices: TI Flash Media PCI74xx/PCI76xx host adapter support
|
||||
(TIFM_7XX1)'.
|
||||
|
||||
To compile this driver as a module, choose M here: the
|
||||
To compile this driver as a module, choose M here: the
|
||||
module will be called tifm_ms.
|
||||
|
||||
config MEMSTICK_JMICRON_38X
|
||||
|
@ -29,7 +29,7 @@ config MEMSTICK_JMICRON_38X
|
|||
Say Y here if you want to be able to access MemoryStick cards with
|
||||
the JMicron(R) JMB38X MemoryStick card reader.
|
||||
|
||||
To compile this driver as a module, choose M here: the
|
||||
To compile this driver as a module, choose M here: the
|
||||
module will be called jmb38x_ms.
|
||||
|
||||
config MEMSTICK_R592
|
||||
|
|
|
@ -433,13 +433,13 @@ static int jmb38x_ms_issue_cmd(struct memstick_host *msh)
|
|||
writel(((1 << 16) & BLOCK_COUNT_MASK)
|
||||
| (data_len & BLOCK_SIZE_MASK),
|
||||
host->addr + BLOCK);
|
||||
t_val = readl(host->addr + INT_STATUS_ENABLE);
|
||||
t_val |= host->req->data_dir == READ
|
||||
? INT_STATUS_FIFO_RRDY
|
||||
: INT_STATUS_FIFO_WRDY;
|
||||
t_val = readl(host->addr + INT_STATUS_ENABLE);
|
||||
t_val |= host->req->data_dir == READ
|
||||
? INT_STATUS_FIFO_RRDY
|
||||
: INT_STATUS_FIFO_WRDY;
|
||||
|
||||
writel(t_val, host->addr + INT_STATUS_ENABLE);
|
||||
writel(t_val, host->addr + INT_SIGNAL_ENABLE);
|
||||
writel(t_val, host->addr + INT_STATUS_ENABLE);
|
||||
writel(t_val, host->addr + INT_SIGNAL_ENABLE);
|
||||
} else {
|
||||
cmd &= ~(TPC_DATA_SEL | 0xf);
|
||||
host->cmd_flags |= REG_DATA;
|
||||
|
|
|
@ -408,38 +408,6 @@ static int mmc_blk_ioctl_copy_to_user(struct mmc_ioc_cmd __user *ic_ptr,
|
|||
return 0;
|
||||
}
|
||||
|
||||
static int ioctl_rpmb_card_status_poll(struct mmc_card *card, u32 *status,
|
||||
u32 retries_max)
|
||||
{
|
||||
int err;
|
||||
u32 retry_count = 0;
|
||||
|
||||
if (!status || !retries_max)
|
||||
return -EINVAL;
|
||||
|
||||
do {
|
||||
err = __mmc_send_status(card, status, 5);
|
||||
if (err)
|
||||
break;
|
||||
|
||||
if (!R1_STATUS(*status) &&
|
||||
(R1_CURRENT_STATE(*status) != R1_STATE_PRG))
|
||||
break; /* RPMB programming operation complete */
|
||||
|
||||
/*
|
||||
* Rechedule to give the MMC device a chance to continue
|
||||
* processing the previous command without being polled too
|
||||
* frequently.
|
||||
*/
|
||||
usleep_range(1000, 5000);
|
||||
} while (++retry_count < retries_max);
|
||||
|
||||
if (retry_count == retries_max)
|
||||
err = -EPERM;
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
static int ioctl_do_sanitize(struct mmc_card *card)
|
||||
{
|
||||
int err;
|
||||
|
@ -468,6 +436,58 @@ static int ioctl_do_sanitize(struct mmc_card *card)
|
|||
return err;
|
||||
}
|
||||
|
||||
static inline bool mmc_blk_in_tran_state(u32 status)
|
||||
{
|
||||
/*
|
||||
* Some cards mishandle the status bits, so make sure to check both the
|
||||
* busy indication and the card state.
|
||||
*/
|
||||
return status & R1_READY_FOR_DATA &&
|
||||
(R1_CURRENT_STATE(status) == R1_STATE_TRAN);
|
||||
}
|
||||
|
||||
static int card_busy_detect(struct mmc_card *card, unsigned int timeout_ms,
|
||||
u32 *resp_errs)
|
||||
{
|
||||
unsigned long timeout = jiffies + msecs_to_jiffies(timeout_ms);
|
||||
int err = 0;
|
||||
u32 status;
|
||||
|
||||
do {
|
||||
bool done = time_after(jiffies, timeout);
|
||||
|
||||
err = __mmc_send_status(card, &status, 5);
|
||||
if (err) {
|
||||
dev_err(mmc_dev(card->host),
|
||||
"error %d requesting status\n", err);
|
||||
return err;
|
||||
}
|
||||
|
||||
/* Accumulate any response error bits seen */
|
||||
if (resp_errs)
|
||||
*resp_errs |= status;
|
||||
|
||||
/*
|
||||
* Timeout if the device never becomes ready for data and never
|
||||
* leaves the program state.
|
||||
*/
|
||||
if (done) {
|
||||
dev_err(mmc_dev(card->host),
|
||||
"Card stuck in wrong state! %s status: %#x\n",
|
||||
__func__, status);
|
||||
return -ETIMEDOUT;
|
||||
}
|
||||
|
||||
/*
|
||||
* Some cards mishandle the status bits,
|
||||
* so make sure to check both the busy
|
||||
* indication and the card state.
|
||||
*/
|
||||
} while (!mmc_blk_in_tran_state(status));
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
static int __mmc_blk_ioctl_cmd(struct mmc_card *card, struct mmc_blk_data *md,
|
||||
struct mmc_blk_ioc_data *idata)
|
||||
{
|
||||
|
@ -477,7 +497,6 @@ static int __mmc_blk_ioctl_cmd(struct mmc_card *card, struct mmc_blk_data *md,
|
|||
struct scatterlist sg;
|
||||
int err;
|
||||
unsigned int target_part;
|
||||
u32 status = 0;
|
||||
|
||||
if (!card || !md || !idata)
|
||||
return -EINVAL;
|
||||
|
@ -611,16 +630,12 @@ static int __mmc_blk_ioctl_cmd(struct mmc_card *card, struct mmc_blk_data *md,
|
|||
|
||||
memcpy(&(idata->ic.response), cmd.resp, sizeof(cmd.resp));
|
||||
|
||||
if (idata->rpmb) {
|
||||
if (idata->rpmb || (cmd.flags & MMC_RSP_R1B)) {
|
||||
/*
|
||||
* Ensure RPMB command has completed by polling CMD13
|
||||
* Ensure RPMB/R1B command has completed by polling CMD13
|
||||
* "Send Status".
|
||||
*/
|
||||
err = ioctl_rpmb_card_status_poll(card, &status, 5);
|
||||
if (err)
|
||||
dev_err(mmc_dev(card->host),
|
||||
"%s: Card Status=0x%08X, error %d\n",
|
||||
__func__, status, err);
|
||||
err = card_busy_detect(card, MMC_BLK_TIMEOUT_MS, NULL);
|
||||
}
|
||||
|
||||
return err;
|
||||
|
@ -970,58 +985,6 @@ static unsigned int mmc_blk_data_timeout_ms(struct mmc_host *host,
|
|||
return ms;
|
||||
}
|
||||
|
||||
static inline bool mmc_blk_in_tran_state(u32 status)
|
||||
{
|
||||
/*
|
||||
* Some cards mishandle the status bits, so make sure to check both the
|
||||
* busy indication and the card state.
|
||||
*/
|
||||
return status & R1_READY_FOR_DATA &&
|
||||
(R1_CURRENT_STATE(status) == R1_STATE_TRAN);
|
||||
}
|
||||
|
||||
static int card_busy_detect(struct mmc_card *card, unsigned int timeout_ms,
|
||||
struct request *req, u32 *resp_errs)
|
||||
{
|
||||
unsigned long timeout = jiffies + msecs_to_jiffies(timeout_ms);
|
||||
int err = 0;
|
||||
u32 status;
|
||||
|
||||
do {
|
||||
bool done = time_after(jiffies, timeout);
|
||||
|
||||
err = __mmc_send_status(card, &status, 5);
|
||||
if (err) {
|
||||
pr_err("%s: error %d requesting status\n",
|
||||
req->rq_disk->disk_name, err);
|
||||
return err;
|
||||
}
|
||||
|
||||
/* Accumulate any response error bits seen */
|
||||
if (resp_errs)
|
||||
*resp_errs |= status;
|
||||
|
||||
/*
|
||||
* Timeout if the device never becomes ready for data and never
|
||||
* leaves the program state.
|
||||
*/
|
||||
if (done) {
|
||||
pr_err("%s: Card stuck in wrong state! %s %s status: %#x\n",
|
||||
mmc_hostname(card->host),
|
||||
req->rq_disk->disk_name, __func__, status);
|
||||
return -ETIMEDOUT;
|
||||
}
|
||||
|
||||
/*
|
||||
* Some cards mishandle the status bits,
|
||||
* so make sure to check both the busy
|
||||
* indication and the card state.
|
||||
*/
|
||||
} while (!mmc_blk_in_tran_state(status));
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
static int mmc_blk_reset(struct mmc_blk_data *md, struct mmc_host *host,
|
||||
int type)
|
||||
{
|
||||
|
@ -1671,7 +1634,7 @@ static int mmc_blk_fix_state(struct mmc_card *card, struct request *req)
|
|||
|
||||
mmc_blk_send_stop(card, timeout);
|
||||
|
||||
err = card_busy_detect(card, timeout, req, NULL);
|
||||
err = card_busy_detect(card, timeout, NULL);
|
||||
|
||||
mmc_retune_release(card->host);
|
||||
|
||||
|
@ -1895,7 +1858,7 @@ static int mmc_blk_card_busy(struct mmc_card *card, struct request *req)
|
|||
if (mmc_host_is_spi(card->host) || rq_data_dir(req) == READ)
|
||||
return 0;
|
||||
|
||||
err = card_busy_detect(card, MMC_BLK_TIMEOUT_MS, req, &status);
|
||||
err = card_busy_detect(card, MMC_BLK_TIMEOUT_MS, &status);
|
||||
|
||||
/*
|
||||
* Do not assume data transferred correctly if there are any error bits
|
||||
|
|
|
@ -1469,8 +1469,7 @@ void mmc_detach_bus(struct mmc_host *host)
|
|||
mmc_bus_put(host);
|
||||
}
|
||||
|
||||
static void _mmc_detect_change(struct mmc_host *host, unsigned long delay,
|
||||
bool cd_irq)
|
||||
void _mmc_detect_change(struct mmc_host *host, unsigned long delay, bool cd_irq)
|
||||
{
|
||||
/*
|
||||
* If the device is configured as wakeup, we prevent a new sleep for
|
||||
|
@ -2129,7 +2128,7 @@ int mmc_hw_reset(struct mmc_host *host)
|
|||
ret = host->bus_ops->hw_reset(host);
|
||||
mmc_bus_put(host);
|
||||
|
||||
if (ret)
|
||||
if (ret < 0)
|
||||
pr_warn("%s: tried to HW reset card, got error %d\n",
|
||||
mmc_hostname(host), ret);
|
||||
|
||||
|
@ -2297,11 +2296,8 @@ void mmc_rescan(struct work_struct *work)
|
|||
|
||||
mmc_bus_get(host);
|
||||
|
||||
/*
|
||||
* if there is a _removable_ card registered, check whether it is
|
||||
* still present
|
||||
*/
|
||||
if (host->bus_ops && !host->bus_dead && mmc_card_is_removable(host))
|
||||
/* Verify a registered card to be functional, else remove it. */
|
||||
if (host->bus_ops && !host->bus_dead)
|
||||
host->bus_ops->detect(host);
|
||||
|
||||
host->detect_change = 0;
|
||||
|
|
|
@ -70,6 +70,8 @@ void mmc_rescan(struct work_struct *work);
|
|||
void mmc_start_host(struct mmc_host *host);
|
||||
void mmc_stop_host(struct mmc_host *host);
|
||||
|
||||
void _mmc_detect_change(struct mmc_host *host, unsigned long delay,
|
||||
bool cd_irq);
|
||||
int _mmc_detect_card_removed(struct mmc_host *host);
|
||||
int mmc_detect_card_removed(struct mmc_host *host);
|
||||
|
||||
|
|
|
@ -297,7 +297,7 @@ static void mmc_manage_enhanced_area(struct mmc_card *card, u8 *ext_csd)
|
|||
}
|
||||
}
|
||||
|
||||
static void mmc_part_add(struct mmc_card *card, unsigned int size,
|
||||
static void mmc_part_add(struct mmc_card *card, u64 size,
|
||||
unsigned int part_cfg, char *name, int idx, bool ro,
|
||||
int area_type)
|
||||
{
|
||||
|
@ -313,7 +313,7 @@ static void mmc_manage_gp_partitions(struct mmc_card *card, u8 *ext_csd)
|
|||
{
|
||||
int idx;
|
||||
u8 hc_erase_grp_sz, hc_wp_grp_sz;
|
||||
unsigned int part_size;
|
||||
u64 part_size;
|
||||
|
||||
/*
|
||||
* General purpose partition feature support --
|
||||
|
@ -343,8 +343,7 @@ static void mmc_manage_gp_partitions(struct mmc_card *card, u8 *ext_csd)
|
|||
(ext_csd[EXT_CSD_GP_SIZE_MULT + idx * 3 + 1]
|
||||
<< 8) +
|
||||
ext_csd[EXT_CSD_GP_SIZE_MULT + idx * 3];
|
||||
part_size *= (size_t)(hc_erase_grp_sz *
|
||||
hc_wp_grp_sz);
|
||||
part_size *= (hc_erase_grp_sz * hc_wp_grp_sz);
|
||||
mmc_part_add(card, part_size << 19,
|
||||
EXT_CSD_PART_CONFIG_ACC_GP0 + idx,
|
||||
"gp%d", idx, false,
|
||||
|
@ -362,7 +361,7 @@ static void mmc_manage_gp_partitions(struct mmc_card *card, u8 *ext_csd)
|
|||
static int mmc_decode_ext_csd(struct mmc_card *card, u8 *ext_csd)
|
||||
{
|
||||
int err = 0, idx;
|
||||
unsigned int part_size;
|
||||
u64 part_size;
|
||||
struct device_node *np;
|
||||
bool broken_hpi = false;
|
||||
|
||||
|
|
|
@ -119,7 +119,14 @@ static const struct mmc_fixup mmc_ext_csd_fixups[] = {
|
|||
END_FIXUP
|
||||
};
|
||||
|
||||
|
||||
static const struct mmc_fixup sdio_fixup_methods[] = {
|
||||
SDIO_FIXUP(SDIO_VENDOR_ID_TI_WL1251, SDIO_DEVICE_ID_TI_WL1251,
|
||||
add_quirk, MMC_QUIRK_NONSTD_FUNC_IF),
|
||||
|
||||
SDIO_FIXUP(SDIO_VENDOR_ID_TI_WL1251, SDIO_DEVICE_ID_TI_WL1251,
|
||||
add_quirk, MMC_QUIRK_DISABLE_CD),
|
||||
|
||||
SDIO_FIXUP(SDIO_VENDOR_ID_TI, SDIO_DEVICE_ID_TI_WL1271,
|
||||
add_quirk, MMC_QUIRK_NONSTD_FUNC_IF),
|
||||
|
||||
|
|
|
@ -1048,9 +1048,35 @@ static int mmc_sdio_runtime_resume(struct mmc_host *host)
|
|||
return ret;
|
||||
}
|
||||
|
||||
/*
|
||||
* SDIO HW reset
|
||||
*
|
||||
* Returns 0 if the HW reset was executed synchronously, returns 1 if the HW
|
||||
* reset was asynchronously scheduled, else a negative error code.
|
||||
*/
|
||||
static int mmc_sdio_hw_reset(struct mmc_host *host)
|
||||
{
|
||||
mmc_power_cycle(host, host->card->ocr);
|
||||
struct mmc_card *card = host->card;
|
||||
|
||||
/*
|
||||
* In case the card is shared among multiple func drivers, reset the
|
||||
* card through a rescan work. In this way it will be removed and
|
||||
* re-detected, thus all func drivers becomes informed about it.
|
||||
*/
|
||||
if (atomic_read(&card->sdio_funcs_probed) > 1) {
|
||||
if (mmc_card_removed(card))
|
||||
return 1;
|
||||
host->rescan_entered = 0;
|
||||
mmc_card_set_removed(card);
|
||||
_mmc_detect_change(host, 0, false);
|
||||
return 1;
|
||||
}
|
||||
|
||||
/*
|
||||
* A single func driver has been probed, then let's skip the heavy
|
||||
* hotplug dance above and execute the reset immediately.
|
||||
*/
|
||||
mmc_power_cycle(host, card->ocr);
|
||||
return mmc_sdio_reinit_card(host);
|
||||
}
|
||||
|
||||
|
|
|
@ -138,6 +138,8 @@ static int sdio_bus_probe(struct device *dev)
|
|||
if (ret)
|
||||
return ret;
|
||||
|
||||
atomic_inc(&func->card->sdio_funcs_probed);
|
||||
|
||||
/* Unbound SDIO functions are always suspended.
|
||||
* During probe, the function is set active and the usage count
|
||||
* is incremented. If the driver supports runtime PM,
|
||||
|
@ -153,7 +155,10 @@ static int sdio_bus_probe(struct device *dev)
|
|||
/* Set the default block size so the driver is sure it's something
|
||||
* sensible. */
|
||||
sdio_claim_host(func);
|
||||
ret = sdio_set_block_size(func, 0);
|
||||
if (mmc_card_removed(func->card))
|
||||
ret = -ENOMEDIUM;
|
||||
else
|
||||
ret = sdio_set_block_size(func, 0);
|
||||
sdio_release_host(func);
|
||||
if (ret)
|
||||
goto disable_runtimepm;
|
||||
|
@ -165,6 +170,7 @@ static int sdio_bus_probe(struct device *dev)
|
|||
return 0;
|
||||
|
||||
disable_runtimepm:
|
||||
atomic_dec(&func->card->sdio_funcs_probed);
|
||||
if (func->card->host->caps & MMC_CAP_POWER_OFF_CARD)
|
||||
pm_runtime_put_noidle(dev);
|
||||
dev_pm_domain_detach(dev, false);
|
||||
|
@ -181,6 +187,7 @@ static int sdio_bus_remove(struct device *dev)
|
|||
pm_runtime_get_sync(dev);
|
||||
|
||||
drv->remove(func);
|
||||
atomic_dec(&func->card->sdio_funcs_probed);
|
||||
|
||||
if (func->irq_handler) {
|
||||
pr_warn("WARNING: driver %s did not remove its interrupt handler!\n",
|
||||
|
|
|
@ -159,6 +159,7 @@ config MMC_SDHCI_OF_ASPEED
|
|||
tristate "SDHCI OF support for the ASPEED SDHCI controller"
|
||||
depends on MMC_SDHCI_PLTFM
|
||||
depends on OF && OF_ADDRESS
|
||||
select MMC_SDHCI_IO_ACCESSORS
|
||||
help
|
||||
This selects the ASPEED Secure Digital Host Controller Interface.
|
||||
|
||||
|
@ -368,6 +369,17 @@ config MMC_SDHCI_F_SDH30
|
|||
|
||||
If unsure, say N.
|
||||
|
||||
config MMC_SDHCI_MILBEAUT
|
||||
tristate "SDHCI support for Socionext Milbeaut Serieas using F_SDH30"
|
||||
depends on MMC_SDHCI_PLTFM
|
||||
depends on OF
|
||||
help
|
||||
This selects the Secure Digital Host Controller Interface (SDHCI)
|
||||
Needed by Milbeaut SoC for MMC / SD / SDIO support.
|
||||
If you have a controller with this interface, say Y or M here.
|
||||
|
||||
If unsure, say N.
|
||||
|
||||
config MMC_SDHCI_IPROC
|
||||
tristate "SDHCI support for the BCM2835 & iProc SD/MMC Controller"
|
||||
depends on ARCH_BCM2835 || ARCH_BCM_IPROC || COMPILE_TEST
|
||||
|
@ -1011,6 +1023,7 @@ config MMC_SDHCI_AM654
|
|||
tristate "Support for the SDHCI Controller in TI's AM654 SOCs"
|
||||
depends on MMC_SDHCI_PLTFM && OF && REGMAP_MMIO
|
||||
select MMC_SDHCI_IO_ACCESSORS
|
||||
select MMC_CQHCI
|
||||
help
|
||||
This selects the Secure Digital Host Controller Interface (SDHCI)
|
||||
support present in TI's AM654 SOCs. The controller supports
|
||||
|
@ -1019,3 +1032,11 @@ config MMC_SDHCI_AM654
|
|||
If you have a controller with this interface, say Y or M here.
|
||||
|
||||
If unsure, say N.
|
||||
|
||||
config MMC_OWL
|
||||
tristate "Actions Semi Owl SD/MMC Host Controller support"
|
||||
depends on HAS_DMA
|
||||
depends on ARCH_ACTIONS || COMPILE_TEST
|
||||
help
|
||||
This selects support for the SD/MMC Host Controller on
|
||||
Actions Semi Owl SoCs.
|
||||
|
|
|
@ -21,6 +21,7 @@ obj-$(CONFIG_MMC_SDHCI_PXAV2) += sdhci-pxav2.o
|
|||
obj-$(CONFIG_MMC_SDHCI_S3C) += sdhci-s3c.o
|
||||
obj-$(CONFIG_MMC_SDHCI_SIRF) += sdhci-sirf.o
|
||||
obj-$(CONFIG_MMC_SDHCI_F_SDH30) += sdhci_f_sdh30.o
|
||||
obj-$(CONFIG_MMC_SDHCI_MILBEAUT) += sdhci-milbeaut.o
|
||||
obj-$(CONFIG_MMC_SDHCI_SPEAR) += sdhci-spear.o
|
||||
obj-$(CONFIG_MMC_SDHCI_AM654) += sdhci_am654.o
|
||||
obj-$(CONFIG_MMC_WBSD) += wbsd.o
|
||||
|
@ -73,6 +74,7 @@ obj-$(CONFIG_MMC_SUNXI) += sunxi-mmc.o
|
|||
obj-$(CONFIG_MMC_USDHI6ROL0) += usdhi6rol0.o
|
||||
obj-$(CONFIG_MMC_TOSHIBA_PCI) += toshsd.o
|
||||
obj-$(CONFIG_MMC_BCM2835) += bcm2835.o
|
||||
obj-$(CONFIG_MMC_OWL) += owl-mmc.o
|
||||
|
||||
obj-$(CONFIG_MMC_REALTEK_PCI) += rtsx_pci_sdmmc.o
|
||||
obj-$(CONFIG_MMC_REALTEK_USB) += rtsx_usb_sdmmc.o
|
||||
|
|
|
@ -2347,8 +2347,7 @@ static void atmci_cleanup_slot(struct atmel_mci_slot *slot,
|
|||
|
||||
static int atmci_configure_dma(struct atmel_mci *host)
|
||||
{
|
||||
host->dma.chan = dma_request_slave_channel_reason(&host->pdev->dev,
|
||||
"rxtx");
|
||||
host->dma.chan = dma_request_chan(&host->pdev->dev, "rxtx");
|
||||
|
||||
if (PTR_ERR(host->dma.chan) == -ENODEV) {
|
||||
struct mci_platform_data *pdata = host->pdev->dev.platform_data;
|
||||
|
|
|
@ -1357,7 +1357,6 @@ static int bcm2835_probe(struct platform_device *pdev)
|
|||
{
|
||||
struct device *dev = &pdev->dev;
|
||||
struct clk *clk;
|
||||
struct resource *iomem;
|
||||
struct bcm2835_host *host;
|
||||
struct mmc_host *mmc;
|
||||
const __be32 *regaddr_p;
|
||||
|
@ -1373,8 +1372,7 @@ static int bcm2835_probe(struct platform_device *pdev)
|
|||
host->pdev = pdev;
|
||||
spin_lock_init(&host->lock);
|
||||
|
||||
iomem = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
||||
host->ioaddr = devm_ioremap_resource(dev, iomem);
|
||||
host->ioaddr = devm_platform_ioremap_resource(pdev, 0);
|
||||
if (IS_ERR(host->ioaddr)) {
|
||||
ret = PTR_ERR(host->ioaddr);
|
||||
goto err;
|
||||
|
|
|
@ -148,7 +148,6 @@ static int octeon_mmc_probe(struct platform_device *pdev)
|
|||
{
|
||||
struct device_node *cn, *node = pdev->dev.of_node;
|
||||
struct cvm_mmc_host *host;
|
||||
struct resource *res;
|
||||
void __iomem *base;
|
||||
int mmc_irq[9];
|
||||
int i, ret = 0;
|
||||
|
@ -205,23 +204,13 @@ static int octeon_mmc_probe(struct platform_device *pdev)
|
|||
|
||||
host->last_slot = -1;
|
||||
|
||||
res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
||||
if (!res) {
|
||||
dev_err(&pdev->dev, "Platform resource[0] is missing\n");
|
||||
return -ENXIO;
|
||||
}
|
||||
base = devm_ioremap_resource(&pdev->dev, res);
|
||||
base = devm_platform_ioremap_resource(pdev, 0);
|
||||
if (IS_ERR(base))
|
||||
return PTR_ERR(base);
|
||||
host->base = (void __iomem *)base;
|
||||
host->reg_off = 0;
|
||||
|
||||
res = platform_get_resource(pdev, IORESOURCE_MEM, 1);
|
||||
if (!res) {
|
||||
dev_err(&pdev->dev, "Platform resource[1] is missing\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
base = devm_ioremap_resource(&pdev->dev, res);
|
||||
base = devm_platform_ioremap_resource(pdev, 1);
|
||||
if (IS_ERR(base))
|
||||
return PTR_ERR(base);
|
||||
host->dma_base = (void __iomem *)base;
|
||||
|
|
|
@ -3441,8 +3441,8 @@ int dw_mci_runtime_resume(struct device *dev)
|
|||
* Restore the initial value at FIFOTH register
|
||||
* And Invalidate the prev_blksz with zero
|
||||
*/
|
||||
mci_writel(host, FIFOTH, host->fifoth_val);
|
||||
host->prev_blksz = 0;
|
||||
mci_writel(host, FIFOTH, host->fifoth_val);
|
||||
host->prev_blksz = 0;
|
||||
|
||||
/* Put in max timeout */
|
||||
mci_writel(host, TMOUT, 0xFFFFFFFF);
|
||||
|
|
|
@ -41,6 +41,7 @@
|
|||
#define JZ_REG_MMC_RESP_FIFO 0x34
|
||||
#define JZ_REG_MMC_RXFIFO 0x38
|
||||
#define JZ_REG_MMC_TXFIFO 0x3C
|
||||
#define JZ_REG_MMC_LPM 0x40
|
||||
#define JZ_REG_MMC_DMAC 0x44
|
||||
|
||||
#define JZ_MMC_STRPCL_EXIT_MULTIPLE BIT(7)
|
||||
|
@ -77,6 +78,8 @@
|
|||
|
||||
#define JZ_MMC_CMDAT_IO_ABORT BIT(11)
|
||||
#define JZ_MMC_CMDAT_BUS_WIDTH_4BIT BIT(10)
|
||||
#define JZ_MMC_CMDAT_BUS_WIDTH_8BIT (BIT(10) | BIT(9))
|
||||
#define JZ_MMC_CMDAT_BUS_WIDTH_MASK (BIT(10) | BIT(9))
|
||||
#define JZ_MMC_CMDAT_DMA_EN BIT(8)
|
||||
#define JZ_MMC_CMDAT_INIT BIT(7)
|
||||
#define JZ_MMC_CMDAT_BUSY BIT(6)
|
||||
|
@ -98,12 +101,20 @@
|
|||
#define JZ_MMC_DMAC_DMA_SEL BIT(1)
|
||||
#define JZ_MMC_DMAC_DMA_EN BIT(0)
|
||||
|
||||
#define JZ_MMC_LPM_DRV_RISING BIT(31)
|
||||
#define JZ_MMC_LPM_DRV_RISING_QTR_PHASE_DLY BIT(31)
|
||||
#define JZ_MMC_LPM_DRV_RISING_1NS_DLY BIT(30)
|
||||
#define JZ_MMC_LPM_SMP_RISING_QTR_OR_HALF_PHASE_DLY BIT(29)
|
||||
#define JZ_MMC_LPM_LOW_POWER_MODE_EN BIT(0)
|
||||
|
||||
#define JZ_MMC_CLK_RATE 24000000
|
||||
|
||||
enum jz4740_mmc_version {
|
||||
JZ_MMC_JZ4740,
|
||||
JZ_MMC_JZ4725B,
|
||||
JZ_MMC_JZ4760,
|
||||
JZ_MMC_JZ4780,
|
||||
JZ_MMC_X1000,
|
||||
};
|
||||
|
||||
enum jz4740_mmc_state {
|
||||
|
@ -852,6 +863,22 @@ static int jz4740_mmc_set_clock_rate(struct jz4740_mmc_host *host, int rate)
|
|||
}
|
||||
|
||||
writew(div, host->base + JZ_REG_MMC_CLKRT);
|
||||
|
||||
if (real_rate > 25000000) {
|
||||
if (host->version >= JZ_MMC_X1000) {
|
||||
writel(JZ_MMC_LPM_DRV_RISING_QTR_PHASE_DLY |
|
||||
JZ_MMC_LPM_SMP_RISING_QTR_OR_HALF_PHASE_DLY |
|
||||
JZ_MMC_LPM_LOW_POWER_MODE_EN,
|
||||
host->base + JZ_REG_MMC_LPM);
|
||||
} else if (host->version >= JZ_MMC_JZ4760) {
|
||||
writel(JZ_MMC_LPM_DRV_RISING |
|
||||
JZ_MMC_LPM_LOW_POWER_MODE_EN,
|
||||
host->base + JZ_REG_MMC_LPM);
|
||||
} else if (host->version >= JZ_MMC_JZ4725B)
|
||||
writel(JZ_MMC_LPM_LOW_POWER_MODE_EN,
|
||||
host->base + JZ_REG_MMC_LPM);
|
||||
}
|
||||
|
||||
return real_rate;
|
||||
}
|
||||
|
||||
|
@ -895,11 +922,16 @@ static void jz4740_mmc_set_ios(struct mmc_host *mmc, struct mmc_ios *ios)
|
|||
|
||||
switch (ios->bus_width) {
|
||||
case MMC_BUS_WIDTH_1:
|
||||
host->cmdat &= ~JZ_MMC_CMDAT_BUS_WIDTH_4BIT;
|
||||
host->cmdat &= ~JZ_MMC_CMDAT_BUS_WIDTH_MASK;
|
||||
break;
|
||||
case MMC_BUS_WIDTH_4:
|
||||
host->cmdat &= ~JZ_MMC_CMDAT_BUS_WIDTH_MASK;
|
||||
host->cmdat |= JZ_MMC_CMDAT_BUS_WIDTH_4BIT;
|
||||
break;
|
||||
case MMC_BUS_WIDTH_8:
|
||||
host->cmdat &= ~JZ_MMC_CMDAT_BUS_WIDTH_MASK;
|
||||
host->cmdat |= JZ_MMC_CMDAT_BUS_WIDTH_8BIT;
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
|
@ -924,7 +956,9 @@ static const struct mmc_host_ops jz4740_mmc_ops = {
|
|||
static const struct of_device_id jz4740_mmc_of_match[] = {
|
||||
{ .compatible = "ingenic,jz4740-mmc", .data = (void *) JZ_MMC_JZ4740 },
|
||||
{ .compatible = "ingenic,jz4725b-mmc", .data = (void *)JZ_MMC_JZ4725B },
|
||||
{ .compatible = "ingenic,jz4760-mmc", .data = (void *) JZ_MMC_JZ4760 },
|
||||
{ .compatible = "ingenic,jz4780-mmc", .data = (void *) JZ_MMC_JZ4780 },
|
||||
{ .compatible = "ingenic,x1000-mmc", .data = (void *) JZ_MMC_X1000 },
|
||||
{},
|
||||
};
|
||||
MODULE_DEVICE_TABLE(of, jz4740_mmc_of_match);
|
||||
|
@ -1025,11 +1059,12 @@ static int jz4740_mmc_probe(struct platform_device* pdev)
|
|||
dev_err(&pdev->dev, "Failed to add mmc host: %d\n", ret);
|
||||
goto err_release_dma;
|
||||
}
|
||||
dev_info(&pdev->dev, "JZ SD/MMC card driver registered\n");
|
||||
dev_info(&pdev->dev, "Ingenic SD/MMC card driver registered\n");
|
||||
|
||||
dev_info(&pdev->dev, "Using %s, %d-bit mode\n",
|
||||
host->use_dma ? "DMA" : "PIO",
|
||||
(mmc->caps & MMC_CAP_4_BIT_DATA) ? 4 : 1);
|
||||
(mmc->caps & MMC_CAP_8_BIT_DATA) ? 8 :
|
||||
((mmc->caps & MMC_CAP_4_BIT_DATA) ? 4 : 1));
|
||||
|
||||
return 0;
|
||||
|
||||
|
|
|
@ -1421,7 +1421,7 @@ static int mmc_spi_probe(struct spi_device *spi)
|
|||
* Index 0 is card detect
|
||||
* Old boardfiles were specifying 1 ms as debounce
|
||||
*/
|
||||
status = mmc_gpiod_request_cd(mmc, NULL, 0, false, 1, NULL);
|
||||
status = mmc_gpiod_request_cd(mmc, NULL, 0, false, 1000, NULL);
|
||||
if (status == -EPROBE_DEFER)
|
||||
goto fail_add_host;
|
||||
if (!status) {
|
||||
|
|
|
@ -44,6 +44,7 @@
|
|||
#define DRIVER_NAME "mmci-pl18x"
|
||||
|
||||
static void mmci_variant_init(struct mmci_host *host);
|
||||
static void ux500_variant_init(struct mmci_host *host);
|
||||
static void ux500v2_variant_init(struct mmci_host *host);
|
||||
|
||||
static unsigned int fmax = 515633;
|
||||
|
@ -184,7 +185,7 @@ static struct variant_data variant_ux500 = {
|
|||
.irq_pio_mask = MCI_IRQ_PIO_MASK,
|
||||
.start_err = MCI_STARTBITERR,
|
||||
.opendrain = MCI_OD,
|
||||
.init = mmci_variant_init,
|
||||
.init = ux500_variant_init,
|
||||
};
|
||||
|
||||
static struct variant_data variant_ux500v2 = {
|
||||
|
@ -261,6 +262,10 @@ static struct variant_data variant_stm32_sdmmc = {
|
|||
.datalength_bits = 25,
|
||||
.datactrl_blocksz = 14,
|
||||
.stm32_idmabsize_mask = GENMASK(12, 5),
|
||||
.busy_timeout = true,
|
||||
.busy_detect = true,
|
||||
.busy_detect_flag = MCI_STM32_BUSYD0,
|
||||
.busy_detect_mask = MCI_STM32_BUSYD0ENDMASK,
|
||||
.init = sdmmc_variant_init,
|
||||
};
|
||||
|
||||
|
@ -419,7 +424,7 @@ static void mmci_set_clkreg(struct mmci_host *host, unsigned int desired)
|
|||
mmci_write_clkreg(host, clk);
|
||||
}
|
||||
|
||||
void mmci_dma_release(struct mmci_host *host)
|
||||
static void mmci_dma_release(struct mmci_host *host)
|
||||
{
|
||||
if (host->ops && host->ops->dma_release)
|
||||
host->ops->dma_release(host);
|
||||
|
@ -427,7 +432,7 @@ void mmci_dma_release(struct mmci_host *host)
|
|||
host->use_dma = false;
|
||||
}
|
||||
|
||||
void mmci_dma_setup(struct mmci_host *host)
|
||||
static void mmci_dma_setup(struct mmci_host *host)
|
||||
{
|
||||
if (!host->ops || !host->ops->dma_setup)
|
||||
return;
|
||||
|
@ -462,7 +467,7 @@ static int mmci_validate_data(struct mmci_host *host,
|
|||
return 0;
|
||||
}
|
||||
|
||||
int mmci_prep_data(struct mmci_host *host, struct mmc_data *data, bool next)
|
||||
static int mmci_prep_data(struct mmci_host *host, struct mmc_data *data, bool next)
|
||||
{
|
||||
int err;
|
||||
|
||||
|
@ -478,7 +483,7 @@ int mmci_prep_data(struct mmci_host *host, struct mmc_data *data, bool next)
|
|||
return err;
|
||||
}
|
||||
|
||||
void mmci_unprep_data(struct mmci_host *host, struct mmc_data *data,
|
||||
static void mmci_unprep_data(struct mmci_host *host, struct mmc_data *data,
|
||||
int err)
|
||||
{
|
||||
if (host->ops && host->ops->unprep_data)
|
||||
|
@ -487,7 +492,7 @@ void mmci_unprep_data(struct mmci_host *host, struct mmc_data *data,
|
|||
data->host_cookie = 0;
|
||||
}
|
||||
|
||||
void mmci_get_next_data(struct mmci_host *host, struct mmc_data *data)
|
||||
static void mmci_get_next_data(struct mmci_host *host, struct mmc_data *data)
|
||||
{
|
||||
WARN_ON(data->host_cookie && data->host_cookie != host->next_cookie);
|
||||
|
||||
|
@ -495,7 +500,7 @@ void mmci_get_next_data(struct mmci_host *host, struct mmc_data *data)
|
|||
host->ops->get_next_data(host, data);
|
||||
}
|
||||
|
||||
int mmci_dma_start(struct mmci_host *host, unsigned int datactrl)
|
||||
static int mmci_dma_start(struct mmci_host *host, unsigned int datactrl)
|
||||
{
|
||||
struct mmc_data *data = host->data;
|
||||
int ret;
|
||||
|
@ -530,7 +535,7 @@ int mmci_dma_start(struct mmci_host *host, unsigned int datactrl)
|
|||
return 0;
|
||||
}
|
||||
|
||||
void mmci_dma_finalize(struct mmci_host *host, struct mmc_data *data)
|
||||
static void mmci_dma_finalize(struct mmci_host *host, struct mmc_data *data)
|
||||
{
|
||||
if (!host->use_dma)
|
||||
return;
|
||||
|
@ -539,7 +544,7 @@ void mmci_dma_finalize(struct mmci_host *host, struct mmc_data *data)
|
|||
host->ops->dma_finalize(host, data);
|
||||
}
|
||||
|
||||
void mmci_dma_error(struct mmci_host *host)
|
||||
static void mmci_dma_error(struct mmci_host *host)
|
||||
{
|
||||
if (!host->use_dma)
|
||||
return;
|
||||
|
@ -610,6 +615,67 @@ static u32 ux500v2_get_dctrl_cfg(struct mmci_host *host)
|
|||
return MCI_DPSM_ENABLE | (host->data->blksz << 16);
|
||||
}
|
||||
|
||||
static bool ux500_busy_complete(struct mmci_host *host, u32 status, u32 err_msk)
|
||||
{
|
||||
void __iomem *base = host->base;
|
||||
|
||||
/*
|
||||
* Before unmasking for the busy end IRQ, confirm that the
|
||||
* command was sent successfully. To keep track of having a
|
||||
* command in-progress, waiting for busy signaling to end,
|
||||
* store the status in host->busy_status.
|
||||
*
|
||||
* Note that, the card may need a couple of clock cycles before
|
||||
* it starts signaling busy on DAT0, hence re-read the
|
||||
* MMCISTATUS register here, to allow the busy bit to be set.
|
||||
* Potentially we may even need to poll the register for a
|
||||
* while, to allow it to be set, but tests indicates that it
|
||||
* isn't needed.
|
||||
*/
|
||||
if (!host->busy_status && !(status & err_msk) &&
|
||||
(readl(base + MMCISTATUS) & host->variant->busy_detect_flag)) {
|
||||
writel(readl(base + MMCIMASK0) |
|
||||
host->variant->busy_detect_mask,
|
||||
base + MMCIMASK0);
|
||||
|
||||
host->busy_status = status & (MCI_CMDSENT | MCI_CMDRESPEND);
|
||||
return false;
|
||||
}
|
||||
|
||||
/*
|
||||
* If there is a command in-progress that has been successfully
|
||||
* sent, then bail out if busy status is set and wait for the
|
||||
* busy end IRQ.
|
||||
*
|
||||
* Note that, the HW triggers an IRQ on both edges while
|
||||
* monitoring DAT0 for busy completion, but there is only one
|
||||
* status bit in MMCISTATUS for the busy state. Therefore
|
||||
* both the start and the end interrupts needs to be cleared,
|
||||
* one after the other. So, clear the busy start IRQ here.
|
||||
*/
|
||||
if (host->busy_status &&
|
||||
(status & host->variant->busy_detect_flag)) {
|
||||
writel(host->variant->busy_detect_mask, base + MMCICLEAR);
|
||||
return false;
|
||||
}
|
||||
|
||||
/*
|
||||
* If there is a command in-progress that has been successfully
|
||||
* sent and the busy bit isn't set, it means we have received
|
||||
* the busy end IRQ. Clear and mask the IRQ, then continue to
|
||||
* process the command.
|
||||
*/
|
||||
if (host->busy_status) {
|
||||
writel(host->variant->busy_detect_mask, base + MMCICLEAR);
|
||||
|
||||
writel(readl(base + MMCIMASK0) &
|
||||
~host->variant->busy_detect_mask, base + MMCIMASK0);
|
||||
host->busy_status = 0;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/*
|
||||
* All the DMA operation mode stuff goes inside this ifdef.
|
||||
* This assumes that you have a generic DMA device interface,
|
||||
|
@ -948,14 +1014,21 @@ static struct mmci_host_ops mmci_variant_ops = {
|
|||
};
|
||||
#endif
|
||||
|
||||
void mmci_variant_init(struct mmci_host *host)
|
||||
static void mmci_variant_init(struct mmci_host *host)
|
||||
{
|
||||
host->ops = &mmci_variant_ops;
|
||||
}
|
||||
|
||||
void ux500v2_variant_init(struct mmci_host *host)
|
||||
static void ux500_variant_init(struct mmci_host *host)
|
||||
{
|
||||
host->ops = &mmci_variant_ops;
|
||||
host->ops->busy_complete = ux500_busy_complete;
|
||||
}
|
||||
|
||||
static void ux500v2_variant_init(struct mmci_host *host)
|
||||
{
|
||||
host->ops = &mmci_variant_ops;
|
||||
host->ops->busy_complete = ux500_busy_complete;
|
||||
host->ops->get_datactrl_cfg = ux500v2_get_dctrl_cfg;
|
||||
}
|
||||
|
||||
|
@ -1075,6 +1148,7 @@ static void
|
|||
mmci_start_command(struct mmci_host *host, struct mmc_command *cmd, u32 c)
|
||||
{
|
||||
void __iomem *base = host->base;
|
||||
unsigned long long clks;
|
||||
|
||||
dev_dbg(mmc_dev(host->mmc), "op %02x arg %08x flags %08x\n",
|
||||
cmd->opcode, cmd->arg, cmd->flags);
|
||||
|
@ -1097,6 +1171,16 @@ mmci_start_command(struct mmci_host *host, struct mmc_command *cmd, u32 c)
|
|||
else
|
||||
c |= host->variant->cmdreg_srsp;
|
||||
}
|
||||
|
||||
if (host->variant->busy_timeout && cmd->flags & MMC_RSP_BUSY) {
|
||||
if (!cmd->busy_timeout)
|
||||
cmd->busy_timeout = 10 * MSEC_PER_SEC;
|
||||
|
||||
clks = (unsigned long long)cmd->busy_timeout * host->cclk;
|
||||
do_div(clks, MSEC_PER_SEC);
|
||||
writel_relaxed(clks, host->base + MMCIDATATIMER);
|
||||
}
|
||||
|
||||
if (/*interrupt*/0)
|
||||
c |= MCI_CPSM_INTERRUPT;
|
||||
|
||||
|
@ -1201,6 +1285,7 @@ static void
|
|||
mmci_cmd_irq(struct mmci_host *host, struct mmc_command *cmd,
|
||||
unsigned int status)
|
||||
{
|
||||
u32 err_msk = MCI_CMDCRCFAIL | MCI_CMDTIMEOUT;
|
||||
void __iomem *base = host->base;
|
||||
bool sbc, busy_resp;
|
||||
|
||||
|
@ -1215,74 +1300,17 @@ mmci_cmd_irq(struct mmci_host *host, struct mmc_command *cmd,
|
|||
* handling. Note that we tag on any latent IRQs postponed
|
||||
* due to waiting for busy status.
|
||||
*/
|
||||
if (!((status|host->busy_status) &
|
||||
(MCI_CMDCRCFAIL|MCI_CMDTIMEOUT|MCI_CMDSENT|MCI_CMDRESPEND)))
|
||||
if (host->variant->busy_timeout && busy_resp)
|
||||
err_msk |= MCI_DATATIMEOUT;
|
||||
|
||||
if (!((status | host->busy_status) &
|
||||
(err_msk | MCI_CMDSENT | MCI_CMDRESPEND)))
|
||||
return;
|
||||
|
||||
/* Handle busy detection on DAT0 if the variant supports it. */
|
||||
if (busy_resp && host->variant->busy_detect) {
|
||||
|
||||
/*
|
||||
* Before unmasking for the busy end IRQ, confirm that the
|
||||
* command was sent successfully. To keep track of having a
|
||||
* command in-progress, waiting for busy signaling to end,
|
||||
* store the status in host->busy_status.
|
||||
*
|
||||
* Note that, the card may need a couple of clock cycles before
|
||||
* it starts signaling busy on DAT0, hence re-read the
|
||||
* MMCISTATUS register here, to allow the busy bit to be set.
|
||||
* Potentially we may even need to poll the register for a
|
||||
* while, to allow it to be set, but tests indicates that it
|
||||
* isn't needed.
|
||||
*/
|
||||
if (!host->busy_status &&
|
||||
!(status & (MCI_CMDCRCFAIL|MCI_CMDTIMEOUT)) &&
|
||||
(readl(base + MMCISTATUS) & host->variant->busy_detect_flag)) {
|
||||
|
||||
writel(readl(base + MMCIMASK0) |
|
||||
host->variant->busy_detect_mask,
|
||||
base + MMCIMASK0);
|
||||
|
||||
host->busy_status =
|
||||
status & (MCI_CMDSENT|MCI_CMDRESPEND);
|
||||
if (busy_resp && host->variant->busy_detect)
|
||||
if (!host->ops->busy_complete(host, status, err_msk))
|
||||
return;
|
||||
}
|
||||
|
||||
/*
|
||||
* If there is a command in-progress that has been successfully
|
||||
* sent, then bail out if busy status is set and wait for the
|
||||
* busy end IRQ.
|
||||
*
|
||||
* Note that, the HW triggers an IRQ on both edges while
|
||||
* monitoring DAT0 for busy completion, but there is only one
|
||||
* status bit in MMCISTATUS for the busy state. Therefore
|
||||
* both the start and the end interrupts needs to be cleared,
|
||||
* one after the other. So, clear the busy start IRQ here.
|
||||
*/
|
||||
if (host->busy_status &&
|
||||
(status & host->variant->busy_detect_flag)) {
|
||||
writel(host->variant->busy_detect_mask,
|
||||
host->base + MMCICLEAR);
|
||||
return;
|
||||
}
|
||||
|
||||
/*
|
||||
* If there is a command in-progress that has been successfully
|
||||
* sent and the busy bit isn't set, it means we have received
|
||||
* the busy end IRQ. Clear and mask the IRQ, then continue to
|
||||
* process the command.
|
||||
*/
|
||||
if (host->busy_status) {
|
||||
|
||||
writel(host->variant->busy_detect_mask,
|
||||
host->base + MMCICLEAR);
|
||||
|
||||
writel(readl(base + MMCIMASK0) &
|
||||
~host->variant->busy_detect_mask,
|
||||
base + MMCIMASK0);
|
||||
host->busy_status = 0;
|
||||
}
|
||||
}
|
||||
|
||||
host->cmd = NULL;
|
||||
|
||||
|
@ -1290,6 +1318,9 @@ mmci_cmd_irq(struct mmci_host *host, struct mmc_command *cmd,
|
|||
cmd->error = -ETIMEDOUT;
|
||||
} else if (status & MCI_CMDCRCFAIL && cmd->flags & MMC_RSP_CRC) {
|
||||
cmd->error = -EILSEQ;
|
||||
} else if (host->variant->busy_timeout && busy_resp &&
|
||||
status & MCI_DATATIMEOUT) {
|
||||
cmd->error = -ETIMEDOUT;
|
||||
} else {
|
||||
cmd->resp[0] = readl(base + MMCIRESPONSE0);
|
||||
cmd->resp[1] = readl(base + MMCIRESPONSE1);
|
||||
|
@ -1583,6 +1614,20 @@ static void mmci_request(struct mmc_host *mmc, struct mmc_request *mrq)
|
|||
spin_unlock_irqrestore(&host->lock, flags);
|
||||
}
|
||||
|
||||
static void mmci_set_max_busy_timeout(struct mmc_host *mmc)
|
||||
{
|
||||
struct mmci_host *host = mmc_priv(mmc);
|
||||
u32 max_busy_timeout = 0;
|
||||
|
||||
if (!host->variant->busy_detect)
|
||||
return;
|
||||
|
||||
if (host->variant->busy_timeout && mmc->actual_clock)
|
||||
max_busy_timeout = ~0UL / (mmc->actual_clock / MSEC_PER_SEC);
|
||||
|
||||
mmc->max_busy_timeout = max_busy_timeout;
|
||||
}
|
||||
|
||||
static void mmci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios)
|
||||
{
|
||||
struct mmci_host *host = mmc_priv(mmc);
|
||||
|
@ -1687,6 +1732,8 @@ static void mmci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios)
|
|||
else
|
||||
mmci_set_clkreg(host, ios->clock);
|
||||
|
||||
mmci_set_max_busy_timeout(mmc);
|
||||
|
||||
if (host->ops && host->ops->set_pwrreg)
|
||||
host->ops->set_pwrreg(host, pwr);
|
||||
else
|
||||
|
@ -1957,7 +2004,6 @@ static int mmci_probe(struct amba_device *dev,
|
|||
mmci_write_datactrlreg(host,
|
||||
host->variant->busy_dpsm_flag);
|
||||
mmc->caps |= MMC_CAP_WAIT_WHILE_BUSY;
|
||||
mmc->max_busy_timeout = 0;
|
||||
}
|
||||
|
||||
/* Prepare a CMD12 - needed to clear the DPSM on some variants. */
|
||||
|
|
|
@ -164,6 +164,7 @@
|
|||
#define MCI_ST_CARDBUSY (1 << 24)
|
||||
/* Extended status bits for the STM32 variants */
|
||||
#define MCI_STM32_BUSYD0 BIT(20)
|
||||
#define MCI_STM32_BUSYD0END BIT(21)
|
||||
|
||||
#define MMCICLEAR 0x038
|
||||
#define MCI_CMDCRCFAILCLR (1 << 0)
|
||||
|
@ -287,6 +288,8 @@ struct mmci_host;
|
|||
* @signal_direction: input/out direction of bus signals can be indicated
|
||||
* @pwrreg_clkgate: MMCIPOWER register must be used to gate the clock
|
||||
* @busy_detect: true if the variant supports busy detection on DAT0.
|
||||
* @busy_timeout: true if the variant starts data timer when the DPSM
|
||||
* enter in Wait_R or Busy state.
|
||||
* @busy_dpsm_flag: bitmask enabling busy detection in the DPSM
|
||||
* @busy_detect_flag: bitmask identifying the bit in the MMCISTATUS register
|
||||
* indicating that the card is busy
|
||||
|
@ -333,6 +336,7 @@ struct variant_data {
|
|||
u8 signal_direction:1;
|
||||
u8 pwrreg_clkgate:1;
|
||||
u8 busy_detect:1;
|
||||
u8 busy_timeout:1;
|
||||
u32 busy_dpsm_flag;
|
||||
u32 busy_detect_flag;
|
||||
u32 busy_detect_mask;
|
||||
|
@ -366,6 +370,7 @@ struct mmci_host_ops {
|
|||
void (*dma_error)(struct mmci_host *host);
|
||||
void (*set_clkreg)(struct mmci_host *host, unsigned int desired);
|
||||
void (*set_pwrreg)(struct mmci_host *host, unsigned int pwr);
|
||||
bool (*busy_complete)(struct mmci_host *host, u32 status, u32 err_msk);
|
||||
};
|
||||
|
||||
struct mmci_host {
|
||||
|
|
|
@ -25,8 +25,8 @@ struct sdmmc_priv {
|
|||
void *sg_cpu;
|
||||
};
|
||||
|
||||
int sdmmc_idma_validate_data(struct mmci_host *host,
|
||||
struct mmc_data *data)
|
||||
static int sdmmc_idma_validate_data(struct mmci_host *host,
|
||||
struct mmc_data *data)
|
||||
{
|
||||
struct scatterlist *sg;
|
||||
int i;
|
||||
|
@ -282,6 +282,47 @@ static u32 sdmmc_get_dctrl_cfg(struct mmci_host *host)
|
|||
return datactrl;
|
||||
}
|
||||
|
||||
static bool sdmmc_busy_complete(struct mmci_host *host, u32 status, u32 err_msk)
|
||||
{
|
||||
void __iomem *base = host->base;
|
||||
u32 busy_d0, busy_d0end, mask, sdmmc_status;
|
||||
|
||||
mask = readl_relaxed(base + MMCIMASK0);
|
||||
sdmmc_status = readl_relaxed(base + MMCISTATUS);
|
||||
busy_d0end = sdmmc_status & MCI_STM32_BUSYD0END;
|
||||
busy_d0 = sdmmc_status & MCI_STM32_BUSYD0;
|
||||
|
||||
/* complete if there is an error or busy_d0end */
|
||||
if ((status & err_msk) || busy_d0end)
|
||||
goto complete;
|
||||
|
||||
/*
|
||||
* On response the busy signaling is reflected in the BUSYD0 flag.
|
||||
* if busy_d0 is in-progress we must activate busyd0end interrupt
|
||||
* to wait this completion. Else this request has no busy step.
|
||||
*/
|
||||
if (busy_d0) {
|
||||
if (!host->busy_status) {
|
||||
writel_relaxed(mask | host->variant->busy_detect_mask,
|
||||
base + MMCIMASK0);
|
||||
host->busy_status = status &
|
||||
(MCI_CMDSENT | MCI_CMDRESPEND);
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
complete:
|
||||
if (host->busy_status) {
|
||||
writel_relaxed(mask & ~host->variant->busy_detect_mask,
|
||||
base + MMCIMASK0);
|
||||
writel_relaxed(host->variant->busy_detect_mask,
|
||||
base + MMCICLEAR);
|
||||
host->busy_status = 0;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
static struct mmci_host_ops sdmmc_variant_ops = {
|
||||
.validate_data = sdmmc_idma_validate_data,
|
||||
.prep_data = sdmmc_idma_prep_data,
|
||||
|
@ -292,6 +333,7 @@ static struct mmci_host_ops sdmmc_variant_ops = {
|
|||
.dma_finalize = sdmmc_idma_finalize,
|
||||
.set_clkreg = mmci_sdmmc_set_clkreg,
|
||||
.set_pwrreg = mmci_sdmmc_set_pwrreg,
|
||||
.busy_complete = sdmmc_busy_complete,
|
||||
};
|
||||
|
||||
void sdmmc_variant_init(struct mmci_host *host)
|
||||
|
|
|
@ -608,8 +608,8 @@ static int moxart_probe(struct platform_device *pdev)
|
|||
host->timeout = msecs_to_jiffies(1000);
|
||||
host->sysclk = clk_get_rate(clk);
|
||||
host->fifo_width = readl(host->base + REG_FEATURE) << 2;
|
||||
host->dma_chan_tx = dma_request_slave_channel_reason(dev, "tx");
|
||||
host->dma_chan_rx = dma_request_slave_channel_reason(dev, "rx");
|
||||
host->dma_chan_tx = dma_request_chan(dev, "tx");
|
||||
host->dma_chan_rx = dma_request_chan(dev, "rx");
|
||||
|
||||
spin_lock_init(&host->lock);
|
||||
|
||||
|
|
|
@ -1510,8 +1510,35 @@ static void omap_hsmmc_init_card(struct mmc_host *mmc, struct mmc_card *card)
|
|||
{
|
||||
struct omap_hsmmc_host *host = mmc_priv(mmc);
|
||||
|
||||
if (mmc_pdata(host)->init_card)
|
||||
mmc_pdata(host)->init_card(card);
|
||||
if (card->type == MMC_TYPE_SDIO || card->type == MMC_TYPE_SD_COMBO) {
|
||||
struct device_node *np = mmc_dev(mmc)->of_node;
|
||||
|
||||
/*
|
||||
* REVISIT: should be moved to sdio core and made more
|
||||
* general e.g. by expanding the DT bindings of child nodes
|
||||
* to provide a mechanism to provide this information:
|
||||
* Documentation/devicetree/bindings/mmc/mmc-card.txt
|
||||
*/
|
||||
|
||||
np = of_get_compatible_child(np, "ti,wl1251");
|
||||
if (np) {
|
||||
/*
|
||||
* We have TI wl1251 attached to MMC3. Pass this
|
||||
* information to the SDIO core because it can't be
|
||||
* probed by normal methods.
|
||||
*/
|
||||
|
||||
dev_info(host->dev, "found wl1251\n");
|
||||
card->quirks |= MMC_QUIRK_NONSTD_SDIO;
|
||||
card->cccr.wide_bus = 1;
|
||||
card->cis.vendor = 0x104c;
|
||||
card->cis.device = 0x9066;
|
||||
card->cis.blksize = 512;
|
||||
card->cis.max_dtr = 24000000;
|
||||
card->ocr = 0x80;
|
||||
of_node_put(np);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static void omap_hsmmc_enable_sdio_irq(struct mmc_host *mmc, int enable)
|
||||
|
|
|
@ -0,0 +1,696 @@
|
|||
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
/*
|
||||
* Actions Semi Owl SoCs SD/MMC driver
|
||||
*
|
||||
* Copyright (c) 2014 Actions Semi Inc.
|
||||
* Copyright (c) 2019 Manivannan Sadhasivam <manivannan.sadhasivam@linaro.org>
|
||||
*
|
||||
* TODO: SDIO support
|
||||
*/
|
||||
|
||||
#include <linux/clk.h>
|
||||
#include <linux/delay.h>
|
||||
#include <linux/dmaengine.h>
|
||||
#include <linux/dma-direction.h>
|
||||
#include <linux/dma-mapping.h>
|
||||
#include <linux/interrupt.h>
|
||||
#include <linux/mmc/host.h>
|
||||
#include <linux/mmc/slot-gpio.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/of_platform.h>
|
||||
#include <linux/reset.h>
|
||||
#include <linux/spinlock.h>
|
||||
|
||||
/*
|
||||
* SDC registers
|
||||
*/
|
||||
#define OWL_REG_SD_EN 0x0000
|
||||
#define OWL_REG_SD_CTL 0x0004
|
||||
#define OWL_REG_SD_STATE 0x0008
|
||||
#define OWL_REG_SD_CMD 0x000c
|
||||
#define OWL_REG_SD_ARG 0x0010
|
||||
#define OWL_REG_SD_RSPBUF0 0x0014
|
||||
#define OWL_REG_SD_RSPBUF1 0x0018
|
||||
#define OWL_REG_SD_RSPBUF2 0x001c
|
||||
#define OWL_REG_SD_RSPBUF3 0x0020
|
||||
#define OWL_REG_SD_RSPBUF4 0x0024
|
||||
#define OWL_REG_SD_DAT 0x0028
|
||||
#define OWL_REG_SD_BLK_SIZE 0x002c
|
||||
#define OWL_REG_SD_BLK_NUM 0x0030
|
||||
#define OWL_REG_SD_BUF_SIZE 0x0034
|
||||
|
||||
/* SD_EN Bits */
|
||||
#define OWL_SD_EN_RANE BIT(31)
|
||||
#define OWL_SD_EN_RAN_SEED(x) (((x) & 0x3f) << 24)
|
||||
#define OWL_SD_EN_S18EN BIT(12)
|
||||
#define OWL_SD_EN_RESE BIT(10)
|
||||
#define OWL_SD_EN_DAT1_S BIT(9)
|
||||
#define OWL_SD_EN_CLK_S BIT(8)
|
||||
#define OWL_SD_ENABLE BIT(7)
|
||||
#define OWL_SD_EN_BSEL BIT(6)
|
||||
#define OWL_SD_EN_SDIOEN BIT(3)
|
||||
#define OWL_SD_EN_DDREN BIT(2)
|
||||
#define OWL_SD_EN_DATAWID(x) (((x) & 0x3) << 0)
|
||||
|
||||
/* SD_CTL Bits */
|
||||
#define OWL_SD_CTL_TOUTEN BIT(31)
|
||||
#define OWL_SD_CTL_TOUTCNT(x) (((x) & 0x7f) << 24)
|
||||
#define OWL_SD_CTL_DELAY_MSK GENMASK(23, 16)
|
||||
#define OWL_SD_CTL_RDELAY(x) (((x) & 0xf) << 20)
|
||||
#define OWL_SD_CTL_WDELAY(x) (((x) & 0xf) << 16)
|
||||
#define OWL_SD_CTL_CMDLEN BIT(13)
|
||||
#define OWL_SD_CTL_SCC BIT(12)
|
||||
#define OWL_SD_CTL_TCN(x) (((x) & 0xf) << 8)
|
||||
#define OWL_SD_CTL_TS BIT(7)
|
||||
#define OWL_SD_CTL_LBE BIT(6)
|
||||
#define OWL_SD_CTL_C7EN BIT(5)
|
||||
#define OWL_SD_CTL_TM(x) (((x) & 0xf) << 0)
|
||||
|
||||
#define OWL_SD_DELAY_LOW_CLK 0x0f
|
||||
#define OWL_SD_DELAY_MID_CLK 0x0a
|
||||
#define OWL_SD_DELAY_HIGH_CLK 0x09
|
||||
#define OWL_SD_RDELAY_DDR50 0x0a
|
||||
#define OWL_SD_WDELAY_DDR50 0x08
|
||||
|
||||
/* SD_STATE Bits */
|
||||
#define OWL_SD_STATE_DAT1BS BIT(18)
|
||||
#define OWL_SD_STATE_SDIOB_P BIT(17)
|
||||
#define OWL_SD_STATE_SDIOB_EN BIT(16)
|
||||
#define OWL_SD_STATE_TOUTE BIT(15)
|
||||
#define OWL_SD_STATE_BAEP BIT(14)
|
||||
#define OWL_SD_STATE_MEMRDY BIT(12)
|
||||
#define OWL_SD_STATE_CMDS BIT(11)
|
||||
#define OWL_SD_STATE_DAT1AS BIT(10)
|
||||
#define OWL_SD_STATE_SDIOA_P BIT(9)
|
||||
#define OWL_SD_STATE_SDIOA_EN BIT(8)
|
||||
#define OWL_SD_STATE_DAT0S BIT(7)
|
||||
#define OWL_SD_STATE_TEIE BIT(6)
|
||||
#define OWL_SD_STATE_TEI BIT(5)
|
||||
#define OWL_SD_STATE_CLNR BIT(4)
|
||||
#define OWL_SD_STATE_CLC BIT(3)
|
||||
#define OWL_SD_STATE_WC16ER BIT(2)
|
||||
#define OWL_SD_STATE_RC16ER BIT(1)
|
||||
#define OWL_SD_STATE_CRC7ER BIT(0)
|
||||
|
||||
struct owl_mmc_host {
|
||||
struct device *dev;
|
||||
struct reset_control *reset;
|
||||
void __iomem *base;
|
||||
struct clk *clk;
|
||||
struct completion sdc_complete;
|
||||
spinlock_t lock;
|
||||
int irq;
|
||||
u32 clock;
|
||||
bool ddr_50;
|
||||
|
||||
enum dma_data_direction dma_dir;
|
||||
struct dma_chan *dma;
|
||||
struct dma_async_tx_descriptor *desc;
|
||||
struct dma_slave_config dma_cfg;
|
||||
struct completion dma_complete;
|
||||
|
||||
struct mmc_host *mmc;
|
||||
struct mmc_request *mrq;
|
||||
struct mmc_command *cmd;
|
||||
struct mmc_data *data;
|
||||
};
|
||||
|
||||
static void owl_mmc_update_reg(void __iomem *reg, unsigned int val, bool state)
|
||||
{
|
||||
unsigned int regval;
|
||||
|
||||
regval = readl(reg);
|
||||
|
||||
if (state)
|
||||
regval |= val;
|
||||
else
|
||||
regval &= ~val;
|
||||
|
||||
writel(regval, reg);
|
||||
}
|
||||
|
||||
static irqreturn_t owl_irq_handler(int irq, void *devid)
|
||||
{
|
||||
struct owl_mmc_host *owl_host = devid;
|
||||
unsigned long flags;
|
||||
u32 state;
|
||||
|
||||
spin_lock_irqsave(&owl_host->lock, flags);
|
||||
|
||||
state = readl(owl_host->base + OWL_REG_SD_STATE);
|
||||
if (state & OWL_SD_STATE_TEI) {
|
||||
state = readl(owl_host->base + OWL_REG_SD_STATE);
|
||||
state |= OWL_SD_STATE_TEI;
|
||||
writel(state, owl_host->base + OWL_REG_SD_STATE);
|
||||
complete(&owl_host->sdc_complete);
|
||||
}
|
||||
|
||||
spin_unlock_irqrestore(&owl_host->lock, flags);
|
||||
|
||||
return IRQ_HANDLED;
|
||||
}
|
||||
|
||||
static void owl_mmc_finish_request(struct owl_mmc_host *owl_host)
|
||||
{
|
||||
struct mmc_request *mrq = owl_host->mrq;
|
||||
struct mmc_data *data = mrq->data;
|
||||
|
||||
/* Should never be NULL */
|
||||
WARN_ON(!mrq);
|
||||
|
||||
owl_host->mrq = NULL;
|
||||
|
||||
if (data)
|
||||
dma_unmap_sg(owl_host->dma->device->dev, data->sg, data->sg_len,
|
||||
owl_host->dma_dir);
|
||||
|
||||
/* Finally finish request */
|
||||
mmc_request_done(owl_host->mmc, mrq);
|
||||
}
|
||||
|
||||
static void owl_mmc_send_cmd(struct owl_mmc_host *owl_host,
|
||||
struct mmc_command *cmd,
|
||||
struct mmc_data *data)
|
||||
{
|
||||
u32 mode, state, resp[2];
|
||||
u32 cmd_rsp_mask = 0;
|
||||
|
||||
init_completion(&owl_host->sdc_complete);
|
||||
|
||||
switch (mmc_resp_type(cmd)) {
|
||||
case MMC_RSP_NONE:
|
||||
mode = OWL_SD_CTL_TM(0);
|
||||
break;
|
||||
|
||||
case MMC_RSP_R1:
|
||||
if (data) {
|
||||
if (data->flags & MMC_DATA_READ)
|
||||
mode = OWL_SD_CTL_TM(4);
|
||||
else
|
||||
mode = OWL_SD_CTL_TM(5);
|
||||
} else {
|
||||
mode = OWL_SD_CTL_TM(1);
|
||||
}
|
||||
cmd_rsp_mask = OWL_SD_STATE_CLNR | OWL_SD_STATE_CRC7ER;
|
||||
|
||||
break;
|
||||
|
||||
case MMC_RSP_R1B:
|
||||
mode = OWL_SD_CTL_TM(3);
|
||||
cmd_rsp_mask = OWL_SD_STATE_CLNR | OWL_SD_STATE_CRC7ER;
|
||||
break;
|
||||
|
||||
case MMC_RSP_R2:
|
||||
mode = OWL_SD_CTL_TM(2);
|
||||
cmd_rsp_mask = OWL_SD_STATE_CLNR | OWL_SD_STATE_CRC7ER;
|
||||
break;
|
||||
|
||||
case MMC_RSP_R3:
|
||||
mode = OWL_SD_CTL_TM(1);
|
||||
cmd_rsp_mask = OWL_SD_STATE_CLNR;
|
||||
break;
|
||||
|
||||
default:
|
||||
dev_warn(owl_host->dev, "Unknown MMC command\n");
|
||||
cmd->error = -EINVAL;
|
||||
return;
|
||||
}
|
||||
|
||||
/* Keep current WDELAY and RDELAY */
|
||||
mode |= (readl(owl_host->base + OWL_REG_SD_CTL) & (0xff << 16));
|
||||
|
||||
/* Start to send corresponding command type */
|
||||
writel(cmd->arg, owl_host->base + OWL_REG_SD_ARG);
|
||||
writel(cmd->opcode, owl_host->base + OWL_REG_SD_CMD);
|
||||
|
||||
/* Set LBE to send clk at the end of last read block */
|
||||
if (data) {
|
||||
mode |= (OWL_SD_CTL_TS | OWL_SD_CTL_LBE | 0x64000000);
|
||||
} else {
|
||||
mode &= ~(OWL_SD_CTL_TOUTEN | OWL_SD_CTL_LBE);
|
||||
mode |= OWL_SD_CTL_TS;
|
||||
}
|
||||
|
||||
owl_host->cmd = cmd;
|
||||
|
||||
/* Start transfer */
|
||||
writel(mode, owl_host->base + OWL_REG_SD_CTL);
|
||||
|
||||
if (data)
|
||||
return;
|
||||
|
||||
if (!wait_for_completion_timeout(&owl_host->sdc_complete, 30 * HZ)) {
|
||||
dev_err(owl_host->dev, "CMD interrupt timeout\n");
|
||||
cmd->error = -ETIMEDOUT;
|
||||
return;
|
||||
}
|
||||
|
||||
state = readl(owl_host->base + OWL_REG_SD_STATE);
|
||||
if (mmc_resp_type(cmd) & MMC_RSP_PRESENT) {
|
||||
if (cmd_rsp_mask & state) {
|
||||
if (state & OWL_SD_STATE_CLNR) {
|
||||
dev_err(owl_host->dev, "Error CMD_NO_RSP\n");
|
||||
cmd->error = -EILSEQ;
|
||||
return;
|
||||
}
|
||||
|
||||
if (state & OWL_SD_STATE_CRC7ER) {
|
||||
dev_err(owl_host->dev, "Error CMD_RSP_CRC\n");
|
||||
cmd->error = -EILSEQ;
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
if (mmc_resp_type(cmd) & MMC_RSP_136) {
|
||||
cmd->resp[3] = readl(owl_host->base + OWL_REG_SD_RSPBUF0);
|
||||
cmd->resp[2] = readl(owl_host->base + OWL_REG_SD_RSPBUF1);
|
||||
cmd->resp[1] = readl(owl_host->base + OWL_REG_SD_RSPBUF2);
|
||||
cmd->resp[0] = readl(owl_host->base + OWL_REG_SD_RSPBUF3);
|
||||
} else {
|
||||
resp[0] = readl(owl_host->base + OWL_REG_SD_RSPBUF0);
|
||||
resp[1] = readl(owl_host->base + OWL_REG_SD_RSPBUF1);
|
||||
cmd->resp[0] = resp[1] << 24 | resp[0] >> 8;
|
||||
cmd->resp[1] = resp[1] >> 8;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static void owl_mmc_dma_complete(void *param)
|
||||
{
|
||||
struct owl_mmc_host *owl_host = param;
|
||||
struct mmc_data *data = owl_host->data;
|
||||
|
||||
if (data)
|
||||
complete(&owl_host->dma_complete);
|
||||
}
|
||||
|
||||
static int owl_mmc_prepare_data(struct owl_mmc_host *owl_host,
|
||||
struct mmc_data *data)
|
||||
{
|
||||
u32 total;
|
||||
|
||||
owl_mmc_update_reg(owl_host->base + OWL_REG_SD_EN, OWL_SD_EN_BSEL,
|
||||
true);
|
||||
writel(data->blocks, owl_host->base + OWL_REG_SD_BLK_NUM);
|
||||
writel(data->blksz, owl_host->base + OWL_REG_SD_BLK_SIZE);
|
||||
total = data->blksz * data->blocks;
|
||||
|
||||
if (total < 512)
|
||||
writel(total, owl_host->base + OWL_REG_SD_BUF_SIZE);
|
||||
else
|
||||
writel(512, owl_host->base + OWL_REG_SD_BUF_SIZE);
|
||||
|
||||
if (data->flags & MMC_DATA_WRITE) {
|
||||
owl_host->dma_dir = DMA_TO_DEVICE;
|
||||
owl_host->dma_cfg.direction = DMA_MEM_TO_DEV;
|
||||
} else {
|
||||
owl_host->dma_dir = DMA_FROM_DEVICE;
|
||||
owl_host->dma_cfg.direction = DMA_DEV_TO_MEM;
|
||||
}
|
||||
|
||||
dma_map_sg(owl_host->dma->device->dev, data->sg,
|
||||
data->sg_len, owl_host->dma_dir);
|
||||
|
||||
dmaengine_slave_config(owl_host->dma, &owl_host->dma_cfg);
|
||||
owl_host->desc = dmaengine_prep_slave_sg(owl_host->dma, data->sg,
|
||||
data->sg_len,
|
||||
owl_host->dma_cfg.direction,
|
||||
DMA_PREP_INTERRUPT |
|
||||
DMA_CTRL_ACK);
|
||||
if (!owl_host->desc) {
|
||||
dev_err(owl_host->dev, "Can't prepare slave sg\n");
|
||||
return -EBUSY;
|
||||
}
|
||||
|
||||
owl_host->data = data;
|
||||
|
||||
owl_host->desc->callback = owl_mmc_dma_complete;
|
||||
owl_host->desc->callback_param = (void *)owl_host;
|
||||
data->error = 0;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void owl_mmc_request(struct mmc_host *mmc, struct mmc_request *mrq)
|
||||
{
|
||||
struct owl_mmc_host *owl_host = mmc_priv(mmc);
|
||||
struct mmc_data *data = mrq->data;
|
||||
int ret;
|
||||
|
||||
owl_host->mrq = mrq;
|
||||
if (mrq->data) {
|
||||
ret = owl_mmc_prepare_data(owl_host, data);
|
||||
if (ret < 0) {
|
||||
data->error = ret;
|
||||
goto err_out;
|
||||
}
|
||||
|
||||
init_completion(&owl_host->dma_complete);
|
||||
dmaengine_submit(owl_host->desc);
|
||||
dma_async_issue_pending(owl_host->dma);
|
||||
}
|
||||
|
||||
owl_mmc_send_cmd(owl_host, mrq->cmd, data);
|
||||
|
||||
if (data) {
|
||||
if (!wait_for_completion_timeout(&owl_host->sdc_complete,
|
||||
10 * HZ)) {
|
||||
dev_err(owl_host->dev, "CMD interrupt timeout\n");
|
||||
mrq->cmd->error = -ETIMEDOUT;
|
||||
dmaengine_terminate_all(owl_host->dma);
|
||||
goto err_out;
|
||||
}
|
||||
|
||||
if (!wait_for_completion_timeout(&owl_host->dma_complete,
|
||||
5 * HZ)) {
|
||||
dev_err(owl_host->dev, "DMA interrupt timeout\n");
|
||||
mrq->cmd->error = -ETIMEDOUT;
|
||||
dmaengine_terminate_all(owl_host->dma);
|
||||
goto err_out;
|
||||
}
|
||||
|
||||
if (data->stop)
|
||||
owl_mmc_send_cmd(owl_host, data->stop, NULL);
|
||||
|
||||
data->bytes_xfered = data->blocks * data->blksz;
|
||||
}
|
||||
|
||||
err_out:
|
||||
owl_mmc_finish_request(owl_host);
|
||||
}
|
||||
|
||||
static int owl_mmc_set_clk_rate(struct owl_mmc_host *owl_host,
|
||||
unsigned int rate)
|
||||
{
|
||||
unsigned long clk_rate;
|
||||
int ret;
|
||||
u32 reg;
|
||||
|
||||
reg = readl(owl_host->base + OWL_REG_SD_CTL);
|
||||
reg &= ~OWL_SD_CTL_DELAY_MSK;
|
||||
|
||||
/* Set RDELAY and WDELAY based on the clock */
|
||||
if (rate <= 1000000) {
|
||||
writel(reg | OWL_SD_CTL_RDELAY(OWL_SD_DELAY_LOW_CLK) |
|
||||
OWL_SD_CTL_WDELAY(OWL_SD_DELAY_LOW_CLK),
|
||||
owl_host->base + OWL_REG_SD_CTL);
|
||||
} else if ((rate > 1000000) && (rate <= 26000000)) {
|
||||
writel(reg | OWL_SD_CTL_RDELAY(OWL_SD_DELAY_MID_CLK) |
|
||||
OWL_SD_CTL_WDELAY(OWL_SD_DELAY_MID_CLK),
|
||||
owl_host->base + OWL_REG_SD_CTL);
|
||||
} else if ((rate > 26000000) && (rate <= 52000000) && !owl_host->ddr_50) {
|
||||
writel(reg | OWL_SD_CTL_RDELAY(OWL_SD_DELAY_HIGH_CLK) |
|
||||
OWL_SD_CTL_WDELAY(OWL_SD_DELAY_HIGH_CLK),
|
||||
owl_host->base + OWL_REG_SD_CTL);
|
||||
/* DDR50 mode has special delay chain */
|
||||
} else if ((rate > 26000000) && (rate <= 52000000) && owl_host->ddr_50) {
|
||||
writel(reg | OWL_SD_CTL_RDELAY(OWL_SD_RDELAY_DDR50) |
|
||||
OWL_SD_CTL_WDELAY(OWL_SD_WDELAY_DDR50),
|
||||
owl_host->base + OWL_REG_SD_CTL);
|
||||
} else {
|
||||
dev_err(owl_host->dev, "SD clock rate not supported\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
clk_rate = clk_round_rate(owl_host->clk, rate << 1);
|
||||
ret = clk_set_rate(owl_host->clk, clk_rate);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void owl_mmc_set_clk(struct owl_mmc_host *owl_host, struct mmc_ios *ios)
|
||||
{
|
||||
if (!ios->clock)
|
||||
return;
|
||||
|
||||
owl_host->clock = ios->clock;
|
||||
owl_mmc_set_clk_rate(owl_host, ios->clock);
|
||||
}
|
||||
|
||||
static void owl_mmc_set_bus_width(struct owl_mmc_host *owl_host,
|
||||
struct mmc_ios *ios)
|
||||
{
|
||||
u32 reg;
|
||||
|
||||
reg = readl(owl_host->base + OWL_REG_SD_EN);
|
||||
reg &= ~0x03;
|
||||
switch (ios->bus_width) {
|
||||
case MMC_BUS_WIDTH_1:
|
||||
break;
|
||||
case MMC_BUS_WIDTH_4:
|
||||
reg |= OWL_SD_EN_DATAWID(1);
|
||||
break;
|
||||
case MMC_BUS_WIDTH_8:
|
||||
reg |= OWL_SD_EN_DATAWID(2);
|
||||
break;
|
||||
}
|
||||
|
||||
writel(reg, owl_host->base + OWL_REG_SD_EN);
|
||||
}
|
||||
|
||||
static void owl_mmc_ctr_reset(struct owl_mmc_host *owl_host)
|
||||
{
|
||||
reset_control_assert(owl_host->reset);
|
||||
udelay(20);
|
||||
reset_control_deassert(owl_host->reset);
|
||||
}
|
||||
|
||||
static void owl_mmc_power_on(struct owl_mmc_host *owl_host)
|
||||
{
|
||||
u32 mode;
|
||||
|
||||
init_completion(&owl_host->sdc_complete);
|
||||
|
||||
/* Enable transfer end IRQ */
|
||||
owl_mmc_update_reg(owl_host->base + OWL_REG_SD_STATE,
|
||||
OWL_SD_STATE_TEIE, true);
|
||||
|
||||
/* Send init clk */
|
||||
mode = (readl(owl_host->base + OWL_REG_SD_CTL) & (0xff << 16));
|
||||
mode |= OWL_SD_CTL_TS | OWL_SD_CTL_TCN(5) | OWL_SD_CTL_TM(8);
|
||||
writel(mode, owl_host->base + OWL_REG_SD_CTL);
|
||||
|
||||
if (!wait_for_completion_timeout(&owl_host->sdc_complete, HZ)) {
|
||||
dev_err(owl_host->dev, "CMD interrupt timeout\n");
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
static void owl_mmc_set_ios(struct mmc_host *mmc, struct mmc_ios *ios)
|
||||
{
|
||||
struct owl_mmc_host *owl_host = mmc_priv(mmc);
|
||||
|
||||
switch (ios->power_mode) {
|
||||
case MMC_POWER_UP:
|
||||
dev_dbg(owl_host->dev, "Powering card up\n");
|
||||
|
||||
/* Reset the SDC controller to clear all previous states */
|
||||
owl_mmc_ctr_reset(owl_host);
|
||||
clk_prepare_enable(owl_host->clk);
|
||||
writel(OWL_SD_ENABLE | OWL_SD_EN_RESE,
|
||||
owl_host->base + OWL_REG_SD_EN);
|
||||
|
||||
break;
|
||||
|
||||
case MMC_POWER_ON:
|
||||
dev_dbg(owl_host->dev, "Powering card on\n");
|
||||
owl_mmc_power_on(owl_host);
|
||||
|
||||
break;
|
||||
|
||||
case MMC_POWER_OFF:
|
||||
dev_dbg(owl_host->dev, "Powering card off\n");
|
||||
clk_disable_unprepare(owl_host->clk);
|
||||
|
||||
return;
|
||||
|
||||
default:
|
||||
dev_dbg(owl_host->dev, "Ignoring unknown card power state\n");
|
||||
break;
|
||||
}
|
||||
|
||||
if (ios->clock != owl_host->clock)
|
||||
owl_mmc_set_clk(owl_host, ios);
|
||||
|
||||
owl_mmc_set_bus_width(owl_host, ios);
|
||||
|
||||
/* Enable DDR mode if requested */
|
||||
if (ios->timing == MMC_TIMING_UHS_DDR50) {
|
||||
owl_host->ddr_50 = 1;
|
||||
owl_mmc_update_reg(owl_host->base + OWL_REG_SD_EN,
|
||||
OWL_SD_EN_DDREN, true);
|
||||
} else {
|
||||
owl_host->ddr_50 = 0;
|
||||
}
|
||||
}
|
||||
|
||||
static int owl_mmc_start_signal_voltage_switch(struct mmc_host *mmc,
|
||||
struct mmc_ios *ios)
|
||||
{
|
||||
struct owl_mmc_host *owl_host = mmc_priv(mmc);
|
||||
|
||||
/* It is enough to change the pad ctrl bit for voltage switch */
|
||||
switch (ios->signal_voltage) {
|
||||
case MMC_SIGNAL_VOLTAGE_330:
|
||||
owl_mmc_update_reg(owl_host->base + OWL_REG_SD_EN,
|
||||
OWL_SD_EN_S18EN, false);
|
||||
break;
|
||||
case MMC_SIGNAL_VOLTAGE_180:
|
||||
owl_mmc_update_reg(owl_host->base + OWL_REG_SD_EN,
|
||||
OWL_SD_EN_S18EN, true);
|
||||
break;
|
||||
default:
|
||||
return -ENOTSUPP;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const struct mmc_host_ops owl_mmc_ops = {
|
||||
.request = owl_mmc_request,
|
||||
.set_ios = owl_mmc_set_ios,
|
||||
.get_ro = mmc_gpio_get_ro,
|
||||
.get_cd = mmc_gpio_get_cd,
|
||||
.start_signal_voltage_switch = owl_mmc_start_signal_voltage_switch,
|
||||
};
|
||||
|
||||
static int owl_mmc_probe(struct platform_device *pdev)
|
||||
{
|
||||
struct owl_mmc_host *owl_host;
|
||||
struct mmc_host *mmc;
|
||||
struct resource *res;
|
||||
int ret;
|
||||
|
||||
mmc = mmc_alloc_host(sizeof(struct owl_mmc_host), &pdev->dev);
|
||||
if (!mmc) {
|
||||
dev_err(&pdev->dev, "mmc alloc host failed\n");
|
||||
return -ENOMEM;
|
||||
}
|
||||
platform_set_drvdata(pdev, mmc);
|
||||
|
||||
owl_host = mmc_priv(mmc);
|
||||
owl_host->dev = &pdev->dev;
|
||||
owl_host->mmc = mmc;
|
||||
spin_lock_init(&owl_host->lock);
|
||||
|
||||
res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
||||
owl_host->base = devm_ioremap_resource(&pdev->dev, res);
|
||||
if (IS_ERR(owl_host->base)) {
|
||||
dev_err(&pdev->dev, "Failed to remap registers\n");
|
||||
ret = PTR_ERR(owl_host->base);
|
||||
goto err_free_host;
|
||||
}
|
||||
|
||||
owl_host->clk = devm_clk_get(&pdev->dev, NULL);
|
||||
if (IS_ERR(owl_host->clk)) {
|
||||
dev_err(&pdev->dev, "No clock defined\n");
|
||||
ret = PTR_ERR(owl_host->clk);
|
||||
goto err_free_host;
|
||||
}
|
||||
|
||||
owl_host->reset = devm_reset_control_get_exclusive(&pdev->dev, NULL);
|
||||
if (IS_ERR(owl_host->reset)) {
|
||||
dev_err(&pdev->dev, "Could not get reset control\n");
|
||||
ret = PTR_ERR(owl_host->reset);
|
||||
goto err_free_host;
|
||||
}
|
||||
|
||||
mmc->ops = &owl_mmc_ops;
|
||||
mmc->max_blk_count = 512;
|
||||
mmc->max_blk_size = 512;
|
||||
mmc->max_segs = 256;
|
||||
mmc->max_seg_size = 262144;
|
||||
mmc->max_req_size = 262144;
|
||||
/* 100kHz ~ 52MHz */
|
||||
mmc->f_min = 100000;
|
||||
mmc->f_max = 52000000;
|
||||
mmc->caps |= MMC_CAP_MMC_HIGHSPEED | MMC_CAP_SD_HIGHSPEED |
|
||||
MMC_CAP_4_BIT_DATA;
|
||||
mmc->caps2 = (MMC_CAP2_BOOTPART_NOACC | MMC_CAP2_NO_SDIO);
|
||||
mmc->ocr_avail = MMC_VDD_32_33 | MMC_VDD_33_34 |
|
||||
MMC_VDD_165_195;
|
||||
|
||||
ret = mmc_of_parse(mmc);
|
||||
if (ret)
|
||||
goto err_free_host;
|
||||
|
||||
pdev->dev.coherent_dma_mask = DMA_BIT_MASK(32);
|
||||
pdev->dev.dma_mask = &pdev->dev.coherent_dma_mask;
|
||||
owl_host->dma = dma_request_slave_channel(&pdev->dev, "mmc");
|
||||
if (!owl_host->dma) {
|
||||
dev_err(owl_host->dev, "Failed to get external DMA channel.\n");
|
||||
ret = -ENXIO;
|
||||
goto err_free_host;
|
||||
}
|
||||
|
||||
dev_info(&pdev->dev, "Using %s for DMA transfers\n",
|
||||
dma_chan_name(owl_host->dma));
|
||||
|
||||
owl_host->dma_cfg.src_addr = res->start + OWL_REG_SD_DAT;
|
||||
owl_host->dma_cfg.dst_addr = res->start + OWL_REG_SD_DAT;
|
||||
owl_host->dma_cfg.src_addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES;
|
||||
owl_host->dma_cfg.dst_addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES;
|
||||
owl_host->dma_cfg.device_fc = false;
|
||||
|
||||
owl_host->irq = platform_get_irq(pdev, 0);
|
||||
if (owl_host->irq < 0) {
|
||||
ret = -EINVAL;
|
||||
goto err_free_host;
|
||||
}
|
||||
|
||||
ret = devm_request_irq(&pdev->dev, owl_host->irq, owl_irq_handler,
|
||||
0, dev_name(&pdev->dev), owl_host);
|
||||
if (ret) {
|
||||
dev_err(&pdev->dev, "Failed to request irq %d\n",
|
||||
owl_host->irq);
|
||||
goto err_free_host;
|
||||
}
|
||||
|
||||
ret = mmc_add_host(mmc);
|
||||
if (ret) {
|
||||
dev_err(&pdev->dev, "Failed to add host\n");
|
||||
goto err_free_host;
|
||||
}
|
||||
|
||||
dev_dbg(&pdev->dev, "Owl MMC Controller Initialized\n");
|
||||
|
||||
return 0;
|
||||
|
||||
err_free_host:
|
||||
mmc_free_host(mmc);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int owl_mmc_remove(struct platform_device *pdev)
|
||||
{
|
||||
struct mmc_host *mmc = platform_get_drvdata(pdev);
|
||||
struct owl_mmc_host *owl_host = mmc_priv(mmc);
|
||||
|
||||
mmc_remove_host(mmc);
|
||||
disable_irq(owl_host->irq);
|
||||
mmc_free_host(mmc);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const struct of_device_id owl_mmc_of_match[] = {
|
||||
{.compatible = "actions,owl-mmc",},
|
||||
{ /* sentinel */ }
|
||||
};
|
||||
MODULE_DEVICE_TABLE(of, owl_mmc_of_match);
|
||||
|
||||
static struct platform_driver owl_mmc_driver = {
|
||||
.driver = {
|
||||
.name = "owl_mmc",
|
||||
.of_match_table = of_match_ptr(owl_mmc_of_match),
|
||||
},
|
||||
.probe = owl_mmc_probe,
|
||||
.remove = owl_mmc_remove,
|
||||
};
|
||||
module_platform_driver(owl_mmc_driver);
|
||||
|
||||
MODULE_DESCRIPTION("Actions Semi Owl SoCs SD/MMC Driver");
|
||||
MODULE_AUTHOR("Actions Semi");
|
||||
MODULE_AUTHOR("Manivannan Sadhasivam <manivannan.sadhasivam@linaro.org>");
|
||||
MODULE_LICENSE("GPL");
|
|
@ -308,6 +308,7 @@ static const struct soc_device_attribute soc_whitelist[] = {
|
|||
.data = (void *)BIT(SDHI_INTERNAL_DMAC_ONE_RX_ONLY) },
|
||||
/* generic ones */
|
||||
{ .soc_id = "r8a774a1" },
|
||||
{ .soc_id = "r8a774b1" },
|
||||
{ .soc_id = "r8a774c0" },
|
||||
{ .soc_id = "r8a77470" },
|
||||
{ .soc_id = "r8a7795" },
|
||||
|
|
|
@ -51,6 +51,11 @@
|
|||
#define ESDHC_CLOCK_HCKEN 0x00000002
|
||||
#define ESDHC_CLOCK_IPGEN 0x00000001
|
||||
|
||||
/* System Control 2 Register */
|
||||
#define ESDHC_SYSTEM_CONTROL_2 0x3c
|
||||
#define ESDHC_SMPCLKSEL 0x00800000
|
||||
#define ESDHC_EXTN 0x00400000
|
||||
|
||||
/* Host Controller Capabilities Register 2 */
|
||||
#define ESDHC_CAPABILITIES_1 0x114
|
||||
|
||||
|
@ -59,7 +64,16 @@
|
|||
#define ESDHC_HS400_WNDW_ADJUST 0x00000040
|
||||
#define ESDHC_HS400_MODE 0x00000010
|
||||
#define ESDHC_TB_EN 0x00000004
|
||||
#define ESDHC_TB_MODE_MASK 0x00000003
|
||||
#define ESDHC_TB_MODE_SW 0x00000003
|
||||
#define ESDHC_TB_MODE_3 0x00000002
|
||||
|
||||
#define ESDHC_TBSTAT 0x124
|
||||
|
||||
#define ESDHC_TBPTR 0x128
|
||||
#define ESDHC_WNDW_STRT_PTR_SHIFT 8
|
||||
#define ESDHC_WNDW_STRT_PTR_MASK (0x7f << 8)
|
||||
#define ESDHC_WNDW_END_PTR_MASK 0x7f
|
||||
|
||||
/* SD Clock Control Register */
|
||||
#define ESDHC_SDCLKCTL 0x144
|
||||
|
|
|
@ -0,0 +1,362 @@
|
|||
// SPDX-License-Identifier: GPL-2.0
|
||||
/*
|
||||
* Copyright (C) 2013 - 2015 Fujitsu Semiconductor, Ltd
|
||||
* Vincent Yang <vincent.yang@tw.fujitsu.com>
|
||||
* Copyright (C) 2015 Linaro Ltd Andy Green <andy.green@linaro.org>
|
||||
* Copyright (C) 2019 Socionext Inc.
|
||||
* Takao Orito <orito.takao@socionext.com>
|
||||
*/
|
||||
|
||||
#include <linux/bits.h>
|
||||
#include <linux/clk.h>
|
||||
#include <linux/delay.h>
|
||||
#include <linux/err.h>
|
||||
#include <linux/gpio/consumer.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/of.h>
|
||||
#include <linux/property.h>
|
||||
|
||||
#include "sdhci-pltfm.h"
|
||||
#include "sdhci_f_sdh30.h"
|
||||
|
||||
/* milbeaut bridge controller register */
|
||||
#define MLB_SOFT_RESET 0x0200
|
||||
#define MLB_SOFT_RESET_RSTX BIT(0)
|
||||
|
||||
#define MLB_WP_CD_LED_SET 0x0210
|
||||
#define MLB_WP_CD_LED_SET_LED_INV BIT(2)
|
||||
|
||||
#define MLB_CR_SET 0x0220
|
||||
#define MLB_CR_SET_CR_TOCLKUNIT BIT(24)
|
||||
#define MLB_CR_SET_CR_TOCLKFREQ_SFT (16)
|
||||
#define MLB_CR_SET_CR_TOCLKFREQ_MASK (0x3F << MLB_CR_SET_CR_TOCLKFREQ_SFT)
|
||||
#define MLB_CR_SET_CR_BCLKFREQ_SFT (8)
|
||||
#define MLB_CR_SET_CR_BCLKFREQ_MASK (0xFF << MLB_CR_SET_CR_BCLKFREQ_SFT)
|
||||
#define MLB_CR_SET_CR_RTUNTIMER_SFT (4)
|
||||
#define MLB_CR_SET_CR_RTUNTIMER_MASK (0xF << MLB_CR_SET_CR_RTUNTIMER_SFT)
|
||||
|
||||
#define MLB_SD_TOCLK_I_DIV 16
|
||||
#define MLB_TOCLKFREQ_UNIT_THRES 16000000
|
||||
#define MLB_CAL_TOCLKFREQ_MHZ(rate) (rate / MLB_SD_TOCLK_I_DIV / 1000000)
|
||||
#define MLB_CAL_TOCLKFREQ_KHZ(rate) (rate / MLB_SD_TOCLK_I_DIV / 1000)
|
||||
#define MLB_TOCLKFREQ_MAX 63
|
||||
#define MLB_TOCLKFREQ_MIN 1
|
||||
|
||||
#define MLB_SD_BCLK_I_DIV 4
|
||||
#define MLB_CAL_BCLKFREQ(rate) (rate / MLB_SD_BCLK_I_DIV / 1000000)
|
||||
#define MLB_BCLKFREQ_MAX 255
|
||||
#define MLB_BCLKFREQ_MIN 1
|
||||
|
||||
#define MLB_CDR_SET 0x0230
|
||||
#define MLB_CDR_SET_CLK2POW16 3
|
||||
|
||||
struct f_sdhost_priv {
|
||||
struct clk *clk_iface;
|
||||
struct clk *clk;
|
||||
struct device *dev;
|
||||
bool enable_cmd_dat_delay;
|
||||
};
|
||||
|
||||
static void sdhci_milbeaut_soft_voltage_switch(struct sdhci_host *host)
|
||||
{
|
||||
u32 ctrl = 0;
|
||||
|
||||
usleep_range(2500, 3000);
|
||||
ctrl = sdhci_readl(host, F_SDH30_IO_CONTROL2);
|
||||
ctrl |= F_SDH30_CRES_O_DN;
|
||||
sdhci_writel(host, ctrl, F_SDH30_IO_CONTROL2);
|
||||
ctrl |= F_SDH30_MSEL_O_1_8;
|
||||
sdhci_writel(host, ctrl, F_SDH30_IO_CONTROL2);
|
||||
|
||||
ctrl &= ~F_SDH30_CRES_O_DN;
|
||||
sdhci_writel(host, ctrl, F_SDH30_IO_CONTROL2);
|
||||
usleep_range(2500, 3000);
|
||||
|
||||
ctrl = sdhci_readl(host, F_SDH30_TUNING_SETTING);
|
||||
ctrl |= F_SDH30_CMD_CHK_DIS;
|
||||
sdhci_writel(host, ctrl, F_SDH30_TUNING_SETTING);
|
||||
}
|
||||
|
||||
static unsigned int sdhci_milbeaut_get_min_clock(struct sdhci_host *host)
|
||||
{
|
||||
return F_SDH30_MIN_CLOCK;
|
||||
}
|
||||
|
||||
static void sdhci_milbeaut_reset(struct sdhci_host *host, u8 mask)
|
||||
{
|
||||
struct f_sdhost_priv *priv = sdhci_priv(host);
|
||||
u16 clk;
|
||||
u32 ctl;
|
||||
ktime_t timeout;
|
||||
|
||||
clk = sdhci_readw(host, SDHCI_CLOCK_CONTROL);
|
||||
clk = (clk & ~SDHCI_CLOCK_CARD_EN) | SDHCI_CLOCK_INT_EN;
|
||||
sdhci_writew(host, clk, SDHCI_CLOCK_CONTROL);
|
||||
|
||||
sdhci_reset(host, mask);
|
||||
|
||||
clk |= SDHCI_CLOCK_CARD_EN;
|
||||
sdhci_writew(host, clk, SDHCI_CLOCK_CONTROL);
|
||||
|
||||
timeout = ktime_add_ms(ktime_get(), 10);
|
||||
while (1) {
|
||||
bool timedout = ktime_after(ktime_get(), timeout);
|
||||
|
||||
clk = sdhci_readw(host, SDHCI_CLOCK_CONTROL);
|
||||
if (clk & SDHCI_CLOCK_INT_STABLE)
|
||||
break;
|
||||
if (timedout) {
|
||||
pr_err("%s: Internal clock never stabilised.\n",
|
||||
mmc_hostname(host->mmc));
|
||||
sdhci_dumpregs(host);
|
||||
return;
|
||||
}
|
||||
udelay(10);
|
||||
}
|
||||
|
||||
if (priv->enable_cmd_dat_delay) {
|
||||
ctl = sdhci_readl(host, F_SDH30_ESD_CONTROL);
|
||||
ctl |= F_SDH30_CMD_DAT_DELAY;
|
||||
sdhci_writel(host, ctl, F_SDH30_ESD_CONTROL);
|
||||
}
|
||||
}
|
||||
|
||||
static void sdhci_milbeaut_set_power(struct sdhci_host *host,
|
||||
unsigned char mode, unsigned short vdd)
|
||||
{
|
||||
if (!IS_ERR(host->mmc->supply.vmmc)) {
|
||||
struct mmc_host *mmc = host->mmc;
|
||||
|
||||
mmc_regulator_set_ocr(mmc, mmc->supply.vmmc, vdd);
|
||||
}
|
||||
sdhci_set_power_noreg(host, mode, vdd);
|
||||
}
|
||||
|
||||
static const struct sdhci_ops sdhci_milbeaut_ops = {
|
||||
.voltage_switch = sdhci_milbeaut_soft_voltage_switch,
|
||||
.get_min_clock = sdhci_milbeaut_get_min_clock,
|
||||
.reset = sdhci_milbeaut_reset,
|
||||
.set_clock = sdhci_set_clock,
|
||||
.set_bus_width = sdhci_set_bus_width,
|
||||
.set_uhs_signaling = sdhci_set_uhs_signaling,
|
||||
.set_power = sdhci_milbeaut_set_power,
|
||||
};
|
||||
|
||||
static void sdhci_milbeaut_bridge_reset(struct sdhci_host *host,
|
||||
int reset_flag)
|
||||
{
|
||||
if (reset_flag)
|
||||
sdhci_writel(host, 0, MLB_SOFT_RESET);
|
||||
else
|
||||
sdhci_writel(host, MLB_SOFT_RESET_RSTX, MLB_SOFT_RESET);
|
||||
}
|
||||
|
||||
static void sdhci_milbeaut_bridge_init(struct sdhci_host *host,
|
||||
int rate)
|
||||
{
|
||||
u32 val, clk;
|
||||
|
||||
/* IO_SDIO_CR_SET should be set while reset */
|
||||
val = sdhci_readl(host, MLB_CR_SET);
|
||||
val &= ~(MLB_CR_SET_CR_TOCLKFREQ_MASK | MLB_CR_SET_CR_TOCLKUNIT |
|
||||
MLB_CR_SET_CR_BCLKFREQ_MASK);
|
||||
if (rate >= MLB_TOCLKFREQ_UNIT_THRES) {
|
||||
clk = MLB_CAL_TOCLKFREQ_MHZ(rate);
|
||||
clk = min_t(u32, MLB_TOCLKFREQ_MAX, clk);
|
||||
val |= MLB_CR_SET_CR_TOCLKUNIT |
|
||||
(clk << MLB_CR_SET_CR_TOCLKFREQ_SFT);
|
||||
} else {
|
||||
clk = MLB_CAL_TOCLKFREQ_KHZ(rate);
|
||||
clk = min_t(u32, MLB_TOCLKFREQ_MAX, clk);
|
||||
clk = max_t(u32, MLB_TOCLKFREQ_MIN, clk);
|
||||
val |= clk << MLB_CR_SET_CR_TOCLKFREQ_SFT;
|
||||
}
|
||||
|
||||
clk = MLB_CAL_BCLKFREQ(rate);
|
||||
clk = min_t(u32, MLB_BCLKFREQ_MAX, clk);
|
||||
clk = max_t(u32, MLB_BCLKFREQ_MIN, clk);
|
||||
val |= clk << MLB_CR_SET_CR_BCLKFREQ_SFT;
|
||||
val &= ~MLB_CR_SET_CR_RTUNTIMER_MASK;
|
||||
sdhci_writel(host, val, MLB_CR_SET);
|
||||
|
||||
sdhci_writel(host, MLB_CDR_SET_CLK2POW16, MLB_CDR_SET);
|
||||
|
||||
sdhci_writel(host, MLB_WP_CD_LED_SET_LED_INV, MLB_WP_CD_LED_SET);
|
||||
}
|
||||
|
||||
static void sdhci_milbeaut_vendor_init(struct sdhci_host *host)
|
||||
{
|
||||
struct f_sdhost_priv *priv = sdhci_priv(host);
|
||||
u32 ctl;
|
||||
|
||||
ctl = sdhci_readl(host, F_SDH30_IO_CONTROL2);
|
||||
ctl |= F_SDH30_CRES_O_DN;
|
||||
sdhci_writel(host, ctl, F_SDH30_IO_CONTROL2);
|
||||
ctl &= ~F_SDH30_MSEL_O_1_8;
|
||||
sdhci_writel(host, ctl, F_SDH30_IO_CONTROL2);
|
||||
ctl &= ~F_SDH30_CRES_O_DN;
|
||||
sdhci_writel(host, ctl, F_SDH30_IO_CONTROL2);
|
||||
|
||||
ctl = sdhci_readw(host, F_SDH30_AHB_CONFIG);
|
||||
ctl |= F_SDH30_SIN | F_SDH30_AHB_INCR_16 | F_SDH30_AHB_INCR_8 |
|
||||
F_SDH30_AHB_INCR_4;
|
||||
ctl &= ~(F_SDH30_AHB_BIGED | F_SDH30_BUSLOCK_EN);
|
||||
sdhci_writew(host, ctl, F_SDH30_AHB_CONFIG);
|
||||
|
||||
if (priv->enable_cmd_dat_delay) {
|
||||
ctl = sdhci_readl(host, F_SDH30_ESD_CONTROL);
|
||||
ctl |= F_SDH30_CMD_DAT_DELAY;
|
||||
sdhci_writel(host, ctl, F_SDH30_ESD_CONTROL);
|
||||
}
|
||||
}
|
||||
|
||||
static const struct of_device_id mlb_dt_ids[] = {
|
||||
{
|
||||
.compatible = "socionext,milbeaut-m10v-sdhci-3.0",
|
||||
},
|
||||
{ /* sentinel */ }
|
||||
};
|
||||
MODULE_DEVICE_TABLE(of, mlb_dt_ids);
|
||||
|
||||
static void sdhci_milbeaut_init(struct sdhci_host *host)
|
||||
{
|
||||
struct f_sdhost_priv *priv = sdhci_priv(host);
|
||||
int rate = clk_get_rate(priv->clk);
|
||||
u16 ctl;
|
||||
|
||||
sdhci_milbeaut_bridge_reset(host, 0);
|
||||
|
||||
ctl = sdhci_readw(host, SDHCI_CLOCK_CONTROL);
|
||||
ctl &= ~(SDHCI_CLOCK_CARD_EN | SDHCI_CLOCK_INT_EN);
|
||||
sdhci_writew(host, ctl, SDHCI_CLOCK_CONTROL);
|
||||
|
||||
sdhci_milbeaut_bridge_reset(host, 1);
|
||||
|
||||
sdhci_milbeaut_bridge_init(host, rate);
|
||||
sdhci_milbeaut_bridge_reset(host, 0);
|
||||
|
||||
sdhci_milbeaut_vendor_init(host);
|
||||
}
|
||||
|
||||
static int sdhci_milbeaut_probe(struct platform_device *pdev)
|
||||
{
|
||||
struct sdhci_host *host;
|
||||
struct device *dev = &pdev->dev;
|
||||
struct resource *res;
|
||||
int irq, ret = 0;
|
||||
struct f_sdhost_priv *priv;
|
||||
|
||||
irq = platform_get_irq(pdev, 0);
|
||||
if (irq < 0) {
|
||||
dev_err(dev, "%s: no irq specified\n", __func__);
|
||||
return irq;
|
||||
}
|
||||
|
||||
host = sdhci_alloc_host(dev, sizeof(struct f_sdhost_priv));
|
||||
if (IS_ERR(host))
|
||||
return PTR_ERR(host);
|
||||
|
||||
priv = sdhci_priv(host);
|
||||
priv->dev = dev;
|
||||
|
||||
host->quirks = SDHCI_QUIRK_NO_ENDATTR_IN_NOPDESC |
|
||||
SDHCI_QUIRK_INVERTED_WRITE_PROTECT |
|
||||
SDHCI_QUIRK_CLOCK_BEFORE_RESET |
|
||||
SDHCI_QUIRK_DELAY_AFTER_POWER;
|
||||
host->quirks2 = SDHCI_QUIRK2_SUPPORT_SINGLE |
|
||||
SDHCI_QUIRK2_TUNING_WORK_AROUND |
|
||||
SDHCI_QUIRK2_PRESET_VALUE_BROKEN;
|
||||
|
||||
priv->enable_cmd_dat_delay = device_property_read_bool(dev,
|
||||
"fujitsu,cmd-dat-delay-select");
|
||||
|
||||
ret = mmc_of_parse(host->mmc);
|
||||
if (ret)
|
||||
goto err;
|
||||
|
||||
platform_set_drvdata(pdev, host);
|
||||
|
||||
host->hw_name = "f_sdh30";
|
||||
host->ops = &sdhci_milbeaut_ops;
|
||||
host->irq = irq;
|
||||
|
||||
res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
||||
host->ioaddr = devm_ioremap_resource(&pdev->dev, res);
|
||||
if (IS_ERR(host->ioaddr)) {
|
||||
ret = PTR_ERR(host->ioaddr);
|
||||
goto err;
|
||||
}
|
||||
|
||||
if (dev_of_node(dev)) {
|
||||
sdhci_get_of_property(pdev);
|
||||
|
||||
priv->clk_iface = devm_clk_get(&pdev->dev, "iface");
|
||||
if (IS_ERR(priv->clk_iface)) {
|
||||
ret = PTR_ERR(priv->clk_iface);
|
||||
goto err;
|
||||
}
|
||||
|
||||
ret = clk_prepare_enable(priv->clk_iface);
|
||||
if (ret)
|
||||
goto err;
|
||||
|
||||
priv->clk = devm_clk_get(&pdev->dev, "core");
|
||||
if (IS_ERR(priv->clk)) {
|
||||
ret = PTR_ERR(priv->clk);
|
||||
goto err_clk;
|
||||
}
|
||||
|
||||
ret = clk_prepare_enable(priv->clk);
|
||||
if (ret)
|
||||
goto err_clk;
|
||||
}
|
||||
|
||||
sdhci_milbeaut_init(host);
|
||||
|
||||
ret = sdhci_add_host(host);
|
||||
if (ret)
|
||||
goto err_add_host;
|
||||
|
||||
return 0;
|
||||
|
||||
err_add_host:
|
||||
clk_disable_unprepare(priv->clk);
|
||||
err_clk:
|
||||
clk_disable_unprepare(priv->clk_iface);
|
||||
err:
|
||||
sdhci_free_host(host);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int sdhci_milbeaut_remove(struct platform_device *pdev)
|
||||
{
|
||||
struct sdhci_host *host = platform_get_drvdata(pdev);
|
||||
struct f_sdhost_priv *priv = sdhci_priv(host);
|
||||
|
||||
sdhci_remove_host(host, readl(host->ioaddr + SDHCI_INT_STATUS) ==
|
||||
0xffffffff);
|
||||
|
||||
clk_disable_unprepare(priv->clk_iface);
|
||||
clk_disable_unprepare(priv->clk);
|
||||
|
||||
sdhci_free_host(host);
|
||||
platform_set_drvdata(pdev, NULL);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static struct platform_driver sdhci_milbeaut_driver = {
|
||||
.driver = {
|
||||
.name = "sdhci-milbeaut",
|
||||
.of_match_table = of_match_ptr(mlb_dt_ids),
|
||||
},
|
||||
.probe = sdhci_milbeaut_probe,
|
||||
.remove = sdhci_milbeaut_remove,
|
||||
};
|
||||
|
||||
module_platform_driver(sdhci_milbeaut_driver);
|
||||
|
||||
MODULE_DESCRIPTION("MILBEAUT SD Card Controller driver");
|
||||
MODULE_AUTHOR("Takao Orito <orito.takao@socionext.com>");
|
||||
MODULE_LICENSE("GPL v2");
|
||||
MODULE_ALIAS("platform:sdhci-milbeaut");
|
|
@ -22,6 +22,7 @@
|
|||
#include <linux/phy/phy.h>
|
||||
#include <linux/regmap.h>
|
||||
#include <linux/of.h>
|
||||
#include <linux/firmware/xlnx-zynqmp.h>
|
||||
|
||||
#include "cqhci.h"
|
||||
#include "sdhci-pltfm.h"
|
||||
|
@ -32,6 +33,10 @@
|
|||
|
||||
#define PHY_CLK_TOO_SLOW_HZ 400000
|
||||
|
||||
/* Default settings for ZynqMP Clock Phases */
|
||||
#define ZYNQMP_ICLK_PHASE {0, 63, 63, 0, 63, 0, 0, 183, 54, 0, 0}
|
||||
#define ZYNQMP_OCLK_PHASE {0, 72, 60, 0, 60, 72, 135, 48, 72, 135, 0}
|
||||
|
||||
/*
|
||||
* On some SoCs the syscon area has a feature where the upper 16-bits of
|
||||
* each 32-bit register act as a write mask for the lower 16-bits. This allows
|
||||
|
@ -71,14 +76,39 @@ struct sdhci_arasan_soc_ctl_map {
|
|||
bool hiword_update;
|
||||
};
|
||||
|
||||
/**
|
||||
* struct sdhci_arasan_clk_data
|
||||
* @sdcardclk_hw: Struct for the clock we might provide to a PHY.
|
||||
* @sdcardclk: Pointer to normal 'struct clock' for sdcardclk_hw.
|
||||
* @sampleclk_hw: Struct for the clock we might provide to a PHY.
|
||||
* @sampleclk: Pointer to normal 'struct clock' for sampleclk_hw.
|
||||
* @clk_phase_in: Array of Input Clock Phase Delays for all speed modes
|
||||
* @clk_phase_out: Array of Output Clock Phase Delays for all speed modes
|
||||
* @set_clk_delays: Function pointer for setting Clock Delays
|
||||
* @clk_of_data: Platform specific runtime clock data storage pointer
|
||||
*/
|
||||
struct sdhci_arasan_clk_data {
|
||||
struct clk_hw sdcardclk_hw;
|
||||
struct clk *sdcardclk;
|
||||
struct clk_hw sampleclk_hw;
|
||||
struct clk *sampleclk;
|
||||
int clk_phase_in[MMC_TIMING_MMC_HS400 + 1];
|
||||
int clk_phase_out[MMC_TIMING_MMC_HS400 + 1];
|
||||
void (*set_clk_delays)(struct sdhci_host *host);
|
||||
void *clk_of_data;
|
||||
};
|
||||
|
||||
struct sdhci_arasan_zynqmp_clk_data {
|
||||
const struct zynqmp_eemi_ops *eemi_ops;
|
||||
};
|
||||
|
||||
/**
|
||||
* struct sdhci_arasan_data
|
||||
* @host: Pointer to the main SDHCI host structure.
|
||||
* @clk_ahb: Pointer to the AHB clock
|
||||
* @phy: Pointer to the generic phy
|
||||
* @is_phy_on: True if the PHY is on; false if not.
|
||||
* @sdcardclk_hw: Struct for the clock we might provide to a PHY.
|
||||
* @sdcardclk: Pointer to normal 'struct clock' for sdcardclk_hw.
|
||||
* @clk_data: Struct for the Arasan Controller Clock Data.
|
||||
* @soc_ctl_base: Pointer to regmap for syscon for soc_ctl registers.
|
||||
* @soc_ctl_map: Map to get offsets into soc_ctl registers.
|
||||
*/
|
||||
|
@ -89,8 +119,7 @@ struct sdhci_arasan_data {
|
|||
bool is_phy_on;
|
||||
|
||||
bool has_cqe;
|
||||
struct clk_hw sdcardclk_hw;
|
||||
struct clk *sdcardclk;
|
||||
struct sdhci_arasan_clk_data clk_data;
|
||||
|
||||
struct regmap *soc_ctl_base;
|
||||
const struct sdhci_arasan_soc_ctl_map *soc_ctl_map;
|
||||
|
@ -120,6 +149,12 @@ static const struct sdhci_arasan_soc_ctl_map intel_lgm_emmc_soc_ctl_map = {
|
|||
.hiword_update = false,
|
||||
};
|
||||
|
||||
static const struct sdhci_arasan_soc_ctl_map intel_lgm_sdxc_soc_ctl_map = {
|
||||
.baseclkfreq = { .reg = 0x80, .width = 8, .shift = 2 },
|
||||
.clockmultiplier = { .reg = 0, .width = -1, .shift = -1 },
|
||||
.hiword_update = false,
|
||||
};
|
||||
|
||||
/**
|
||||
* sdhci_arasan_syscon_write - Write to a field in soc_ctl registers
|
||||
*
|
||||
|
@ -174,6 +209,7 @@ static void sdhci_arasan_set_clock(struct sdhci_host *host, unsigned int clock)
|
|||
{
|
||||
struct sdhci_pltfm_host *pltfm_host = sdhci_priv(host);
|
||||
struct sdhci_arasan_data *sdhci_arasan = sdhci_pltfm_priv(pltfm_host);
|
||||
struct sdhci_arasan_clk_data *clk_data = &sdhci_arasan->clk_data;
|
||||
bool ctrl_phy = false;
|
||||
|
||||
if (!IS_ERR(sdhci_arasan->phy)) {
|
||||
|
@ -215,6 +251,10 @@ static void sdhci_arasan_set_clock(struct sdhci_host *host, unsigned int clock)
|
|||
sdhci_arasan->is_phy_on = false;
|
||||
}
|
||||
|
||||
/* Set the Input and Output Clock Phase Delays */
|
||||
if (clk_data->set_clk_delays)
|
||||
clk_data->set_clk_delays(host);
|
||||
|
||||
sdhci_set_clock(host, clock);
|
||||
|
||||
if (sdhci_arasan->quirks & SDHCI_ARASAN_QUIRK_CLOCK_UNSTABLE)
|
||||
|
@ -384,6 +424,11 @@ static struct sdhci_arasan_of_data intel_lgm_emmc_data = {
|
|||
.pdata = &sdhci_arasan_cqe_pdata,
|
||||
};
|
||||
|
||||
static struct sdhci_arasan_of_data intel_lgm_sdxc_data = {
|
||||
.soc_ctl_map = &intel_lgm_sdxc_soc_ctl_map,
|
||||
.pdata = &sdhci_arasan_cqe_pdata,
|
||||
};
|
||||
|
||||
#ifdef CONFIG_PM_SLEEP
|
||||
/**
|
||||
* sdhci_arasan_suspend - Suspend method for the driver
|
||||
|
@ -489,6 +534,10 @@ static const struct of_device_id sdhci_arasan_of_match[] = {
|
|||
.compatible = "intel,lgm-sdhci-5.1-emmc",
|
||||
.data = &intel_lgm_emmc_data,
|
||||
},
|
||||
{
|
||||
.compatible = "intel,lgm-sdhci-5.1-sdxc",
|
||||
.data = &intel_lgm_sdxc_data,
|
||||
},
|
||||
/* Generic compatible below here */
|
||||
{
|
||||
.compatible = "arasan,sdhci-8.9a",
|
||||
|
@ -502,6 +551,10 @@ static const struct of_device_id sdhci_arasan_of_match[] = {
|
|||
.compatible = "arasan,sdhci-4.9a",
|
||||
.data = &sdhci_arasan_data,
|
||||
},
|
||||
{
|
||||
.compatible = "xlnx,zynqmp-8.9a",
|
||||
.data = &sdhci_arasan_data,
|
||||
},
|
||||
{ /* sentinel */ }
|
||||
};
|
||||
MODULE_DEVICE_TABLE(of, sdhci_arasan_of_match);
|
||||
|
@ -520,8 +573,10 @@ static unsigned long sdhci_arasan_sdcardclk_recalc_rate(struct clk_hw *hw,
|
|||
unsigned long parent_rate)
|
||||
|
||||
{
|
||||
struct sdhci_arasan_clk_data *clk_data =
|
||||
container_of(hw, struct sdhci_arasan_clk_data, sdcardclk_hw);
|
||||
struct sdhci_arasan_data *sdhci_arasan =
|
||||
container_of(hw, struct sdhci_arasan_data, sdcardclk_hw);
|
||||
container_of(clk_data, struct sdhci_arasan_data, clk_data);
|
||||
struct sdhci_host *host = sdhci_arasan->host;
|
||||
|
||||
return host->mmc->actual_clock;
|
||||
|
@ -531,6 +586,177 @@ static const struct clk_ops arasan_sdcardclk_ops = {
|
|||
.recalc_rate = sdhci_arasan_sdcardclk_recalc_rate,
|
||||
};
|
||||
|
||||
/**
|
||||
* sdhci_arasan_sampleclk_recalc_rate - Return the sampling clock rate
|
||||
*
|
||||
* Return the current actual rate of the sampling clock. This can be used
|
||||
* to communicate with out PHY.
|
||||
*
|
||||
* @hw: Pointer to the hardware clock structure.
|
||||
* @parent_rate The parent rate (should be rate of clk_xin).
|
||||
* Returns the sample clock rate.
|
||||
*/
|
||||
static unsigned long sdhci_arasan_sampleclk_recalc_rate(struct clk_hw *hw,
|
||||
unsigned long parent_rate)
|
||||
|
||||
{
|
||||
struct sdhci_arasan_clk_data *clk_data =
|
||||
container_of(hw, struct sdhci_arasan_clk_data, sampleclk_hw);
|
||||
struct sdhci_arasan_data *sdhci_arasan =
|
||||
container_of(clk_data, struct sdhci_arasan_data, clk_data);
|
||||
struct sdhci_host *host = sdhci_arasan->host;
|
||||
|
||||
return host->mmc->actual_clock;
|
||||
}
|
||||
|
||||
static const struct clk_ops arasan_sampleclk_ops = {
|
||||
.recalc_rate = sdhci_arasan_sampleclk_recalc_rate,
|
||||
};
|
||||
|
||||
/**
|
||||
* sdhci_zynqmp_sdcardclk_set_phase - Set the SD Output Clock Tap Delays
|
||||
*
|
||||
* Set the SD Output Clock Tap Delays for Output path
|
||||
*
|
||||
* @hw: Pointer to the hardware clock structure.
|
||||
* @degrees The clock phase shift between 0 - 359.
|
||||
* Return: 0 on success and error value on error
|
||||
*/
|
||||
static int sdhci_zynqmp_sdcardclk_set_phase(struct clk_hw *hw, int degrees)
|
||||
|
||||
{
|
||||
struct sdhci_arasan_clk_data *clk_data =
|
||||
container_of(hw, struct sdhci_arasan_clk_data, sdcardclk_hw);
|
||||
struct sdhci_arasan_data *sdhci_arasan =
|
||||
container_of(clk_data, struct sdhci_arasan_data, clk_data);
|
||||
struct sdhci_host *host = sdhci_arasan->host;
|
||||
struct sdhci_arasan_zynqmp_clk_data *zynqmp_clk_data =
|
||||
clk_data->clk_of_data;
|
||||
const struct zynqmp_eemi_ops *eemi_ops = zynqmp_clk_data->eemi_ops;
|
||||
const char *clk_name = clk_hw_get_name(hw);
|
||||
u32 node_id = !strcmp(clk_name, "clk_out_sd0") ? NODE_SD_0 : NODE_SD_1;
|
||||
u8 tap_delay, tap_max = 0;
|
||||
int ret;
|
||||
|
||||
/*
|
||||
* This is applicable for SDHCI_SPEC_300 and above
|
||||
* ZynqMP does not set phase for <=25MHz clock.
|
||||
* If degrees is zero, no need to do anything.
|
||||
*/
|
||||
if (host->version < SDHCI_SPEC_300 ||
|
||||
host->timing == MMC_TIMING_LEGACY ||
|
||||
host->timing == MMC_TIMING_UHS_SDR12 || !degrees)
|
||||
return 0;
|
||||
|
||||
switch (host->timing) {
|
||||
case MMC_TIMING_MMC_HS:
|
||||
case MMC_TIMING_SD_HS:
|
||||
case MMC_TIMING_UHS_SDR25:
|
||||
case MMC_TIMING_UHS_DDR50:
|
||||
case MMC_TIMING_MMC_DDR52:
|
||||
/* For 50MHz clock, 30 Taps are available */
|
||||
tap_max = 30;
|
||||
break;
|
||||
case MMC_TIMING_UHS_SDR50:
|
||||
/* For 100MHz clock, 15 Taps are available */
|
||||
tap_max = 15;
|
||||
break;
|
||||
case MMC_TIMING_UHS_SDR104:
|
||||
case MMC_TIMING_MMC_HS200:
|
||||
/* For 200MHz clock, 8 Taps are available */
|
||||
tap_max = 8;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
|
||||
tap_delay = (degrees * tap_max) / 360;
|
||||
|
||||
/* Set the Clock Phase */
|
||||
ret = eemi_ops->ioctl(node_id, IOCTL_SET_SD_TAPDELAY,
|
||||
PM_TAPDELAY_OUTPUT, tap_delay, NULL);
|
||||
if (ret)
|
||||
pr_err("Error setting Output Tap Delay\n");
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static const struct clk_ops zynqmp_sdcardclk_ops = {
|
||||
.recalc_rate = sdhci_arasan_sdcardclk_recalc_rate,
|
||||
.set_phase = sdhci_zynqmp_sdcardclk_set_phase,
|
||||
};
|
||||
|
||||
/**
|
||||
* sdhci_zynqmp_sampleclk_set_phase - Set the SD Input Clock Tap Delays
|
||||
*
|
||||
* Set the SD Input Clock Tap Delays for Input path
|
||||
*
|
||||
* @hw: Pointer to the hardware clock structure.
|
||||
* @degrees The clock phase shift between 0 - 359.
|
||||
* Return: 0 on success and error value on error
|
||||
*/
|
||||
static int sdhci_zynqmp_sampleclk_set_phase(struct clk_hw *hw, int degrees)
|
||||
|
||||
{
|
||||
struct sdhci_arasan_clk_data *clk_data =
|
||||
container_of(hw, struct sdhci_arasan_clk_data, sampleclk_hw);
|
||||
struct sdhci_arasan_data *sdhci_arasan =
|
||||
container_of(clk_data, struct sdhci_arasan_data, clk_data);
|
||||
struct sdhci_host *host = sdhci_arasan->host;
|
||||
struct sdhci_arasan_zynqmp_clk_data *zynqmp_clk_data =
|
||||
clk_data->clk_of_data;
|
||||
const struct zynqmp_eemi_ops *eemi_ops = zynqmp_clk_data->eemi_ops;
|
||||
const char *clk_name = clk_hw_get_name(hw);
|
||||
u32 node_id = !strcmp(clk_name, "clk_in_sd0") ? NODE_SD_0 : NODE_SD_1;
|
||||
u8 tap_delay, tap_max = 0;
|
||||
int ret;
|
||||
|
||||
/*
|
||||
* This is applicable for SDHCI_SPEC_300 and above
|
||||
* ZynqMP does not set phase for <=25MHz clock.
|
||||
* If degrees is zero, no need to do anything.
|
||||
*/
|
||||
if (host->version < SDHCI_SPEC_300 ||
|
||||
host->timing == MMC_TIMING_LEGACY ||
|
||||
host->timing == MMC_TIMING_UHS_SDR12 || !degrees)
|
||||
return 0;
|
||||
|
||||
switch (host->timing) {
|
||||
case MMC_TIMING_MMC_HS:
|
||||
case MMC_TIMING_SD_HS:
|
||||
case MMC_TIMING_UHS_SDR25:
|
||||
case MMC_TIMING_UHS_DDR50:
|
||||
case MMC_TIMING_MMC_DDR52:
|
||||
/* For 50MHz clock, 120 Taps are available */
|
||||
tap_max = 120;
|
||||
break;
|
||||
case MMC_TIMING_UHS_SDR50:
|
||||
/* For 100MHz clock, 60 Taps are available */
|
||||
tap_max = 60;
|
||||
break;
|
||||
case MMC_TIMING_UHS_SDR104:
|
||||
case MMC_TIMING_MMC_HS200:
|
||||
/* For 200MHz clock, 30 Taps are available */
|
||||
tap_max = 30;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
|
||||
tap_delay = (degrees * tap_max) / 360;
|
||||
|
||||
/* Set the Clock Phase */
|
||||
ret = eemi_ops->ioctl(node_id, IOCTL_SET_SD_TAPDELAY,
|
||||
PM_TAPDELAY_INPUT, tap_delay, NULL);
|
||||
if (ret)
|
||||
pr_err("Error setting Input Tap Delay\n");
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static const struct clk_ops zynqmp_sampleclk_ops = {
|
||||
.recalc_rate = sdhci_arasan_sampleclk_recalc_rate,
|
||||
.set_phase = sdhci_zynqmp_sampleclk_set_phase,
|
||||
};
|
||||
|
||||
/**
|
||||
* sdhci_arasan_update_clockmultiplier - Set corecfg_clockmultiplier
|
||||
*
|
||||
|
@ -609,8 +835,229 @@ static void sdhci_arasan_update_baseclkfreq(struct sdhci_host *host)
|
|||
sdhci_arasan_syscon_write(host, &soc_ctl_map->baseclkfreq, mhz);
|
||||
}
|
||||
|
||||
static void sdhci_arasan_set_clk_delays(struct sdhci_host *host)
|
||||
{
|
||||
struct sdhci_pltfm_host *pltfm_host = sdhci_priv(host);
|
||||
struct sdhci_arasan_data *sdhci_arasan = sdhci_pltfm_priv(pltfm_host);
|
||||
struct sdhci_arasan_clk_data *clk_data = &sdhci_arasan->clk_data;
|
||||
|
||||
clk_set_phase(clk_data->sampleclk,
|
||||
clk_data->clk_phase_in[host->timing]);
|
||||
clk_set_phase(clk_data->sdcardclk,
|
||||
clk_data->clk_phase_out[host->timing]);
|
||||
}
|
||||
|
||||
static void arasan_dt_read_clk_phase(struct device *dev,
|
||||
struct sdhci_arasan_clk_data *clk_data,
|
||||
unsigned int timing, const char *prop)
|
||||
{
|
||||
struct device_node *np = dev->of_node;
|
||||
|
||||
int clk_phase[2] = {0};
|
||||
|
||||
/*
|
||||
* Read Tap Delay values from DT, if the DT does not contain the
|
||||
* Tap Values then use the pre-defined values.
|
||||
*/
|
||||
if (of_property_read_variable_u32_array(np, prop, &clk_phase[0],
|
||||
2, 0)) {
|
||||
dev_dbg(dev, "Using predefined clock phase for %s = %d %d\n",
|
||||
prop, clk_data->clk_phase_in[timing],
|
||||
clk_data->clk_phase_out[timing]);
|
||||
return;
|
||||
}
|
||||
|
||||
/* The values read are Input and Output Clock Delays in order */
|
||||
clk_data->clk_phase_in[timing] = clk_phase[0];
|
||||
clk_data->clk_phase_out[timing] = clk_phase[1];
|
||||
}
|
||||
|
||||
/**
|
||||
* sdhci_arasan_register_sdclk - Register the sdclk for a PHY to use
|
||||
* arasan_dt_parse_clk_phases - Read Clock Delay values from DT
|
||||
*
|
||||
* Called at initialization to parse the values of Clock Delays.
|
||||
*
|
||||
* @dev: Pointer to our struct device.
|
||||
* @clk_data: Pointer to the Clock Data structure
|
||||
*/
|
||||
static void arasan_dt_parse_clk_phases(struct device *dev,
|
||||
struct sdhci_arasan_clk_data *clk_data)
|
||||
{
|
||||
int *iclk_phase, *oclk_phase;
|
||||
u32 mio_bank = 0;
|
||||
int i;
|
||||
|
||||
/*
|
||||
* This has been kept as a pointer and is assigned a function here.
|
||||
* So that different controller variants can assign their own handling
|
||||
* function.
|
||||
*/
|
||||
clk_data->set_clk_delays = sdhci_arasan_set_clk_delays;
|
||||
|
||||
if (of_device_is_compatible(dev->of_node, "xlnx,zynqmp-8.9a")) {
|
||||
iclk_phase = (int [MMC_TIMING_MMC_HS400 + 1]) ZYNQMP_ICLK_PHASE;
|
||||
oclk_phase = (int [MMC_TIMING_MMC_HS400 + 1]) ZYNQMP_OCLK_PHASE;
|
||||
|
||||
of_property_read_u32(dev->of_node, "xlnx,mio-bank", &mio_bank);
|
||||
if (mio_bank == 2) {
|
||||
oclk_phase[MMC_TIMING_UHS_SDR104] = 90;
|
||||
oclk_phase[MMC_TIMING_MMC_HS200] = 90;
|
||||
}
|
||||
|
||||
for (i = 0; i <= MMC_TIMING_MMC_HS400; i++) {
|
||||
clk_data->clk_phase_in[i] = iclk_phase[i];
|
||||
clk_data->clk_phase_out[i] = oclk_phase[i];
|
||||
}
|
||||
}
|
||||
|
||||
arasan_dt_read_clk_phase(dev, clk_data, MMC_TIMING_LEGACY,
|
||||
"clk-phase-legacy");
|
||||
arasan_dt_read_clk_phase(dev, clk_data, MMC_TIMING_MMC_HS,
|
||||
"clk-phase-mmc-hs");
|
||||
arasan_dt_read_clk_phase(dev, clk_data, MMC_TIMING_SD_HS,
|
||||
"clk-phase-sd-hs");
|
||||
arasan_dt_read_clk_phase(dev, clk_data, MMC_TIMING_UHS_SDR12,
|
||||
"clk-phase-uhs-sdr12");
|
||||
arasan_dt_read_clk_phase(dev, clk_data, MMC_TIMING_UHS_SDR25,
|
||||
"clk-phase-uhs-sdr25");
|
||||
arasan_dt_read_clk_phase(dev, clk_data, MMC_TIMING_UHS_SDR50,
|
||||
"clk-phase-uhs-sdr50");
|
||||
arasan_dt_read_clk_phase(dev, clk_data, MMC_TIMING_UHS_SDR104,
|
||||
"clk-phase-uhs-sdr104");
|
||||
arasan_dt_read_clk_phase(dev, clk_data, MMC_TIMING_UHS_DDR50,
|
||||
"clk-phase-uhs-ddr50");
|
||||
arasan_dt_read_clk_phase(dev, clk_data, MMC_TIMING_MMC_DDR52,
|
||||
"clk-phase-mmc-ddr52");
|
||||
arasan_dt_read_clk_phase(dev, clk_data, MMC_TIMING_MMC_HS200,
|
||||
"clk-phase-mmc-hs200");
|
||||
arasan_dt_read_clk_phase(dev, clk_data, MMC_TIMING_MMC_HS400,
|
||||
"clk-phase-mmc-hs400");
|
||||
}
|
||||
|
||||
/**
|
||||
* sdhci_arasan_register_sdcardclk - Register the sdcardclk for a PHY to use
|
||||
*
|
||||
* Some PHY devices need to know what the actual card clock is. In order for
|
||||
* them to find out, we'll provide a clock through the common clock framework
|
||||
* for them to query.
|
||||
*
|
||||
* @sdhci_arasan: Our private data structure.
|
||||
* @clk_xin: Pointer to the functional clock
|
||||
* @dev: Pointer to our struct device.
|
||||
* Returns 0 on success and error value on error
|
||||
*/
|
||||
static int
|
||||
sdhci_arasan_register_sdcardclk(struct sdhci_arasan_data *sdhci_arasan,
|
||||
struct clk *clk_xin,
|
||||
struct device *dev)
|
||||
{
|
||||
struct sdhci_arasan_clk_data *clk_data = &sdhci_arasan->clk_data;
|
||||
struct device_node *np = dev->of_node;
|
||||
struct clk_init_data sdcardclk_init;
|
||||
const char *parent_clk_name;
|
||||
int ret;
|
||||
|
||||
ret = of_property_read_string_index(np, "clock-output-names", 0,
|
||||
&sdcardclk_init.name);
|
||||
if (ret) {
|
||||
dev_err(dev, "DT has #clock-cells but no clock-output-names\n");
|
||||
return ret;
|
||||
}
|
||||
|
||||
parent_clk_name = __clk_get_name(clk_xin);
|
||||
sdcardclk_init.parent_names = &parent_clk_name;
|
||||
sdcardclk_init.num_parents = 1;
|
||||
sdcardclk_init.flags = CLK_GET_RATE_NOCACHE;
|
||||
if (of_device_is_compatible(np, "xlnx,zynqmp-8.9a"))
|
||||
sdcardclk_init.ops = &zynqmp_sdcardclk_ops;
|
||||
else
|
||||
sdcardclk_init.ops = &arasan_sdcardclk_ops;
|
||||
|
||||
clk_data->sdcardclk_hw.init = &sdcardclk_init;
|
||||
clk_data->sdcardclk =
|
||||
devm_clk_register(dev, &clk_data->sdcardclk_hw);
|
||||
clk_data->sdcardclk_hw.init = NULL;
|
||||
|
||||
ret = of_clk_add_provider(np, of_clk_src_simple_get,
|
||||
clk_data->sdcardclk);
|
||||
if (ret)
|
||||
dev_err(dev, "Failed to add sdcard clock provider\n");
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
/**
|
||||
* sdhci_arasan_register_sampleclk - Register the sampleclk for a PHY to use
|
||||
*
|
||||
* Some PHY devices need to know what the actual card clock is. In order for
|
||||
* them to find out, we'll provide a clock through the common clock framework
|
||||
* for them to query.
|
||||
*
|
||||
* @sdhci_arasan: Our private data structure.
|
||||
* @clk_xin: Pointer to the functional clock
|
||||
* @dev: Pointer to our struct device.
|
||||
* Returns 0 on success and error value on error
|
||||
*/
|
||||
static int
|
||||
sdhci_arasan_register_sampleclk(struct sdhci_arasan_data *sdhci_arasan,
|
||||
struct clk *clk_xin,
|
||||
struct device *dev)
|
||||
{
|
||||
struct sdhci_arasan_clk_data *clk_data = &sdhci_arasan->clk_data;
|
||||
struct device_node *np = dev->of_node;
|
||||
struct clk_init_data sampleclk_init;
|
||||
const char *parent_clk_name;
|
||||
int ret;
|
||||
|
||||
ret = of_property_read_string_index(np, "clock-output-names", 1,
|
||||
&sampleclk_init.name);
|
||||
if (ret) {
|
||||
dev_err(dev, "DT has #clock-cells but no clock-output-names\n");
|
||||
return ret;
|
||||
}
|
||||
|
||||
parent_clk_name = __clk_get_name(clk_xin);
|
||||
sampleclk_init.parent_names = &parent_clk_name;
|
||||
sampleclk_init.num_parents = 1;
|
||||
sampleclk_init.flags = CLK_GET_RATE_NOCACHE;
|
||||
if (of_device_is_compatible(np, "xlnx,zynqmp-8.9a"))
|
||||
sampleclk_init.ops = &zynqmp_sampleclk_ops;
|
||||
else
|
||||
sampleclk_init.ops = &arasan_sampleclk_ops;
|
||||
|
||||
clk_data->sampleclk_hw.init = &sampleclk_init;
|
||||
clk_data->sampleclk =
|
||||
devm_clk_register(dev, &clk_data->sampleclk_hw);
|
||||
clk_data->sampleclk_hw.init = NULL;
|
||||
|
||||
ret = of_clk_add_provider(np, of_clk_src_simple_get,
|
||||
clk_data->sampleclk);
|
||||
if (ret)
|
||||
dev_err(dev, "Failed to add sample clock provider\n");
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
/**
|
||||
* sdhci_arasan_unregister_sdclk - Undoes sdhci_arasan_register_sdclk()
|
||||
*
|
||||
* Should be called any time we're exiting and sdhci_arasan_register_sdclk()
|
||||
* returned success.
|
||||
*
|
||||
* @dev: Pointer to our struct device.
|
||||
*/
|
||||
static void sdhci_arasan_unregister_sdclk(struct device *dev)
|
||||
{
|
||||
struct device_node *np = dev->of_node;
|
||||
|
||||
if (!of_find_property(np, "#clock-cells", NULL))
|
||||
return;
|
||||
|
||||
of_clk_del_provider(dev->of_node);
|
||||
}
|
||||
|
||||
/**
|
||||
* sdhci_arasan_register_sdclk - Register the sdcardclk for a PHY to use
|
||||
*
|
||||
* Some PHY devices need to know what the actual card clock is. In order for
|
||||
* them to find out, we'll provide a clock through the common clock framework
|
||||
|
@ -634,56 +1081,27 @@ static int sdhci_arasan_register_sdclk(struct sdhci_arasan_data *sdhci_arasan,
|
|||
struct device *dev)
|
||||
{
|
||||
struct device_node *np = dev->of_node;
|
||||
struct clk_init_data sdcardclk_init;
|
||||
const char *parent_clk_name;
|
||||
u32 num_clks = 0;
|
||||
int ret;
|
||||
|
||||
/* Providing a clock to the PHY is optional; no error if missing */
|
||||
if (!of_find_property(np, "#clock-cells", NULL))
|
||||
if (of_property_read_u32(np, "#clock-cells", &num_clks) < 0)
|
||||
return 0;
|
||||
|
||||
ret = of_property_read_string_index(np, "clock-output-names", 0,
|
||||
&sdcardclk_init.name);
|
||||
if (ret) {
|
||||
dev_err(dev, "DT has #clock-cells but no clock-output-names\n");
|
||||
ret = sdhci_arasan_register_sdcardclk(sdhci_arasan, clk_xin, dev);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
if (num_clks) {
|
||||
ret = sdhci_arasan_register_sampleclk(sdhci_arasan, clk_xin,
|
||||
dev);
|
||||
if (ret) {
|
||||
sdhci_arasan_unregister_sdclk(dev);
|
||||
return ret;
|
||||
}
|
||||
}
|
||||
|
||||
parent_clk_name = __clk_get_name(clk_xin);
|
||||
sdcardclk_init.parent_names = &parent_clk_name;
|
||||
sdcardclk_init.num_parents = 1;
|
||||
sdcardclk_init.flags = CLK_GET_RATE_NOCACHE;
|
||||
sdcardclk_init.ops = &arasan_sdcardclk_ops;
|
||||
|
||||
sdhci_arasan->sdcardclk_hw.init = &sdcardclk_init;
|
||||
sdhci_arasan->sdcardclk =
|
||||
devm_clk_register(dev, &sdhci_arasan->sdcardclk_hw);
|
||||
sdhci_arasan->sdcardclk_hw.init = NULL;
|
||||
|
||||
ret = of_clk_add_provider(np, of_clk_src_simple_get,
|
||||
sdhci_arasan->sdcardclk);
|
||||
if (ret)
|
||||
dev_err(dev, "Failed to add clock provider\n");
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
/**
|
||||
* sdhci_arasan_unregister_sdclk - Undoes sdhci_arasan_register_sdclk()
|
||||
*
|
||||
* Should be called any time we're exiting and sdhci_arasan_register_sdclk()
|
||||
* returned success.
|
||||
*
|
||||
* @dev: Pointer to our struct device.
|
||||
*/
|
||||
static void sdhci_arasan_unregister_sdclk(struct device *dev)
|
||||
{
|
||||
struct device_node *np = dev->of_node;
|
||||
|
||||
if (!of_find_property(np, "#clock-cells", NULL))
|
||||
return;
|
||||
|
||||
of_clk_del_provider(dev->of_node);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int sdhci_arasan_add_host(struct sdhci_arasan_data *sdhci_arasan)
|
||||
|
@ -814,6 +1232,25 @@ static int sdhci_arasan_probe(struct platform_device *pdev)
|
|||
if (ret)
|
||||
goto clk_disable_all;
|
||||
|
||||
if (of_device_is_compatible(np, "xlnx,zynqmp-8.9a")) {
|
||||
struct sdhci_arasan_zynqmp_clk_data *zynqmp_clk_data;
|
||||
const struct zynqmp_eemi_ops *eemi_ops;
|
||||
|
||||
zynqmp_clk_data = devm_kzalloc(&pdev->dev,
|
||||
sizeof(*zynqmp_clk_data),
|
||||
GFP_KERNEL);
|
||||
eemi_ops = zynqmp_pm_get_eemi_ops();
|
||||
if (IS_ERR(eemi_ops)) {
|
||||
ret = PTR_ERR(eemi_ops);
|
||||
goto unreg_clk;
|
||||
}
|
||||
|
||||
zynqmp_clk_data->eemi_ops = eemi_ops;
|
||||
sdhci_arasan->clk_data.clk_of_data = zynqmp_clk_data;
|
||||
}
|
||||
|
||||
arasan_dt_parse_clk_phases(&pdev->dev, &sdhci_arasan->clk_data);
|
||||
|
||||
ret = mmc_of_parse(host->mmc);
|
||||
if (ret) {
|
||||
if (ret != -EPROBE_DEFER)
|
||||
|
|
|
@ -111,7 +111,19 @@ static void aspeed_sdhci_set_bus_width(struct sdhci_host *host, int width)
|
|||
sdhci_writeb(host, ctrl, SDHCI_HOST_CONTROL);
|
||||
}
|
||||
|
||||
static u32 aspeed_sdhci_readl(struct sdhci_host *host, int reg)
|
||||
{
|
||||
u32 val = readl(host->ioaddr + reg);
|
||||
|
||||
if (unlikely(reg == SDHCI_PRESENT_STATE) &&
|
||||
(host->mmc->caps2 & MMC_CAP2_CD_ACTIVE_HIGH))
|
||||
val ^= SDHCI_CARD_PRESENT;
|
||||
|
||||
return val;
|
||||
}
|
||||
|
||||
static const struct sdhci_ops aspeed_sdhci_ops = {
|
||||
.read_l = aspeed_sdhci_readl,
|
||||
.set_clock = aspeed_sdhci_set_clock,
|
||||
.get_max_clock = aspeed_sdhci_get_max_clock,
|
||||
.set_bus_width = aspeed_sdhci_set_bus_width,
|
||||
|
|
|
@ -27,6 +27,9 @@
|
|||
#define SDMMC_CACR 0x230
|
||||
#define SDMMC_CACR_CAPWREN BIT(0)
|
||||
#define SDMMC_CACR_KEY (0x46 << 8)
|
||||
#define SDMMC_CALCR 0x240
|
||||
#define SDMMC_CALCR_EN BIT(0)
|
||||
#define SDMMC_CALCR_ALWYSON BIT(4)
|
||||
|
||||
#define SDHCI_AT91_PRESET_COMMON_CONF 0x400 /* drv type B, programmable clock mode */
|
||||
|
||||
|
@ -35,6 +38,7 @@ struct sdhci_at91_priv {
|
|||
struct clk *gck;
|
||||
struct clk *mainck;
|
||||
bool restore_needed;
|
||||
bool cal_always_on;
|
||||
};
|
||||
|
||||
static void sdhci_at91_set_force_card_detect(struct sdhci_host *host)
|
||||
|
@ -116,10 +120,17 @@ static void sdhci_at91_set_uhs_signaling(struct sdhci_host *host,
|
|||
|
||||
static void sdhci_at91_reset(struct sdhci_host *host, u8 mask)
|
||||
{
|
||||
struct sdhci_pltfm_host *pltfm_host = sdhci_priv(host);
|
||||
struct sdhci_at91_priv *priv = sdhci_pltfm_priv(pltfm_host);
|
||||
|
||||
sdhci_reset(host, mask);
|
||||
|
||||
if (host->mmc->caps & MMC_CAP_NONREMOVABLE)
|
||||
sdhci_at91_set_force_card_detect(host);
|
||||
|
||||
if (priv->cal_always_on && (mask & SDHCI_RESET_ALL))
|
||||
sdhci_writel(host, SDMMC_CALCR_ALWYSON | SDMMC_CALCR_EN,
|
||||
SDMMC_CALCR);
|
||||
}
|
||||
|
||||
static const struct sdhci_ops sdhci_at91_sama5d2_ops = {
|
||||
|
@ -345,6 +356,14 @@ static int sdhci_at91_probe(struct platform_device *pdev)
|
|||
|
||||
priv->restore_needed = false;
|
||||
|
||||
/*
|
||||
* if SDCAL pin is wrongly connected, we must enable
|
||||
* the analog calibration cell permanently.
|
||||
*/
|
||||
priv->cal_always_on =
|
||||
device_property_read_bool(&pdev->dev,
|
||||
"microchip,sdcal-inverted");
|
||||
|
||||
ret = mmc_of_parse(host->mmc);
|
||||
if (ret)
|
||||
goto clocks_disable_unprepare;
|
||||
|
|
|
@ -77,8 +77,10 @@ struct sdhci_esdhc {
|
|||
bool quirk_incorrect_hostver;
|
||||
bool quirk_limited_clk_division;
|
||||
bool quirk_unreliable_pulse_detection;
|
||||
bool quirk_fixup_tuning;
|
||||
bool quirk_tuning_erratum_type1;
|
||||
bool quirk_tuning_erratum_type2;
|
||||
bool quirk_ignore_data_inhibit;
|
||||
bool in_sw_tuning;
|
||||
unsigned int peripheral_clock;
|
||||
const struct esdhc_clk_fixup *clk_fixup;
|
||||
u32 div_ratio;
|
||||
|
@ -408,6 +410,8 @@ static void esdhc_le_writel(struct sdhci_host *host, u32 val, int reg)
|
|||
|
||||
static void esdhc_be_writew(struct sdhci_host *host, u16 val, int reg)
|
||||
{
|
||||
struct sdhci_pltfm_host *pltfm_host = sdhci_priv(host);
|
||||
struct sdhci_esdhc *esdhc = sdhci_pltfm_priv(pltfm_host);
|
||||
int base = reg & ~0x3;
|
||||
u32 value;
|
||||
u32 ret;
|
||||
|
@ -416,10 +420,24 @@ static void esdhc_be_writew(struct sdhci_host *host, u16 val, int reg)
|
|||
ret = esdhc_writew_fixup(host, reg, val, value);
|
||||
if (reg != SDHCI_TRANSFER_MODE)
|
||||
iowrite32be(ret, host->ioaddr + base);
|
||||
|
||||
/* Starting SW tuning requires ESDHC_SMPCLKSEL to be set
|
||||
* 1us later after ESDHC_EXTN is set.
|
||||
*/
|
||||
if (base == ESDHC_SYSTEM_CONTROL_2) {
|
||||
if (!(value & ESDHC_EXTN) && (ret & ESDHC_EXTN) &&
|
||||
esdhc->in_sw_tuning) {
|
||||
udelay(1);
|
||||
ret |= ESDHC_SMPCLKSEL;
|
||||
iowrite32be(ret, host->ioaddr + base);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static void esdhc_le_writew(struct sdhci_host *host, u16 val, int reg)
|
||||
{
|
||||
struct sdhci_pltfm_host *pltfm_host = sdhci_priv(host);
|
||||
struct sdhci_esdhc *esdhc = sdhci_pltfm_priv(pltfm_host);
|
||||
int base = reg & ~0x3;
|
||||
u32 value;
|
||||
u32 ret;
|
||||
|
@ -428,6 +446,18 @@ static void esdhc_le_writew(struct sdhci_host *host, u16 val, int reg)
|
|||
ret = esdhc_writew_fixup(host, reg, val, value);
|
||||
if (reg != SDHCI_TRANSFER_MODE)
|
||||
iowrite32(ret, host->ioaddr + base);
|
||||
|
||||
/* Starting SW tuning requires ESDHC_SMPCLKSEL to be set
|
||||
* 1us later after ESDHC_EXTN is set.
|
||||
*/
|
||||
if (base == ESDHC_SYSTEM_CONTROL_2) {
|
||||
if (!(value & ESDHC_EXTN) && (ret & ESDHC_EXTN) &&
|
||||
esdhc->in_sw_tuning) {
|
||||
udelay(1);
|
||||
ret |= ESDHC_SMPCLKSEL;
|
||||
iowrite32(ret, host->ioaddr + base);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static void esdhc_be_writeb(struct sdhci_host *host, u8 val, int reg)
|
||||
|
@ -560,6 +590,32 @@ static void esdhc_clock_enable(struct sdhci_host *host, bool enable)
|
|||
}
|
||||
}
|
||||
|
||||
static void esdhc_flush_async_fifo(struct sdhci_host *host)
|
||||
{
|
||||
ktime_t timeout;
|
||||
u32 val;
|
||||
|
||||
val = sdhci_readl(host, ESDHC_DMA_SYSCTL);
|
||||
val |= ESDHC_FLUSH_ASYNC_FIFO;
|
||||
sdhci_writel(host, val, ESDHC_DMA_SYSCTL);
|
||||
|
||||
/* Wait max 20 ms */
|
||||
timeout = ktime_add_ms(ktime_get(), 20);
|
||||
while (1) {
|
||||
bool timedout = ktime_after(ktime_get(), timeout);
|
||||
|
||||
if (!(sdhci_readl(host, ESDHC_DMA_SYSCTL) &
|
||||
ESDHC_FLUSH_ASYNC_FIFO))
|
||||
break;
|
||||
if (timedout) {
|
||||
pr_err("%s: flushing asynchronous FIFO timeout.\n",
|
||||
mmc_hostname(host->mmc));
|
||||
break;
|
||||
}
|
||||
usleep_range(10, 20);
|
||||
}
|
||||
}
|
||||
|
||||
static void esdhc_of_set_clock(struct sdhci_host *host, unsigned int clock)
|
||||
{
|
||||
struct sdhci_pltfm_host *pltfm_host = sdhci_priv(host);
|
||||
|
@ -652,9 +708,7 @@ static void esdhc_of_set_clock(struct sdhci_host *host, unsigned int clock)
|
|||
sdhci_writel(host, temp | ESDHC_HS400_WNDW_ADJUST, ESDHC_TBCTL);
|
||||
|
||||
esdhc_clock_enable(host, false);
|
||||
temp = sdhci_readl(host, ESDHC_DMA_SYSCTL);
|
||||
temp |= ESDHC_FLUSH_ASYNC_FIFO;
|
||||
sdhci_writel(host, temp, ESDHC_DMA_SYSCTL);
|
||||
esdhc_flush_async_fifo(host);
|
||||
}
|
||||
|
||||
/* Wait max 20 ms */
|
||||
|
@ -796,16 +850,21 @@ static int esdhc_signal_voltage_switch(struct mmc_host *mmc,
|
|||
}
|
||||
}
|
||||
|
||||
static struct soc_device_attribute soc_fixup_tuning[] = {
|
||||
static struct soc_device_attribute soc_tuning_erratum_type1[] = {
|
||||
{ .family = "QorIQ T1023", .revision = "1.0", },
|
||||
{ .family = "QorIQ T1040", .revision = "1.0", },
|
||||
{ .family = "QorIQ T2080", .revision = "1.0", },
|
||||
{ .family = "QorIQ T1023", .revision = "1.0", },
|
||||
{ .family = "QorIQ LS1021A", .revision = "1.0", },
|
||||
{ .family = "QorIQ LS1080A", .revision = "1.0", },
|
||||
{ .family = "QorIQ LS2080A", .revision = "1.0", },
|
||||
{ },
|
||||
};
|
||||
|
||||
static struct soc_device_attribute soc_tuning_erratum_type2[] = {
|
||||
{ .family = "QorIQ LS1012A", .revision = "1.0", },
|
||||
{ .family = "QorIQ LS1043A", .revision = "1.*", },
|
||||
{ .family = "QorIQ LS1046A", .revision = "1.0", },
|
||||
{ .family = "QorIQ LS1080A", .revision = "1.0", },
|
||||
{ .family = "QorIQ LS2080A", .revision = "1.0", },
|
||||
{ .family = "QorIQ LA1575A", .revision = "1.0", },
|
||||
{ },
|
||||
};
|
||||
|
||||
|
@ -814,10 +873,7 @@ static void esdhc_tuning_block_enable(struct sdhci_host *host, bool enable)
|
|||
u32 val;
|
||||
|
||||
esdhc_clock_enable(host, false);
|
||||
|
||||
val = sdhci_readl(host, ESDHC_DMA_SYSCTL);
|
||||
val |= ESDHC_FLUSH_ASYNC_FIFO;
|
||||
sdhci_writel(host, val, ESDHC_DMA_SYSCTL);
|
||||
esdhc_flush_async_fifo(host);
|
||||
|
||||
val = sdhci_readl(host, ESDHC_TBCTL);
|
||||
if (enable)
|
||||
|
@ -829,15 +885,97 @@ static void esdhc_tuning_block_enable(struct sdhci_host *host, bool enable)
|
|||
esdhc_clock_enable(host, true);
|
||||
}
|
||||
|
||||
static void esdhc_prepare_sw_tuning(struct sdhci_host *host, u8 *window_start,
|
||||
u8 *window_end)
|
||||
{
|
||||
struct sdhci_pltfm_host *pltfm_host = sdhci_priv(host);
|
||||
struct sdhci_esdhc *esdhc = sdhci_pltfm_priv(pltfm_host);
|
||||
u8 tbstat_15_8, tbstat_7_0;
|
||||
u32 val;
|
||||
|
||||
if (esdhc->quirk_tuning_erratum_type1) {
|
||||
*window_start = 5 * esdhc->div_ratio;
|
||||
*window_end = 3 * esdhc->div_ratio;
|
||||
return;
|
||||
}
|
||||
|
||||
/* Write TBCTL[11:8]=4'h8 */
|
||||
val = sdhci_readl(host, ESDHC_TBCTL);
|
||||
val &= ~(0xf << 8);
|
||||
val |= 8 << 8;
|
||||
sdhci_writel(host, val, ESDHC_TBCTL);
|
||||
|
||||
mdelay(1);
|
||||
|
||||
/* Read TBCTL[31:0] register and rewrite again */
|
||||
val = sdhci_readl(host, ESDHC_TBCTL);
|
||||
sdhci_writel(host, val, ESDHC_TBCTL);
|
||||
|
||||
mdelay(1);
|
||||
|
||||
/* Read the TBSTAT[31:0] register twice */
|
||||
val = sdhci_readl(host, ESDHC_TBSTAT);
|
||||
val = sdhci_readl(host, ESDHC_TBSTAT);
|
||||
|
||||
/* Reset data lines by setting ESDHCCTL[RSTD] */
|
||||
sdhci_reset(host, SDHCI_RESET_DATA);
|
||||
/* Write 32'hFFFF_FFFF to IRQSTAT register */
|
||||
sdhci_writel(host, 0xFFFFFFFF, SDHCI_INT_STATUS);
|
||||
|
||||
/* If TBSTAT[15:8]-TBSTAT[7:0] > 4 * div_ratio
|
||||
* or TBSTAT[7:0]-TBSTAT[15:8] > 4 * div_ratio,
|
||||
* then program TBPTR[TB_WNDW_END_PTR] = 4 * div_ratio
|
||||
* and program TBPTR[TB_WNDW_START_PTR] = 8 * div_ratio.
|
||||
*/
|
||||
tbstat_7_0 = val & 0xff;
|
||||
tbstat_15_8 = (val >> 8) & 0xff;
|
||||
|
||||
if (abs(tbstat_15_8 - tbstat_7_0) > (4 * esdhc->div_ratio)) {
|
||||
*window_start = 8 * esdhc->div_ratio;
|
||||
*window_end = 4 * esdhc->div_ratio;
|
||||
} else {
|
||||
*window_start = 5 * esdhc->div_ratio;
|
||||
*window_end = 3 * esdhc->div_ratio;
|
||||
}
|
||||
}
|
||||
|
||||
static int esdhc_execute_sw_tuning(struct mmc_host *mmc, u32 opcode,
|
||||
u8 window_start, u8 window_end)
|
||||
{
|
||||
struct sdhci_host *host = mmc_priv(mmc);
|
||||
struct sdhci_pltfm_host *pltfm_host = sdhci_priv(host);
|
||||
struct sdhci_esdhc *esdhc = sdhci_pltfm_priv(pltfm_host);
|
||||
u32 val;
|
||||
int ret;
|
||||
|
||||
/* Program TBPTR[TB_WNDW_END_PTR] and TBPTR[TB_WNDW_START_PTR] */
|
||||
val = ((u32)window_start << ESDHC_WNDW_STRT_PTR_SHIFT) &
|
||||
ESDHC_WNDW_STRT_PTR_MASK;
|
||||
val |= window_end & ESDHC_WNDW_END_PTR_MASK;
|
||||
sdhci_writel(host, val, ESDHC_TBPTR);
|
||||
|
||||
/* Program the software tuning mode by setting TBCTL[TB_MODE]=2'h3 */
|
||||
val = sdhci_readl(host, ESDHC_TBCTL);
|
||||
val &= ~ESDHC_TB_MODE_MASK;
|
||||
val |= ESDHC_TB_MODE_SW;
|
||||
sdhci_writel(host, val, ESDHC_TBCTL);
|
||||
|
||||
esdhc->in_sw_tuning = true;
|
||||
ret = sdhci_execute_tuning(mmc, opcode);
|
||||
esdhc->in_sw_tuning = false;
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int esdhc_execute_tuning(struct mmc_host *mmc, u32 opcode)
|
||||
{
|
||||
struct sdhci_host *host = mmc_priv(mmc);
|
||||
struct sdhci_pltfm_host *pltfm_host = sdhci_priv(host);
|
||||
struct sdhci_esdhc *esdhc = sdhci_pltfm_priv(pltfm_host);
|
||||
u8 window_start, window_end;
|
||||
int ret, retries = 1;
|
||||
bool hs400_tuning;
|
||||
unsigned int clk;
|
||||
u32 val;
|
||||
int ret;
|
||||
|
||||
/* For tuning mode, the sd clock divisor value
|
||||
* must be larger than 3 according to reference manual.
|
||||
|
@ -846,39 +984,73 @@ static int esdhc_execute_tuning(struct mmc_host *mmc, u32 opcode)
|
|||
if (host->clock > clk)
|
||||
esdhc_of_set_clock(host, clk);
|
||||
|
||||
if (esdhc->quirk_limited_clk_division &&
|
||||
host->flags & SDHCI_HS400_TUNING)
|
||||
esdhc_of_set_clock(host, host->clock);
|
||||
|
||||
esdhc_tuning_block_enable(host, true);
|
||||
|
||||
hs400_tuning = host->flags & SDHCI_HS400_TUNING;
|
||||
ret = sdhci_execute_tuning(mmc, opcode);
|
||||
|
||||
if (hs400_tuning) {
|
||||
do {
|
||||
if (esdhc->quirk_limited_clk_division &&
|
||||
hs400_tuning)
|
||||
esdhc_of_set_clock(host, host->clock);
|
||||
|
||||
/* Do HW tuning */
|
||||
val = sdhci_readl(host, ESDHC_TBCTL);
|
||||
val &= ~ESDHC_TB_MODE_MASK;
|
||||
val |= ESDHC_TB_MODE_3;
|
||||
sdhci_writel(host, val, ESDHC_TBCTL);
|
||||
|
||||
ret = sdhci_execute_tuning(mmc, opcode);
|
||||
if (ret)
|
||||
break;
|
||||
|
||||
/* If HW tuning fails and triggers erratum,
|
||||
* try workaround.
|
||||
*/
|
||||
ret = host->tuning_err;
|
||||
if (ret == -EAGAIN &&
|
||||
(esdhc->quirk_tuning_erratum_type1 ||
|
||||
esdhc->quirk_tuning_erratum_type2)) {
|
||||
/* Recover HS400 tuning flag */
|
||||
if (hs400_tuning)
|
||||
host->flags |= SDHCI_HS400_TUNING;
|
||||
pr_info("%s: Hold on to use fixed sampling clock. Try SW tuning!\n",
|
||||
mmc_hostname(mmc));
|
||||
/* Do SW tuning */
|
||||
esdhc_prepare_sw_tuning(host, &window_start,
|
||||
&window_end);
|
||||
ret = esdhc_execute_sw_tuning(mmc, opcode,
|
||||
window_start,
|
||||
window_end);
|
||||
if (ret)
|
||||
break;
|
||||
|
||||
/* Retry both HW/SW tuning with reduced clock. */
|
||||
ret = host->tuning_err;
|
||||
if (ret == -EAGAIN && retries) {
|
||||
/* Recover HS400 tuning flag */
|
||||
if (hs400_tuning)
|
||||
host->flags |= SDHCI_HS400_TUNING;
|
||||
|
||||
clk = host->max_clk / (esdhc->div_ratio + 1);
|
||||
esdhc_of_set_clock(host, clk);
|
||||
pr_info("%s: Hold on to use fixed sampling clock. Try tuning with reduced clock!\n",
|
||||
mmc_hostname(mmc));
|
||||
} else {
|
||||
break;
|
||||
}
|
||||
} else {
|
||||
break;
|
||||
}
|
||||
} while (retries--);
|
||||
|
||||
if (ret) {
|
||||
esdhc_tuning_block_enable(host, false);
|
||||
} else if (hs400_tuning) {
|
||||
val = sdhci_readl(host, ESDHC_SDTIMNGCTL);
|
||||
val |= ESDHC_FLW_CTL_BG;
|
||||
sdhci_writel(host, val, ESDHC_SDTIMNGCTL);
|
||||
}
|
||||
|
||||
if (host->tuning_err == -EAGAIN && esdhc->quirk_fixup_tuning) {
|
||||
|
||||
/* program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO and
|
||||
* program TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO
|
||||
*/
|
||||
val = sdhci_readl(host, ESDHC_TBPTR);
|
||||
val = (val & ~((0x7f << 8) | 0x7f)) |
|
||||
(3 * esdhc->div_ratio) | ((5 * esdhc->div_ratio) << 8);
|
||||
sdhci_writel(host, val, ESDHC_TBPTR);
|
||||
|
||||
/* program the software tuning mode by setting
|
||||
* TBCTL[TB_MODE]=2'h3
|
||||
*/
|
||||
val = sdhci_readl(host, ESDHC_TBCTL);
|
||||
val |= 0x3;
|
||||
sdhci_writel(host, val, ESDHC_TBCTL);
|
||||
sdhci_execute_tuning(mmc, opcode);
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
|
@ -1114,10 +1286,15 @@ static int sdhci_esdhc_probe(struct platform_device *pdev)
|
|||
|
||||
pltfm_host = sdhci_priv(host);
|
||||
esdhc = sdhci_pltfm_priv(pltfm_host);
|
||||
if (soc_device_match(soc_fixup_tuning))
|
||||
esdhc->quirk_fixup_tuning = true;
|
||||
if (soc_device_match(soc_tuning_erratum_type1))
|
||||
esdhc->quirk_tuning_erratum_type1 = true;
|
||||
else
|
||||
esdhc->quirk_fixup_tuning = false;
|
||||
esdhc->quirk_tuning_erratum_type1 = false;
|
||||
|
||||
if (soc_device_match(soc_tuning_erratum_type2))
|
||||
esdhc->quirk_tuning_erratum_type2 = true;
|
||||
else
|
||||
esdhc->quirk_tuning_erratum_type2 = false;
|
||||
|
||||
if (esdhc->vendor_ver == VENDOR_V_22)
|
||||
host->quirks2 |= SDHCI_QUIRK2_HOST_NO_CMD23;
|
||||
|
|
|
@ -21,6 +21,7 @@
|
|||
#include <linux/mmc/mmc.h>
|
||||
#include <linux/scatterlist.h>
|
||||
#include <linux/io.h>
|
||||
#include <linux/iopoll.h>
|
||||
#include <linux/gpio.h>
|
||||
#include <linux/pm_runtime.h>
|
||||
#include <linux/mmc/slot-gpio.h>
|
||||
|
@ -1590,11 +1591,59 @@ static int amd_probe(struct sdhci_pci_chip *chip)
|
|||
return 0;
|
||||
}
|
||||
|
||||
static u32 sdhci_read_present_state(struct sdhci_host *host)
|
||||
{
|
||||
return sdhci_readl(host, SDHCI_PRESENT_STATE);
|
||||
}
|
||||
|
||||
static void amd_sdhci_reset(struct sdhci_host *host, u8 mask)
|
||||
{
|
||||
struct sdhci_pci_slot *slot = sdhci_priv(host);
|
||||
struct pci_dev *pdev = slot->chip->pdev;
|
||||
u32 present_state;
|
||||
|
||||
/*
|
||||
* SDHC 0x7906 requires a hard reset to clear all internal state.
|
||||
* Otherwise it can get into a bad state where the DATA lines are always
|
||||
* read as zeros.
|
||||
*/
|
||||
if (pdev->device == 0x7906 && (mask & SDHCI_RESET_ALL)) {
|
||||
pci_clear_master(pdev);
|
||||
|
||||
pci_save_state(pdev);
|
||||
|
||||
pci_set_power_state(pdev, PCI_D3cold);
|
||||
pr_debug("%s: power_state=%u\n", mmc_hostname(host->mmc),
|
||||
pdev->current_state);
|
||||
pci_set_power_state(pdev, PCI_D0);
|
||||
|
||||
pci_restore_state(pdev);
|
||||
|
||||
/*
|
||||
* SDHCI_RESET_ALL says the card detect logic should not be
|
||||
* reset, but since we need to reset the entire controller
|
||||
* we should wait until the card detect logic has stabilized.
|
||||
*
|
||||
* This normally takes about 40ms.
|
||||
*/
|
||||
readx_poll_timeout(
|
||||
sdhci_read_present_state,
|
||||
host,
|
||||
present_state,
|
||||
present_state & SDHCI_CD_STABLE,
|
||||
10000,
|
||||
100000
|
||||
);
|
||||
}
|
||||
|
||||
return sdhci_reset(host, mask);
|
||||
}
|
||||
|
||||
static const struct sdhci_ops amd_sdhci_pci_ops = {
|
||||
.set_clock = sdhci_set_clock,
|
||||
.enable_dma = sdhci_pci_enable_dma,
|
||||
.set_bus_width = sdhci_set_bus_width,
|
||||
.reset = sdhci_reset,
|
||||
.reset = amd_sdhci_reset,
|
||||
.set_uhs_signaling = sdhci_set_uhs_signaling,
|
||||
};
|
||||
|
||||
|
@ -1673,6 +1722,8 @@ static const struct pci_device_id pci_ids[] = {
|
|||
SDHCI_PCI_DEVICE(INTEL, CML_EMMC, intel_glk_emmc),
|
||||
SDHCI_PCI_DEVICE(INTEL, CML_SD, intel_byt_sd),
|
||||
SDHCI_PCI_DEVICE(INTEL, CMLH_SD, intel_byt_sd),
|
||||
SDHCI_PCI_DEVICE(INTEL, JSL_EMMC, intel_glk_emmc),
|
||||
SDHCI_PCI_DEVICE(INTEL, JSL_SD, intel_byt_sd),
|
||||
SDHCI_PCI_DEVICE(O2, 8120, o2),
|
||||
SDHCI_PCI_DEVICE(O2, 8220, o2),
|
||||
SDHCI_PCI_DEVICE(O2, 8221, o2),
|
||||
|
|
|
@ -55,6 +55,8 @@
|
|||
#define PCI_DEVICE_ID_INTEL_CML_EMMC 0x02c4
|
||||
#define PCI_DEVICE_ID_INTEL_CML_SD 0x02f5
|
||||
#define PCI_DEVICE_ID_INTEL_CMLH_SD 0x06f5
|
||||
#define PCI_DEVICE_ID_INTEL_JSL_EMMC 0x4dc4
|
||||
#define PCI_DEVICE_ID_INTEL_JSL_SD 0x4df8
|
||||
|
||||
#define PCI_DEVICE_ID_SYSKONNECT_8000 0x8000
|
||||
#define PCI_DEVICE_ID_VIA_95D0 0x95d0
|
||||
|
|
|
@ -337,8 +337,19 @@ static void sdhci_init(struct sdhci_host *host, int soft)
|
|||
|
||||
static void sdhci_reinit(struct sdhci_host *host)
|
||||
{
|
||||
u32 cd = host->ier & (SDHCI_INT_CARD_REMOVE | SDHCI_INT_CARD_INSERT);
|
||||
|
||||
sdhci_init(host, 0);
|
||||
sdhci_enable_card_detection(host);
|
||||
|
||||
/*
|
||||
* A change to the card detect bits indicates a change in present state,
|
||||
* refer sdhci_set_card_detection(). A card detect interrupt might have
|
||||
* been missed while the host controller was being reset, so trigger a
|
||||
* rescan to check.
|
||||
*/
|
||||
if (cd != (host->ier & (SDHCI_INT_CARD_REMOVE | SDHCI_INT_CARD_INSERT)))
|
||||
mmc_detect_change(host->mmc, msecs_to_jiffies(200));
|
||||
}
|
||||
|
||||
static void __sdhci_led_activate(struct sdhci_host *host)
|
||||
|
@ -2202,7 +2213,7 @@ int sdhci_start_signal_voltage_switch(struct mmc_host *mmc,
|
|||
if (!(ctrl & SDHCI_CTRL_VDD_180))
|
||||
return 0;
|
||||
|
||||
pr_warn("%s: 3.3V regulator output did not became stable\n",
|
||||
pr_warn("%s: 3.3V regulator output did not become stable\n",
|
||||
mmc_hostname(mmc));
|
||||
|
||||
return -EAGAIN;
|
||||
|
@ -2234,7 +2245,7 @@ int sdhci_start_signal_voltage_switch(struct mmc_host *mmc,
|
|||
if (ctrl & SDHCI_CTRL_VDD_180)
|
||||
return 0;
|
||||
|
||||
pr_warn("%s: 1.8V regulator output did not became stable\n",
|
||||
pr_warn("%s: 1.8V regulator output did not become stable\n",
|
||||
mmc_hostname(mmc));
|
||||
|
||||
return -EAGAIN;
|
||||
|
|
|
@ -12,6 +12,7 @@
|
|||
#include <linux/property.h>
|
||||
#include <linux/regmap.h>
|
||||
|
||||
#include "cqhci.h"
|
||||
#include "sdhci-pltfm.h"
|
||||
|
||||
/* CTL_CFG Registers */
|
||||
|
@ -68,6 +69,9 @@
|
|||
|
||||
#define CLOCK_TOO_SLOW_HZ 400000
|
||||
|
||||
/* Command Queue Host Controller Interface Base address */
|
||||
#define SDHCI_AM654_CQE_BASE_ADDR 0x200
|
||||
|
||||
static struct regmap_config sdhci_am654_regmap_config = {
|
||||
.reg_bits = 32,
|
||||
.val_bits = 32,
|
||||
|
@ -259,6 +263,19 @@ static const struct sdhci_am654_driver_data sdhci_am654_drvdata = {
|
|||
.flags = IOMUX_PRESENT | FREQSEL_2_BIT | STRBSEL_4_BIT | DLL_PRESENT,
|
||||
};
|
||||
|
||||
static u32 sdhci_am654_cqhci_irq(struct sdhci_host *host, u32 intmask)
|
||||
{
|
||||
int cmd_error = 0;
|
||||
int data_error = 0;
|
||||
|
||||
if (!sdhci_cqe_irq(host, intmask, &cmd_error, &data_error))
|
||||
return intmask;
|
||||
|
||||
cqhci_irq(host->mmc, intmask, cmd_error, data_error);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static struct sdhci_ops sdhci_j721e_8bit_ops = {
|
||||
.get_max_clock = sdhci_pltfm_clk_get_max_clock,
|
||||
.get_timeout_clock = sdhci_pltfm_clk_get_max_clock,
|
||||
|
@ -267,6 +284,7 @@ static struct sdhci_ops sdhci_j721e_8bit_ops = {
|
|||
.set_power = sdhci_am654_set_power,
|
||||
.set_clock = sdhci_am654_set_clock,
|
||||
.write_b = sdhci_am654_write_b,
|
||||
.irq = sdhci_am654_cqhci_irq,
|
||||
.reset = sdhci_reset,
|
||||
};
|
||||
|
||||
|
@ -290,6 +308,7 @@ static struct sdhci_ops sdhci_j721e_4bit_ops = {
|
|||
.set_power = sdhci_am654_set_power,
|
||||
.set_clock = sdhci_j721e_4bit_set_clock,
|
||||
.write_b = sdhci_am654_write_b,
|
||||
.irq = sdhci_am654_cqhci_irq,
|
||||
.reset = sdhci_reset,
|
||||
};
|
||||
|
||||
|
@ -304,6 +323,40 @@ static const struct sdhci_am654_driver_data sdhci_j721e_4bit_drvdata = {
|
|||
.pdata = &sdhci_j721e_4bit_pdata,
|
||||
.flags = IOMUX_PRESENT,
|
||||
};
|
||||
|
||||
static void sdhci_am654_dumpregs(struct mmc_host *mmc)
|
||||
{
|
||||
sdhci_dumpregs(mmc_priv(mmc));
|
||||
}
|
||||
|
||||
static const struct cqhci_host_ops sdhci_am654_cqhci_ops = {
|
||||
.enable = sdhci_cqe_enable,
|
||||
.disable = sdhci_cqe_disable,
|
||||
.dumpregs = sdhci_am654_dumpregs,
|
||||
};
|
||||
|
||||
static int sdhci_am654_cqe_add_host(struct sdhci_host *host)
|
||||
{
|
||||
struct cqhci_host *cq_host;
|
||||
int ret;
|
||||
|
||||
cq_host = devm_kzalloc(host->mmc->parent, sizeof(struct cqhci_host),
|
||||
GFP_KERNEL);
|
||||
if (!cq_host)
|
||||
return -ENOMEM;
|
||||
|
||||
cq_host->mmio = host->ioaddr + SDHCI_AM654_CQE_BASE_ADDR;
|
||||
cq_host->quirks |= CQHCI_QUIRK_SHORT_TXFR_DESC_SZ;
|
||||
cq_host->caps |= CQHCI_TASK_DESC_SZ_128;
|
||||
cq_host->ops = &sdhci_am654_cqhci_ops;
|
||||
|
||||
host->mmc->caps2 |= MMC_CAP2_CQE;
|
||||
|
||||
ret = cqhci_init(cq_host, host->mmc, 1);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int sdhci_am654_init(struct sdhci_host *host)
|
||||
{
|
||||
struct sdhci_pltfm_host *pltfm_host = sdhci_priv(host);
|
||||
|
@ -344,7 +397,23 @@ static int sdhci_am654_init(struct sdhci_host *host)
|
|||
regmap_update_bits(sdhci_am654->base, CTL_CFG_2, SLOTTYPE_MASK,
|
||||
ctl_cfg_2);
|
||||
|
||||
return sdhci_add_host(host);
|
||||
ret = sdhci_setup_host(host);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
ret = sdhci_am654_cqe_add_host(host);
|
||||
if (ret)
|
||||
goto err_cleanup_host;
|
||||
|
||||
ret = __sdhci_add_host(host);
|
||||
if (ret)
|
||||
goto err_cleanup_host;
|
||||
|
||||
return 0;
|
||||
|
||||
err_cleanup_host:
|
||||
sdhci_cleanup_host(host);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int sdhci_am654_get_of_property(struct platform_device *pdev,
|
||||
|
|
|
@ -16,31 +16,7 @@
|
|||
#include <linux/clk.h>
|
||||
|
||||
#include "sdhci-pltfm.h"
|
||||
|
||||
/* F_SDH30 extended Controller registers */
|
||||
#define F_SDH30_AHB_CONFIG 0x100
|
||||
#define F_SDH30_AHB_BIGED 0x00000040
|
||||
#define F_SDH30_BUSLOCK_DMA 0x00000020
|
||||
#define F_SDH30_BUSLOCK_EN 0x00000010
|
||||
#define F_SDH30_SIN 0x00000008
|
||||
#define F_SDH30_AHB_INCR_16 0x00000004
|
||||
#define F_SDH30_AHB_INCR_8 0x00000002
|
||||
#define F_SDH30_AHB_INCR_4 0x00000001
|
||||
|
||||
#define F_SDH30_TUNING_SETTING 0x108
|
||||
#define F_SDH30_CMD_CHK_DIS 0x00010000
|
||||
|
||||
#define F_SDH30_IO_CONTROL2 0x114
|
||||
#define F_SDH30_CRES_O_DN 0x00080000
|
||||
#define F_SDH30_MSEL_O_1_8 0x00040000
|
||||
|
||||
#define F_SDH30_ESD_CONTROL 0x124
|
||||
#define F_SDH30_EMMC_RST 0x00000002
|
||||
#define F_SDH30_EMMC_HS200 0x01000000
|
||||
|
||||
#define F_SDH30_CMD_DAT_DELAY 0x200
|
||||
|
||||
#define F_SDH30_MIN_CLOCK 400000
|
||||
#include "sdhci_f_sdh30.h"
|
||||
|
||||
struct f_sdhost_priv {
|
||||
struct clk *clk_iface;
|
||||
|
|
|
@ -0,0 +1,32 @@
|
|||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
/*
|
||||
* Copyright (C) 2013 - 2015 Fujitsu Semiconductor, Ltd
|
||||
* Vincent Yang <vincent.yang@tw.fujitsu.com>
|
||||
* Copyright (C) 2015 Linaro Ltd Andy Green <andy.green@linaro.org>
|
||||
* Copyright (C) 2019 Socionext Inc.
|
||||
*
|
||||
*/
|
||||
|
||||
/* F_SDH30 extended Controller registers */
|
||||
#define F_SDH30_AHB_CONFIG 0x100
|
||||
#define F_SDH30_AHB_BIGED BIT(6)
|
||||
#define F_SDH30_BUSLOCK_DMA BIT(5)
|
||||
#define F_SDH30_BUSLOCK_EN BIT(4)
|
||||
#define F_SDH30_SIN BIT(3)
|
||||
#define F_SDH30_AHB_INCR_16 BIT(2)
|
||||
#define F_SDH30_AHB_INCR_8 BIT(1)
|
||||
#define F_SDH30_AHB_INCR_4 BIT(0)
|
||||
|
||||
#define F_SDH30_TUNING_SETTING 0x108
|
||||
#define F_SDH30_CMD_CHK_DIS BIT(16)
|
||||
|
||||
#define F_SDH30_IO_CONTROL2 0x114
|
||||
#define F_SDH30_CRES_O_DN BIT(19)
|
||||
#define F_SDH30_MSEL_O_1_8 BIT(18)
|
||||
|
||||
#define F_SDH30_ESD_CONTROL 0x124
|
||||
#define F_SDH30_EMMC_RST BIT(1)
|
||||
#define F_SDH30_CMD_DAT_DELAY BIT(9)
|
||||
#define F_SDH30_EMMC_HS200 BIT(24)
|
||||
|
||||
#define F_SDH30_MIN_CLOCK 400000
|
|
@ -1185,7 +1185,7 @@ int tmio_mmc_host_probe(struct tmio_mmc_host *_host)
|
|||
if (ret == -EPROBE_DEFER)
|
||||
return ret;
|
||||
|
||||
mmc->caps |= MMC_CAP_4_BIT_DATA | pdata->capabilities;
|
||||
mmc->caps |= MMC_CAP_ERASE | MMC_CAP_4_BIT_DATA | pdata->capabilities;
|
||||
mmc->caps2 |= pdata->capabilities2;
|
||||
mmc->max_segs = pdata->max_segs ? : 32;
|
||||
mmc->max_blk_size = TMIO_MAX_BLK_SIZE;
|
||||
|
|
|
@ -2070,18 +2070,11 @@ static void vub300_enable_sdio_irq(struct mmc_host *mmc, int enable)
|
|||
kref_put(&vub300->kref, vub300_delete);
|
||||
}
|
||||
|
||||
static void vub300_init_card(struct mmc_host *mmc, struct mmc_card *card)
|
||||
{ /* NOT irq */
|
||||
struct vub300_mmc_host *vub300 = mmc_priv(mmc);
|
||||
dev_info(&vub300->udev->dev, "NO host QUIRKS for this card\n");
|
||||
}
|
||||
|
||||
static const struct mmc_host_ops vub300_mmc_ops = {
|
||||
.request = vub300_mmc_request,
|
||||
.set_ios = vub300_mmc_set_ios,
|
||||
.get_ro = vub300_mmc_get_ro,
|
||||
.enable_sdio_irq = vub300_enable_sdio_irq,
|
||||
.init_card = vub300_init_card,
|
||||
};
|
||||
|
||||
static int vub300_probe(struct usb_interface *interface,
|
||||
|
|
|
@ -631,6 +631,7 @@ static int _mwifiex_fw_dpc(const struct firmware *firmware, void *context)
|
|||
|
||||
mwifiex_drv_get_driver_version(adapter, fmt, sizeof(fmt) - 1);
|
||||
mwifiex_dbg(adapter, MSG, "driver_version = %s\n", fmt);
|
||||
adapter->is_up = true;
|
||||
goto done;
|
||||
|
||||
err_add_intf:
|
||||
|
@ -1469,6 +1470,7 @@ int mwifiex_shutdown_sw(struct mwifiex_adapter *adapter)
|
|||
mwifiex_deauthenticate(priv, NULL);
|
||||
|
||||
mwifiex_uninit_sw(adapter);
|
||||
adapter->is_up = false;
|
||||
|
||||
if (adapter->if_ops.down_dev)
|
||||
adapter->if_ops.down_dev(adapter);
|
||||
|
@ -1730,7 +1732,8 @@ int mwifiex_remove_card(struct mwifiex_adapter *adapter)
|
|||
if (!adapter)
|
||||
return 0;
|
||||
|
||||
mwifiex_uninit_sw(adapter);
|
||||
if (adapter->is_up)
|
||||
mwifiex_uninit_sw(adapter);
|
||||
|
||||
if (adapter->irq_wakeup >= 0)
|
||||
device_init_wakeup(adapter->dev, false);
|
||||
|
|
|
@ -1017,6 +1017,7 @@ struct mwifiex_adapter {
|
|||
|
||||
/* For synchronizing FW initialization with device lifecycle. */
|
||||
struct completion *fw_done;
|
||||
bool is_up;
|
||||
|
||||
bool ext_scan;
|
||||
u8 fw_api_ver;
|
||||
|
|
|
@ -444,6 +444,9 @@ static int mwifiex_sdio_suspend(struct device *dev)
|
|||
return 0;
|
||||
}
|
||||
|
||||
if (!adapter->is_up)
|
||||
return -EBUSY;
|
||||
|
||||
mwifiex_enable_wake(adapter);
|
||||
|
||||
/* Enable the Host Sleep */
|
||||
|
@ -2220,22 +2223,30 @@ static void mwifiex_sdio_card_reset_work(struct mwifiex_adapter *adapter)
|
|||
struct sdio_func *func = card->func;
|
||||
int ret;
|
||||
|
||||
/* Prepare the adapter for the reset. */
|
||||
mwifiex_shutdown_sw(adapter);
|
||||
|
||||
/* power cycle the adapter */
|
||||
sdio_claim_host(func);
|
||||
mmc_hw_reset(func->card->host);
|
||||
sdio_release_host(func);
|
||||
|
||||
/* Previous save_adapter won't be valid after this. We will cancel
|
||||
* pending work requests.
|
||||
*/
|
||||
clear_bit(MWIFIEX_IFACE_WORK_DEVICE_DUMP, &card->work_flags);
|
||||
clear_bit(MWIFIEX_IFACE_WORK_CARD_RESET, &card->work_flags);
|
||||
|
||||
ret = mwifiex_reinit_sw(adapter);
|
||||
if (ret)
|
||||
dev_err(&func->dev, "reinit failed: %d\n", ret);
|
||||
/* Run a HW reset of the SDIO interface. */
|
||||
sdio_claim_host(func);
|
||||
ret = mmc_hw_reset(func->card->host);
|
||||
sdio_release_host(func);
|
||||
|
||||
switch (ret) {
|
||||
case 1:
|
||||
dev_dbg(&func->dev, "SDIO HW reset asynchronous\n");
|
||||
complete_all(adapter->fw_done);
|
||||
break;
|
||||
case 0:
|
||||
ret = mwifiex_reinit_sw(adapter);
|
||||
if (ret)
|
||||
dev_err(&func->dev, "reinit failed: %d\n", ret);
|
||||
break;
|
||||
default:
|
||||
dev_err(&func->dev, "SDIO HW reset failed: %d\n", ret);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
/* This function read/write firmware */
|
||||
|
|
|
@ -16,17 +16,12 @@
|
|||
#include <linux/irq.h>
|
||||
#include <linux/pm_runtime.h>
|
||||
#include <linux/gpio.h>
|
||||
#include <linux/of.h>
|
||||
#include <linux/of_gpio.h>
|
||||
#include <linux/of_irq.h>
|
||||
|
||||
#include "wl1251.h"
|
||||
|
||||
#ifndef SDIO_VENDOR_ID_TI
|
||||
#define SDIO_VENDOR_ID_TI 0x104c
|
||||
#endif
|
||||
|
||||
#ifndef SDIO_DEVICE_ID_TI_WL1251
|
||||
#define SDIO_DEVICE_ID_TI_WL1251 0x9066
|
||||
#endif
|
||||
|
||||
struct wl1251_sdio {
|
||||
struct sdio_func *func;
|
||||
u32 elp_val;
|
||||
|
@ -49,7 +44,7 @@ static void wl1251_sdio_interrupt(struct sdio_func *func)
|
|||
}
|
||||
|
||||
static const struct sdio_device_id wl1251_devices[] = {
|
||||
{ SDIO_DEVICE(SDIO_VENDOR_ID_TI, SDIO_DEVICE_ID_TI_WL1251) },
|
||||
{ SDIO_DEVICE(SDIO_VENDOR_ID_TI_WL1251, SDIO_DEVICE_ID_TI_WL1251) },
|
||||
{}
|
||||
};
|
||||
MODULE_DEVICE_TABLE(sdio, wl1251_devices);
|
||||
|
@ -217,6 +212,7 @@ static int wl1251_sdio_probe(struct sdio_func *func,
|
|||
struct ieee80211_hw *hw;
|
||||
struct wl1251_sdio *wl_sdio;
|
||||
const struct wl1251_platform_data *wl1251_board_data;
|
||||
struct device_node *np = func->dev.of_node;
|
||||
|
||||
hw = wl1251_alloc_hw();
|
||||
if (IS_ERR(hw))
|
||||
|
@ -248,6 +244,17 @@ static int wl1251_sdio_probe(struct sdio_func *func,
|
|||
wl->power_gpio = wl1251_board_data->power_gpio;
|
||||
wl->irq = wl1251_board_data->irq;
|
||||
wl->use_eeprom = wl1251_board_data->use_eeprom;
|
||||
} else if (np) {
|
||||
wl->use_eeprom = of_property_read_bool(np,
|
||||
"ti,wl1251-has-eeprom");
|
||||
wl->power_gpio = of_get_named_gpio(np, "ti,power-gpio", 0);
|
||||
wl->irq = of_irq_get(np, 0);
|
||||
|
||||
if (wl->power_gpio == -EPROBE_DEFER ||
|
||||
wl->irq == -EPROBE_DEFER) {
|
||||
ret = -EPROBE_DEFER;
|
||||
goto disable;
|
||||
}
|
||||
}
|
||||
|
||||
if (gpio_is_valid(wl->power_gpio)) {
|
||||
|
|
|
@ -26,14 +26,6 @@
|
|||
#include "wl12xx_80211.h"
|
||||
#include "io.h"
|
||||
|
||||
#ifndef SDIO_VENDOR_ID_TI
|
||||
#define SDIO_VENDOR_ID_TI 0x0097
|
||||
#endif
|
||||
|
||||
#ifndef SDIO_DEVICE_ID_TI_WL1271
|
||||
#define SDIO_DEVICE_ID_TI_WL1271 0x4076
|
||||
#endif
|
||||
|
||||
static bool dump = false;
|
||||
|
||||
struct wl12xx_sdio_glue {
|
||||
|
|
|
@ -91,7 +91,8 @@ enum pm_ret_status {
|
|||
};
|
||||
|
||||
enum pm_ioctl_id {
|
||||
IOCTL_SET_PLL_FRAC_MODE = 8,
|
||||
IOCTL_SET_SD_TAPDELAY = 7,
|
||||
IOCTL_SET_PLL_FRAC_MODE,
|
||||
IOCTL_GET_PLL_FRAC_MODE,
|
||||
IOCTL_SET_PLL_FRAC_DATA,
|
||||
IOCTL_GET_PLL_FRAC_DATA,
|
||||
|
@ -250,6 +251,16 @@ enum zynqmp_pm_request_ack {
|
|||
ZYNQMP_PM_REQUEST_ACK_NON_BLOCKING,
|
||||
};
|
||||
|
||||
enum pm_node_id {
|
||||
NODE_SD_0 = 39,
|
||||
NODE_SD_1,
|
||||
};
|
||||
|
||||
enum tap_delay_type {
|
||||
PM_TAPDELAY_INPUT = 0,
|
||||
PM_TAPDELAY_OUTPUT,
|
||||
};
|
||||
|
||||
/**
|
||||
* struct zynqmp_pm_query_data - PM query data
|
||||
* @qid: query ID
|
||||
|
|
|
@ -226,7 +226,7 @@ struct mmc_queue_req;
|
|||
* MMC Physical partitions
|
||||
*/
|
||||
struct mmc_part {
|
||||
unsigned int size; /* partition size (in bytes) */
|
||||
u64 size; /* partition size (in bytes) */
|
||||
unsigned int part_cfg; /* partition type */
|
||||
char name[MAX_MMC_PART_NAME_LEN];
|
||||
bool force_ro; /* to make boot parts RO by default */
|
||||
|
@ -291,6 +291,7 @@ struct mmc_card {
|
|||
struct sd_switch_caps sw_caps; /* switch (CMD6) caps */
|
||||
|
||||
unsigned int sdio_funcs; /* number of SDIO functions */
|
||||
atomic_t sdio_funcs_probed; /* number of probed SDIO funcs */
|
||||
struct sdio_cccr cccr; /* common card info */
|
||||
struct sdio_cis cis; /* common tuple info */
|
||||
struct sdio_func *sdio_func[SDIO_MAX_FUNCS]; /* SDIO functions (devices) */
|
||||
|
|
|
@ -71,6 +71,8 @@
|
|||
|
||||
#define SDIO_VENDOR_ID_TI 0x0097
|
||||
#define SDIO_DEVICE_ID_TI_WL1271 0x4076
|
||||
#define SDIO_VENDOR_ID_TI_WL1251 0x104c
|
||||
#define SDIO_DEVICE_ID_TI_WL1251 0x9066
|
||||
|
||||
#define SDIO_VENDOR_ID_STE 0x0020
|
||||
#define SDIO_DEVICE_ID_STE_CW1200 0x2280
|
||||
|
|
|
@ -67,9 +67,6 @@ struct omap_hsmmc_platform_data {
|
|||
/* string specifying a particular variant of hardware */
|
||||
char *version;
|
||||
|
||||
/* if we have special card, init it using this callback */
|
||||
void (*init_card)(struct mmc_card *card);
|
||||
|
||||
const char *name;
|
||||
u32 ocr_mask;
|
||||
};
|
||||
|
|
Loading…
Reference in New Issue