You've already forked linux-apfs
mirror of
https://github.com/linux-apfs/linux-apfs.git
synced 2026-05-01 15:00:59 -07:00
Merge tag 'armsoc-drivers' of git://git.kernel.org/pub/scm/linux/kernel/git/arm/arm-soc
Pull ARM SoC driver updates from Arnd Bergmann:
"This branch contains platform-related driver updates for ARM and
ARM64, these are the areas that bring the changes:
New drivers:
- driver support for Renesas R-Car V3M (R8A77970)
- power management support for Amlogic GX
- a new driver for the Tegra BPMP thermal sensor
- a new bus driver for Technologic Systems NBUS
Changes for subsystems that prefer to merge through arm-soc:
- the usual updates for reset controller drivers from Philipp Zabel,
with five added drivers for SoCs in the arc, meson, socfpa,
uniphier and mediatek families
- updates to the ARM SCPI and PSCI frameworks, from Sudeep Holla,
Heiner Kallweit and Lorenzo Pieralisi
Changes specific to some ARM-based SoC
- the Freescale/NXP DPAA QBMan drivers from PowerPC can now work on
ARM as well
- several changes for power management on Broadcom SoCs
- various improvements on Qualcomm, Broadcom, Amlogic, Atmel,
Mediatek
- minor Cleanups for Samsung, TI OMAP SoCs"
[ NOTE! This doesn't work without the previous ARM SoC device-tree pull,
because the R8A77970 driver is missing a header file that came from
that pull.
The fact that this got merged afterwards only fixes it at this point,
and bisection of that driver will fail if/when you walk into the
history of that driver. - Linus ]
* tag 'armsoc-drivers' of git://git.kernel.org/pub/scm/linux/kernel/git/arm/arm-soc: (96 commits)
soc: amlogic: meson-gx-pwrc-vpu: fix power-off when powered by bootloader
bus: add driver for the Technologic Systems NBUS
memory: omap-gpmc: Remove deprecated gpmc_update_nand_reg()
soc: qcom: remove unused label
soc: amlogic: gx pm domain: add PM and OF dependencies
drivers/firmware: psci_checker: Add missing destroy_timer_on_stack()
dt-bindings: power: add amlogic meson power domain bindings
soc: amlogic: add Meson GX VPU Domains driver
soc: qcom: Remote filesystem memory driver
dt-binding: soc: qcom: Add binding for rmtfs memory
of: reserved_mem: Accessor for acquiring reserved_mem
of/platform: Generalize /reserved-memory handling
soc: mediatek: pwrap: fix fatal compiler error
soc: mediatek: pwrap: fix compiler errors
arm64: mediatek: cleanup message for platform selection
soc: Allow test-building of MediaTek drivers
soc: mediatek: place Kconfig for all SoC drivers under menu
soc: mediatek: pwrap: add support for MT7622 SoC
soc: mediatek: pwrap: add common way for setup CS timing extenstion
soc: mediatek: pwrap: add MediaTek MT6380 as one slave of pwrap
..
This commit is contained in:
@@ -164,6 +164,8 @@ Control registers for this memory controller's DDR PHY.
|
|||||||
|
|
||||||
Required properties:
|
Required properties:
|
||||||
- compatible : should contain one of these
|
- compatible : should contain one of these
|
||||||
|
"brcm,brcmstb-ddr-phy-v71.1"
|
||||||
|
"brcm,brcmstb-ddr-phy-v72.0"
|
||||||
"brcm,brcmstb-ddr-phy-v225.1"
|
"brcm,brcmstb-ddr-phy-v225.1"
|
||||||
"brcm,brcmstb-ddr-phy-v240.1"
|
"brcm,brcmstb-ddr-phy-v240.1"
|
||||||
"brcm,brcmstb-ddr-phy-v240.2"
|
"brcm,brcmstb-ddr-phy-v240.2"
|
||||||
@@ -184,7 +186,9 @@ Sequencer DRAM parameters and control registers. Used for Self-Refresh
|
|||||||
Power-Down (SRPD), among other things.
|
Power-Down (SRPD), among other things.
|
||||||
|
|
||||||
Required properties:
|
Required properties:
|
||||||
- compatible : should contain "brcm,brcmstb-memc-ddr"
|
- compatible : should contain one of these
|
||||||
|
"brcm,brcmstb-memc-ddr-rev-b.2.2"
|
||||||
|
"brcm,brcmstb-memc-ddr"
|
||||||
- reg : the MEMC DDR register range
|
- reg : the MEMC DDR register range
|
||||||
|
|
||||||
Example:
|
Example:
|
||||||
|
|||||||
@@ -4,7 +4,6 @@ Properties:
|
|||||||
- compatible : should contain two values. First value must be one from following list:
|
- compatible : should contain two values. First value must be one from following list:
|
||||||
- "samsung,exynos3250-pmu" - for Exynos3250 SoC,
|
- "samsung,exynos3250-pmu" - for Exynos3250 SoC,
|
||||||
- "samsung,exynos4210-pmu" - for Exynos4210 SoC,
|
- "samsung,exynos4210-pmu" - for Exynos4210 SoC,
|
||||||
- "samsung,exynos4212-pmu" - for Exynos4212 SoC,
|
|
||||||
- "samsung,exynos4412-pmu" - for Exynos4412 SoC,
|
- "samsung,exynos4412-pmu" - for Exynos4412 SoC,
|
||||||
- "samsung,exynos5250-pmu" - for Exynos5250 SoC,
|
- "samsung,exynos5250-pmu" - for Exynos5250 SoC,
|
||||||
- "samsung,exynos5260-pmu" - for Exynos5260 SoC.
|
- "samsung,exynos5260-pmu" - for Exynos5260 SoC.
|
||||||
|
|||||||
@@ -18,6 +18,8 @@ Required properties:
|
|||||||
* Core, iface, and bus clocks required for "qcom,scm"
|
* Core, iface, and bus clocks required for "qcom,scm"
|
||||||
- clock-names: Must contain "core" for the core clock, "iface" for the interface
|
- clock-names: Must contain "core" for the core clock, "iface" for the interface
|
||||||
clock and "bus" for the bus clock per the requirements of the compatible.
|
clock and "bus" for the bus clock per the requirements of the compatible.
|
||||||
|
- qcom,dload-mode: phandle to the TCSR hardware block and offset of the
|
||||||
|
download mode control register (optional)
|
||||||
|
|
||||||
Example for MSM8916:
|
Example for MSM8916:
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,27 @@
|
|||||||
|
DDR PHY Front End (DPFE) for Broadcom STB
|
||||||
|
=========================================
|
||||||
|
|
||||||
|
DPFE and the DPFE firmware provide an interface for the host CPU to
|
||||||
|
communicate with the DCPU, which resides inside the DDR PHY.
|
||||||
|
|
||||||
|
There are three memory regions for interacting with the DCPU. These are
|
||||||
|
specified in a single reg property.
|
||||||
|
|
||||||
|
Required properties:
|
||||||
|
- compatible: must be "brcm,bcm7271-dpfe-cpu", "brcm,bcm7268-dpfe-cpu"
|
||||||
|
or "brcm,dpfe-cpu"
|
||||||
|
- reg: must reference three register ranges
|
||||||
|
- start address and length of the DCPU register space
|
||||||
|
- start address and length of the DCPU data memory space
|
||||||
|
- start address and length of the DCPU instruction memory space
|
||||||
|
- reg-names: must contain "dpfe-cpu", "dpfe-dmem", and "dpfe-imem";
|
||||||
|
they must be in the same order as the register declarations
|
||||||
|
|
||||||
|
Example:
|
||||||
|
dpfe_cpu0: dpfe-cpu@f1132000 {
|
||||||
|
compatible = "brcm,bcm7271-dpfe-cpu", "brcm,dpfe-cpu";
|
||||||
|
reg = <0xf1132000 0x180
|
||||||
|
0xf1134000 0x1000
|
||||||
|
0xf1138000 0x4000>;
|
||||||
|
reg-names = "dpfe-cpu", "dpfe-dmem", "dpfe-imem";
|
||||||
|
};
|
||||||
@@ -11,3 +11,156 @@ Required properties:
|
|||||||
|
|
||||||
The experimental -viper variants are for running Linux on the 3384's
|
The experimental -viper variants are for running Linux on the 3384's
|
||||||
BMIPS4355 cable modem CPU instead of the BMIPS5000 application processor.
|
BMIPS4355 cable modem CPU instead of the BMIPS5000 application processor.
|
||||||
|
|
||||||
|
Power management
|
||||||
|
----------------
|
||||||
|
|
||||||
|
For power management (particularly, S2/S3/S5 system suspend), the following SoC
|
||||||
|
components are needed:
|
||||||
|
|
||||||
|
= Always-On control block (AON CTRL)
|
||||||
|
|
||||||
|
This hardware provides control registers for the "always-on" (even in low-power
|
||||||
|
modes) hardware, such as the Power Management State Machine (PMSM).
|
||||||
|
|
||||||
|
Required properties:
|
||||||
|
- compatible : should be one of
|
||||||
|
"brcm,bcm7425-aon-ctrl"
|
||||||
|
"brcm,bcm7429-aon-ctrl"
|
||||||
|
"brcm,bcm7435-aon-ctrl" and
|
||||||
|
"brcm,brcmstb-aon-ctrl"
|
||||||
|
- reg : the register start and length for the AON CTRL block
|
||||||
|
|
||||||
|
Example:
|
||||||
|
|
||||||
|
syscon@410000 {
|
||||||
|
compatible = "brcm,bcm7425-aon-ctrl", "brcm,brcmstb-aon-ctrl";
|
||||||
|
reg = <0x410000 0x400>;
|
||||||
|
};
|
||||||
|
|
||||||
|
= Memory controllers
|
||||||
|
|
||||||
|
A Broadcom STB SoC typically has a number of independent memory controllers,
|
||||||
|
each of which may have several associated hardware blocks, which are versioned
|
||||||
|
independently (control registers, DDR PHYs, etc.). One might consider
|
||||||
|
describing these controllers as a parent "memory controllers" block, which
|
||||||
|
contains N sub-nodes (one for each controller in the system), each of which is
|
||||||
|
associated with a number of hardware register resources (e.g., its PHY.
|
||||||
|
|
||||||
|
== MEMC (MEMory Controller)
|
||||||
|
|
||||||
|
Represents a single memory controller instance.
|
||||||
|
|
||||||
|
Required properties:
|
||||||
|
- compatible : should contain "brcm,brcmstb-memc" and "simple-bus"
|
||||||
|
- ranges : should contain the child address in the parent address
|
||||||
|
space, must be 0 here, and the register start and length of
|
||||||
|
the entire memory controller (including all sub nodes: DDR PHY,
|
||||||
|
arbiter, etc.)
|
||||||
|
- #address-cells : must be 1
|
||||||
|
- #size-cells : must be 1
|
||||||
|
|
||||||
|
Example:
|
||||||
|
|
||||||
|
memory-controller@0 {
|
||||||
|
compatible = "brcm,brcmstb-memc", "simple-bus";
|
||||||
|
ranges = <0x0 0x0 0xa000>;
|
||||||
|
#address-cells = <1>;
|
||||||
|
#size-cells = <1>;
|
||||||
|
|
||||||
|
memc-arb@1000 {
|
||||||
|
...
|
||||||
|
};
|
||||||
|
|
||||||
|
memc-ddr@2000 {
|
||||||
|
...
|
||||||
|
};
|
||||||
|
|
||||||
|
ddr-phy@6000 {
|
||||||
|
...
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
Should contain subnodes for any of the following relevant hardware resources:
|
||||||
|
|
||||||
|
== DDR PHY control
|
||||||
|
|
||||||
|
Control registers for this memory controller's DDR PHY.
|
||||||
|
|
||||||
|
Required properties:
|
||||||
|
- compatible : should contain one of these
|
||||||
|
"brcm,brcmstb-ddr-phy-v64.5"
|
||||||
|
"brcm,brcmstb-ddr-phy"
|
||||||
|
|
||||||
|
- reg : the DDR PHY register range and length
|
||||||
|
|
||||||
|
Example:
|
||||||
|
|
||||||
|
ddr-phy@6000 {
|
||||||
|
compatible = "brcm,brcmstb-ddr-phy-v64.5";
|
||||||
|
reg = <0x6000 0xc8>;
|
||||||
|
};
|
||||||
|
|
||||||
|
== DDR memory controller sequencer
|
||||||
|
|
||||||
|
Control registers for this memory controller's DDR memory sequencer
|
||||||
|
|
||||||
|
Required properties:
|
||||||
|
- compatible : should contain one of these
|
||||||
|
"brcm,bcm7425-memc-ddr"
|
||||||
|
"brcm,bcm7429-memc-ddr"
|
||||||
|
"brcm,bcm7435-memc-ddr" and
|
||||||
|
"brcm,brcmstb-memc-ddr"
|
||||||
|
|
||||||
|
- reg : the DDR sequencer register range and length
|
||||||
|
|
||||||
|
Example:
|
||||||
|
|
||||||
|
memc-ddr@2000 {
|
||||||
|
compatible = "brcm,bcm7425-memc-ddr", "brcm,brcmstb-memc-ddr";
|
||||||
|
reg = <0x2000 0x300>;
|
||||||
|
};
|
||||||
|
|
||||||
|
== MEMC Arbiter
|
||||||
|
|
||||||
|
The memory controller arbiter is responsible for memory clients allocation
|
||||||
|
(bandwidth, priorities etc.) and needs to have its contents restored during
|
||||||
|
deep sleep states (S3).
|
||||||
|
|
||||||
|
Required properties:
|
||||||
|
|
||||||
|
- compatible : should contain one of these
|
||||||
|
"brcm,brcmstb-memc-arb-v10.0.0.0"
|
||||||
|
"brcm,brcmstb-memc-arb"
|
||||||
|
|
||||||
|
- reg : the DDR Arbiter register range and length
|
||||||
|
|
||||||
|
Example:
|
||||||
|
|
||||||
|
memc-arb@1000 {
|
||||||
|
compatible = "brcm,brcmstb-memc-arb-v10.0.0.0";
|
||||||
|
reg = <0x1000 0x248>;
|
||||||
|
};
|
||||||
|
|
||||||
|
== Timers
|
||||||
|
|
||||||
|
The Broadcom STB chips contain a timer block with several general purpose
|
||||||
|
timers that can be used.
|
||||||
|
|
||||||
|
Required properties:
|
||||||
|
|
||||||
|
- compatible : should contain one of:
|
||||||
|
"brcm,bcm7425-timers"
|
||||||
|
"brcm,bcm7429-timers"
|
||||||
|
"brcm,bcm7435-timers and
|
||||||
|
"brcm,brcmstb-timers"
|
||||||
|
- reg : the timers register range
|
||||||
|
- interrupts : the interrupt line for this timer block
|
||||||
|
|
||||||
|
Example:
|
||||||
|
|
||||||
|
timers: timer@4067c0 {
|
||||||
|
compatible = "brcm,bcm7425-timers", "brcm,brcmstb-timers";
|
||||||
|
reg = <0x4067c0 0x40>;
|
||||||
|
interrupts = <&periph_intc 19>;
|
||||||
|
};
|
||||||
|
|||||||
@@ -0,0 +1,61 @@
|
|||||||
|
Amlogic Meson Power Controller
|
||||||
|
==============================
|
||||||
|
|
||||||
|
The Amlogic Meson SoCs embeds an internal Power domain controller.
|
||||||
|
|
||||||
|
VPU Power Domain
|
||||||
|
----------------
|
||||||
|
|
||||||
|
The Video Processing Unit power domain is controlled by this power controller,
|
||||||
|
but the domain requires some external resources to meet the correct power
|
||||||
|
sequences.
|
||||||
|
The bindings must respect the power domain bindings as described in the file
|
||||||
|
power_domain.txt
|
||||||
|
|
||||||
|
Device Tree Bindings:
|
||||||
|
---------------------
|
||||||
|
|
||||||
|
Required properties:
|
||||||
|
- compatible: should be "amlogic,meson-gx-pwrc-vpu" for the Meson GX SoCs
|
||||||
|
- #power-domain-cells: should be 0
|
||||||
|
- amlogic,hhi-sysctrl: phandle to the HHI sysctrl node
|
||||||
|
- resets: phandles to the reset lines needed for this power demain sequence
|
||||||
|
as described in ../reset/reset.txt
|
||||||
|
- clocks: from common clock binding: handle to VPU and VAPB clocks
|
||||||
|
- clock-names: from common clock binding: must contain "vpu", "vapb"
|
||||||
|
corresponding to entry in the clocks property.
|
||||||
|
|
||||||
|
Parent node should have the following properties :
|
||||||
|
- compatible: "amlogic,meson-gx-ao-sysctrl", "syscon", "simple-mfd"
|
||||||
|
- reg: base address and size of the AO system control register space.
|
||||||
|
|
||||||
|
Example:
|
||||||
|
-------
|
||||||
|
|
||||||
|
ao_sysctrl: sys-ctrl@0 {
|
||||||
|
compatible = "amlogic,meson-gx-ao-sysctrl", "syscon", "simple-mfd";
|
||||||
|
reg = <0x0 0x0 0x0 0x100>;
|
||||||
|
|
||||||
|
pwrc_vpu: power-controller-vpu {
|
||||||
|
compatible = "amlogic,meson-gx-pwrc-vpu";
|
||||||
|
#power-domain-cells = <0>;
|
||||||
|
amlogic,hhi-sysctrl = <&sysctrl>;
|
||||||
|
resets = <&reset RESET_VIU>,
|
||||||
|
<&reset RESET_VENC>,
|
||||||
|
<&reset RESET_VCBUS>,
|
||||||
|
<&reset RESET_BT656>,
|
||||||
|
<&reset RESET_DVIN_RESET>,
|
||||||
|
<&reset RESET_RDMA>,
|
||||||
|
<&reset RESET_VENCI>,
|
||||||
|
<&reset RESET_VENCP>,
|
||||||
|
<&reset RESET_VDAC>,
|
||||||
|
<&reset RESET_VDI6>,
|
||||||
|
<&reset RESET_VENCL>,
|
||||||
|
<&reset RESET_VID_LOCK>;
|
||||||
|
clocks = <&clkc CLKID_VPU>,
|
||||||
|
<&clkc CLKID_VAPB>;
|
||||||
|
clock-names = "vpu", "vapb";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
@@ -17,6 +17,7 @@ Required properties:
|
|||||||
- "renesas,r8a7794-sysc" (R-Car E2)
|
- "renesas,r8a7794-sysc" (R-Car E2)
|
||||||
- "renesas,r8a7795-sysc" (R-Car H3)
|
- "renesas,r8a7795-sysc" (R-Car H3)
|
||||||
- "renesas,r8a7796-sysc" (R-Car M3-W)
|
- "renesas,r8a7796-sysc" (R-Car M3-W)
|
||||||
|
- "renesas,r8a77970-sysc" (R-Car V3M)
|
||||||
- "renesas,r8a77995-sysc" (R-Car D3)
|
- "renesas,r8a77995-sysc" (R-Car D3)
|
||||||
- reg: Address start and address range for the device.
|
- reg: Address start and address range for the device.
|
||||||
- #power-domain-cells: Must be 1.
|
- #power-domain-cells: Must be 1.
|
||||||
|
|||||||
@@ -0,0 +1,51 @@
|
|||||||
|
Qualcomm Remote File System Memory binding
|
||||||
|
|
||||||
|
This binding describes the Qualcomm remote filesystem memory, which serves the
|
||||||
|
purpose of describing the shared memory region used for remote processors to
|
||||||
|
access block device data using the Remote Filesystem protocol.
|
||||||
|
|
||||||
|
- compatible:
|
||||||
|
Usage: required
|
||||||
|
Value type: <stringlist>
|
||||||
|
Definition: must be:
|
||||||
|
"qcom,rmtfs-mem"
|
||||||
|
|
||||||
|
- reg:
|
||||||
|
Usage: required for static allocation
|
||||||
|
Value type: <prop-encoded-array>
|
||||||
|
Definition: must specify base address and size of the memory region,
|
||||||
|
as described in reserved-memory.txt
|
||||||
|
|
||||||
|
- size:
|
||||||
|
Usage: required for dynamic allocation
|
||||||
|
Value type: <prop-encoded-array>
|
||||||
|
Definition: must specify a size of the memory region, as described in
|
||||||
|
reserved-memory.txt
|
||||||
|
|
||||||
|
- qcom,client-id:
|
||||||
|
Usage: required
|
||||||
|
Value type: <u32>
|
||||||
|
Definition: identifier of the client to use this region for buffers.
|
||||||
|
|
||||||
|
- qcom,vmid:
|
||||||
|
Usage: optional
|
||||||
|
Value type: <u32>
|
||||||
|
Definition: vmid of the remote processor, to set up memory protection.
|
||||||
|
|
||||||
|
= EXAMPLE
|
||||||
|
The following example shows the remote filesystem memory setup for APQ8016,
|
||||||
|
with the rmtfs region for the Hexagon DSP (id #1) located at 0x86700000.
|
||||||
|
|
||||||
|
reserved-memory {
|
||||||
|
#address-cells = <2>;
|
||||||
|
#size-cells = <2>;
|
||||||
|
ranges;
|
||||||
|
|
||||||
|
rmtfs@86700000 {
|
||||||
|
compatible = "qcom,rmtfs-mem";
|
||||||
|
reg = <0x0 0x86700000 0x0 0xe0000>;
|
||||||
|
no-map;
|
||||||
|
|
||||||
|
qcom,client-id = <1>;
|
||||||
|
};
|
||||||
|
};
|
||||||
@@ -26,6 +26,7 @@ Required properties:
|
|||||||
- "renesas,r8a7794-rst" (R-Car E2)
|
- "renesas,r8a7794-rst" (R-Car E2)
|
||||||
- "renesas,r8a7795-rst" (R-Car H3)
|
- "renesas,r8a7795-rst" (R-Car H3)
|
||||||
- "renesas,r8a7796-rst" (R-Car M3-W)
|
- "renesas,r8a7796-rst" (R-Car M3-W)
|
||||||
|
- "renesas,r8a77970-rst" (R-Car V3M)
|
||||||
- "renesas,r8a77995-rst" (R-Car D3)
|
- "renesas,r8a77995-rst" (R-Car D3)
|
||||||
- reg: Address start and address range for the device.
|
- reg: Address start and address range for the device.
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,33 @@
|
|||||||
|
Binding for the AXS10x reset controller
|
||||||
|
|
||||||
|
This binding describes the ARC AXS10x boards custom IP-block which allows
|
||||||
|
to control reset signals of selected peripherals. For example DW GMAC, etc...
|
||||||
|
This block is controlled via memory-mapped register (AKA CREG) which
|
||||||
|
represents up-to 32 reset lines.
|
||||||
|
|
||||||
|
As of today only the following lines are used:
|
||||||
|
- DW GMAC - line 5
|
||||||
|
|
||||||
|
This binding uses the common reset binding[1].
|
||||||
|
|
||||||
|
[1] Documentation/devicetree/bindings/reset/reset.txt
|
||||||
|
|
||||||
|
Required properties:
|
||||||
|
- compatible: should be "snps,axs10x-reset".
|
||||||
|
- reg: should always contain pair address - length: for creg reset
|
||||||
|
bits register.
|
||||||
|
- #reset-cells: from common reset binding; Should always be set to 1.
|
||||||
|
|
||||||
|
Example:
|
||||||
|
reset: reset-controller@11220 {
|
||||||
|
compatible = "snps,axs10x-reset";
|
||||||
|
#reset-cells = <1>;
|
||||||
|
reg = <0x11220 0x4>;
|
||||||
|
};
|
||||||
|
|
||||||
|
Specifying reset lines connected to IP modules:
|
||||||
|
ethernet@.... {
|
||||||
|
....
|
||||||
|
resets = <&reset 5>;
|
||||||
|
....
|
||||||
|
};
|
||||||
@@ -13,6 +13,7 @@ Required properties:
|
|||||||
"socionext,uniphier-pxs2-reset" - for PXs2/LD6b SoC
|
"socionext,uniphier-pxs2-reset" - for PXs2/LD6b SoC
|
||||||
"socionext,uniphier-ld11-reset" - for LD11 SoC
|
"socionext,uniphier-ld11-reset" - for LD11 SoC
|
||||||
"socionext,uniphier-ld20-reset" - for LD20 SoC
|
"socionext,uniphier-ld20-reset" - for LD20 SoC
|
||||||
|
"socionext,uniphier-pxs3-reset" - for PXs3 SoC
|
||||||
- #reset-cells: should be 1.
|
- #reset-cells: should be 1.
|
||||||
|
|
||||||
Example:
|
Example:
|
||||||
@@ -44,6 +45,7 @@ Required properties:
|
|||||||
"socionext,uniphier-ld11-mio-reset" - for LD11 SoC (MIO)
|
"socionext,uniphier-ld11-mio-reset" - for LD11 SoC (MIO)
|
||||||
"socionext,uniphier-ld11-sd-reset" - for LD11 SoC (SD)
|
"socionext,uniphier-ld11-sd-reset" - for LD11 SoC (SD)
|
||||||
"socionext,uniphier-ld20-sd-reset" - for LD20 SoC
|
"socionext,uniphier-ld20-sd-reset" - for LD20 SoC
|
||||||
|
"socionext,uniphier-pxs3-sd-reset" - for PXs3 SoC
|
||||||
- #reset-cells: should be 1.
|
- #reset-cells: should be 1.
|
||||||
|
|
||||||
Example:
|
Example:
|
||||||
@@ -74,6 +76,7 @@ Required properties:
|
|||||||
"socionext,uniphier-pxs2-peri-reset" - for PXs2/LD6b SoC
|
"socionext,uniphier-pxs2-peri-reset" - for PXs2/LD6b SoC
|
||||||
"socionext,uniphier-ld11-peri-reset" - for LD11 SoC
|
"socionext,uniphier-ld11-peri-reset" - for LD11 SoC
|
||||||
"socionext,uniphier-ld20-peri-reset" - for LD20 SoC
|
"socionext,uniphier-ld20-peri-reset" - for LD20 SoC
|
||||||
|
"socionext,uniphier-pxs3-peri-reset" - for PXs3 SoC
|
||||||
- #reset-cells: should be 1.
|
- #reset-cells: should be 1.
|
||||||
|
|
||||||
Example:
|
Example:
|
||||||
|
|||||||
@@ -65,8 +65,8 @@ to the respective BMan instance
|
|||||||
BMan Private Memory Node
|
BMan Private Memory Node
|
||||||
|
|
||||||
BMan requires a contiguous range of physical memory used for the backing store
|
BMan requires a contiguous range of physical memory used for the backing store
|
||||||
for BMan Free Buffer Proxy Records (FBPR). This memory is reserved/allocated as a
|
for BMan Free Buffer Proxy Records (FBPR). This memory is reserved/allocated as
|
||||||
node under the /reserved-memory node
|
a node under the /reserved-memory node.
|
||||||
|
|
||||||
The BMan FBPR memory node must be named "bman-fbpr"
|
The BMan FBPR memory node must be named "bman-fbpr"
|
||||||
|
|
||||||
@@ -75,7 +75,9 @@ PROPERTIES
|
|||||||
- compatible
|
- compatible
|
||||||
Usage: required
|
Usage: required
|
||||||
Value type: <stringlist>
|
Value type: <stringlist>
|
||||||
Definition: Must inclide "fsl,bman-fbpr"
|
Definition: PPC platforms: Must include "fsl,bman-fbpr"
|
||||||
|
ARM platforms: Must include "shared-dma-pool"
|
||||||
|
as well as the "no-map" property
|
||||||
|
|
||||||
The following constraints are relevant to the FBPR private memory:
|
The following constraints are relevant to the FBPR private memory:
|
||||||
- The size must be 2^(size + 1), with size = 11..33. That is 4 KiB to
|
- The size must be 2^(size + 1), with size = 11..33. That is 4 KiB to
|
||||||
@@ -100,10 +102,10 @@ The example below shows a BMan FBPR dynamic allocation memory node
|
|||||||
ranges;
|
ranges;
|
||||||
|
|
||||||
bman_fbpr: bman-fbpr {
|
bman_fbpr: bman-fbpr {
|
||||||
compatible = "fsl,bman-fbpr";
|
compatible = "shared-mem-pool";
|
||||||
alloc-ranges = <0 0 0x10 0>;
|
|
||||||
size = <0 0x1000000>;
|
size = <0 0x1000000>;
|
||||||
alignment = <0 0x1000000>;
|
alignment = <0 0x1000000>;
|
||||||
|
no-map;
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -60,6 +60,12 @@ are located at offsets 0xbf8 and 0xbfc
|
|||||||
Value type: <prop-encoded-array>
|
Value type: <prop-encoded-array>
|
||||||
Definition: Reference input clock. Its frequency is half of the
|
Definition: Reference input clock. Its frequency is half of the
|
||||||
platform clock
|
platform clock
|
||||||
|
- memory-regions
|
||||||
|
Usage: Required for ARM
|
||||||
|
Value type: <phandle array>
|
||||||
|
Definition: List of phandles referencing the QMan private memory
|
||||||
|
nodes (described below). The qman-fqd node must be
|
||||||
|
first followed by qman-pfdr node. Only used on ARM
|
||||||
|
|
||||||
Devices connected to a QMan instance via Direct Connect Portals (DCP) must link
|
Devices connected to a QMan instance via Direct Connect Portals (DCP) must link
|
||||||
to the respective QMan instance
|
to the respective QMan instance
|
||||||
@@ -74,7 +80,9 @@ QMan Private Memory Nodes
|
|||||||
|
|
||||||
QMan requires two contiguous range of physical memory used for the backing store
|
QMan requires two contiguous range of physical memory used for the backing store
|
||||||
for QMan Frame Queue Descriptor (FQD) and Packed Frame Descriptor Record (PFDR).
|
for QMan Frame Queue Descriptor (FQD) and Packed Frame Descriptor Record (PFDR).
|
||||||
This memory is reserved/allocated as a nodes under the /reserved-memory node
|
This memory is reserved/allocated as a node under the /reserved-memory node.
|
||||||
|
|
||||||
|
For additional details about reserved memory regions see reserved-memory.txt
|
||||||
|
|
||||||
The QMan FQD memory node must be named "qman-fqd"
|
The QMan FQD memory node must be named "qman-fqd"
|
||||||
|
|
||||||
@@ -83,7 +91,9 @@ PROPERTIES
|
|||||||
- compatible
|
- compatible
|
||||||
Usage: required
|
Usage: required
|
||||||
Value type: <stringlist>
|
Value type: <stringlist>
|
||||||
Definition: Must inclide "fsl,qman-fqd"
|
Definition: PPC platforms: Must include "fsl,qman-fqd"
|
||||||
|
ARM platforms: Must include "shared-dma-pool"
|
||||||
|
as well as the "no-map" property
|
||||||
|
|
||||||
The QMan PFDR memory node must be named "qman-pfdr"
|
The QMan PFDR memory node must be named "qman-pfdr"
|
||||||
|
|
||||||
@@ -92,7 +102,9 @@ PROPERTIES
|
|||||||
- compatible
|
- compatible
|
||||||
Usage: required
|
Usage: required
|
||||||
Value type: <stringlist>
|
Value type: <stringlist>
|
||||||
Definition: Must inclide "fsl,qman-pfdr"
|
Definition: PPC platforms: Must include "fsl,qman-pfdr"
|
||||||
|
ARM platforms: Must include "shared-dma-pool"
|
||||||
|
as well as the "no-map" property
|
||||||
|
|
||||||
The following constraints are relevant to the FQD and PFDR private memory:
|
The following constraints are relevant to the FQD and PFDR private memory:
|
||||||
- The size must be 2^(size + 1), with size = 11..29. That is 4 KiB to
|
- The size must be 2^(size + 1), with size = 11..29. That is 4 KiB to
|
||||||
@@ -117,16 +129,16 @@ The example below shows a QMan FQD and a PFDR dynamic allocation memory nodes
|
|||||||
ranges;
|
ranges;
|
||||||
|
|
||||||
qman_fqd: qman-fqd {
|
qman_fqd: qman-fqd {
|
||||||
compatible = "fsl,qman-fqd";
|
compatible = "shared-dma-pool";
|
||||||
alloc-ranges = <0 0 0x10 0>;
|
|
||||||
size = <0 0x400000>;
|
size = <0 0x400000>;
|
||||||
alignment = <0 0x400000>;
|
alignment = <0 0x400000>;
|
||||||
|
no-map;
|
||||||
};
|
};
|
||||||
qman_pfdr: qman-pfdr {
|
qman_pfdr: qman-pfdr {
|
||||||
compatible = "fsl,qman-pfdr";
|
compatible = "shared-dma-pool";
|
||||||
alloc-ranges = <0 0 0x10 0>;
|
|
||||||
size = <0 0x2000000>;
|
size = <0 0x2000000>;
|
||||||
alignment = <0 0x2000000>;
|
alignment = <0 0x2000000>;
|
||||||
|
no-map;
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -19,6 +19,7 @@ IP Pairing
|
|||||||
Required properties in pwrap device node.
|
Required properties in pwrap device node.
|
||||||
- compatible:
|
- compatible:
|
||||||
"mediatek,mt2701-pwrap" for MT2701/7623 SoCs
|
"mediatek,mt2701-pwrap" for MT2701/7623 SoCs
|
||||||
|
"mediatek,mt7622-pwrap" for MT7622 SoCs
|
||||||
"mediatek,mt8135-pwrap" for MT8135 SoCs
|
"mediatek,mt8135-pwrap" for MT8135 SoCs
|
||||||
"mediatek,mt8173-pwrap" for MT8173 SoCs
|
"mediatek,mt8173-pwrap" for MT8173 SoCs
|
||||||
- interrupts: IRQ for pwrap in SOC
|
- interrupts: IRQ for pwrap in SOC
|
||||||
@@ -36,9 +37,12 @@ Required properties in pwrap device node.
|
|||||||
- clocks: Must contain an entry for each entry in clock-names.
|
- clocks: Must contain an entry for each entry in clock-names.
|
||||||
|
|
||||||
Optional properities:
|
Optional properities:
|
||||||
- pmic: Mediatek PMIC MFD is the child device of pwrap
|
- pmic: Using either MediaTek PMIC MFD as the child device of pwrap
|
||||||
See the following for child node definitions:
|
See the following for child node definitions:
|
||||||
Documentation/devicetree/bindings/mfd/mt6397.txt
|
Documentation/devicetree/bindings/mfd/mt6397.txt
|
||||||
|
or the regulator-only device as the child device of pwrap, such as MT6380.
|
||||||
|
See the following definitions for such kinds of devices.
|
||||||
|
Documentation/devicetree/bindings/regulator/mt6380-regulator.txt
|
||||||
|
|
||||||
Example:
|
Example:
|
||||||
pwrap: pwrap@1000f000 {
|
pwrap: pwrap@1000f000 {
|
||||||
|
|||||||
+19
-1
@@ -1219,6 +1219,8 @@ L: linux-arm-kernel@lists.infradead.org (moderated for non-subscribers)
|
|||||||
W: http://www.linux4sam.org
|
W: http://www.linux4sam.org
|
||||||
T: git git://git.kernel.org/pub/scm/linux/kernel/git/nferre/linux-at91.git
|
T: git git://git.kernel.org/pub/scm/linux/kernel/git/nferre/linux-at91.git
|
||||||
S: Supported
|
S: Supported
|
||||||
|
N: at91
|
||||||
|
N: atmel
|
||||||
F: arch/arm/mach-at91/
|
F: arch/arm/mach-at91/
|
||||||
F: include/soc/at91/
|
F: include/soc/at91/
|
||||||
F: arch/arm/boot/dts/at91*.dts
|
F: arch/arm/boot/dts/at91*.dts
|
||||||
@@ -1227,6 +1229,9 @@ F: arch/arm/boot/dts/sama*.dts
|
|||||||
F: arch/arm/boot/dts/sama*.dtsi
|
F: arch/arm/boot/dts/sama*.dtsi
|
||||||
F: arch/arm/include/debug/at91.S
|
F: arch/arm/include/debug/at91.S
|
||||||
F: drivers/memory/atmel*
|
F: drivers/memory/atmel*
|
||||||
|
F: drivers/watchdog/sama5d4_wdt.c
|
||||||
|
X: drivers/input/touchscreen/atmel_mxt_ts.c
|
||||||
|
X: drivers/net/wireless/atmel/
|
||||||
|
|
||||||
ARM/CALXEDA HIGHBANK ARCHITECTURE
|
ARM/CALXEDA HIGHBANK ARCHITECTURE
|
||||||
M: Rob Herring <robh@kernel.org>
|
M: Rob Herring <robh@kernel.org>
|
||||||
@@ -2141,7 +2146,6 @@ F: drivers/gpio/gpio-zx.c
|
|||||||
F: drivers/i2c/busses/i2c-zx2967.c
|
F: drivers/i2c/busses/i2c-zx2967.c
|
||||||
F: drivers/mmc/host/dw_mmc-zx.*
|
F: drivers/mmc/host/dw_mmc-zx.*
|
||||||
F: drivers/pinctrl/zte/
|
F: drivers/pinctrl/zte/
|
||||||
F: drivers/reset/reset-zx2967.c
|
|
||||||
F: drivers/soc/zte/
|
F: drivers/soc/zte/
|
||||||
F: drivers/thermal/zx2967_thermal.c
|
F: drivers/thermal/zx2967_thermal.c
|
||||||
F: drivers/watchdog/zx2967_wdt.c
|
F: drivers/watchdog/zx2967_wdt.c
|
||||||
@@ -2990,6 +2994,14 @@ L: bcm-kernel-feedback-list@broadcom.com
|
|||||||
S: Maintained
|
S: Maintained
|
||||||
F: drivers/mtd/nand/brcmnand/
|
F: drivers/mtd/nand/brcmnand/
|
||||||
|
|
||||||
|
BROADCOM STB DPFE DRIVER
|
||||||
|
M: Markus Mayer <mmayer@broadcom.com>
|
||||||
|
M: bcm-kernel-feedback-list@broadcom.com
|
||||||
|
L: linux-arm-kernel@lists.infradead.org (moderated for non-subscribers)
|
||||||
|
S: Maintained
|
||||||
|
F: Documentation/devicetree/bindings/memory-controllers/brcm,dpfe-cpu.txt
|
||||||
|
F: drivers/memory/brcmstb_dpfe.c
|
||||||
|
|
||||||
BROADCOM SYSTEMPORT ETHERNET DRIVER
|
BROADCOM SYSTEMPORT ETHERNET DRIVER
|
||||||
M: Florian Fainelli <f.fainelli@gmail.com>
|
M: Florian Fainelli <f.fainelli@gmail.com>
|
||||||
L: netdev@vger.kernel.org
|
L: netdev@vger.kernel.org
|
||||||
@@ -13004,6 +13016,12 @@ F: arch/arc/plat-axs10x
|
|||||||
F: arch/arc/boot/dts/ax*
|
F: arch/arc/boot/dts/ax*
|
||||||
F: Documentation/devicetree/bindings/arc/axs10*
|
F: Documentation/devicetree/bindings/arc/axs10*
|
||||||
|
|
||||||
|
SYNOPSYS AXS10x RESET CONTROLLER DRIVER
|
||||||
|
M: Eugeniy Paltsev <Eugeniy.Paltsev@synopsys.com>
|
||||||
|
S: Supported
|
||||||
|
F: drivers/reset/reset-axs10x.c
|
||||||
|
F: Documentation/devicetree/bindings/reset/snps,axs10x-reset.txt
|
||||||
|
|
||||||
SYNOPSYS DESIGNWARE APB GPIO DRIVER
|
SYNOPSYS DESIGNWARE APB GPIO DRIVER
|
||||||
M: Hoan Tran <hotran@apm.com>
|
M: Hoan Tran <hotran@apm.com>
|
||||||
L: linux-gpio@vger.kernel.org
|
L: linux-gpio@vger.kernel.org
|
||||||
|
|||||||
@@ -54,12 +54,14 @@ static const struct of_device_id mtk_tz_smp_boot_infos[] __initconst = {
|
|||||||
{ .compatible = "mediatek,mt8135", .data = &mtk_mt8135_tz_boot },
|
{ .compatible = "mediatek,mt8135", .data = &mtk_mt8135_tz_boot },
|
||||||
{ .compatible = "mediatek,mt8127", .data = &mtk_mt8135_tz_boot },
|
{ .compatible = "mediatek,mt8127", .data = &mtk_mt8135_tz_boot },
|
||||||
{ .compatible = "mediatek,mt2701", .data = &mtk_mt8135_tz_boot },
|
{ .compatible = "mediatek,mt2701", .data = &mtk_mt8135_tz_boot },
|
||||||
|
{},
|
||||||
};
|
};
|
||||||
|
|
||||||
static const struct of_device_id mtk_smp_boot_infos[] __initconst = {
|
static const struct of_device_id mtk_smp_boot_infos[] __initconst = {
|
||||||
{ .compatible = "mediatek,mt6589", .data = &mtk_mt6589_boot },
|
{ .compatible = "mediatek,mt6589", .data = &mtk_mt6589_boot },
|
||||||
{ .compatible = "mediatek,mt7623", .data = &mtk_mt7623_boot },
|
{ .compatible = "mediatek,mt7623", .data = &mtk_mt7623_boot },
|
||||||
{ .compatible = "mediatek,mt7623a", .data = &mtk_mt7623_boot },
|
{ .compatible = "mediatek,mt7623a", .data = &mtk_mt7623_boot },
|
||||||
|
{},
|
||||||
};
|
};
|
||||||
|
|
||||||
static void __iomem *mtk_smp_base;
|
static void __iomem *mtk_smp_base;
|
||||||
|
|||||||
@@ -91,12 +91,13 @@ config ARCH_HISI
|
|||||||
This enables support for Hisilicon ARMv8 SoC family
|
This enables support for Hisilicon ARMv8 SoC family
|
||||||
|
|
||||||
config ARCH_MEDIATEK
|
config ARCH_MEDIATEK
|
||||||
bool "Mediatek MT65xx & MT81xx ARMv8 SoC"
|
bool "MediaTek SoC Family"
|
||||||
select ARM_GIC
|
select ARM_GIC
|
||||||
select PINCTRL
|
select PINCTRL
|
||||||
select MTK_TIMER
|
select MTK_TIMER
|
||||||
help
|
help
|
||||||
Support for Mediatek MT65xx & MT81xx ARMv8 SoCs
|
This enables support for MediaTek MT27xx, MT65xx, MT76xx
|
||||||
|
& MT81xx ARMv8 SoCs
|
||||||
|
|
||||||
config ARCH_MESON
|
config ARCH_MESON
|
||||||
bool "Amlogic Platforms"
|
bool "Amlogic Platforms"
|
||||||
|
|||||||
@@ -165,6 +165,14 @@ config TI_SYSC
|
|||||||
Generic driver for Texas Instruments interconnect target module
|
Generic driver for Texas Instruments interconnect target module
|
||||||
found on many TI SoCs.
|
found on many TI SoCs.
|
||||||
|
|
||||||
|
config TS_NBUS
|
||||||
|
tristate "Technologic Systems NBUS Driver"
|
||||||
|
depends on SOC_IMX28
|
||||||
|
depends on OF_GPIO && PWM
|
||||||
|
help
|
||||||
|
Driver for the Technologic Systems NBUS which is used to interface
|
||||||
|
with the peripherals in the FPGA of the TS-4600 SoM.
|
||||||
|
|
||||||
config UNIPHIER_SYSTEM_BUS
|
config UNIPHIER_SYSTEM_BUS
|
||||||
tristate "UniPhier System Bus driver"
|
tristate "UniPhier System Bus driver"
|
||||||
depends on ARCH_UNIPHIER && OF
|
depends on ARCH_UNIPHIER && OF
|
||||||
|
|||||||
@@ -22,6 +22,7 @@ obj-$(CONFIG_SIMPLE_PM_BUS) += simple-pm-bus.o
|
|||||||
obj-$(CONFIG_TEGRA_ACONNECT) += tegra-aconnect.o
|
obj-$(CONFIG_TEGRA_ACONNECT) += tegra-aconnect.o
|
||||||
obj-$(CONFIG_TEGRA_GMI) += tegra-gmi.o
|
obj-$(CONFIG_TEGRA_GMI) += tegra-gmi.o
|
||||||
obj-$(CONFIG_TI_SYSC) += ti-sysc.o
|
obj-$(CONFIG_TI_SYSC) += ti-sysc.o
|
||||||
|
obj-$(CONFIG_TS_NBUS) += ts-nbus.o
|
||||||
obj-$(CONFIG_UNIPHIER_SYSTEM_BUS) += uniphier-system-bus.o
|
obj-$(CONFIG_UNIPHIER_SYSTEM_BUS) += uniphier-system-bus.o
|
||||||
obj-$(CONFIG_VEXPRESS_CONFIG) += vexpress-config.o
|
obj-$(CONFIG_VEXPRESS_CONFIG) += vexpress-config.o
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,375 @@
|
|||||||
|
/*
|
||||||
|
* NBUS driver for TS-4600 based boards
|
||||||
|
*
|
||||||
|
* Copyright (c) 2016 - Savoir-faire Linux
|
||||||
|
* Author: Sebastien Bourdelin <sebastien.bourdelin@savoirfairelinux.com>
|
||||||
|
*
|
||||||
|
* This file is licensed under the terms of the GNU General Public
|
||||||
|
* License version 2. This program is licensed "as is" without any
|
||||||
|
* warranty of any kind, whether express or implied.
|
||||||
|
*
|
||||||
|
* This driver implements a GPIOs bit-banged bus, called the NBUS by Technologic
|
||||||
|
* Systems. It is used to communicate with the peripherals in the FPGA on the
|
||||||
|
* TS-4600 SoM.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <linux/bitops.h>
|
||||||
|
#include <linux/gpio/consumer.h>
|
||||||
|
#include <linux/kernel.h>
|
||||||
|
#include <linux/module.h>
|
||||||
|
#include <linux/mutex.h>
|
||||||
|
#include <linux/of_platform.h>
|
||||||
|
#include <linux/platform_device.h>
|
||||||
|
#include <linux/pwm.h>
|
||||||
|
#include <linux/ts-nbus.h>
|
||||||
|
|
||||||
|
#define TS_NBUS_DIRECTION_IN 0
|
||||||
|
#define TS_NBUS_DIRECTION_OUT 1
|
||||||
|
#define TS_NBUS_WRITE_ADR 0
|
||||||
|
#define TS_NBUS_WRITE_VAL 1
|
||||||
|
|
||||||
|
struct ts_nbus {
|
||||||
|
struct pwm_device *pwm;
|
||||||
|
struct gpio_descs *data;
|
||||||
|
struct gpio_desc *csn;
|
||||||
|
struct gpio_desc *txrx;
|
||||||
|
struct gpio_desc *strobe;
|
||||||
|
struct gpio_desc *ale;
|
||||||
|
struct gpio_desc *rdy;
|
||||||
|
struct mutex lock;
|
||||||
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
* request all gpios required by the bus.
|
||||||
|
*/
|
||||||
|
static int ts_nbus_init_pdata(struct platform_device *pdev, struct ts_nbus
|
||||||
|
*ts_nbus)
|
||||||
|
{
|
||||||
|
ts_nbus->data = devm_gpiod_get_array(&pdev->dev, "ts,data",
|
||||||
|
GPIOD_OUT_HIGH);
|
||||||
|
if (IS_ERR(ts_nbus->data)) {
|
||||||
|
dev_err(&pdev->dev, "failed to retrieve ts,data-gpio from dts\n");
|
||||||
|
return PTR_ERR(ts_nbus->data);
|
||||||
|
}
|
||||||
|
|
||||||
|
ts_nbus->csn = devm_gpiod_get(&pdev->dev, "ts,csn", GPIOD_OUT_HIGH);
|
||||||
|
if (IS_ERR(ts_nbus->csn)) {
|
||||||
|
dev_err(&pdev->dev, "failed to retrieve ts,csn-gpio from dts\n");
|
||||||
|
return PTR_ERR(ts_nbus->csn);
|
||||||
|
}
|
||||||
|
|
||||||
|
ts_nbus->txrx = devm_gpiod_get(&pdev->dev, "ts,txrx", GPIOD_OUT_HIGH);
|
||||||
|
if (IS_ERR(ts_nbus->txrx)) {
|
||||||
|
dev_err(&pdev->dev, "failed to retrieve ts,txrx-gpio from dts\n");
|
||||||
|
return PTR_ERR(ts_nbus->txrx);
|
||||||
|
}
|
||||||
|
|
||||||
|
ts_nbus->strobe = devm_gpiod_get(&pdev->dev, "ts,strobe", GPIOD_OUT_HIGH);
|
||||||
|
if (IS_ERR(ts_nbus->strobe)) {
|
||||||
|
dev_err(&pdev->dev, "failed to retrieve ts,strobe-gpio from dts\n");
|
||||||
|
return PTR_ERR(ts_nbus->strobe);
|
||||||
|
}
|
||||||
|
|
||||||
|
ts_nbus->ale = devm_gpiod_get(&pdev->dev, "ts,ale", GPIOD_OUT_HIGH);
|
||||||
|
if (IS_ERR(ts_nbus->ale)) {
|
||||||
|
dev_err(&pdev->dev, "failed to retrieve ts,ale-gpio from dts\n");
|
||||||
|
return PTR_ERR(ts_nbus->ale);
|
||||||
|
}
|
||||||
|
|
||||||
|
ts_nbus->rdy = devm_gpiod_get(&pdev->dev, "ts,rdy", GPIOD_IN);
|
||||||
|
if (IS_ERR(ts_nbus->rdy)) {
|
||||||
|
dev_err(&pdev->dev, "failed to retrieve ts,rdy-gpio from dts\n");
|
||||||
|
return PTR_ERR(ts_nbus->rdy);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* the data gpios are used for reading and writing values, their directions
|
||||||
|
* should be adjusted accordingly.
|
||||||
|
*/
|
||||||
|
static void ts_nbus_set_direction(struct ts_nbus *ts_nbus, int direction)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i < 8; i++) {
|
||||||
|
if (direction == TS_NBUS_DIRECTION_IN)
|
||||||
|
gpiod_direction_input(ts_nbus->data->desc[i]);
|
||||||
|
else
|
||||||
|
/* when used as output the default state of the data
|
||||||
|
* lines are set to high */
|
||||||
|
gpiod_direction_output(ts_nbus->data->desc[i], 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* reset the bus in its initial state.
|
||||||
|
* The data, csn, strobe and ale lines must be zero'ed to let the FPGA knows a
|
||||||
|
* new transaction can be process.
|
||||||
|
*/
|
||||||
|
static void ts_nbus_reset_bus(struct ts_nbus *ts_nbus)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
int values[8];
|
||||||
|
|
||||||
|
for (i = 0; i < 8; i++)
|
||||||
|
values[i] = 0;
|
||||||
|
|
||||||
|
gpiod_set_array_value_cansleep(8, ts_nbus->data->desc, values);
|
||||||
|
gpiod_set_value_cansleep(ts_nbus->csn, 0);
|
||||||
|
gpiod_set_value_cansleep(ts_nbus->strobe, 0);
|
||||||
|
gpiod_set_value_cansleep(ts_nbus->ale, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* let the FPGA knows it can process.
|
||||||
|
*/
|
||||||
|
static void ts_nbus_start_transaction(struct ts_nbus *ts_nbus)
|
||||||
|
{
|
||||||
|
gpiod_set_value_cansleep(ts_nbus->strobe, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* read a byte value from the data gpios.
|
||||||
|
* return 0 on success or negative errno on failure.
|
||||||
|
*/
|
||||||
|
static int ts_nbus_read_byte(struct ts_nbus *ts_nbus, u8 *val)
|
||||||
|
{
|
||||||
|
struct gpio_descs *gpios = ts_nbus->data;
|
||||||
|
int ret, i;
|
||||||
|
|
||||||
|
*val = 0;
|
||||||
|
for (i = 0; i < 8; i++) {
|
||||||
|
ret = gpiod_get_value_cansleep(gpios->desc[i]);
|
||||||
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
|
if (ret)
|
||||||
|
*val |= BIT(i);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* set the data gpios accordingly to the byte value.
|
||||||
|
*/
|
||||||
|
static void ts_nbus_write_byte(struct ts_nbus *ts_nbus, u8 byte)
|
||||||
|
{
|
||||||
|
struct gpio_descs *gpios = ts_nbus->data;
|
||||||
|
int i;
|
||||||
|
int values[8];
|
||||||
|
|
||||||
|
for (i = 0; i < 8; i++)
|
||||||
|
if (byte & BIT(i))
|
||||||
|
values[i] = 1;
|
||||||
|
else
|
||||||
|
values[i] = 0;
|
||||||
|
|
||||||
|
gpiod_set_array_value_cansleep(8, gpios->desc, values);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* reading the bus consists of resetting the bus, then notifying the FPGA to
|
||||||
|
* send the data in the data gpios and return the read value.
|
||||||
|
* return 0 on success or negative errno on failure.
|
||||||
|
*/
|
||||||
|
static int ts_nbus_read_bus(struct ts_nbus *ts_nbus, u8 *val)
|
||||||
|
{
|
||||||
|
ts_nbus_reset_bus(ts_nbus);
|
||||||
|
ts_nbus_start_transaction(ts_nbus);
|
||||||
|
|
||||||
|
return ts_nbus_read_byte(ts_nbus, val);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* writing to the bus consists of resetting the bus, then define the type of
|
||||||
|
* command (address/value), write the data and notify the FPGA to retrieve the
|
||||||
|
* value in the data gpios.
|
||||||
|
*/
|
||||||
|
static void ts_nbus_write_bus(struct ts_nbus *ts_nbus, int cmd, u8 val)
|
||||||
|
{
|
||||||
|
ts_nbus_reset_bus(ts_nbus);
|
||||||
|
|
||||||
|
if (cmd == TS_NBUS_WRITE_ADR)
|
||||||
|
gpiod_set_value_cansleep(ts_nbus->ale, 1);
|
||||||
|
|
||||||
|
ts_nbus_write_byte(ts_nbus, val);
|
||||||
|
ts_nbus_start_transaction(ts_nbus);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* read the value in the FPGA register at the given address.
|
||||||
|
* return 0 on success or negative errno on failure.
|
||||||
|
*/
|
||||||
|
int ts_nbus_read(struct ts_nbus *ts_nbus, u8 adr, u16 *val)
|
||||||
|
{
|
||||||
|
int ret, i;
|
||||||
|
u8 byte;
|
||||||
|
|
||||||
|
/* bus access must be atomic */
|
||||||
|
mutex_lock(&ts_nbus->lock);
|
||||||
|
|
||||||
|
/* set the bus in read mode */
|
||||||
|
gpiod_set_value_cansleep(ts_nbus->txrx, 0);
|
||||||
|
|
||||||
|
/* write address */
|
||||||
|
ts_nbus_write_bus(ts_nbus, TS_NBUS_WRITE_ADR, adr);
|
||||||
|
|
||||||
|
/* set the data gpios direction as input before reading */
|
||||||
|
ts_nbus_set_direction(ts_nbus, TS_NBUS_DIRECTION_IN);
|
||||||
|
|
||||||
|
/* reading value MSB first */
|
||||||
|
do {
|
||||||
|
*val = 0;
|
||||||
|
byte = 0;
|
||||||
|
for (i = 1; i >= 0; i--) {
|
||||||
|
/* read a byte from the bus, leave on error */
|
||||||
|
ret = ts_nbus_read_bus(ts_nbus, &byte);
|
||||||
|
if (ret < 0)
|
||||||
|
goto err;
|
||||||
|
|
||||||
|
/* append the byte read to the final value */
|
||||||
|
*val |= byte << (i * 8);
|
||||||
|
}
|
||||||
|
gpiod_set_value_cansleep(ts_nbus->csn, 1);
|
||||||
|
ret = gpiod_get_value_cansleep(ts_nbus->rdy);
|
||||||
|
} while (ret);
|
||||||
|
|
||||||
|
err:
|
||||||
|
/* restore the data gpios direction as output after reading */
|
||||||
|
ts_nbus_set_direction(ts_nbus, TS_NBUS_DIRECTION_OUT);
|
||||||
|
|
||||||
|
mutex_unlock(&ts_nbus->lock);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL_GPL(ts_nbus_read);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* write the desired value in the FPGA register at the given address.
|
||||||
|
*/
|
||||||
|
int ts_nbus_write(struct ts_nbus *ts_nbus, u8 adr, u16 val)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
/* bus access must be atomic */
|
||||||
|
mutex_lock(&ts_nbus->lock);
|
||||||
|
|
||||||
|
/* set the bus in write mode */
|
||||||
|
gpiod_set_value_cansleep(ts_nbus->txrx, 1);
|
||||||
|
|
||||||
|
/* write address */
|
||||||
|
ts_nbus_write_bus(ts_nbus, TS_NBUS_WRITE_ADR, adr);
|
||||||
|
|
||||||
|
/* writing value MSB first */
|
||||||
|
for (i = 1; i >= 0; i--)
|
||||||
|
ts_nbus_write_bus(ts_nbus, TS_NBUS_WRITE_VAL, (u8)(val >> (i * 8)));
|
||||||
|
|
||||||
|
/* wait for completion */
|
||||||
|
gpiod_set_value_cansleep(ts_nbus->csn, 1);
|
||||||
|
while (gpiod_get_value_cansleep(ts_nbus->rdy) != 0) {
|
||||||
|
gpiod_set_value_cansleep(ts_nbus->csn, 0);
|
||||||
|
gpiod_set_value_cansleep(ts_nbus->csn, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
mutex_unlock(&ts_nbus->lock);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL_GPL(ts_nbus_write);
|
||||||
|
|
||||||
|
static int ts_nbus_probe(struct platform_device *pdev)
|
||||||
|
{
|
||||||
|
struct pwm_device *pwm;
|
||||||
|
struct pwm_args pargs;
|
||||||
|
struct device *dev = &pdev->dev;
|
||||||
|
struct ts_nbus *ts_nbus;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
ts_nbus = devm_kzalloc(dev, sizeof(*ts_nbus), GFP_KERNEL);
|
||||||
|
if (!ts_nbus)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
mutex_init(&ts_nbus->lock);
|
||||||
|
|
||||||
|
ret = ts_nbus_init_pdata(pdev, ts_nbus);
|
||||||
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
pwm = devm_pwm_get(dev, NULL);
|
||||||
|
if (IS_ERR(pwm)) {
|
||||||
|
ret = PTR_ERR(pwm);
|
||||||
|
if (ret != -EPROBE_DEFER)
|
||||||
|
dev_err(dev, "unable to request PWM\n");
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
pwm_get_args(pwm, &pargs);
|
||||||
|
if (!pargs.period) {
|
||||||
|
dev_err(&pdev->dev, "invalid PWM period\n");
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* FIXME: pwm_apply_args() should be removed when switching to
|
||||||
|
* the atomic PWM API.
|
||||||
|
*/
|
||||||
|
pwm_apply_args(pwm);
|
||||||
|
ret = pwm_config(pwm, pargs.period, pargs.period);
|
||||||
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* we can now start the FPGA and populate the peripherals.
|
||||||
|
*/
|
||||||
|
pwm_enable(pwm);
|
||||||
|
ts_nbus->pwm = pwm;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* let the child nodes retrieve this instance of the ts-nbus.
|
||||||
|
*/
|
||||||
|
dev_set_drvdata(dev, ts_nbus);
|
||||||
|
|
||||||
|
ret = of_platform_populate(dev->of_node, NULL, NULL, dev);
|
||||||
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
dev_info(dev, "initialized\n");
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int ts_nbus_remove(struct platform_device *pdev)
|
||||||
|
{
|
||||||
|
struct ts_nbus *ts_nbus = dev_get_drvdata(&pdev->dev);
|
||||||
|
|
||||||
|
/* shutdown the FPGA */
|
||||||
|
mutex_lock(&ts_nbus->lock);
|
||||||
|
pwm_disable(ts_nbus->pwm);
|
||||||
|
mutex_unlock(&ts_nbus->lock);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const struct of_device_id ts_nbus_of_match[] = {
|
||||||
|
{ .compatible = "technologic,ts-nbus", },
|
||||||
|
{ },
|
||||||
|
};
|
||||||
|
MODULE_DEVICE_TABLE(of, ts_nbus_of_match);
|
||||||
|
|
||||||
|
static struct platform_driver ts_nbus_driver = {
|
||||||
|
.probe = ts_nbus_probe,
|
||||||
|
.remove = ts_nbus_remove,
|
||||||
|
.driver = {
|
||||||
|
.name = "ts_nbus",
|
||||||
|
.of_match_table = ts_nbus_of_match,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
module_platform_driver(ts_nbus_driver);
|
||||||
|
|
||||||
|
MODULE_ALIAS("platform:ts_nbus");
|
||||||
|
MODULE_AUTHOR("Sebastien Bourdelin <sebastien.bourdelin@savoirfairelinux.com>");
|
||||||
|
MODULE_DESCRIPTION("Technologic Systems NBUS");
|
||||||
|
MODULE_LICENSE("GPL v2");
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user