Commit d0a231f0 authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'pci-v5.17-changes' of git://git.kernel.org/pub/scm/linux/kernel/git/helgaas/pci

Pull pci updates from Bjorn Helgaas:
 "Enumeration:
   - Use pci_find_vsec_capability() instead of open-coding it (Andy
     Shevchenko)
   - Convert pci_dev_present() stub from macro to static inline to avoid
     'unused variable' errors (Hans de Goede)
   - Convert sysfs slot attributes from default_attrs to default_groups
     (Greg Kroah-Hartman)
   - Use DWORD accesses for LTR, L1 SS to avoid BayHub OZ711LV2 erratum
     (Rajat Jain)
   - Remove unnecessary initialization of static variables (Longji Guo)

  Resource management:
   - Always write Intel I210 ROM BAR on update to work around device
     defect (Bjorn Helgaas)

  PCIe native device hotplug:
   - Fix pciehp lockdep errors on Thunderbolt undock (Hans de Goede)
   - Fix infinite loop in pciehp IRQ handler on power fault (Lukas
     Wunner)

  Power management:
   - Convert amd64-agp, sis-agp, via-agp from legacy PCI power
     management to generic power management (Vaibhav Gupta)

  IOMMU:
   - Add function 1 DMA alias quirk for Marvell 88SE9125 SATA controller
     so it can work with an IOMMU (Yifeng Li)

  Error handling:
   - Add PCI_ERROR_RESPONSE and related definitions for signaling and
     checking for transaction errors on PCI (Naveen Naidu)
   - Fabricate PCI_ERROR_RESPONSE data (~0) in config read wrappers,
     instead of in host controller drivers, when transactions fail on
     PCI (Naveen Naidu)
   - Use PCI_POSSIBLE_ERROR() to check for possible failure of config
     reads (Naveen Naidu)

  Peer-to-peer DMA:
   - Add Logan Gunthorpe as P2PDMA maintainer (Bjorn Helgaas)

  ASPM:
   - Calculate link L0s and L1 exit latencies when needed instead of
     caching them (Saheed O. Bolarinwa)
   - Calculate device L0s and L1 acceptable exit latencies when needed
     instead of caching them (Saheed O. Bolarinwa)
   - Remove struct aspm_latency since it's no longer needed (Saheed O.
     Bolarinwa)

  APM X-Gene PCIe controller driver:
   - Fix IB window setup, which was broken by the fact that IB resources
     are now sorted in address order instead of DT dma-ranges order (Rob
     Herring)

  Apple PCIe controller driver:
   - Enable clock gating to save power (Hector Martin)
   - Fix REFCLK1 enable/poll logic (Hector Martin)

  Broadcom STB PCIe controller driver:
   - Declare bitmap correctly for use by bitmap interfaces (Christophe
     JAILLET)
   - Clean up computation of legacy and non-legacy MSI bitmasks (Florian
     Fainelli)
   - Update suspend/resume/remove error handling to warn about errors
     and not fail the operation (Jim Quinlan)
   - Correct the "pcie" and "msi" interrupt descriptions in DT binding
     (Jim Quinlan)
   - Add DT bindings for endpoint voltage regulators (Jim Quinlan)
   - Split brcm_pcie_setup() into two functions (Jim Quinlan)
   - Add mechanism for turning on voltage regulators for connected
     devices (Jim Quinlan)
   - Turn voltage regulators for connected devices on/off when bus is
     added or removed (Jim Quinlan)
   - When suspending, don't turn off voltage regulators for wakeup
     devices (Jim Quinlan)

  Freescale i.MX6 PCIe controller driver:
   - Add i.MX8MM support (Richard Zhu)

  Freescale Layerscape PCIe controller driver:
   - Use DWC common ops instead of layerscape-specific link-up functions
     (Hou Zhiqiang)

  Intel VMD host bridge driver:
   - Honor platform ACPI _OSC feature negotiation for Root Ports below
     VMD (Kai-Heng Feng)
   - Add support for Raptor Lake SKUs (Karthik L Gopalakrishnan)
   - Reset everything below VMD before enumerating to work around
     failure to enumerate NVMe devices when guest OS reboots (Nirmal
     Patel)

  Bridge emulation (used by Marvell Aardvark and MVEBU):
   - Make emulated ROM BAR read-only by default (Pali Rohár)
   - Make some emulated legacy PCI bits read-only for PCIe devices (Pali
     Rohár)
   - Update reserved bits in emulated PCIe Capability (Pali Rohár)
   - Allow drivers to emulate different PCIe Capability versions (Pali
     Rohár)
   - Set emulated Capabilities List bit for all PCIe devices, since they
     must have at least a PCIe Capability (Pali Rohár)

  Marvell Aardvark PCIe controller driver:
   - Add bridge emulation definitions for PCIe DEVCAP2, DEVCTL2,
     DEVSTA2, LNKCAP2, LNKCTL2, LNKSTA2, SLTCAP2, SLTCTL2, SLTSTA2 (Pali
     Rohár)
   - Add aardvark support for DEVCAP2, DEVCTL2, LNKCAP2 and LNKCTL2
     registers (Pali Rohár)
   - Clear all MSIs at setup to avoid spurious interrupts (Pali Rohár)
   - Disable bus mastering when unbinding host controller driver (Pali
     Rohár)
   - Mask all interrupts when unbinding host controller driver (Pali
     Rohár)
   - Fix memory leak in host controller unbind (Pali Rohár)
   - Assert PERST# when unbinding host controller driver (Pali Rohár)
   - Disable link training when unbinding host controller driver (Pali
     Rohár)
   - Disable common PHY when unbinding host controller driver (Pali
     Rohár)
   - Fix resource type checking to check only IORESOURCE_MEM, not
     IORESOURCE_MEM_64, which is a flavor of IORESOURCE_MEM (Pali Rohár)

  Marvell MVEBU PCIe controller driver:
   - Implement pci_remap_iospace() for ARM so mvebu can use
     devm_pci_remap_iospace() instead of the previous ARM-specific
     pci_ioremap_io() interface (Pali Rohár)
   - Use the standard pci_host_probe() instead of the device-specific
     mvebu_pci_host_probe() (Pali Rohár)
   - Replace all uses of ARM-specific pci_ioremap_io() with the ARM
     implementation of the standard pci_remap_iospace() interface and
     remove pci_ioremap_io() (Pali Rohár)
   - Skip initializing invalid Root Ports (Pali Rohár)
   - Check for errors from pci_bridge_emul_init() (Pali Rohár)
   - Ignore any bridges at non-zero function numbers (Pali Rohár)
   - Return ~0 data for invalid config read size (Pali Rohár)
   - Disallow mapping interrupts on emulated bridges (Pali Rohár)
   - Clear Root Port Memory & I/O Space Enable and Bus Master Enable at
     initialization (Pali Rohár)
   - Make type bits in Root Port I/O Base register read-only (Pali
     Rohár)
   - Disable Root Port windows when base/limit set to invalid values
     (Pali Rohár)
   - Set controller to Root Complex mode (Pali Rohár)
   - Set Root Port Class Code to PCI Bridge (Pali Rohár)
   - Update emulated Root Port secondary bus numbers to better reflect
     the actual topology (Pali Rohár)
   - Add PCI_BRIDGE_CTL_BUS_RESET support to emulated Root Ports so
     pci_reset_secondary_bus() can reset connected devices (Pali Rohár)
   - Add PCI_EXP_DEVCTL Error Reporting Enable support to emulated Root
     Ports (Pali Rohár)
   - Add PCI_EXP_RTSTA PME Status bit support to emulated Root Ports
     (Pali Rohár)
   - Add DEVCAP2, DEVCTL2 and LNKCTL2 support to emulated Root Ports on
     Armada XP and newer devices (Pali Rohár)
   - Export mvebu-mbus.c symbols to allow pci-mvebu.c to be a module
     (Pali Rohár)
   - Add support for compiling as a module (Pali Rohár)

  MediaTek PCIe controller driver:
   - Assert PERST# for 100ms to allow power and clock to stabilize
     (qizhong cheng)

  MediaTek PCIe Gen3 controller driver:
   - Disable Mediatek DVFSRC voltage request since lack of DVFSRC to
     respond to the request causes failure to exit L1 PM Substate
     (Jianjun Wang)

  MediaTek MT7621 PCIe controller driver:
   - Declare mt7621_pci_ops static (Sergio Paracuellos)
   - Give pcibios_root_bridge_prepare() access to host bridge windows
     (Sergio Paracuellos)
   - Move MIPS I/O coherency unit setup from driver to
     pcibios_root_bridge_prepare() (Sergio Paracuellos)
   - Add missing MODULE_LICENSE() (Sergio Paracuellos)
   - Allow COMPILE_TEST for all arches (Sergio Paracuellos)

  Microsoft Hyper-V host bridge driver:
   - Add hv-internal interfaces to encapsulate arch IRQ dependencies
     (Sunil Muthuswamy)
   - Add arm64 Hyper-V vPCI support (Sunil Muthuswamy)

  Qualcomm PCIe controller driver:
   - Undo PM setup in qcom_pcie_probe() error handling path (Christophe
     JAILLET)
   - Use __be16 type to store return value from cpu_to_be16()
     (Manivannan Sadhasivam)
   - Constify static dw_pcie_ep_ops (Rikard Falkeborn)

  Renesas R-Car PCIe controller driver:
   - Fix aarch32 abort handler so it doesn't check the wrong bus clock
     before accessing the host controller (Marek Vasut)

  TI Keystone PCIe controller driver:
   - Add register offset for ti,syscon-pcie-id and ti,syscon-pcie-mode
     DT properties (Kishon Vijay Abraham I)

  MicroSemi Switchtec management driver:
   - Add Gen4 automotive device IDs (Kelvin Cao)
   - Declare state_names[] as static so it's not allocated and
     initialized for every call (Kelvin Cao)

  Host controller driver cleanups:
   - Use of_device_get_match_data(), not of_match_device(), when we only
     need the device data in altera, artpec6, cadence, designware-plat,
     dra7xx, keystone, kirin (Fan Fei)
   - Drop pointless of_device_get_match_data() cast in j721e (Bjorn
     Helgaas)
   - Drop redundant struct device * from j721e since struct cdns_pcie
     already has one (Bjorn Helgaas)
   - Rename driver structs to *_pcie in intel-gw, iproc, ls-gen4,
     mediatek-gen3, microchip, mt7621, rcar-gen2, tegra194, uniphier,
     xgene, xilinx, xilinx-cpm for consistency across drivers (Fan Fei)
   - Fix invalid address space conversions in hisi, spear13xx (Bjorn
     Helgaas)

  Miscellaneous:
   - Sort Intel Device IDs by value (Andy Shevchenko)
   - Change Capability offsets to hex to match spec (Baruch Siach)
   - Correct misspellings (Krzysztof Wilczyński)
   - Terminate statement with semicolon in pci_endpoint_test.c (Ming
     Wang)"

* tag 'pci-v5.17-changes' of git://git.kernel.org/pub/scm/linux/kernel/git/helgaas/pci: (151 commits)
  PCI: mt7621: Allow COMPILE_TEST for all arches
  PCI: mt7621: Add missing MODULE_LICENSE()
  PCI: mt7621: Move MIPS setup to pcibios_root_bridge_prepare()
  PCI: Let pcibios_root_bridge_prepare() access bridge->windows
  PCI: mt7621: Declare mt7621_pci_ops static
  PCI: brcmstb: Do not turn off WOL regulators on suspend
  PCI: brcmstb: Add control of subdevice voltage regulators
  PCI: brcmstb: Add mechanism to turn on subdev regulators
  PCI: brcmstb: Split brcm_pcie_setup() into two funcs
  dt-bindings: PCI: Add bindings for Brcmstb EP voltage regulators
  dt-bindings: PCI: Correct brcmstb interrupts, interrupt-map.
  PCI: brcmstb: Fix function return value handling
  PCI: brcmstb: Do not use __GENMASK
  PCI: brcmstb: Declare 'used' as bitmap, not unsigned long
  PCI: hv: Add arm64 Hyper-V vPCI support
  PCI: hv: Make the code arch neutral by adding arch specific interfaces
  PCI: pciehp: Use down_read/write_nested(reset_lock) to fix lockdep errors
  x86/PCI: Remove initialization of static variables to false
  PCI: Use DWORD accesses for LTR, L1 SS to avoid erratum
  misc: pci_endpoint_test: Terminate statement with semicolon
  ...
parents 88db8458 87c71931
...@@ -146,11 +146,15 @@ examples: ...@@ -146,11 +146,15 @@ examples:
#address-cells = <3>; #address-cells = <3>;
#size-cells = <2>; #size-cells = <2>;
#interrupt-cells = <1>; #interrupt-cells = <1>;
interrupts = <GIC_SPI 148 IRQ_TYPE_LEVEL_HIGH>, interrupts = <GIC_SPI 147 IRQ_TYPE_LEVEL_HIGH>,
<GIC_SPI 148 IRQ_TYPE_LEVEL_HIGH>; <GIC_SPI 148 IRQ_TYPE_LEVEL_HIGH>;
interrupt-names = "pcie", "msi"; interrupt-names = "pcie", "msi";
interrupt-map-mask = <0x0 0x0 0x0 0x7>; interrupt-map-mask = <0x0 0x0 0x0 0x7>;
interrupt-map = <0 0 0 1 &gicv2 GIC_SPI 143 IRQ_TYPE_LEVEL_HIGH>; interrupt-map = <0 0 0 1 &gicv2 GIC_SPI 143 IRQ_TYPE_LEVEL_HIGH
0 0 0 2 &gicv2 GIC_SPI 144 IRQ_TYPE_LEVEL_HIGH
0 0 0 3 &gicv2 GIC_SPI 145 IRQ_TYPE_LEVEL_HIGH
0 0 0 4 &gicv2 GIC_SPI 146 IRQ_TYPE_LEVEL_HIGH>;
msi-parent = <&pcie0>; msi-parent = <&pcie0>;
msi-controller; msi-controller;
ranges = <0x02000000 0x0 0xf8000000 0x6 0x00000000 0x0 0x04000000>; ranges = <0x02000000 0x0 0xf8000000 0x6 0x00000000 0x0 0x04000000>;
...@@ -158,5 +162,24 @@ examples: ...@@ -158,5 +162,24 @@ examples:
<0x42000000 0x1 0x80000000 0x3 0x00000000 0x0 0x80000000>; <0x42000000 0x1 0x80000000 0x3 0x00000000 0x0 0x80000000>;
brcm,enable-ssc; brcm,enable-ssc;
brcm,scb-sizes = <0x0000000080000000 0x0000000080000000>; brcm,scb-sizes = <0x0000000080000000 0x0000000080000000>;
/* PCIe bridge, Root Port */
pci@0,0 {
#address-cells = <3>;
#size-cells = <2>;
reg = <0x0 0x0 0x0 0x0 0x0>;
compatible = "pciclass,0604";
device_type = "pci";
vpcie3v3-supply = <&vreg7>;
ranges;
/* PCIe endpoint */
pci-ep@0,0 {
assigned-addresses =
<0x82010000 0x0 0xf8000000 0x6 0x00000000 0x0 0x2000>;
reg = <0x0 0x0 0x0 0x0 0x0>;
compatible = "pci14e4,1688";
};
};
}; };
}; };
...@@ -127,6 +127,12 @@ properties: ...@@ -127,6 +127,12 @@ properties:
enum: [1, 2, 3, 4] enum: [1, 2, 3, 4]
default: 1 default: 1
phys:
maxItems: 1
phy-names:
const: pcie-phy
reset-gpio: reset-gpio:
description: Should specify the GPIO for controlling the PCI bus device description: Should specify the GPIO for controlling the PCI bus device
reset signal. It's not polarity aware and defaults to active-low reset reset signal. It's not polarity aware and defaults to active-low reset
......
...@@ -32,8 +32,12 @@ properties: ...@@ -32,8 +32,12 @@ properties:
maxItems: 1 maxItems: 1
ti,syscon-pcie-mode: ti,syscon-pcie-mode:
$ref: /schemas/types.yaml#/definitions/phandle-array
items:
- items:
- description: Phandle to the SYSCON entry
- description: pcie_ctrl register offset within SYSCON
description: Phandle to the SYSCON entry required for configuring PCIe in RC or EP mode. description: Phandle to the SYSCON entry required for configuring PCIe in RC or EP mode.
$ref: /schemas/types.yaml#/definitions/phandle
interrupts: interrupts:
minItems: 1 minItems: 1
...@@ -65,7 +69,7 @@ examples: ...@@ -65,7 +69,7 @@ examples:
<0x5506000 0x1000>; <0x5506000 0x1000>;
reg-names = "app", "dbics", "addr_space", "atu"; reg-names = "app", "dbics", "addr_space", "atu";
power-domains = <&k3_pds 120 TI_SCI_PD_EXCLUSIVE>; power-domains = <&k3_pds 120 TI_SCI_PD_EXCLUSIVE>;
ti,syscon-pcie-mode = <&pcie0_mode>; ti,syscon-pcie-mode = <&scm_conf 0x4060>;
max-link-speed = <2>; max-link-speed = <2>;
dma-coherent; dma-coherent;
interrupts = <GIC_SPI 340 IRQ_TYPE_EDGE_RISING>; interrupts = <GIC_SPI 340 IRQ_TYPE_EDGE_RISING>;
......
...@@ -36,12 +36,20 @@ properties: ...@@ -36,12 +36,20 @@ properties:
maxItems: 1 maxItems: 1
ti,syscon-pcie-id: ti,syscon-pcie-id:
$ref: /schemas/types.yaml#/definitions/phandle-array
items:
- items:
- description: Phandle to the SYSCON entry
- description: pcie_device_id register offset within SYSCON
description: Phandle to the SYSCON entry required for getting PCIe device/vendor ID description: Phandle to the SYSCON entry required for getting PCIe device/vendor ID
$ref: /schemas/types.yaml#/definitions/phandle
ti,syscon-pcie-mode: ti,syscon-pcie-mode:
$ref: /schemas/types.yaml#/definitions/phandle-array
items:
- items:
- description: Phandle to the SYSCON entry
- description: pcie_ctrl register offset within SYSCON
description: Phandle to the SYSCON entry required for configuring PCIe in RC or EP mode. description: Phandle to the SYSCON entry required for configuring PCIe in RC or EP mode.
$ref: /schemas/types.yaml#/definitions/phandle
msi-map: true msi-map: true
...@@ -87,8 +95,8 @@ examples: ...@@ -87,8 +95,8 @@ examples:
#size-cells = <2>; #size-cells = <2>;
ranges = <0x81000000 0 0 0x10020000 0 0x00010000>, ranges = <0x81000000 0 0 0x10020000 0 0x00010000>,
<0x82000000 0 0x10030000 0x10030000 0 0x07FD0000>; <0x82000000 0 0x10030000 0x10030000 0 0x07FD0000>;
ti,syscon-pcie-id = <&pcie_devid>; ti,syscon-pcie-id = <&scm_conf 0x0210>;
ti,syscon-pcie-mode = <&pcie0_mode>; ti,syscon-pcie-mode = <&scm_conf 0x4060>;
bus-range = <0x0 0xff>; bus-range = <0x0 0xff>;
max-link-speed = <2>; max-link-speed = <2>;
dma-coherent; dma-coherent;
......
...@@ -14890,6 +14890,19 @@ L: linux-pci@vger.kernel.org ...@@ -14890,6 +14890,19 @@ L: linux-pci@vger.kernel.org
S: Supported S: Supported
F: Documentation/PCI/pci-error-recovery.rst F: Documentation/PCI/pci-error-recovery.rst
PCI PEER-TO-PEER DMA (P2PDMA)
M: Bjorn Helgaas <bhelgaas@google.com>
M: Logan Gunthorpe <logang@deltatee.com>
L: linux-pci@vger.kernel.org
S: Supported
Q: https://patchwork.kernel.org/project/linux-pci/list/
B: https://bugzilla.kernel.org
C: irc://irc.oftc.net/linux-pci
T: git git://git.kernel.org/pub/scm/linux/kernel/git/helgaas/pci.git
F: Documentation/driver-api/pci/p2pdma.rst
F: drivers/pci/p2pdma.c
F: include/linux/pci-p2pdma.h
PCI MSI DRIVER FOR ALTERA MSI IP PCI MSI DRIVER FOR ALTERA MSI IP
M: Joyce Ooi <joyce.ooi@intel.com> M: Joyce Ooi <joyce.ooi@intel.com>
L: linux-pci@vger.kernel.org L: linux-pci@vger.kernel.org
......
...@@ -180,7 +180,10 @@ void pci_ioremap_set_mem_type(int mem_type); ...@@ -180,7 +180,10 @@ void pci_ioremap_set_mem_type(int mem_type);
static inline void pci_ioremap_set_mem_type(int mem_type) {} static inline void pci_ioremap_set_mem_type(int mem_type) {}
#endif #endif
extern int pci_ioremap_io(unsigned int offset, phys_addr_t phys_addr); struct resource;
#define pci_remap_iospace pci_remap_iospace
int pci_remap_iospace(const struct resource *res, phys_addr_t phys_addr);
/* /*
* PCI configuration space mapping function. * PCI configuration space mapping function.
......
...@@ -38,6 +38,7 @@ static int num_pcie_ports; ...@@ -38,6 +38,7 @@ static int num_pcie_ports;
static int __init dove_pcie_setup(int nr, struct pci_sys_data *sys) static int __init dove_pcie_setup(int nr, struct pci_sys_data *sys)
{ {
struct pcie_port *pp; struct pcie_port *pp;
struct resource realio;
if (nr >= num_pcie_ports) if (nr >= num_pcie_ports)
return 0; return 0;
...@@ -53,10 +54,10 @@ static int __init dove_pcie_setup(int nr, struct pci_sys_data *sys) ...@@ -53,10 +54,10 @@ static int __init dove_pcie_setup(int nr, struct pci_sys_data *sys)
orion_pcie_setup(pp->base); orion_pcie_setup(pp->base);
if (pp->index == 0) realio.start = sys->busnr * SZ_64K;
pci_ioremap_io(sys->busnr * SZ_64K, DOVE_PCIE0_IO_PHYS_BASE); realio.end = realio.start + SZ_64K - 1;
else pci_remap_iospace(&realio, pp->index == 0 ? DOVE_PCIE0_IO_PHYS_BASE :
pci_ioremap_io(sys->busnr * SZ_64K, DOVE_PCIE1_IO_PHYS_BASE); DOVE_PCIE1_IO_PHYS_BASE);
/* /*
* IORESOURCE_MEM * IORESOURCE_MEM
......
...@@ -185,6 +185,7 @@ iop3xx_pci_abort(unsigned long addr, unsigned int fsr, struct pt_regs *regs) ...@@ -185,6 +185,7 @@ iop3xx_pci_abort(unsigned long addr, unsigned int fsr, struct pt_regs *regs)
int iop3xx_pci_setup(int nr, struct pci_sys_data *sys) int iop3xx_pci_setup(int nr, struct pci_sys_data *sys)
{ {
struct resource *res; struct resource *res;
struct resource realio;
if (nr != 0) if (nr != 0)
return 0; return 0;
...@@ -206,7 +207,9 @@ int iop3xx_pci_setup(int nr, struct pci_sys_data *sys) ...@@ -206,7 +207,9 @@ int iop3xx_pci_setup(int nr, struct pci_sys_data *sys)
pci_add_resource_offset(&sys->resources, res, sys->mem_offset); pci_add_resource_offset(&sys->resources, res, sys->mem_offset);
pci_ioremap_io(0, IOP3XX_PCI_LOWER_IO_PA); realio.start = 0;
realio.end = realio.start + SZ_64K - 1;
pci_remap_iospace(&realio, IOP3XX_PCI_LOWER_IO_PA);
return 1; return 1;
} }
......
...@@ -101,6 +101,7 @@ static void __init mv78xx0_pcie_preinit(void) ...@@ -101,6 +101,7 @@ static void __init mv78xx0_pcie_preinit(void)
static int __init mv78xx0_pcie_setup(int nr, struct pci_sys_data *sys) static int __init mv78xx0_pcie_setup(int nr, struct pci_sys_data *sys)
{ {
struct pcie_port *pp; struct pcie_port *pp;
struct resource realio;
if (nr >= num_pcie_ports) if (nr >= num_pcie_ports)
return 0; return 0;
...@@ -115,7 +116,9 @@ static int __init mv78xx0_pcie_setup(int nr, struct pci_sys_data *sys) ...@@ -115,7 +116,9 @@ static int __init mv78xx0_pcie_setup(int nr, struct pci_sys_data *sys)
orion_pcie_set_local_bus_nr(pp->base, sys->busnr); orion_pcie_set_local_bus_nr(pp->base, sys->busnr);
orion_pcie_setup(pp->base); orion_pcie_setup(pp->base);
pci_ioremap_io(nr * SZ_64K, MV78XX0_PCIE_IO_PHYS_BASE(nr)); realio.start = nr * SZ_64K;
realio.end = realio.start + SZ_64K - 1;
pci_remap_iospace(&realio, MV78XX0_PCIE_IO_PHYS_BASE(nr));
pci_add_resource_offset(&sys->resources, &pp->res, sys->mem_offset); pci_add_resource_offset(&sys->resources, &pp->res, sys->mem_offset);
......
...@@ -142,6 +142,7 @@ static struct pci_ops pcie_ops = { ...@@ -142,6 +142,7 @@ static struct pci_ops pcie_ops = {
static int __init pcie_setup(struct pci_sys_data *sys) static int __init pcie_setup(struct pci_sys_data *sys)
{ {
struct resource *res; struct resource *res;
struct resource realio;
int dev; int dev;
/* /*
...@@ -164,7 +165,9 @@ static int __init pcie_setup(struct pci_sys_data *sys) ...@@ -164,7 +165,9 @@ static int __init pcie_setup(struct pci_sys_data *sys)
pcie_ops.read = pcie_rd_conf_wa; pcie_ops.read = pcie_rd_conf_wa;
} }
pci_ioremap_io(sys->busnr * SZ_64K, ORION5X_PCIE_IO_PHYS_BASE); realio.start = sys->busnr * SZ_64K;
realio.end = realio.start + SZ_64K - 1;
pci_remap_iospace(&realio, ORION5X_PCIE_IO_PHYS_BASE);
/* /*
* Request resources. * Request resources.
...@@ -466,6 +469,7 @@ static void __init orion5x_setup_pci_wins(void) ...@@ -466,6 +469,7 @@ static void __init orion5x_setup_pci_wins(void)
static int __init pci_setup(struct pci_sys_data *sys) static int __init pci_setup(struct pci_sys_data *sys)
{ {
struct resource *res; struct resource *res;
struct resource realio;
/* /*
* Point PCI unit MBUS decode windows to DRAM space. * Point PCI unit MBUS decode windows to DRAM space.
...@@ -482,7 +486,9 @@ static int __init pci_setup(struct pci_sys_data *sys) ...@@ -482,7 +486,9 @@ static int __init pci_setup(struct pci_sys_data *sys)
*/ */
orion5x_setbits(PCI_CMD, PCI_CMD_HOST_REORDER); orion5x_setbits(PCI_CMD, PCI_CMD_HOST_REORDER);
pci_ioremap_io(sys->busnr * SZ_64K, ORION5X_PCI_IO_PHYS_BASE); realio.start = sys->busnr * SZ_64K;
realio.end = realio.start + SZ_64K - 1;
pci_remap_iospace(&realio, ORION5X_PCI_IO_PHYS_BASE);
/* /*
* Request resources * Request resources
......
...@@ -459,16 +459,20 @@ void pci_ioremap_set_mem_type(int mem_type) ...@@ -459,16 +459,20 @@ void pci_ioremap_set_mem_type(int mem_type)
pci_ioremap_mem_type = mem_type; pci_ioremap_mem_type = mem_type;
} }
int pci_ioremap_io(unsigned int offset, phys_addr_t phys_addr) int pci_remap_iospace(const struct resource *res, phys_addr_t phys_addr)
{ {
BUG_ON(offset + SZ_64K - 1 > IO_SPACE_LIMIT); unsigned long vaddr = (unsigned long)PCI_IOBASE + res->start;
return ioremap_page_range(PCI_IO_VIRT_BASE + offset, if (!(res->flags & IORESOURCE_IO))
PCI_IO_VIRT_BASE + offset + SZ_64K, return -EINVAL;
phys_addr,
if (res->end > IO_SPACE_LIMIT)
return -EINVAL;
return ioremap_page_range(vaddr, vaddr + resource_size(res), phys_addr,
__pgprot(get_mem_type(pci_ioremap_mem_type)->prot_pte)); __pgprot(get_mem_type(pci_ioremap_mem_type)->prot_pte));
} }
EXPORT_SYMBOL_GPL(pci_ioremap_io); EXPORT_SYMBOL(pci_remap_iospace);
void __iomem *pci_remap_cfgspace(resource_size_t res_cookie, size_t size) void __iomem *pci_remap_cfgspace(resource_size_t res_cookie, size_t size)
{ {
......
...@@ -64,6 +64,15 @@ ...@@ -64,6 +64,15 @@
#define HV_REGISTER_STIMER0_CONFIG 0x000B0000 #define HV_REGISTER_STIMER0_CONFIG 0x000B0000
#define HV_REGISTER_STIMER0_COUNT 0x000B0001 #define HV_REGISTER_STIMER0_COUNT 0x000B0001
union hv_msi_entry {
u64 as_uint64[2];
struct {
u64 address;
u32 data;
u32 reserved;
} __packed;
};
#include <asm-generic/hyperv-tlfs.h> #include <asm-generic/hyperv-tlfs.h>
#endif #endif
...@@ -10,6 +10,8 @@ ...@@ -10,6 +10,8 @@
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/sys_soc.h> #include <linux/sys_soc.h>
#include <linux/memblock.h> #include <linux/memblock.h>
#include <linux/pci.h>
#include <linux/bug.h>
#include <asm/bootinfo.h> #include <asm/bootinfo.h>
#include <asm/mipsregs.h> #include <asm/mipsregs.h>
...@@ -22,6 +24,35 @@ ...@@ -22,6 +24,35 @@
static void *detect_magic __initdata = detect_memory_region; static void *detect_magic __initdata = detect_memory_region;
int pcibios_root_bridge_prepare(struct pci_host_bridge *bridge)
{
struct resource_entry *entry;
resource_size_t mask;
entry = resource_list_first_type(&bridge->windows, IORESOURCE_MEM);
if (!entry) {
pr_err("Cannot get memory resource\n");
return -EINVAL;
}
if (mips_cps_numiocu(0)) {
/*
* Hardware doesn't accept mask values with 1s after
* 0s (e.g. 0xffef), so warn if that's happen
*/
mask = ~(entry->res->end - entry->res->start) & CM_GCR_REGn_MASK_ADDRMASK;
WARN_ON(mask && BIT(ffz(~mask)) - 1 != ~mask);
write_gcr_reg1_base(entry->res->start);
write_gcr_reg1_mask(mask | CM_GCR_REGn_MASK_CMTGT_IOCU0);
pr_info("PCI coherence region base: 0x%08llx, mask/settings: 0x%08llx\n",
(unsigned long long)read_gcr_reg1_base(),
(unsigned long long)read_gcr_reg1_mask());
}
return 0;
}
phys_addr_t mips_cpc_default_phys_base(void) phys_addr_t mips_cpc_default_phys_base(void)
{ {
panic("Cannot detect cpc address"); panic("Cannot detect cpc address");
......
...@@ -602,6 +602,39 @@ enum hv_interrupt_type { ...@@ -602,6 +602,39 @@ enum hv_interrupt_type {
HV_X64_INTERRUPT_TYPE_MAXIMUM = 0x000A, HV_X64_INTERRUPT_TYPE_MAXIMUM = 0x000A,
}; };
union hv_msi_address_register {
u32 as_uint32;
struct {
u32 reserved1:2;
u32 destination_mode:1;
u32 redirection_hint:1;
u32 reserved2:8;
u32 destination_id:8;
u32 msi_base:12;
};
} __packed;
union hv_msi_data_register {
u32 as_uint32;
struct {
u32 vector:8;
u32 delivery_mode:3;
u32 reserved1:3;
u32 level_assert:1;
u32 trigger_mode:1;
u32 reserved2:16;
};
} __packed;
/* HvRetargetDeviceInterrupt hypercall */
union hv_msi_entry {
u64 as_uint64;
struct {
union hv_msi_address_register address;
union hv_msi_data_register data;
} __packed;
};
#include <asm-generic/hyperv-tlfs.h> #include <asm-generic/hyperv-tlfs.h>
#endif #endif
...@@ -169,13 +169,6 @@ bool hv_vcpu_is_preempted(int vcpu); ...@@ -169,13 +169,6 @@ bool hv_vcpu_is_preempted(int vcpu);
static inline void hv_apic_init(void) {} static inline void hv_apic_init(void) {}
#endif #endif
static inline void hv_set_msi_entry_from_desc(union hv_msi_entry *msi_entry,
struct msi_desc *msi_desc)
{
msi_entry->address.as_uint32 = msi_desc->msg.address_lo;
msi_entry->data.as_uint32 = msi_desc->msg.data;
}
struct irq_domain *hv_create_pci_msi_domain(void); struct irq_domain *hv_create_pci_msi_domain(void);
int hv_map_ioapic_interrupt(int ioapic_id, bool level, int vcpu, int vector, int hv_map_ioapic_interrupt(int ioapic_id, bool level, int vcpu, int vector,
......
...@@ -20,7 +20,7 @@ struct pci_root_info { ...@@ -20,7 +20,7 @@ struct pci_root_info {
}; };
static bool pci_use_crs = true; static bool pci_use_crs = true;
static bool pci_ignore_seg = false; static bool pci_ignore_seg;
static int __init set_use_crs(const struct dmi_system_id *id) static int __init set_use_crs(const struct dmi_system_id *id)
{ {
......
...@@ -914,6 +914,7 @@ int mvebu_mbus_add_window_remap_by_id(unsigned int target, ...@@ -914,6 +914,7 @@ int mvebu_mbus_add_window_remap_by_id(unsigned int target,
return mvebu_mbus_alloc_window(s, base, size, remap, target, attribute); return mvebu_mbus_alloc_window(s, base, size, remap, target, attribute);
} }
EXPORT_SYMBOL_GPL(mvebu_mbus_add_window_remap_by_id);
int mvebu_mbus_add_window_by_id(unsigned int target, unsigned int attribute, int mvebu_mbus_add_window_by_id(unsigned int target, unsigned int attribute,
phys_addr_t base, size_t size) phys_addr_t base, size_t size)
...@@ -921,6 +922,7 @@ int mvebu_mbus_add_window_by_id(unsigned int target, unsigned int attribute, ...@@ -921,6 +922,7 @@ int mvebu_mbus_add_window_by_id(unsigned int target, unsigned int attribute,
return mvebu_mbus_add_window_remap_by_id(target, attribute, base, return mvebu_mbus_add_window_remap_by_id(target, attribute, base,
size, MVEBU_MBUS_NO_REMAP); size, MVEBU_MBUS_NO_REMAP);
} }
EXPORT_SYMBOL_GPL(mvebu_mbus_add_window_by_id);
int mvebu_mbus_del_window(phys_addr_t base, size_t size) int mvebu_mbus_del_window(phys_addr_t base, size_t size)
{ {
...@@ -933,6 +935,7 @@ int mvebu_mbus_del_window(phys_addr_t base, size_t size) ...@@ -933,6 +935,7 @@ int mvebu_mbus_del_window(phys_addr_t base, size_t size)
mvebu_mbus_disable_window(&mbus_state, win); mvebu_mbus_disable_window(&mbus_state, win);
return 0; return 0;
} }
EXPORT_SYMBOL_GPL(mvebu_mbus_del_window);
void mvebu_mbus_get_pcie_mem_aperture(struct resource *res) void mvebu_mbus_get_pcie_mem_aperture(struct resource *res)
{ {
...@@ -940,6 +943,7 @@ void mvebu_mbus_get_pcie_mem_aperture(struct resource *res) ...@@ -940,6 +943,7 @@ void mvebu_mbus_get_pcie_mem_aperture(struct resource *res)
return; return;
*res = mbus_state.pcie_mem_aperture; *res = mbus_state.pcie_mem_aperture;
} }
EXPORT_SYMBOL_GPL(mvebu_mbus_get_pcie_mem_aperture);
void mvebu_mbus_get_pcie_io_aperture(struct resource *res) void mvebu_mbus_get_pcie_io_aperture(struct resource *res)
{ {
...@@ -947,6 +951,7 @@ void mvebu_mbus_get_pcie_io_aperture(struct resource *res) ...@@ -947,6 +951,7 @@ void mvebu_mbus_get_pcie_io_aperture(struct resource *res)
return; return;
*res = mbus_state.pcie_io_aperture; *res = mbus_state.pcie_io_aperture;
} }
EXPORT_SYMBOL_GPL(mvebu_mbus_get_pcie_io_aperture);
int mvebu_mbus_get_dram_win_info(phys_addr_t phyaddr, u8 *target, u8 *attr) int mvebu_mbus_get_dram_win_info(phys_addr_t phyaddr, u8 *target, u8 *attr)
{ {
......
...@@ -588,20 +588,11 @@ static void agp_amd64_remove(struct pci_dev *pdev) ...@@ -588,20 +588,11 @@ static void agp_amd64_remove(struct pci_dev *pdev)
agp_bridges_found--; agp_bridges_found--;
} }
#ifdef CONFIG_PM #define agp_amd64_suspend NULL
static int agp_amd64_suspend(struct pci_dev *pdev, pm_message_t state) static int __maybe_unused agp_amd64_resume(struct device *dev)
{ {
pci_save_state(pdev); struct pci_dev *pdev = to_pci_dev(dev);
pci_set_power_state(pdev, pci_choose_state(pdev, state));
return 0;
}
static int agp_amd64_resume(struct pci_dev *pdev)
{
pci_set_power_state(pdev, PCI_D0);
pci_restore_state(pdev);
if (pdev->vendor == PCI_VENDOR_ID_NVIDIA) if (pdev->vendor == PCI_VENDOR_ID_NVIDIA)
nforce3_agp_init(pdev); nforce3_agp_init(pdev);
...@@ -609,8 +600,6 @@ static int agp_amd64_resume(struct pci_dev *pdev) ...@@ -609,8 +600,6 @@ static int agp_amd64_resume(struct pci_dev *pdev)
return amd_8151_configure(); return amd_8151_configure();
} }
#endif /* CONFIG_PM */
static const struct pci_device_id agp_amd64_pci_table[] = { static const struct pci_device_id agp_amd64_pci_table[] = {
{ {
.class = (PCI_CLASS_BRIDGE_HOST << 8), .class = (PCI_CLASS_BRIDGE_HOST << 8),
...@@ -738,15 +727,14 @@ static const struct pci_device_id agp_amd64_pci_promisc_table[] = { ...@@ -738,15 +727,14 @@ static const struct pci_device_id agp_amd64_pci_promisc_table[] = {
{ } { }
}; };
static SIMPLE_DEV_PM_OPS(agp_amd64_pm_ops, agp_amd64_suspend, agp_amd64_resume);
static struct pci_driver agp_amd64_pci_driver = { static struct pci_driver agp_amd64_pci_driver = {
.name = "agpgart-amd64", .name = "agpgart-amd64",
.id_table = agp_amd64_pci_table, .id_table = agp_amd64_pci_table,
.probe = agp_amd64_probe, .probe = agp_amd64_probe,
.remove = agp_amd64_remove, .remove = agp_amd64_remove,
#ifdef CONFIG_PM .driver.pm = &agp_amd64_pm_ops,
.suspend = agp_amd64_suspend,
.resume = agp_amd64_resume,
#endif
}; };
......
...@@ -217,26 +217,14 @@ static void agp_sis_remove(struct pci_dev *pdev) ...@@ -217,26 +217,14 @@ static void agp_sis_remove(struct pci_dev *pdev)
agp_put_bridge(bridge); agp_put_bridge(bridge);
} }
#ifdef CONFIG_PM #define agp_sis_suspend NULL
static int agp_sis_suspend(struct pci_dev *pdev, pm_message_t state) static int __maybe_unused agp_sis_resume(
__attribute__((unused)) struct device *dev)
{ {
pci_save_state(pdev);
pci_set_power_state(pdev, pci_choose_state(pdev, state));
return 0;
}
static int agp_sis_resume(struct pci_dev *pdev)
{
pci_set_power_state(pdev, PCI_D0);
pci_restore_state(pdev);
return sis_driver.configure(); return sis_driver.configure();
} }
#endif /* CONFIG_PM */
static const struct pci_device_id agp_sis_pci_table[] = { static const struct pci_device_id agp_sis_pci_table[] = {
{ {
.class = (PCI_CLASS_BRIDGE_HOST << 8), .class = (PCI_CLASS_BRIDGE_HOST << 8),
...@@ -419,15 +407,14 @@ static const struct pci_device_id agp_sis_pci_table[] = { ...@@ -419,15 +407,14 @@ static const struct pci_device_id agp_sis_pci_table[] = {
MODULE_DEVICE_TABLE(pci, agp_sis_pci_table); MODULE_DEVICE_TABLE(pci, agp_sis_pci_table);
static SIMPLE_DEV_PM_OPS(agp_sis_pm_ops, agp_sis_suspend, agp_sis_resume);
static struct pci_driver agp_sis_pci_driver = { static struct pci_driver agp_sis_pci_driver = {
.name = "agpgart-sis", .name = "agpgart-sis",
.id_table = agp_sis_pci_table, .id_table = agp_sis_pci_table,
.probe = agp_sis_probe, .probe = agp_sis_probe,
.remove = agp_sis_remove, .remove = agp_sis_remove,
#ifdef CONFIG_PM .driver.pm = &agp_sis_pm_ops,
.suspend = agp_sis_suspend,
.resume = agp_sis_resume,
#endif
}; };
static int __init agp_sis_init(void) static int __init agp_sis_init(void)
......
...@@ -492,22 +492,11 @@ static void agp_via_remove(struct pci_dev *pdev) ...@@ -492,22 +492,11 @@ static void agp_via_remove(struct pci_dev *pdev)
agp_put_bridge(bridge); agp_put_bridge(bridge);
} }
#ifdef CONFIG_PM #define agp_via_suspend NULL
static int agp_via_suspend(struct pci_dev *pdev, pm_message_t state) static int __maybe_unused agp_via_resume(struct device *dev)
{ {
pci_save_state (pdev); struct agp_bridge_data *bridge = dev_get_drvdata(dev);
pci_set_power_state (pdev, PCI_D3hot);
return 0;
}
static int agp_via_resume(struct pci_dev *pdev)
{
struct agp_bridge_data *bridge = pci_get_drvdata(pdev);
pci_set_power_state (pdev, PCI_D0);
pci_restore_state(pdev);
if (bridge->driver == &via_agp3_driver) if (bridge->driver == &via_agp3_driver)
return via_configure_agp3(); return via_configure_agp3();
...@@ -517,8 +506,6 @@ static int agp_via_resume(struct pci_dev *pdev) ...@@ -517,8 +506,6 @@ static int agp_via_resume(struct pci_dev *pdev)
return 0; return 0;
} }
#endif /* CONFIG_PM */
/* must be the same order as name table above */ /* must be the same order as name table above */
static const struct pci_device_id agp_via_pci_table[] = { static const struct pci_device_id agp_via_pci_table[] = {
#define ID(x) \ #define ID(x) \
...@@ -567,16 +554,14 @@ static const struct pci_device_id agp_via_pci_table[] = { ...@@ -567,16 +554,14 @@ static const struct pci_device_id agp_via_pci_table[] = {
MODULE_DEVICE_TABLE(pci, agp_via_pci_table); MODULE_DEVICE_TABLE(pci, agp_via_pci_table);
static SIMPLE_DEV_PM_OPS(agp_via_pm_ops, agp_via_suspend, agp_via_resume);
static struct pci_driver agp_via_pci_driver = { static struct pci_driver agp_via_pci_driver = {
.name = "agpgart-via", .name = "agpgart-via",
.id_table = agp_via_pci_table, .id_table = agp_via_pci_table,
.probe = agp_via_probe, .probe = agp_via_probe,
.remove = agp_via_remove, .remove = agp_via_remove,
#ifdef CONFIG_PM .driver.pm = &agp_via_pm_ops,
.suspend = agp_via_suspend,
.resume = agp_via_resume,
#endif
}; };
......
...@@ -865,7 +865,7 @@ static int pci_endpoint_test_probe(struct pci_dev *pdev, ...@@ -865,7 +865,7 @@ static int pci_endpoint_test_probe(struct pci_dev *pdev,
goto err_release_irq; goto err_release_irq;
} }
misc_device->parent = &pdev->dev; misc_device->parent = &pdev->dev;
misc_device->fops = &pci_endpoint_test_fops, misc_device->fops = &pci_endpoint_test_fops;
err = misc_register(misc_device); err = misc_register(misc_device);
if (err) { if (err) {
......
...@@ -184,7 +184,7 @@ config PCI_LABEL ...@@ -184,7 +184,7 @@ config PCI_LABEL
config PCI_HYPERV config PCI_HYPERV
tristate "Hyper-V PCI Frontend" tristate "Hyper-V PCI Frontend"
depends on X86_64 && HYPERV && PCI_MSI && PCI_MSI_IRQ_DOMAIN && SYSFS depends on ((X86 && X86_64) || ARM64) && HYPERV && PCI_MSI && PCI_MSI_IRQ_DOMAIN && SYSFS
select PCI_HYPERV_INTERFACE select PCI_HYPERV_INTERFACE
help help
The PCI device frontend driver allows the kernel to import arbitrary The PCI device frontend driver allows the kernel to import arbitrary
......
...@@ -42,6 +42,9 @@ int noinline pci_bus_read_config_##size \ ...@@ -42,6 +42,9 @@ int noinline pci_bus_read_config_##size \
if (PCI_##size##_BAD) return PCIBIOS_BAD_REGISTER_NUMBER; \ if (PCI_##size##_BAD) return PCIBIOS_BAD_REGISTER_NUMBER; \
pci_lock_config(flags); \ pci_lock_config(flags); \
res = bus->ops->read(bus, devfn, pos, len, &data); \ res = bus->ops->read(bus, devfn, pos, len, &data); \
if (res) \
PCI_SET_ERROR_RESPONSE(value); \
else \
*value = (type)data; \ *value = (type)data; \
pci_unlock_config(flags); \ pci_unlock_config(flags); \
return res; \ return res; \
...@@ -80,10 +83,8 @@ int pci_generic_config_read(struct pci_bus *bus, unsigned int devfn, ...@@ -80,10 +83,8 @@ int pci_generic_config_read(struct pci_bus *bus, unsigned int devfn,
void __iomem *addr; void __iomem *addr;
addr = bus->ops->map_bus(bus, devfn, where); addr = bus->ops->map_bus(bus, devfn, where);
if (!addr) { if (!addr)
*val = ~0;
return PCIBIOS_DEVICE_NOT_FOUND; return PCIBIOS_DEVICE_NOT_FOUND;
}
if (size == 1) if (size == 1)
*val = readb(addr); *val = readb(addr);
...@@ -122,10 +123,8 @@ int pci_generic_config_read32(struct pci_bus *bus, unsigned int devfn, ...@@ -122,10 +123,8 @@ int pci_generic_config_read32(struct pci_bus *bus, unsigned int devfn,
void __iomem *addr; void __iomem *addr;
addr = bus->ops->map_bus(bus, devfn, where & ~0x3); addr = bus->ops->map_bus(bus, devfn, where & ~0x3);
if (!addr) { if (!addr)
*val = ~0;
return PCIBIOS_DEVICE_NOT_FOUND; return PCIBIOS_DEVICE_NOT_FOUND;
}
*val = readl(addr); *val = readl(addr);
...@@ -228,6 +227,9 @@ int pci_user_read_config_##size \ ...@@ -228,6 +227,9 @@ int pci_user_read_config_##size \
ret = dev->bus->ops->read(dev->bus, dev->devfn, \ ret = dev->bus->ops->read(dev->bus, dev->devfn, \
pos, sizeof(type), &data); \ pos, sizeof(type), &data); \
raw_spin_unlock_irq(&pci_lock); \ raw_spin_unlock_irq(&pci_lock); \
if (ret) \
PCI_SET_ERROR_RESPONSE(val); \
else \
*val = (type)data; \ *val = (type)data; \
return pcibios_err_to_errno(ret); \ return pcibios_err_to_errno(ret); \
} \ } \
...@@ -410,9 +412,9 @@ int pcie_capability_read_word(struct pci_dev *dev, int pos, u16 *val) ...@@ -410,9 +412,9 @@ int pcie_capability_read_word(struct pci_dev *dev, int pos, u16 *val)
if (pcie_capability_reg_implemented(dev, pos)) { if (pcie_capability_reg_implemented(dev, pos)) {
ret = pci_read_config_word(dev, pci_pcie_cap(dev) + pos, val); ret = pci_read_config_word(dev, pci_pcie_cap(dev) + pos, val);
/* /*
* Reset *val to 0 if pci_read_config_word() fails, it may * Reset *val to 0 if pci_read_config_word() fails; it may
* have been written as 0xFFFF if hardware error happens * have been written as 0xFFFF (PCI_ERROR_RESPONSE) if the
* during pci_read_config_word(). * config read failed on PCI.
*/ */
if (ret) if (ret)
*val = 0; *val = 0;
...@@ -445,9 +447,9 @@ int pcie_capability_read_dword(struct pci_dev *dev, int pos, u32 *val) ...@@ -445,9 +447,9 @@ int pcie_capability_read_dword(struct pci_dev *dev, int pos, u32 *val)
if (pcie_capability_reg_implemented(dev, pos)) { if (pcie_capability_reg_implemented(dev, pos)) {
ret = pci_read_config_dword(dev, pci_pcie_cap(dev) + pos, val); ret = pci_read_config_dword(dev, pci_pcie_cap(dev) + pos, val);
/* /*
* Reset *val to 0 if pci_read_config_dword() fails, it may * Reset *val to 0 if pci_read_config_dword() fails; it may
* have been written as 0xFFFFFFFF if hardware error happens * have been written as 0xFFFFFFFF (PCI_ERROR_RESPONSE) if
* during pci_read_config_dword(). * the config read failed on PCI.
*/ */
if (ret) if (ret)
*val = 0; *val = 0;
...@@ -523,7 +525,7 @@ EXPORT_SYMBOL(pcie_capability_clear_and_set_dword); ...@@ -523,7 +525,7 @@ EXPORT_SYMBOL(pcie_capability_clear_and_set_dword);
int pci_read_config_byte(const struct pci_dev *dev, int where, u8 *val) int pci_read_config_byte(const struct pci_dev *dev, int where, u8 *val)
{ {
if (pci_dev_is_disconnected(dev)) { if (pci_dev_is_disconnected(dev)) {
*val = ~0; PCI_SET_ERROR_RESPONSE(val);
return PCIBIOS_DEVICE_NOT_FOUND; return PCIBIOS_DEVICE_NOT_FOUND;
} }
return pci_bus_read_config_byte(dev->bus, dev->devfn, where, val); return pci_bus_read_config_byte(dev->bus, dev->devfn, where, val);
...@@ -533,7 +535,7 @@ EXPORT_SYMBOL(pci_read_config_byte); ...@@ -533,7 +535,7 @@ EXPORT_SYMBOL(pci_read_config_byte);
int pci_read_config_word(const struct pci_dev *dev, int where, u16 *val) int pci_read_config_word(const struct pci_dev *dev, int where, u16 *val)
{ {
if (pci_dev_is_disconnected(dev)) { if (pci_dev_is_disconnected(dev)) {
*val = ~0; PCI_SET_ERROR_RESPONSE(val);
return PCIBIOS_DEVICE_NOT_FOUND; return PCIBIOS_DEVICE_NOT_FOUND;
} }
return pci_bus_read_config_word(dev->bus, dev->devfn, where, val); return pci_bus_read_config_word(dev->bus, dev->devfn, where, val);
...@@ -544,7 +546,7 @@ int pci_read_config_dword(const struct pci_dev *dev, int where, ...@@ -544,7 +546,7 @@ int pci_read_config_dword(const struct pci_dev *dev, int where,
u32 *val) u32 *val)
{ {
if (pci_dev_is_disconnected(dev)) { if (pci_dev_is_disconnected(dev)) {
*val = ~0; PCI_SET_ERROR_RESPONSE(val);
return PCIBIOS_DEVICE_NOT_FOUND; return PCIBIOS_DEVICE_NOT_FOUND;
} }
return pci_bus_read_config_dword(dev->bus, dev->devfn, where, val); return pci_bus_read_config_dword(dev->bus, dev->devfn, where, val);
......
...@@ -4,7 +4,7 @@ menu "PCI controller drivers" ...@@ -4,7 +4,7 @@ menu "PCI controller drivers"
depends on PCI depends on PCI
config PCI_MVEBU config PCI_MVEBU
bool "Marvell EBU PCIe controller" tristate "Marvell EBU PCIe controller"
depends on ARCH_MVEBU || ARCH_DOVE || COMPILE_TEST depends on ARCH_MVEBU || ARCH_DOVE || COMPILE_TEST
depends on MVEBU_MBUS depends on MVEBU_MBUS
depends on ARM depends on ARM
...@@ -281,7 +281,7 @@ config PCIE_BRCMSTB ...@@ -281,7 +281,7 @@ config PCIE_BRCMSTB
config PCI_HYPERV_INTERFACE config PCI_HYPERV_INTERFACE
tristate "Hyper-V PCI Interface" tristate "Hyper-V PCI Interface"
depends on X86 && HYPERV && PCI_MSI && PCI_MSI_IRQ_DOMAIN && X86_64 depends on ((X86 && X86_64) || ARM64) && HYPERV && PCI_MSI && PCI_MSI_IRQ_DOMAIN
help help
The Hyper-V PCI Interface is a helper driver allows other drivers to The Hyper-V PCI Interface is a helper driver allows other drivers to
have a common interface with the Hyper-V PCI frontend driver. have a common interface with the Hyper-V PCI frontend driver.
...@@ -332,8 +332,8 @@ config PCIE_APPLE ...@@ -332,8 +332,8 @@ config PCIE_APPLE
If unsure, say Y if you have an Apple Silicon system. If unsure, say Y if you have an Apple Silicon system.
config PCIE_MT7621 config PCIE_MT7621
bool "MediaTek MT7621 PCIe Controller" tristate "MediaTek MT7621 PCIe Controller"
depends on SOC_MT7621 || (MIPS && COMPILE_TEST) depends on SOC_MT7621 || COMPILE_TEST
select PHY_MT7621_PCI select PHY_MT7621_PCI
default SOC_MT7621 default SOC_MT7621
help help
......
...@@ -51,11 +51,10 @@ enum link_status { ...@@ -51,11 +51,10 @@ enum link_status {
#define MAX_LANES 2 #define MAX_LANES 2
struct j721e_pcie { struct j721e_pcie {
struct device *dev; struct cdns_pcie *cdns_pcie;
struct clk *refclk; struct clk *refclk;
u32 mode; u32 mode;
u32 num_lanes; u32 num_lanes;
struct cdns_pcie *cdns_pcie;
void __iomem *user_cfg_base; void __iomem *user_cfg_base;
void __iomem *intd_cfg_base; void __iomem *intd_cfg_base;
u32 linkdown_irq_regfield; u32 linkdown_irq_regfield;
...@@ -99,7 +98,7 @@ static inline void j721e_pcie_intd_writel(struct j721e_pcie *pcie, u32 offset, ...@@ -99,7 +98,7 @@ static inline void j721e_pcie_intd_writel(struct j721e_pcie *pcie, u32 offset,
static irqreturn_t j721e_pcie_link_irq_handler(int irq, void *priv) static irqreturn_t j721e_pcie_link_irq_handler(int irq, void *priv)
{ {
struct j721e_pcie *pcie = priv; struct j721e_pcie *pcie = priv;
struct device *dev = pcie->dev; struct device *dev = pcie->cdns_pcie->dev;
u32 reg; u32 reg;
reg = j721e_pcie_intd_readl(pcie, STATUS_REG_SYS_2); reg = j721e_pcie_intd_readl(pcie, STATUS_REG_SYS_2);
...@@ -165,7 +164,7 @@ static const struct cdns_pcie_ops j721e_pcie_ops = { ...@@ -165,7 +164,7 @@ static const struct cdns_pcie_ops j721e_pcie_ops = {
static int j721e_pcie_set_mode(struct j721e_pcie *pcie, struct regmap *syscon, static int j721e_pcie_set_mode(struct j721e_pcie *pcie, struct regmap *syscon,
unsigned int offset) unsigned int offset)
{ {
struct device *dev = pcie->dev; struct device *dev = pcie->cdns_pcie->dev;
u32 mask = J721E_MODE_RC; u32 mask = J721E_MODE_RC;
u32 mode = pcie->mode; u32 mode = pcie->mode;
u32 val = 0; u32 val = 0;
...@@ -184,7 +183,7 @@ static int j721e_pcie_set_mode(struct j721e_pcie *pcie, struct regmap *syscon, ...@@ -184,7 +183,7 @@ static int j721e_pcie_set_mode(struct j721e_pcie *pcie, struct regmap *syscon,
static int j721e_pcie_set_link_speed(struct j721e_pcie *pcie, static int j721e_pcie_set_link_speed(struct j721e_pcie *pcie,
struct regmap *syscon, unsigned int offset) struct regmap *syscon, unsigned int offset)
{ {
struct device *dev = pcie->dev; struct device *dev = pcie->cdns_pcie->dev;
struct device_node *np = dev->of_node; struct device_node *np = dev->of_node;
int link_speed; int link_speed;
u32 val = 0; u32 val = 0;
...@@ -205,7 +204,7 @@ static int j721e_pcie_set_link_speed(struct j721e_pcie *pcie, ...@@ -205,7 +204,7 @@ static int j721e_pcie_set_link_speed(struct j721e_pcie *pcie,
static int j721e_pcie_set_lane_count(struct j721e_pcie *pcie, static int j721e_pcie_set_lane_count(struct j721e_pcie *pcie,
struct regmap *syscon, unsigned int offset) struct regmap *syscon, unsigned int offset)
{ {
struct device *dev = pcie->dev; struct device *dev = pcie->cdns_pcie->dev;
u32 lanes = pcie->num_lanes; u32 lanes = pcie->num_lanes;
u32 val = 0; u32 val = 0;
int ret; int ret;
...@@ -220,7 +219,7 @@ static int j721e_pcie_set_lane_count(struct j721e_pcie *pcie, ...@@ -220,7 +219,7 @@ static int j721e_pcie_set_lane_count(struct j721e_pcie *pcie,
static int j721e_pcie_ctrl_init(struct j721e_pcie *pcie) static int j721e_pcie_ctrl_init(struct j721e_pcie *pcie)
{ {
struct device *dev = pcie->dev; struct device *dev = pcie->cdns_pcie->dev;
struct device_node *node = dev->of_node; struct device_node *node = dev->of_node;
struct of_phandle_args args; struct of_phandle_args args;
unsigned int offset = 0; unsigned int offset = 0;
...@@ -354,7 +353,7 @@ static int j721e_pcie_probe(struct platform_device *pdev) ...@@ -354,7 +353,7 @@ static int j721e_pcie_probe(struct platform_device *pdev)
struct device *dev = &pdev->dev; struct device *dev = &pdev->dev;
struct device_node *node = dev->of_node; struct device_node *node = dev->of_node;
struct pci_host_bridge *bridge; struct pci_host_bridge *bridge;
struct j721e_pcie_data *data; const struct j721e_pcie_data *data;
struct cdns_pcie *cdns_pcie; struct cdns_pcie *cdns_pcie;
struct j721e_pcie *pcie; struct j721e_pcie *pcie;
struct cdns_pcie_rc *rc; struct cdns_pcie_rc *rc;
...@@ -367,7 +366,7 @@ static int j721e_pcie_probe(struct platform_device *pdev) ...@@ -367,7 +366,7 @@ static int j721e_pcie_probe(struct platform_device *pdev)
int ret; int ret;
int irq; int irq;
data = (struct j721e_pcie_data *)of_device_get_match_data(dev); data = of_device_get_match_data(dev);
if (!data) if (!data)
return -EINVAL; return -EINVAL;
...@@ -377,7 +376,6 @@ static int j721e_pcie_probe(struct platform_device *pdev) ...@@ -377,7 +376,6 @@ static int j721e_pcie_probe(struct platform_device *pdev)
if (!pcie) if (!pcie)
return -ENOMEM; return -ENOMEM;
pcie->dev = dev;
pcie->mode = mode; pcie->mode = mode;
pcie->linkdown_irq_regfield = data->linkdown_irq_regfield; pcie->linkdown_irq_regfield = data->linkdown_irq_regfield;
......
...@@ -45,7 +45,6 @@ static int cdns_plat_pcie_probe(struct platform_device *pdev) ...@@ -45,7 +45,6 @@ static int cdns_plat_pcie_probe(struct platform_device *pdev)
{ {
const struct cdns_plat_pcie_of_data *data; const struct cdns_plat_pcie_of_data *data;
struct cdns_plat_pcie *cdns_plat_pcie; struct cdns_plat_pcie *cdns_plat_pcie;
const struct of_device_id *match;
struct device *dev = &pdev->dev; struct device *dev = &pdev->dev;
struct pci_host_bridge *bridge; struct pci_host_bridge *bridge;
struct cdns_pcie_ep *ep; struct cdns_pcie_ep *ep;
...@@ -54,11 +53,10 @@ static int cdns_plat_pcie_probe(struct platform_device *pdev) ...@@ -54,11 +53,10 @@ static int cdns_plat_pcie_probe(struct platform_device *pdev)
bool is_rc; bool is_rc;
int ret; int ret;
match = of_match_device(cdns_plat_pcie_of_match, dev); data = of_device_get_match_data(dev);
if (!match) if (!data)
return -EINVAL; return -EINVAL;
data = (struct cdns_plat_pcie_of_data *)match->data;
is_rc = data->is_rc; is_rc = data->is_rc;
pr_debug(" Started %s with is_rc: %d\n", __func__, is_rc); pr_debug(" Started %s with is_rc: %d\n", __func__, is_rc);
......
...@@ -310,7 +310,7 @@ struct cdns_pcie { ...@@ -310,7 +310,7 @@ struct cdns_pcie {
* single function at a time * single function at a time
* @vendor_id: PCI vendor ID * @vendor_id: PCI vendor ID
* @device_id: PCI device ID * @device_id: PCI device ID
* @avail_ib_bar: Satus of RP_BAR0, RP_BAR1 and RP_NO_BAR if it's free or * @avail_ib_bar: Status of RP_BAR0, RP_BAR1 and RP_NO_BAR if it's free or
* available * available
* @quirk_retrain_flag: Retrain link as quirk for PCIe Gen2 * @quirk_retrain_flag: Retrain link as quirk for PCIe Gen2
* @quirk_detect_quiet_flag: LTSSM Detect Quiet min delay set as quirk * @quirk_detect_quiet_flag: LTSSM Detect Quiet min delay set as quirk
......
...@@ -697,16 +697,14 @@ static int dra7xx_pcie_probe(struct platform_device *pdev) ...@@ -697,16 +697,14 @@ static int dra7xx_pcie_probe(struct platform_device *pdev)
struct device_node *np = dev->of_node; struct device_node *np = dev->of_node;
char name[10]; char name[10];
struct gpio_desc *reset; struct gpio_desc *reset;
const struct of_device_id *match;
const struct dra7xx_pcie_of_data *data; const struct dra7xx_pcie_of_data *data;
enum dw_pcie_device_mode mode; enum dw_pcie_device_mode mode;
u32 b1co_mode_sel_mask; u32 b1co_mode_sel_mask;
match = of_match_device(of_match_ptr(of_dra7xx_pcie_match), dev); data = of_device_get_match_data(dev);
if (!match) if (!data)
return -EINVAL; return -EINVAL;
data = (struct dra7xx_pcie_of_data *)match->data;
mode = (enum dw_pcie_device_mode)data->mode; mode = (enum dw_pcie_device_mode)data->mode;
b1co_mode_sel_mask = data->b1co_mode_sel_mask; b1co_mode_sel_mask = data->b1co_mode_sel_mask;
......
...@@ -217,10 +217,8 @@ static int exynos_pcie_rd_own_conf(struct pci_bus *bus, unsigned int devfn, ...@@ -217,10 +217,8 @@ static int exynos_pcie_rd_own_conf(struct pci_bus *bus, unsigned int devfn,
{ {
struct dw_pcie *pci = to_dw_pcie_from_pp(bus->sysdata); struct dw_pcie *pci = to_dw_pcie_from_pp(bus->sysdata);
if (PCI_SLOT(devfn)) { if (PCI_SLOT(devfn))
*val = ~0;
return PCIBIOS_DEVICE_NOT_FOUND; return PCIBIOS_DEVICE_NOT_FOUND;
}
*val = dw_pcie_read_dbi(pci, where, size); *val = dw_pcie_read_dbi(pci, where, size);
return PCIBIOS_SUCCESSFUL; return PCIBIOS_SUCCESSFUL;
......
...@@ -29,6 +29,7 @@ ...@@ -29,6 +29,7 @@
#include <linux/types.h> #include <linux/types.h>
#include <linux/interrupt.h> #include <linux/interrupt.h>
#include <linux/reset.h> #include <linux/reset.h>
#include <linux/phy/phy.h>
#include <linux/pm_domain.h> #include <linux/pm_domain.h>
#include <linux/pm_runtime.h> #include <linux/pm_runtime.h>
...@@ -49,6 +50,7 @@ enum imx6_pcie_variants { ...@@ -49,6 +50,7 @@ enum imx6_pcie_variants {
IMX6QP, IMX6QP,
IMX7D, IMX7D,
IMX8MQ, IMX8MQ,
IMX8MM,
}; };
#define IMX6_PCIE_FLAG_IMX6_PHY BIT(0) #define IMX6_PCIE_FLAG_IMX6_PHY BIT(0)
...@@ -88,6 +90,7 @@ struct imx6_pcie { ...@@ -88,6 +90,7 @@ struct imx6_pcie {
struct device *pd_pcie; struct device *pd_pcie;
/* power domain for pcie phy */ /* power domain for pcie phy */
struct device *pd_pcie_phy; struct device *pd_pcie_phy;
struct phy *phy;
const struct imx6_pcie_drvdata *drvdata; const struct imx6_pcie_drvdata *drvdata;
}; };
...@@ -372,6 +375,8 @@ static void imx6_pcie_assert_core_reset(struct imx6_pcie *imx6_pcie) ...@@ -372,6 +375,8 @@ static void imx6_pcie_assert_core_reset(struct imx6_pcie *imx6_pcie)
case IMX7D: case IMX7D:
case IMX8MQ: case IMX8MQ:
reset_control_assert(imx6_pcie->pciephy_reset); reset_control_assert(imx6_pcie->pciephy_reset);
fallthrough;
case IMX8MM:
reset_control_assert(imx6_pcie->apps_reset); reset_control_assert(imx6_pcie->apps_reset);
break; break;
case IMX6SX: case IMX6SX:
...@@ -407,7 +412,8 @@ static void imx6_pcie_assert_core_reset(struct imx6_pcie *imx6_pcie) ...@@ -407,7 +412,8 @@ static void imx6_pcie_assert_core_reset(struct imx6_pcie *imx6_pcie)
static unsigned int imx6_pcie_grp_offset(const struct imx6_pcie *imx6_pcie) static unsigned int imx6_pcie_grp_offset(const struct imx6_pcie *imx6_pcie)
{ {
WARN_ON(imx6_pcie->drvdata->variant != IMX8MQ); WARN_ON(imx6_pcie->drvdata->variant != IMX8MQ &&
imx6_pcie->drvdata->variant != IMX8MM);
return imx6_pcie->controller_id == 1 ? IOMUXC_GPR16 : IOMUXC_GPR14; return imx6_pcie->controller_id == 1 ? IOMUXC_GPR16 : IOMUXC_GPR14;
} }
...@@ -446,6 +452,11 @@ static int imx6_pcie_enable_ref_clk(struct imx6_pcie *imx6_pcie) ...@@ -446,6 +452,11 @@ static int imx6_pcie_enable_ref_clk(struct imx6_pcie *imx6_pcie)
break; break;
case IMX7D: case IMX7D:
break; break;
case IMX8MM:
ret = clk_prepare_enable(imx6_pcie->pcie_aux);
if (ret)
dev_err(dev, "unable to enable pcie_aux clock\n");
break;
case IMX8MQ: case IMX8MQ:
ret = clk_prepare_enable(imx6_pcie->pcie_aux); ret = clk_prepare_enable(imx6_pcie->pcie_aux);
if (ret) { if (ret) {
...@@ -522,6 +533,14 @@ static void imx6_pcie_deassert_core_reset(struct imx6_pcie *imx6_pcie) ...@@ -522,6 +533,14 @@ static void imx6_pcie_deassert_core_reset(struct imx6_pcie *imx6_pcie)
goto err_ref_clk; goto err_ref_clk;
} }
switch (imx6_pcie->drvdata->variant) {
case IMX8MM:
if (phy_power_on(imx6_pcie->phy))
dev_err(dev, "unable to power on PHY\n");
break;
default:
break;
}
/* allow the clocks to stabilize */ /* allow the clocks to stabilize */
usleep_range(200, 500); usleep_range(200, 500);
...@@ -538,6 +557,10 @@ static void imx6_pcie_deassert_core_reset(struct imx6_pcie *imx6_pcie) ...@@ -538,6 +557,10 @@ static void imx6_pcie_deassert_core_reset(struct imx6_pcie *imx6_pcie)
case IMX8MQ: case IMX8MQ:
reset_control_deassert(imx6_pcie->pciephy_reset); reset_control_deassert(imx6_pcie->pciephy_reset);
break; break;
case IMX8MM:
if (phy_init(imx6_pcie->phy))
dev_err(dev, "waiting for phy ready timeout!\n");
break;
case IMX7D: case IMX7D:
reset_control_deassert(imx6_pcie->pciephy_reset); reset_control_deassert(imx6_pcie->pciephy_reset);
...@@ -614,6 +637,12 @@ static void imx6_pcie_configure_type(struct imx6_pcie *imx6_pcie) ...@@ -614,6 +637,12 @@ static void imx6_pcie_configure_type(struct imx6_pcie *imx6_pcie)
static void imx6_pcie_init_phy(struct imx6_pcie *imx6_pcie) static void imx6_pcie_init_phy(struct imx6_pcie *imx6_pcie)
{ {
switch (imx6_pcie->drvdata->variant) { switch (imx6_pcie->drvdata->variant) {
case IMX8MM:
/*
* The PHY initialization had been done in the PHY
* driver, break here directly.
*/
break;
case IMX8MQ: case IMX8MQ:
/* /*
* TODO: Currently this code assumes external * TODO: Currently this code assumes external
...@@ -753,6 +782,7 @@ static void imx6_pcie_ltssm_enable(struct device *dev) ...@@ -753,6 +782,7 @@ static void imx6_pcie_ltssm_enable(struct device *dev)
break; break;
case IMX7D: case IMX7D:
case IMX8MQ: case IMX8MQ:
case IMX8MM:
reset_control_deassert(imx6_pcie->apps_reset); reset_control_deassert(imx6_pcie->apps_reset);
break; break;
} }
...@@ -871,6 +901,7 @@ static void imx6_pcie_ltssm_disable(struct device *dev) ...@@ -871,6 +901,7 @@ static void imx6_pcie_ltssm_disable(struct device *dev)
IMX6Q_GPR12_PCIE_CTL_2, 0); IMX6Q_GPR12_PCIE_CTL_2, 0);
break; break;
case IMX7D: case IMX7D:
case IMX8MM:
reset_control_assert(imx6_pcie->apps_reset); reset_control_assert(imx6_pcie->apps_reset);
break; break;
default: default:
...@@ -930,6 +961,7 @@ static void imx6_pcie_clk_disable(struct imx6_pcie *imx6_pcie) ...@@ -930,6 +961,7 @@ static void imx6_pcie_clk_disable(struct imx6_pcie *imx6_pcie)
IMX7D_GPR12_PCIE_PHY_REFCLK_SEL); IMX7D_GPR12_PCIE_PHY_REFCLK_SEL);
break; break;
case IMX8MQ: case IMX8MQ:
case IMX8MM:
clk_disable_unprepare(imx6_pcie->pcie_aux); clk_disable_unprepare(imx6_pcie->pcie_aux);
break; break;
default: default:
...@@ -945,8 +977,16 @@ static int imx6_pcie_suspend_noirq(struct device *dev) ...@@ -945,8 +977,16 @@ static int imx6_pcie_suspend_noirq(struct device *dev)
return 0; return 0;
imx6_pcie_pm_turnoff(imx6_pcie); imx6_pcie_pm_turnoff(imx6_pcie);
imx6_pcie_clk_disable(imx6_pcie);
imx6_pcie_ltssm_disable(dev); imx6_pcie_ltssm_disable(dev);
imx6_pcie_clk_disable(imx6_pcie);
switch (imx6_pcie->drvdata->variant) {
case IMX8MM:
if (phy_power_off(imx6_pcie->phy))
dev_err(dev, "unable to power off PHY\n");
break;
default:
break;
}
return 0; return 0;
} }
...@@ -1043,11 +1083,6 @@ static int imx6_pcie_probe(struct platform_device *pdev) ...@@ -1043,11 +1083,6 @@ static int imx6_pcie_probe(struct platform_device *pdev)
} }
/* Fetch clocks */ /* Fetch clocks */
imx6_pcie->pcie_phy = devm_clk_get(dev, "pcie_phy");
if (IS_ERR(imx6_pcie->pcie_phy))
return dev_err_probe(dev, PTR_ERR(imx6_pcie->pcie_phy),
"pcie_phy clock source missing or invalid\n");
imx6_pcie->pcie_bus = devm_clk_get(dev, "pcie_bus"); imx6_pcie->pcie_bus = devm_clk_get(dev, "pcie_bus");
if (IS_ERR(imx6_pcie->pcie_bus)) if (IS_ERR(imx6_pcie->pcie_bus))
return dev_err_probe(dev, PTR_ERR(imx6_pcie->pcie_bus), return dev_err_probe(dev, PTR_ERR(imx6_pcie->pcie_bus),
...@@ -1089,10 +1124,35 @@ static int imx6_pcie_probe(struct platform_device *pdev) ...@@ -1089,10 +1124,35 @@ static int imx6_pcie_probe(struct platform_device *pdev)
dev_err(dev, "Failed to get PCIE APPS reset control\n"); dev_err(dev, "Failed to get PCIE APPS reset control\n");
return PTR_ERR(imx6_pcie->apps_reset); return PTR_ERR(imx6_pcie->apps_reset);
} }
break;
case IMX8MM:
imx6_pcie->pcie_aux = devm_clk_get(dev, "pcie_aux");
if (IS_ERR(imx6_pcie->pcie_aux))
return dev_err_probe(dev, PTR_ERR(imx6_pcie->pcie_aux),
"pcie_aux clock source missing or invalid\n");
imx6_pcie->apps_reset = devm_reset_control_get_exclusive(dev,
"apps");
if (IS_ERR(imx6_pcie->apps_reset))
return dev_err_probe(dev, PTR_ERR(imx6_pcie->apps_reset),
"failed to get pcie apps reset control\n");
imx6_pcie->phy = devm_phy_get(dev, "pcie-phy");
if (IS_ERR(imx6_pcie->phy))
return dev_err_probe(dev, PTR_ERR(imx6_pcie->phy),
"failed to get pcie phy\n");
break; break;
default: default:
break; break;
} }
/* Don't fetch the pcie_phy clock, if it has abstract PHY driver */
if (imx6_pcie->phy == NULL) {
imx6_pcie->pcie_phy = devm_clk_get(dev, "pcie_phy");
if (IS_ERR(imx6_pcie->pcie_phy))
return dev_err_probe(dev, PTR_ERR(imx6_pcie->pcie_phy),
"pcie_phy clock source missing or invalid\n");
}
/* Grab turnoff reset */ /* Grab turnoff reset */
imx6_pcie->turnoff_reset = devm_reset_control_get_optional_exclusive(dev, "turnoff"); imx6_pcie->turnoff_reset = devm_reset_control_get_optional_exclusive(dev, "turnoff");
...@@ -1202,6 +1262,10 @@ static const struct imx6_pcie_drvdata drvdata[] = { ...@@ -1202,6 +1262,10 @@ static const struct imx6_pcie_drvdata drvdata[] = {
[IMX8MQ] = { [IMX8MQ] = {
.variant = IMX8MQ, .variant = IMX8MQ,
}, },
[IMX8MM] = {
.variant = IMX8MM,
.flags = IMX6_PCIE_FLAG_SUPPORTS_SUSPEND,
},
}; };
static const struct of_device_id imx6_pcie_of_match[] = { static const struct of_device_id imx6_pcie_of_match[] = {
...@@ -1209,7 +1273,8 @@ static const struct of_device_id imx6_pcie_of_match[] = { ...@@ -1209,7 +1273,8 @@ static const struct of_device_id imx6_pcie_of_match[] = {
{ .compatible = "fsl,imx6sx-pcie", .data = &drvdata[IMX6SX], }, { .compatible = "fsl,imx6sx-pcie", .data = &drvdata[IMX6SX], },
{ .compatible = "fsl,imx6qp-pcie", .data = &drvdata[IMX6QP], }, { .compatible = "fsl,imx6qp-pcie", .data = &drvdata[IMX6QP], },
{ .compatible = "fsl,imx7d-pcie", .data = &drvdata[IMX7D], }, { .compatible = "fsl,imx7d-pcie", .data = &drvdata[IMX7D], },
{ .compatible = "fsl,imx8mq-pcie", .data = &drvdata[IMX8MQ], } , { .compatible = "fsl,imx8mq-pcie", .data = &drvdata[IMX8MQ], },
{ .compatible = "fsl,imx8mm-pcie", .data = &drvdata[IMX8MM], },
{}, {},
}; };
......
...@@ -747,9 +747,9 @@ static int ks_pcie_config_legacy_irq(struct keystone_pcie *ks_pcie) ...@@ -747,9 +747,9 @@ static int ks_pcie_config_legacy_irq(struct keystone_pcie *ks_pcie)
#ifdef CONFIG_ARM #ifdef CONFIG_ARM
/* /*
* When a PCI device does not exist during config cycles, keystone host gets a * When a PCI device does not exist during config cycles, keystone host
* bus error instead of returning 0xffffffff. This handler always returns 0 * gets a bus error instead of returning 0xffffffff (PCI_ERROR_RESPONSE).
* for this kind of faults. * This handler always returns 0 for this kind of fault.
*/ */
static int ks_pcie_fault(unsigned long addr, unsigned int fsr, static int ks_pcie_fault(unsigned long addr, unsigned int fsr,
struct pt_regs *regs) struct pt_regs *regs)
...@@ -775,12 +775,19 @@ static int __init ks_pcie_init_id(struct keystone_pcie *ks_pcie) ...@@ -775,12 +775,19 @@ static int __init ks_pcie_init_id(struct keystone_pcie *ks_pcie)
struct dw_pcie *pci = ks_pcie->pci; struct dw_pcie *pci = ks_pcie->pci;
struct device *dev = pci->dev; struct device *dev = pci->dev;
struct device_node *np = dev->of_node; struct device_node *np = dev->of_node;
struct of_phandle_args args;
unsigned int offset = 0;
devctrl_regs = syscon_regmap_lookup_by_phandle(np, "ti,syscon-pcie-id"); devctrl_regs = syscon_regmap_lookup_by_phandle(np, "ti,syscon-pcie-id");
if (IS_ERR(devctrl_regs)) if (IS_ERR(devctrl_regs))
return PTR_ERR(devctrl_regs); return PTR_ERR(devctrl_regs);
ret = regmap_read(devctrl_regs, 0, &id); /* Do not error out to maintain old DT compatibility */
ret = of_parse_phandle_with_fixed_args(np, "ti,syscon-pcie-id", 1, 0, &args);
if (!ret)
offset = args.args[0];
ret = regmap_read(devctrl_regs, offset, &id);
if (ret) if (ret)
return ret; return ret;
...@@ -989,6 +996,8 @@ static int ks_pcie_enable_phy(struct keystone_pcie *ks_pcie) ...@@ -989,6 +996,8 @@ static int ks_pcie_enable_phy(struct keystone_pcie *ks_pcie)
static int ks_pcie_set_mode(struct device *dev) static int ks_pcie_set_mode(struct device *dev)
{ {
struct device_node *np = dev->of_node; struct device_node *np = dev->of_node;
struct of_phandle_args args;
unsigned int offset = 0;
struct regmap *syscon; struct regmap *syscon;
u32 val; u32 val;
u32 mask; u32 mask;
...@@ -998,10 +1007,15 @@ static int ks_pcie_set_mode(struct device *dev) ...@@ -998,10 +1007,15 @@ static int ks_pcie_set_mode(struct device *dev)
if (IS_ERR(syscon)) if (IS_ERR(syscon))
return 0; return 0;
/* Do not error out to maintain old DT compatibility */
ret = of_parse_phandle_with_fixed_args(np, "ti,syscon-pcie-mode", 1, 0, &args);
if (!ret)
offset = args.args[0];
mask = KS_PCIE_DEV_TYPE_MASK | KS_PCIE_SYSCLOCKOUTEN; mask = KS_PCIE_DEV_TYPE_MASK | KS_PCIE_SYSCLOCKOUTEN;
val = KS_PCIE_DEV_TYPE(RC) | KS_PCIE_SYSCLOCKOUTEN; val = KS_PCIE_DEV_TYPE(RC) | KS_PCIE_SYSCLOCKOUTEN;
ret = regmap_update_bits(syscon, 0, mask, val); ret = regmap_update_bits(syscon, offset, mask, val);
if (ret) { if (ret) {
dev_err(dev, "failed to set pcie mode\n"); dev_err(dev, "failed to set pcie mode\n");
return ret; return ret;
...@@ -1014,6 +1028,8 @@ static int ks_pcie_am654_set_mode(struct device *dev, ...@@ -1014,6 +1028,8 @@ static int ks_pcie_am654_set_mode(struct device *dev,
enum dw_pcie_device_mode mode) enum dw_pcie_device_mode mode)
{ {
struct device_node *np = dev->of_node; struct device_node *np = dev->of_node;
struct of_phandle_args args;
unsigned int offset = 0;
struct regmap *syscon; struct regmap *syscon;
u32 val; u32 val;
u32 mask; u32 mask;
...@@ -1023,6 +1039,11 @@ static int ks_pcie_am654_set_mode(struct device *dev, ...@@ -1023,6 +1039,11 @@ static int ks_pcie_am654_set_mode(struct device *dev,
if (IS_ERR(syscon)) if (IS_ERR(syscon))
return 0; return 0;
/* Do not error out to maintain old DT compatibility */
ret = of_parse_phandle_with_fixed_args(np, "ti,syscon-pcie-mode", 1, 0, &args);
if (!ret)
offset = args.args[0];
mask = AM654_PCIE_DEV_TYPE_MASK; mask = AM654_PCIE_DEV_TYPE_MASK;
switch (mode) { switch (mode) {
...@@ -1037,7 +1058,7 @@ static int ks_pcie_am654_set_mode(struct device *dev, ...@@ -1037,7 +1058,7 @@ static int ks_pcie_am654_set_mode(struct device *dev,
return -EINVAL; return -EINVAL;
} }
ret = regmap_update_bits(syscon, 0, mask, val); ret = regmap_update_bits(syscon, offset, mask, val);
if (ret) { if (ret) {
dev_err(dev, "failed to set pcie mode\n"); dev_err(dev, "failed to set pcie mode\n");
return ret; return ret;
...@@ -1087,7 +1108,6 @@ static int __init ks_pcie_probe(struct platform_device *pdev) ...@@ -1087,7 +1108,6 @@ static int __init ks_pcie_probe(struct platform_device *pdev)
struct device *dev = &pdev->dev; struct device *dev = &pdev->dev;
struct device_node *np = dev->of_node; struct device_node *np = dev->of_node;
const struct ks_pcie_of_data *data; const struct ks_pcie_of_data *data;
const struct of_device_id *match;
enum dw_pcie_device_mode mode; enum dw_pcie_device_mode mode;
struct dw_pcie *pci; struct dw_pcie *pci;
struct keystone_pcie *ks_pcie; struct keystone_pcie *ks_pcie;
...@@ -1104,8 +1124,7 @@ static int __init ks_pcie_probe(struct platform_device *pdev) ...@@ -1104,8 +1124,7 @@ static int __init ks_pcie_probe(struct platform_device *pdev)
int irq; int irq;
int i; int i;
match = of_match_device(of_match_ptr(ks_pcie_of_match), dev); data = of_device_get_match_data(dev);
data = (struct ks_pcie_of_data *)match->data;
if (!data) if (!data)
return -EINVAL; return -EINVAL;
......
...@@ -3,6 +3,7 @@ ...@@ -3,6 +3,7 @@
* PCIe host controller driver for Freescale Layerscape SoCs * PCIe host controller driver for Freescale Layerscape SoCs
* *
* Copyright (C) 2014 Freescale Semiconductor. * Copyright (C) 2014 Freescale Semiconductor.
* Copyright 2021 NXP
* *
* Author: Minghuan Lian <Minghuan.Lian@freescale.com> * Author: Minghuan Lian <Minghuan.Lian@freescale.com>
*/ */
...@@ -22,12 +23,6 @@ ...@@ -22,12 +23,6 @@
#include "pcie-designware.h" #include "pcie-designware.h"
/* PEX1/2 Misc Ports Status Register */
#define SCFG_PEXMSCPORTSR(pex_idx) (0x94 + (pex_idx) * 4)
#define LTSSM_STATE_SHIFT 20
#define LTSSM_STATE_MASK 0x3f
#define LTSSM_PCIE_L0 0x11 /* L0 state */
/* PEX Internal Configuration Registers */ /* PEX Internal Configuration Registers */
#define PCIE_STRFMR1 0x71c /* Symbol Timer & Filter Mask Register1 */ #define PCIE_STRFMR1 0x71c /* Symbol Timer & Filter Mask Register1 */
#define PCIE_ABSERR 0x8d0 /* Bridge Slave Error Response Register */ #define PCIE_ABSERR 0x8d0 /* Bridge Slave Error Response Register */
...@@ -35,20 +30,8 @@ ...@@ -35,20 +30,8 @@
#define PCIE_IATU_NUM 6 #define PCIE_IATU_NUM 6
struct ls_pcie_drvdata {
u32 lut_offset;
u32 ltssm_shift;
u32 lut_dbg;
const struct dw_pcie_host_ops *ops;
const struct dw_pcie_ops *dw_pcie_ops;
};
struct ls_pcie { struct ls_pcie {
struct dw_pcie *pci; struct dw_pcie *pci;
void __iomem *lut;
struct regmap *scfg;
const struct ls_pcie_drvdata *drvdata;
int index;
}; };
#define to_ls_pcie(x) dev_get_drvdata((x)->dev) #define to_ls_pcie(x) dev_get_drvdata((x)->dev)
...@@ -83,38 +66,6 @@ static void ls_pcie_drop_msg_tlp(struct ls_pcie *pcie) ...@@ -83,38 +66,6 @@ static void ls_pcie_drop_msg_tlp(struct ls_pcie *pcie)
iowrite32(val, pci->dbi_base + PCIE_STRFMR1); iowrite32(val, pci->dbi_base + PCIE_STRFMR1);
} }
static int ls1021_pcie_link_up(struct dw_pcie *pci)
{
u32 state;
struct ls_pcie *pcie = to_ls_pcie(pci);
if (!pcie->scfg)
return 0;
regmap_read(pcie->scfg, SCFG_PEXMSCPORTSR(pcie->index), &state);
state = (state >> LTSSM_STATE_SHIFT) & LTSSM_STATE_MASK;
if (state < LTSSM_PCIE_L0)
return 0;
return 1;
}
static int ls_pcie_link_up(struct dw_pcie *pci)
{
struct ls_pcie *pcie = to_ls_pcie(pci);
u32 state;
state = (ioread32(pcie->lut + pcie->drvdata->lut_dbg) >>
pcie->drvdata->ltssm_shift) &
LTSSM_STATE_MASK;
if (state < LTSSM_PCIE_L0)
return 0;
return 1;
}
/* Forward error response of outbound non-posted requests */ /* Forward error response of outbound non-posted requests */
static void ls_pcie_fix_error_response(struct ls_pcie *pcie) static void ls_pcie_fix_error_response(struct ls_pcie *pcie)
{ {
...@@ -139,96 +90,20 @@ static int ls_pcie_host_init(struct pcie_port *pp) ...@@ -139,96 +90,20 @@ static int ls_pcie_host_init(struct pcie_port *pp)
return 0; return 0;
} }
static int ls1021_pcie_host_init(struct pcie_port *pp)
{
struct dw_pcie *pci = to_dw_pcie_from_pp(pp);
struct ls_pcie *pcie = to_ls_pcie(pci);
struct device *dev = pci->dev;
u32 index[2];
int ret;
pcie->scfg = syscon_regmap_lookup_by_phandle(dev->of_node,
"fsl,pcie-scfg");
if (IS_ERR(pcie->scfg)) {
ret = PTR_ERR(pcie->scfg);
dev_err(dev, "No syscfg phandle specified\n");
pcie->scfg = NULL;
return ret;
}
if (of_property_read_u32_array(dev->of_node,
"fsl,pcie-scfg", index, 2)) {
pcie->scfg = NULL;
return -EINVAL;
}
pcie->index = index[1];
return ls_pcie_host_init(pp);
}
static const struct dw_pcie_host_ops ls1021_pcie_host_ops = {
.host_init = ls1021_pcie_host_init,
};
static const struct dw_pcie_host_ops ls_pcie_host_ops = { static const struct dw_pcie_host_ops ls_pcie_host_ops = {
.host_init = ls_pcie_host_init, .host_init = ls_pcie_host_init,
}; };
static const struct dw_pcie_ops dw_ls1021_pcie_ops = {
.link_up = ls1021_pcie_link_up,
};
static const struct dw_pcie_ops dw_ls_pcie_ops = {
.link_up = ls_pcie_link_up,
};
static const struct ls_pcie_drvdata ls1021_drvdata = {
.ops = &ls1021_pcie_host_ops,
.dw_pcie_ops = &dw_ls1021_pcie_ops,
};
static const struct ls_pcie_drvdata ls1043_drvdata = {
.lut_offset = 0x10000,
.ltssm_shift = 24,
.lut_dbg = 0x7fc,
.ops = &ls_pcie_host_ops,
.dw_pcie_ops = &dw_ls_pcie_ops,
};
static const struct ls_pcie_drvdata ls1046_drvdata = {
.lut_offset = 0x80000,
.ltssm_shift = 24,
.lut_dbg = 0x407fc,
.ops = &ls_pcie_host_ops,
.dw_pcie_ops = &dw_ls_pcie_ops,
};
static const struct ls_pcie_drvdata ls2080_drvdata = {
.lut_offset = 0x80000,
.ltssm_shift = 0,
.lut_dbg = 0x7fc,
.ops = &ls_pcie_host_ops,
.dw_pcie_ops = &dw_ls_pcie_ops,
};
static const struct ls_pcie_drvdata ls2088_drvdata = {
.lut_offset = 0x80000,
.ltssm_shift = 0,
.lut_dbg = 0x407fc,
.ops = &ls_pcie_host_ops,
.dw_pcie_ops = &dw_ls_pcie_ops,
};
static const struct of_device_id ls_pcie_of_match[] = { static const struct of_device_id ls_pcie_of_match[] = {
{ .compatible = "fsl,ls1012a-pcie", .data = &ls1046_drvdata }, { .compatible = "fsl,ls1012a-pcie", },
{ .compatible = "fsl,ls1021a-pcie", .data = &ls1021_drvdata }, { .compatible = "fsl,ls1021a-pcie", },
{ .compatible = "fsl,ls1028a-pcie", .data = &ls2088_drvdata }, { .compatible = "fsl,ls1028a-pcie", },
{ .compatible = "fsl,ls1043a-pcie", .data = &ls1043_drvdata }, { .compatible = "fsl,ls1043a-pcie", },
{ .compatible = "fsl,ls1046a-pcie", .data = &ls1046_drvdata }, { .compatible = "fsl,ls1046a-pcie", },
{ .compatible = "fsl,ls2080a-pcie", .data = &ls2080_drvdata }, { .compatible = "fsl,ls2080a-pcie", },
{ .compatible = "fsl,ls2085a-pcie", .data = &ls2080_drvdata }, { .compatible = "fsl,ls2085a-pcie", },
{ .compatible = "fsl,ls2088a-pcie", .data = &ls2088_drvdata }, { .compatible = "fsl,ls2088a-pcie", },
{ .compatible = "fsl,ls1088a-pcie", .data = &ls2088_drvdata }, { .compatible = "fsl,ls1088a-pcie", },
{ }, { },
}; };
...@@ -247,11 +122,8 @@ static int ls_pcie_probe(struct platform_device *pdev) ...@@ -247,11 +122,8 @@ static int ls_pcie_probe(struct platform_device *pdev)
if (!pci) if (!pci)
return -ENOMEM; return -ENOMEM;
pcie->drvdata = of_device_get_match_data(dev);
pci->dev = dev; pci->dev = dev;
pci->ops = pcie->drvdata->dw_pcie_ops; pci->pp.ops = &ls_pcie_host_ops;
pci->pp.ops = pcie->drvdata->ops;
pcie->pci = pci; pcie->pci = pci;
...@@ -260,8 +132,6 @@ static int ls_pcie_probe(struct platform_device *pdev) ...@@ -260,8 +132,6 @@ static int ls_pcie_probe(struct platform_device *pdev)
if (IS_ERR(pci->dbi_base)) if (IS_ERR(pci->dbi_base))
return PTR_ERR(pci->dbi_base); return PTR_ERR(pci->dbi_base);
pcie->lut = pci->dbi_base + pcie->drvdata->lut_offset;
if (!ls_pcie_is_bridge(pcie)) if (!ls_pcie_is_bridge(pcie))
return -ENODEV; return -ENODEV;
......
...@@ -380,17 +380,15 @@ static int artpec6_pcie_probe(struct platform_device *pdev) ...@@ -380,17 +380,15 @@ static int artpec6_pcie_probe(struct platform_device *pdev)
struct dw_pcie *pci; struct dw_pcie *pci;
struct artpec6_pcie *artpec6_pcie; struct artpec6_pcie *artpec6_pcie;
int ret; int ret;
const struct of_device_id *match;
const struct artpec_pcie_of_data *data; const struct artpec_pcie_of_data *data;
enum artpec_pcie_variants variant; enum artpec_pcie_variants variant;
enum dw_pcie_device_mode mode; enum dw_pcie_device_mode mode;
u32 val; u32 val;
match = of_match_device(artpec6_pcie_of_match, dev); data = of_device_get_match_data(dev);
if (!match) if (!data)
return -EINVAL; return -EINVAL;
data = (struct artpec_pcie_of_data *)match->data;
variant = (enum artpec_pcie_variants)data->variant; variant = (enum artpec_pcie_variants)data->variant;
mode = (enum dw_pcie_device_mode)data->mode; mode = (enum dw_pcie_device_mode)data->mode;
......
...@@ -122,15 +122,13 @@ static int dw_plat_pcie_probe(struct platform_device *pdev) ...@@ -122,15 +122,13 @@ static int dw_plat_pcie_probe(struct platform_device *pdev)
struct dw_plat_pcie *dw_plat_pcie; struct dw_plat_pcie *dw_plat_pcie;
struct dw_pcie *pci; struct dw_pcie *pci;
int ret; int ret;
const struct of_device_id *match;
const struct dw_plat_pcie_of_data *data; const struct dw_plat_pcie_of_data *data;
enum dw_pcie_device_mode mode; enum dw_pcie_device_mode mode;
match = of_match_device(dw_plat_pcie_of_match, dev); data = of_device_get_match_data(dev);
if (!match) if (!data)
return -EINVAL; return -EINVAL;
data = (struct dw_plat_pcie_of_data *)match->data;
mode = (enum dw_pcie_device_mode)data->mode; mode = (enum dw_pcie_device_mode)data->mode;
dw_plat_pcie = devm_kzalloc(dev, sizeof(*dw_plat_pcie), GFP_KERNEL); dw_plat_pcie = devm_kzalloc(dev, sizeof(*dw_plat_pcie), GFP_KERNEL);
......
...@@ -672,10 +672,11 @@ void dw_pcie_iatu_detect(struct dw_pcie *pci) ...@@ -672,10 +672,11 @@ void dw_pcie_iatu_detect(struct dw_pcie *pci)
if (!pci->atu_base) { if (!pci->atu_base) {
struct resource *res = struct resource *res =
platform_get_resource_byname(pdev, IORESOURCE_MEM, "atu"); platform_get_resource_byname(pdev, IORESOURCE_MEM, "atu");
if (res) if (res) {
pci->atu_size = resource_size(res); pci->atu_size = resource_size(res);
pci->atu_base = devm_ioremap_resource(dev, res); pci->atu_base = devm_ioremap_resource(dev, res);
if (IS_ERR(pci->atu_base)) }
if (!pci->atu_base || IS_ERR(pci->atu_base))
pci->atu_base = pci->dbi_base + DEFAULT_DBI_ATU_OFFSET; pci->atu_base = pci->dbi_base + DEFAULT_DBI_ATU_OFFSET;
} }
......
...@@ -18,6 +18,10 @@ ...@@ -18,6 +18,10 @@
#if defined(CONFIG_PCI_HISI) || (defined(CONFIG_ACPI) && defined(CONFIG_PCI_QUIRKS)) #if defined(CONFIG_PCI_HISI) || (defined(CONFIG_ACPI) && defined(CONFIG_PCI_QUIRKS))
struct hisi_pcie {
void __iomem *reg_base;
};
static int hisi_pcie_rd_conf(struct pci_bus *bus, u32 devfn, int where, static int hisi_pcie_rd_conf(struct pci_bus *bus, u32 devfn, int where,
int size, u32 *val) int size, u32 *val)
{ {
...@@ -58,10 +62,10 @@ static void __iomem *hisi_pcie_map_bus(struct pci_bus *bus, unsigned int devfn, ...@@ -58,10 +62,10 @@ static void __iomem *hisi_pcie_map_bus(struct pci_bus *bus, unsigned int devfn,
int where) int where)
{ {
struct pci_config_window *cfg = bus->sysdata; struct pci_config_window *cfg = bus->sysdata;
void __iomem *reg_base = cfg->priv; struct hisi_pcie *pcie = cfg->priv;
if (bus->number == cfg->busr.start) if (bus->number == cfg->busr.start)
return reg_base + where; return pcie->reg_base + where;
else else
return pci_ecam_map_bus(bus, devfn, where); return pci_ecam_map_bus(bus, devfn, where);
} }
...@@ -71,12 +75,16 @@ static void __iomem *hisi_pcie_map_bus(struct pci_bus *bus, unsigned int devfn, ...@@ -71,12 +75,16 @@ static void __iomem *hisi_pcie_map_bus(struct pci_bus *bus, unsigned int devfn,
static int hisi_pcie_init(struct pci_config_window *cfg) static int hisi_pcie_init(struct pci_config_window *cfg)
{ {
struct device *dev = cfg->parent; struct device *dev = cfg->parent;
struct hisi_pcie *pcie;
struct acpi_device *adev = to_acpi_device(dev); struct acpi_device *adev = to_acpi_device(dev);
struct acpi_pci_root *root = acpi_driver_data(adev); struct acpi_pci_root *root = acpi_driver_data(adev);
struct resource *res; struct resource *res;
void __iomem *reg_base;
int ret; int ret;
pcie = devm_kzalloc(dev, sizeof(*pcie), GFP_KERNEL);
if (!pcie)
return -ENOMEM;
/* /*
* Retrieve RC base and size from a HISI0081 device with _UID * Retrieve RC base and size from a HISI0081 device with _UID
* matching our segment. * matching our segment.
...@@ -91,11 +99,11 @@ static int hisi_pcie_init(struct pci_config_window *cfg) ...@@ -91,11 +99,11 @@ static int hisi_pcie_init(struct pci_config_window *cfg)
return -ENOMEM; return -ENOMEM;
} }
reg_base = devm_pci_remap_cfgspace(dev, res->start, resource_size(res)); pcie->reg_base = devm_pci_remap_cfgspace(dev, res->start, resource_size(res));
if (!reg_base) if (!pcie->reg_base)
return -ENOMEM; return -ENOMEM;
cfg->priv = reg_base; cfg->priv = pcie;
return 0; return 0;
} }
...@@ -115,9 +123,13 @@ const struct pci_ecam_ops hisi_pcie_ops = { ...@@ -115,9 +123,13 @@ const struct pci_ecam_ops hisi_pcie_ops = {
static int hisi_pcie_platform_init(struct pci_config_window *cfg) static int hisi_pcie_platform_init(struct pci_config_window *cfg)
{ {
struct device *dev = cfg->parent; struct device *dev = cfg->parent;
struct hisi_pcie *pcie;
struct platform_device *pdev = to_platform_device(dev); struct platform_device *pdev = to_platform_device(dev);
struct resource *res; struct resource *res;
void __iomem *reg_base;
pcie = devm_kzalloc(dev, sizeof(*pcie), GFP_KERNEL);
if (!pcie)
return -ENOMEM;
res = platform_get_resource(pdev, IORESOURCE_MEM, 1); res = platform_get_resource(pdev, IORESOURCE_MEM, 1);
if (!res) { if (!res) {
...@@ -125,11 +137,11 @@ static int hisi_pcie_platform_init(struct pci_config_window *cfg) ...@@ -125,11 +137,11 @@ static int hisi_pcie_platform_init(struct pci_config_window *cfg)
return -EINVAL; return -EINVAL;
} }
reg_base = devm_pci_remap_cfgspace(dev, res->start, resource_size(res)); pcie->reg_base = devm_pci_remap_cfgspace(dev, res->start, resource_size(res));
if (!reg_base) if (!pcie->reg_base)
return -ENOMEM; return -ENOMEM;
cfg->priv = reg_base; cfg->priv = pcie;
return 0; return 0;
} }
......
...@@ -127,10 +127,8 @@ static int histb_pcie_rd_own_conf(struct pci_bus *bus, unsigned int devfn, ...@@ -127,10 +127,8 @@ static int histb_pcie_rd_own_conf(struct pci_bus *bus, unsigned int devfn,
{ {
struct dw_pcie *pci = to_dw_pcie_from_pp(bus->sysdata); struct dw_pcie *pci = to_dw_pcie_from_pp(bus->sysdata);
if (PCI_SLOT(devfn)) { if (PCI_SLOT(devfn))
*val = ~0;
return PCIBIOS_DEVICE_NOT_FOUND; return PCIBIOS_DEVICE_NOT_FOUND;
}
*val = dw_pcie_read_dbi(pci, where, size); *val = dw_pcie_read_dbi(pci, where, size);
return PCIBIOS_SUCCESSFUL; return PCIBIOS_SUCCESSFUL;
......
This diff is collapsed.
...@@ -530,10 +530,8 @@ static int kirin_pcie_rd_own_conf(struct pci_bus *bus, unsigned int devfn, ...@@ -530,10 +530,8 @@ static int kirin_pcie_rd_own_conf(struct pci_bus *bus, unsigned int devfn,
{ {
struct dw_pcie *pci = to_dw_pcie_from_pp(bus->sysdata); struct dw_pcie *pci = to_dw_pcie_from_pp(bus->sysdata);
if (PCI_SLOT(devfn)) { if (PCI_SLOT(devfn))
*val = ~0;
return PCIBIOS_DEVICE_NOT_FOUND; return PCIBIOS_DEVICE_NOT_FOUND;
}
*val = dw_pcie_read_dbi(pci, where, size); *val = dw_pcie_read_dbi(pci, where, size);
return PCIBIOS_SUCCESSFUL; return PCIBIOS_SUCCESSFUL;
...@@ -773,7 +771,6 @@ static const struct of_device_id kirin_pcie_match[] = { ...@@ -773,7 +771,6 @@ static const struct of_device_id kirin_pcie_match[] = {
static int kirin_pcie_probe(struct platform_device *pdev) static int kirin_pcie_probe(struct platform_device *pdev)
{ {
enum pcie_kirin_phy_type phy_type; enum pcie_kirin_phy_type phy_type;
const struct of_device_id *of_id;
struct device *dev = &pdev->dev; struct device *dev = &pdev->dev;
struct kirin_pcie *kirin_pcie; struct kirin_pcie *kirin_pcie;
struct dw_pcie *pci; struct dw_pcie *pci;
...@@ -784,13 +781,12 @@ static int kirin_pcie_probe(struct platform_device *pdev) ...@@ -784,13 +781,12 @@ static int kirin_pcie_probe(struct platform_device *pdev)
return -EINVAL; return -EINVAL;
} }
of_id = of_match_device(kirin_pcie_match, dev); phy_type = (long)of_device_get_match_data(dev);
if (!of_id) { if (!phy_type) {
dev_err(dev, "OF data missing\n"); dev_err(dev, "OF data missing\n");
return -EINVAL; return -EINVAL;
} }
phy_type = (long)of_id->data;
kirin_pcie = devm_kzalloc(dev, sizeof(struct kirin_pcie), GFP_KERNEL); kirin_pcie = devm_kzalloc(dev, sizeof(struct kirin_pcie), GFP_KERNEL);
if (!kirin_pcie) if (!kirin_pcie)
......
...@@ -553,10 +553,8 @@ static int qcom_pcie_ep_enable_irq_resources(struct platform_device *pdev, ...@@ -553,10 +553,8 @@ static int qcom_pcie_ep_enable_irq_resources(struct platform_device *pdev,
int irq, ret; int irq, ret;
irq = platform_get_irq_byname(pdev, "global"); irq = platform_get_irq_byname(pdev, "global");
if (irq < 0) { if (irq < 0)
dev_err(&pdev->dev, "Failed to get Global IRQ\n");
return irq; return irq;
}
ret = devm_request_threaded_irq(&pdev->dev, irq, NULL, ret = devm_request_threaded_irq(&pdev->dev, irq, NULL,
qcom_pcie_ep_global_irq_thread, qcom_pcie_ep_global_irq_thread,
...@@ -620,7 +618,7 @@ static void qcom_pcie_ep_init(struct dw_pcie_ep *ep) ...@@ -620,7 +618,7 @@ static void qcom_pcie_ep_init(struct dw_pcie_ep *ep)
dw_pcie_ep_reset_bar(pci, bar); dw_pcie_ep_reset_bar(pci, bar);
} }
static struct dw_pcie_ep_ops pci_ep_ops = { static const struct dw_pcie_ep_ops pci_ep_ops = {
.ep_init = qcom_pcie_ep_init, .ep_init = qcom_pcie_ep_init,
.raise_irq = qcom_pcie_ep_raise_irq, .raise_irq = qcom_pcie_ep_raise_irq,
.get_features = qcom_pcie_epc_get_features, .get_features = qcom_pcie_epc_get_features,
......
...@@ -1343,7 +1343,7 @@ static int qcom_pcie_config_sid_sm8250(struct qcom_pcie *pcie) ...@@ -1343,7 +1343,7 @@ static int qcom_pcie_config_sid_sm8250(struct qcom_pcie *pcie)
/* Look for an available entry to hold the mapping */ /* Look for an available entry to hold the mapping */
for (i = 0; i < nr_map; i++) { for (i = 0; i < nr_map; i++) {
u16 bdf_be = cpu_to_be16(map[i].bdf); __be16 bdf_be = cpu_to_be16(map[i].bdf);
u32 val; u32 val;
u8 hash; u8 hash;
...@@ -1534,6 +1534,12 @@ static int qcom_pcie_probe(struct platform_device *pdev) ...@@ -1534,6 +1534,12 @@ static int qcom_pcie_probe(struct platform_device *pdev)
const struct qcom_pcie_cfg *pcie_cfg; const struct qcom_pcie_cfg *pcie_cfg;
int ret; int ret;
pcie_cfg = of_device_get_match_data(dev);
if (!pcie_cfg || !pcie_cfg->ops) {
dev_err(dev, "Invalid platform data\n");
return -EINVAL;
}
pcie = devm_kzalloc(dev, sizeof(*pcie), GFP_KERNEL); pcie = devm_kzalloc(dev, sizeof(*pcie), GFP_KERNEL);
if (!pcie) if (!pcie)
return -ENOMEM; return -ENOMEM;
...@@ -1553,12 +1559,6 @@ static int qcom_pcie_probe(struct platform_device *pdev) ...@@ -1553,12 +1559,6 @@ static int qcom_pcie_probe(struct platform_device *pdev)
pcie->pci = pci; pcie->pci = pci;
pcie_cfg = of_device_get_match_data(dev);
if (!pcie_cfg || !pcie_cfg->ops) {
dev_err(dev, "Invalid platform data\n");
return -EINVAL;
}
pcie->ops = pcie_cfg->ops; pcie->ops = pcie_cfg->ops;
pcie->pipe_clk_need_muxing = pcie_cfg->pipe_clk_need_muxing; pcie->pipe_clk_need_muxing = pcie_cfg->pipe_clk_need_muxing;
......
...@@ -69,7 +69,7 @@ struct pcie_app_reg { ...@@ -69,7 +69,7 @@ struct pcie_app_reg {
static int spear13xx_pcie_start_link(struct dw_pcie *pci) static int spear13xx_pcie_start_link(struct dw_pcie *pci)
{ {
struct spear13xx_pcie *spear13xx_pcie = to_spear13xx_pcie(pci); struct spear13xx_pcie *spear13xx_pcie = to_spear13xx_pcie(pci);
struct pcie_app_reg *app_reg = spear13xx_pcie->app_base; struct pcie_app_reg __iomem *app_reg = spear13xx_pcie->app_base;
/* enable ltssm */ /* enable ltssm */
writel(DEVICE_TYPE_RC | (1 << MISCTRL_EN_ID) writel(DEVICE_TYPE_RC | (1 << MISCTRL_EN_ID)
...@@ -83,7 +83,7 @@ static int spear13xx_pcie_start_link(struct dw_pcie *pci) ...@@ -83,7 +83,7 @@ static int spear13xx_pcie_start_link(struct dw_pcie *pci)
static irqreturn_t spear13xx_pcie_irq_handler(int irq, void *arg) static irqreturn_t spear13xx_pcie_irq_handler(int irq, void *arg)
{ {
struct spear13xx_pcie *spear13xx_pcie = arg; struct spear13xx_pcie *spear13xx_pcie = arg;
struct pcie_app_reg *app_reg = spear13xx_pcie->app_base; struct pcie_app_reg __iomem *app_reg = spear13xx_pcie->app_base;
struct dw_pcie *pci = spear13xx_pcie->pci; struct dw_pcie *pci = spear13xx_pcie->pci;
struct pcie_port *pp = &pci->pp; struct pcie_port *pp = &pci->pp;
unsigned int status; unsigned int status;
...@@ -102,7 +102,7 @@ static irqreturn_t spear13xx_pcie_irq_handler(int irq, void *arg) ...@@ -102,7 +102,7 @@ static irqreturn_t spear13xx_pcie_irq_handler(int irq, void *arg)
static void spear13xx_pcie_enable_interrupts(struct spear13xx_pcie *spear13xx_pcie) static void spear13xx_pcie_enable_interrupts(struct spear13xx_pcie *spear13xx_pcie)
{ {
struct pcie_app_reg *app_reg = spear13xx_pcie->app_base; struct pcie_app_reg __iomem *app_reg = spear13xx_pcie->app_base;
/* Enable MSI interrupt */ /* Enable MSI interrupt */
if (IS_ENABLED(CONFIG_PCI_MSI)) if (IS_ENABLED(CONFIG_PCI_MSI))
...@@ -113,7 +113,7 @@ static void spear13xx_pcie_enable_interrupts(struct spear13xx_pcie *spear13xx_pc ...@@ -113,7 +113,7 @@ static void spear13xx_pcie_enable_interrupts(struct spear13xx_pcie *spear13xx_pc
static int spear13xx_pcie_link_up(struct dw_pcie *pci) static int spear13xx_pcie_link_up(struct dw_pcie *pci)
{ {
struct spear13xx_pcie *spear13xx_pcie = to_spear13xx_pcie(pci); struct spear13xx_pcie *spear13xx_pcie = to_spear13xx_pcie(pci);
struct pcie_app_reg *app_reg = spear13xx_pcie->app_base; struct pcie_app_reg __iomem *app_reg = spear13xx_pcie->app_base;
if (readl(&app_reg->app_status_1) & XMLH_LINK_UP) if (readl(&app_reg->app_status_1) & XMLH_LINK_UP)
return 1; return 1;
......
This diff is collapsed.
This diff is collapsed.
...@@ -34,31 +34,31 @@ ...@@ -34,31 +34,31 @@
#define PF_DBG_WE BIT(31) #define PF_DBG_WE BIT(31)
#define PF_DBG_PABR BIT(27) #define PF_DBG_PABR BIT(27)
#define to_ls_pcie_g4(x) platform_get_drvdata((x)->pdev) #define to_ls_g4_pcie(x) platform_get_drvdata((x)->pdev)
struct ls_pcie_g4 { struct ls_g4_pcie {
struct mobiveil_pcie pci; struct mobiveil_pcie pci;
struct delayed_work dwork; struct delayed_work dwork;
int irq; int irq;
}; };
static inline u32 ls_pcie_g4_pf_readl(struct ls_pcie_g4 *pcie, u32 off) static inline u32 ls_g4_pcie_pf_readl(struct ls_g4_pcie *pcie, u32 off)
{ {
return ioread32(pcie->pci.csr_axi_slave_base + PCIE_PF_OFF + off); return ioread32(pcie->pci.csr_axi_slave_base + PCIE_PF_OFF + off);
} }
static inline void ls_pcie_g4_pf_writel(struct ls_pcie_g4 *pcie, static inline void ls_g4_pcie_pf_writel(struct ls_g4_pcie *pcie,
u32 off, u32 val) u32 off, u32 val)
{ {
iowrite32(val, pcie->pci.csr_axi_slave_base + PCIE_PF_OFF + off); iowrite32(val, pcie->pci.csr_axi_slave_base + PCIE_PF_OFF + off);
} }
static int ls_pcie_g4_link_up(struct mobiveil_pcie *pci) static int ls_g4_pcie_link_up(struct mobiveil_pcie *pci)
{ {
struct ls_pcie_g4 *pcie = to_ls_pcie_g4(pci); struct ls_g4_pcie *pcie = to_ls_g4_pcie(pci);
u32 state; u32 state;
state = ls_pcie_g4_pf_readl(pcie, PCIE_PF_DBG); state = ls_g4_pcie_pf_readl(pcie, PCIE_PF_DBG);
state = state & PF_DBG_LTSSM_MASK; state = state & PF_DBG_LTSSM_MASK;
if (state == PF_DBG_LTSSM_L0) if (state == PF_DBG_LTSSM_L0)
...@@ -67,14 +67,14 @@ static int ls_pcie_g4_link_up(struct mobiveil_pcie *pci) ...@@ -67,14 +67,14 @@ static int ls_pcie_g4_link_up(struct mobiveil_pcie *pci)
return 0; return 0;
} }
static void ls_pcie_g4_disable_interrupt(struct ls_pcie_g4 *pcie) static void ls_g4_pcie_disable_interrupt(struct ls_g4_pcie *pcie)
{ {
struct mobiveil_pcie *mv_pci = &pcie->pci; struct mobiveil_pcie *mv_pci = &pcie->pci;
mobiveil_csr_writel(mv_pci, 0, PAB_INTP_AMBA_MISC_ENB); mobiveil_csr_writel(mv_pci, 0, PAB_INTP_AMBA_MISC_ENB);
} }
static void ls_pcie_g4_enable_interrupt(struct ls_pcie_g4 *pcie) static void ls_g4_pcie_enable_interrupt(struct ls_g4_pcie *pcie)
{ {
struct mobiveil_pcie *mv_pci = &pcie->pci; struct mobiveil_pcie *mv_pci = &pcie->pci;
u32 val; u32 val;
...@@ -87,7 +87,7 @@ static void ls_pcie_g4_enable_interrupt(struct ls_pcie_g4 *pcie) ...@@ -87,7 +87,7 @@ static void ls_pcie_g4_enable_interrupt(struct ls_pcie_g4 *pcie)
mobiveil_csr_writel(mv_pci, val, PAB_INTP_AMBA_MISC_ENB); mobiveil_csr_writel(mv_pci, val, PAB_INTP_AMBA_MISC_ENB);
} }
static int ls_pcie_g4_reinit_hw(struct ls_pcie_g4 *pcie) static int ls_g4_pcie_reinit_hw(struct ls_g4_pcie *pcie)
{ {
struct mobiveil_pcie *mv_pci = &pcie->pci; struct mobiveil_pcie *mv_pci = &pcie->pci;
struct device *dev = &mv_pci->pdev->dev; struct device *dev = &mv_pci->pdev->dev;
...@@ -97,7 +97,7 @@ static int ls_pcie_g4_reinit_hw(struct ls_pcie_g4 *pcie) ...@@ -97,7 +97,7 @@ static int ls_pcie_g4_reinit_hw(struct ls_pcie_g4 *pcie)
/* Poll for pab_csb_reset to set and PAB activity to clear */ /* Poll for pab_csb_reset to set and PAB activity to clear */
do { do {
usleep_range(10, 15); usleep_range(10, 15);
val = ls_pcie_g4_pf_readl(pcie, PCIE_PF_INT_STAT); val = ls_g4_pcie_pf_readl(pcie, PCIE_PF_INT_STAT);
act_stat = mobiveil_csr_readl(mv_pci, PAB_ACTIVITY_STAT); act_stat = mobiveil_csr_readl(mv_pci, PAB_ACTIVITY_STAT);
} while (((val & PF_INT_STAT_PABRST) == 0 || act_stat) && to--); } while (((val & PF_INT_STAT_PABRST) == 0 || act_stat) && to--);
if (to < 0) { if (to < 0) {
...@@ -106,22 +106,22 @@ static int ls_pcie_g4_reinit_hw(struct ls_pcie_g4 *pcie) ...@@ -106,22 +106,22 @@ static int ls_pcie_g4_reinit_hw(struct ls_pcie_g4 *pcie)
} }
/* clear PEX_RESET bit in PEX_PF0_DBG register */ /* clear PEX_RESET bit in PEX_PF0_DBG register */
val = ls_pcie_g4_pf_readl(pcie, PCIE_PF_DBG); val = ls_g4_pcie_pf_readl(pcie, PCIE_PF_DBG);
val |= PF_DBG_WE; val |= PF_DBG_WE;
ls_pcie_g4_pf_writel(pcie, PCIE_PF_DBG, val); ls_g4_pcie_pf_writel(pcie, PCIE_PF_DBG, val);
val = ls_pcie_g4_pf_readl(pcie, PCIE_PF_DBG); val = ls_g4_pcie_pf_readl(pcie, PCIE_PF_DBG);
val |= PF_DBG_PABR; val |= PF_DBG_PABR;
ls_pcie_g4_pf_writel(pcie, PCIE_PF_DBG, val); ls_g4_pcie_pf_writel(pcie, PCIE_PF_DBG, val);
val = ls_pcie_g4_pf_readl(pcie, PCIE_PF_DBG); val = ls_g4_pcie_pf_readl(pcie, PCIE_PF_DBG);
val &= ~PF_DBG_WE; val &= ~PF_DBG_WE;
ls_pcie_g4_pf_writel(pcie, PCIE_PF_DBG, val); ls_g4_pcie_pf_writel(pcie, PCIE_PF_DBG, val);
mobiveil_host_init(mv_pci, true); mobiveil_host_init(mv_pci, true);
to = 100; to = 100;
while (!ls_pcie_g4_link_up(mv_pci) && to--) while (!ls_g4_pcie_link_up(mv_pci) && to--)
usleep_range(200, 250); usleep_range(200, 250);
if (to < 0) { if (to < 0) {
dev_err(dev, "PCIe link training timeout\n"); dev_err(dev, "PCIe link training timeout\n");
...@@ -131,9 +131,9 @@ static int ls_pcie_g4_reinit_hw(struct ls_pcie_g4 *pcie) ...@@ -131,9 +131,9 @@ static int ls_pcie_g4_reinit_hw(struct ls_pcie_g4 *pcie)
return 0; return 0;
} }
static irqreturn_t ls_pcie_g4_isr(int irq, void *dev_id) static irqreturn_t ls_g4_pcie_isr(int irq, void *dev_id)
{ {
struct ls_pcie_g4 *pcie = (struct ls_pcie_g4 *)dev_id; struct ls_g4_pcie *pcie = (struct ls_g4_pcie *)dev_id;
struct mobiveil_pcie *mv_pci = &pcie->pci; struct mobiveil_pcie *mv_pci = &pcie->pci;
u32 val; u32 val;
...@@ -142,7 +142,7 @@ static irqreturn_t ls_pcie_g4_isr(int irq, void *dev_id) ...@@ -142,7 +142,7 @@ static irqreturn_t ls_pcie_g4_isr(int irq, void *dev_id)
return IRQ_NONE; return IRQ_NONE;
if (val & PAB_INTP_RESET) { if (val & PAB_INTP_RESET) {
ls_pcie_g4_disable_interrupt(pcie); ls_g4_pcie_disable_interrupt(pcie);
schedule_delayed_work(&pcie->dwork, msecs_to_jiffies(1)); schedule_delayed_work(&pcie->dwork, msecs_to_jiffies(1));
} }
...@@ -151,9 +151,9 @@ static irqreturn_t ls_pcie_g4_isr(int irq, void *dev_id) ...@@ -151,9 +151,9 @@ static irqreturn_t ls_pcie_g4_isr(int irq, void *dev_id)
return IRQ_HANDLED; return IRQ_HANDLED;
} }
static int ls_pcie_g4_interrupt_init(struct mobiveil_pcie *mv_pci) static int ls_g4_pcie_interrupt_init(struct mobiveil_pcie *mv_pci)
{ {
struct ls_pcie_g4 *pcie = to_ls_pcie_g4(mv_pci); struct ls_g4_pcie *pcie = to_ls_g4_pcie(mv_pci);
struct platform_device *pdev = mv_pci->pdev; struct platform_device *pdev = mv_pci->pdev;
struct device *dev = &pdev->dev; struct device *dev = &pdev->dev;
int ret; int ret;
...@@ -162,7 +162,7 @@ static int ls_pcie_g4_interrupt_init(struct mobiveil_pcie *mv_pci) ...@@ -162,7 +162,7 @@ static int ls_pcie_g4_interrupt_init(struct mobiveil_pcie *mv_pci)
if (pcie->irq < 0) if (pcie->irq < 0)
return pcie->irq; return pcie->irq;
ret = devm_request_irq(dev, pcie->irq, ls_pcie_g4_isr, ret = devm_request_irq(dev, pcie->irq, ls_g4_pcie_isr,
IRQF_SHARED, pdev->name, pcie); IRQF_SHARED, pdev->name, pcie);
if (ret) { if (ret) {
dev_err(dev, "Can't register PCIe IRQ, errno = %d\n", ret); dev_err(dev, "Can't register PCIe IRQ, errno = %d\n", ret);
...@@ -172,11 +172,11 @@ static int ls_pcie_g4_interrupt_init(struct mobiveil_pcie *mv_pci) ...@@ -172,11 +172,11 @@ static int ls_pcie_g4_interrupt_init(struct mobiveil_pcie *mv_pci)
return 0; return 0;
} }
static void ls_pcie_g4_reset(struct work_struct *work) static void ls_g4_pcie_reset(struct work_struct *work)
{ {
struct delayed_work *dwork = container_of(work, struct delayed_work, struct delayed_work *dwork = container_of(work, struct delayed_work,
work); work);
struct ls_pcie_g4 *pcie = container_of(dwork, struct ls_pcie_g4, dwork); struct ls_g4_pcie *pcie = container_of(dwork, struct ls_g4_pcie, dwork);
struct mobiveil_pcie *mv_pci = &pcie->pci; struct mobiveil_pcie *mv_pci = &pcie->pci;
u16 ctrl; u16 ctrl;
...@@ -184,26 +184,26 @@ static void ls_pcie_g4_reset(struct work_struct *work) ...@@ -184,26 +184,26 @@ static void ls_pcie_g4_reset(struct work_struct *work)
ctrl &= ~PCI_BRIDGE_CTL_BUS_RESET; ctrl &= ~PCI_BRIDGE_CTL_BUS_RESET;
mobiveil_csr_writew(mv_pci, ctrl, PCI_BRIDGE_CONTROL); mobiveil_csr_writew(mv_pci, ctrl, PCI_BRIDGE_CONTROL);
if (!ls_pcie_g4_reinit_hw(pcie)) if (!ls_g4_pcie_reinit_hw(pcie))
return; return;
ls_pcie_g4_enable_interrupt(pcie); ls_g4_pcie_enable_interrupt(pcie);
} }
static struct mobiveil_rp_ops ls_pcie_g4_rp_ops = { static struct mobiveil_rp_ops ls_g4_pcie_rp_ops = {
.interrupt_init = ls_pcie_g4_interrupt_init, .interrupt_init = ls_g4_pcie_interrupt_init,
}; };
static const struct mobiveil_pab_ops ls_pcie_g4_pab_ops = { static const struct mobiveil_pab_ops ls_g4_pcie_pab_ops = {
.link_up = ls_pcie_g4_link_up, .link_up = ls_g4_pcie_link_up,
}; };
static int __init ls_pcie_g4_probe(struct platform_device *pdev) static int __init ls_g4_pcie_probe(struct platform_device *pdev)
{ {
struct device *dev = &pdev->dev; struct device *dev = &pdev->dev;
struct pci_host_bridge *bridge; struct pci_host_bridge *bridge;
struct mobiveil_pcie *mv_pci; struct mobiveil_pcie *mv_pci;
struct ls_pcie_g4 *pcie; struct ls_g4_pcie *pcie;
struct device_node *np = dev->of_node; struct device_node *np = dev->of_node;
int ret; int ret;
...@@ -220,13 +220,13 @@ static int __init ls_pcie_g4_probe(struct platform_device *pdev) ...@@ -220,13 +220,13 @@ static int __init ls_pcie_g4_probe(struct platform_device *pdev)
mv_pci = &pcie->pci; mv_pci = &pcie->pci;
mv_pci->pdev = pdev; mv_pci->pdev = pdev;
mv_pci->ops = &ls_pcie_g4_pab_ops; mv_pci->ops = &ls_g4_pcie_pab_ops;
mv_pci->rp.ops = &ls_pcie_g4_rp_ops; mv_pci->rp.ops = &ls_g4_pcie_rp_ops;
mv_pci->rp.bridge = bridge; mv_pci->rp.bridge = bridge;
platform_set_drvdata(pdev, pcie); platform_set_drvdata(pdev, pcie);
INIT_DELAYED_WORK(&pcie->dwork, ls_pcie_g4_reset); INIT_DELAYED_WORK(&pcie->dwork, ls_g4_pcie_reset);
ret = mobiveil_pcie_host_probe(mv_pci); ret = mobiveil_pcie_host_probe(mv_pci);
if (ret) { if (ret) {
...@@ -234,22 +234,22 @@ static int __init ls_pcie_g4_probe(struct platform_device *pdev) ...@@ -234,22 +234,22 @@ static int __init ls_pcie_g4_probe(struct platform_device *pdev)
return ret; return ret;
} }
ls_pcie_g4_enable_interrupt(pcie); ls_g4_pcie_enable_interrupt(pcie);
return 0; return 0;
} }
static const struct of_device_id ls_pcie_g4_of_match[] = { static const struct of_device_id ls_g4_pcie_of_match[] = {
{ .compatible = "fsl,lx2160a-pcie", }, { .compatible = "fsl,lx2160a-pcie", },
{ }, { },
}; };
static struct platform_driver ls_pcie_g4_driver = { static struct platform_driver ls_g4_pcie_driver = {
.driver = { .driver = {
.name = "layerscape-pcie-gen4", .name = "layerscape-pcie-gen4",
.of_match_table = ls_pcie_g4_of_match, .of_match_table = ls_g4_pcie_of_match,
.suppress_bind_attrs = true, .suppress_bind_attrs = true,
}, },
}; };
builtin_platform_driver_probe(ls_pcie_g4_driver, ls_pcie_g4_probe); builtin_platform_driver_probe(ls_g4_pcie_driver, ls_g4_pcie_probe);
...@@ -115,6 +115,7 @@ ...@@ -115,6 +115,7 @@
#define PCIE_MSI_ADDR_HIGH_REG (CONTROL_BASE_ADDR + 0x54) #define PCIE_MSI_ADDR_HIGH_REG (CONTROL_BASE_ADDR + 0x54)
#define PCIE_MSI_STATUS_REG (CONTROL_BASE_ADDR + 0x58) #define PCIE_MSI_STATUS_REG (CONTROL_BASE_ADDR + 0x58)
#define PCIE_MSI_MASK_REG (CONTROL_BASE_ADDR + 0x5C) #define PCIE_MSI_MASK_REG (CONTROL_BASE_ADDR + 0x5C)
#define PCIE_MSI_ALL_MASK GENMASK(31, 0)
#define PCIE_MSI_PAYLOAD_REG (CONTROL_BASE_ADDR + 0x9C) #define PCIE_MSI_PAYLOAD_REG (CONTROL_BASE_ADDR + 0x9C)
#define PCIE_MSI_DATA_MASK GENMASK(15, 0) #define PCIE_MSI_DATA_MASK GENMASK(15, 0)
...@@ -570,6 +571,7 @@ static void advk_pcie_setup_hw(struct advk_pcie *pcie) ...@@ -570,6 +571,7 @@ static void advk_pcie_setup_hw(struct advk_pcie *pcie)
advk_writel(pcie, reg, PCIE_CORE_CTRL2_REG); advk_writel(pcie, reg, PCIE_CORE_CTRL2_REG);
/* Clear all interrupts */ /* Clear all interrupts */
advk_writel(pcie, PCIE_MSI_ALL_MASK, PCIE_MSI_STATUS_REG);
advk_writel(pcie, PCIE_ISR0_ALL_MASK, PCIE_ISR0_REG); advk_writel(pcie, PCIE_ISR0_ALL_MASK, PCIE_ISR0_REG);
advk_writel(pcie, PCIE_ISR1_ALL_MASK, PCIE_ISR1_REG); advk_writel(pcie, PCIE_ISR1_ALL_MASK, PCIE_ISR1_REG);
advk_writel(pcie, PCIE_IRQ_ALL_MASK, HOST_CTRL_INT_STATUS_REG); advk_writel(pcie, PCIE_IRQ_ALL_MASK, HOST_CTRL_INT_STATUS_REG);
...@@ -582,7 +584,7 @@ static void advk_pcie_setup_hw(struct advk_pcie *pcie) ...@@ -582,7 +584,7 @@ static void advk_pcie_setup_hw(struct advk_pcie *pcie)
advk_writel(pcie, PCIE_ISR1_ALL_MASK, PCIE_ISR1_MASK_REG); advk_writel(pcie, PCIE_ISR1_ALL_MASK, PCIE_ISR1_MASK_REG);
/* Unmask all MSIs */ /* Unmask all MSIs */
advk_writel(pcie, 0, PCIE_MSI_MASK_REG); advk_writel(pcie, ~(u32)PCIE_MSI_ALL_MASK, PCIE_MSI_MASK_REG);
/* Enable summary interrupt for GIC SPI source */ /* Enable summary interrupt for GIC SPI source */
reg = PCIE_IRQ_ALL_MASK & (~PCIE_IRQ_ENABLE_INTS_MASK); reg = PCIE_IRQ_ALL_MASK & (~PCIE_IRQ_ENABLE_INTS_MASK);
...@@ -872,11 +874,15 @@ advk_pci_bridge_emul_pcie_conf_read(struct pci_bridge_emul *bridge, ...@@ -872,11 +874,15 @@ advk_pci_bridge_emul_pcie_conf_read(struct pci_bridge_emul *bridge,
return PCI_BRIDGE_EMUL_HANDLED; return PCI_BRIDGE_EMUL_HANDLED;
} }
case PCI_CAP_LIST_ID:
case PCI_EXP_DEVCAP: case PCI_EXP_DEVCAP:
case PCI_EXP_DEVCTL: case PCI_EXP_DEVCTL:
case PCI_EXP_DEVCAP2:
case PCI_EXP_DEVCTL2:
case PCI_EXP_LNKCAP2:
case PCI_EXP_LNKCTL2:
*value = advk_readl(pcie, PCIE_CORE_PCIEXP_CAP + reg); *value = advk_readl(pcie, PCIE_CORE_PCIEXP_CAP + reg);
return PCI_BRIDGE_EMUL_HANDLED; return PCI_BRIDGE_EMUL_HANDLED;
default: default:
return PCI_BRIDGE_EMUL_NOT_HANDLED; return PCI_BRIDGE_EMUL_NOT_HANDLED;
} }
...@@ -890,10 +896,6 @@ advk_pci_bridge_emul_pcie_conf_write(struct pci_bridge_emul *bridge, ...@@ -890,10 +896,6 @@ advk_pci_bridge_emul_pcie_conf_write(struct pci_bridge_emul *bridge,
struct advk_pcie *pcie = bridge->data; struct advk_pcie *pcie = bridge->data;
switch (reg) { switch (reg) {
case PCI_EXP_DEVCTL:
advk_writel(pcie, new, PCIE_CORE_PCIEXP_CAP + reg);
break;
case PCI_EXP_LNKCTL: case PCI_EXP_LNKCTL:
advk_writel(pcie, new, PCIE_CORE_PCIEXP_CAP + reg); advk_writel(pcie, new, PCIE_CORE_PCIEXP_CAP + reg);
if (new & PCI_EXP_LNKCTL_RL) if (new & PCI_EXP_LNKCTL_RL)
...@@ -915,6 +917,12 @@ advk_pci_bridge_emul_pcie_conf_write(struct pci_bridge_emul *bridge, ...@@ -915,6 +917,12 @@ advk_pci_bridge_emul_pcie_conf_write(struct pci_bridge_emul *bridge,
advk_writel(pcie, new, PCIE_ISR0_REG); advk_writel(pcie, new, PCIE_ISR0_REG);
break; break;
case PCI_EXP_DEVCTL:
case PCI_EXP_DEVCTL2:
case PCI_EXP_LNKCTL2:
advk_writel(pcie, new, PCIE_CORE_PCIEXP_CAP + reg);
break;
default: default:
break; break;
} }
...@@ -953,6 +961,9 @@ static int advk_sw_pci_bridge_init(struct advk_pcie *pcie) ...@@ -953,6 +961,9 @@ static int advk_sw_pci_bridge_init(struct advk_pcie *pcie)
/* Support interrupt A for MSI feature */ /* Support interrupt A for MSI feature */
bridge->conf.intpin = PCIE_CORE_INT_A_ASSERT_ENABLE; bridge->conf.intpin = PCIE_CORE_INT_A_ASSERT_ENABLE;
/* Aardvark HW provides PCIe Capability structure in version 2 */
bridge->pcie_conf.cap = cpu_to_le16(2);
/* Indicates supports for Completion Retry Status */ /* Indicates supports for Completion Retry Status */
bridge->pcie_conf.rootcap = cpu_to_le16(PCI_EXP_RTCAP_CRSVIS); bridge->pcie_conf.rootcap = cpu_to_le16(PCI_EXP_RTCAP_CRSVIS);
...@@ -1017,10 +1028,8 @@ static int advk_pcie_rd_conf(struct pci_bus *bus, u32 devfn, ...@@ -1017,10 +1028,8 @@ static int advk_pcie_rd_conf(struct pci_bus *bus, u32 devfn,
u32 reg; u32 reg;
int ret; int ret;
if (!advk_pcie_valid_device(pcie, bus, devfn)) { if (!advk_pcie_valid_device(pcie, bus, devfn))
*val = 0xffffffff;
return PCIBIOS_DEVICE_NOT_FOUND; return PCIBIOS_DEVICE_NOT_FOUND;
}
if (pci_is_root_bus(bus)) if (pci_is_root_bus(bus))
return pci_bridge_emul_conf_read(&pcie->bridge, where, return pci_bridge_emul_conf_read(&pcie->bridge, where,
...@@ -1383,7 +1392,7 @@ static void advk_pcie_handle_msi(struct advk_pcie *pcie) ...@@ -1383,7 +1392,7 @@ static void advk_pcie_handle_msi(struct advk_pcie *pcie)
msi_mask = advk_readl(pcie, PCIE_MSI_MASK_REG); msi_mask = advk_readl(pcie, PCIE_MSI_MASK_REG);
msi_val = advk_readl(pcie, PCIE_MSI_STATUS_REG); msi_val = advk_readl(pcie, PCIE_MSI_STATUS_REG);
msi_status = msi_val & ~msi_mask; msi_status = msi_val & ((~msi_mask) & PCIE_MSI_ALL_MASK);
for (msi_idx = 0; msi_idx < MSI_IRQ_NUM; msi_idx++) { for (msi_idx = 0; msi_idx < MSI_IRQ_NUM; msi_idx++) {
if (!(BIT(msi_idx) & msi_status)) if (!(BIT(msi_idx) & msi_status))
...@@ -1535,8 +1544,7 @@ static int advk_pcie_probe(struct platform_device *pdev) ...@@ -1535,8 +1544,7 @@ static int advk_pcie_probe(struct platform_device *pdev)
* only PIO for issuing configuration transfers which does * only PIO for issuing configuration transfers which does
* not use PCIe window configuration. * not use PCIe window configuration.
*/ */
if (type != IORESOURCE_MEM && type != IORESOURCE_MEM_64 && if (type != IORESOURCE_MEM && type != IORESOURCE_IO)
type != IORESOURCE_IO)
continue; continue;
/* /*
...@@ -1544,8 +1552,7 @@ static int advk_pcie_probe(struct platform_device *pdev) ...@@ -1544,8 +1552,7 @@ static int advk_pcie_probe(struct platform_device *pdev)
* configuration is set to transparent memory access so it * configuration is set to transparent memory access so it
* does not need window configuration. * does not need window configuration.
*/ */
if ((type == IORESOURCE_MEM || type == IORESOURCE_MEM_64) && if (type == IORESOURCE_MEM && entry->offset == 0)
entry->offset == 0)
continue; continue;
/* /*
...@@ -1677,20 +1684,64 @@ static int advk_pcie_remove(struct platform_device *pdev) ...@@ -1677,20 +1684,64 @@ static int advk_pcie_remove(struct platform_device *pdev)
{ {
struct advk_pcie *pcie = platform_get_drvdata(pdev); struct advk_pcie *pcie = platform_get_drvdata(pdev);
struct pci_host_bridge *bridge = pci_host_bridge_from_priv(pcie); struct pci_host_bridge *bridge = pci_host_bridge_from_priv(pcie);
u32 val;
int i; int i;
/* Remove PCI bus with all devices */
pci_lock_rescan_remove(); pci_lock_rescan_remove();
pci_stop_root_bus(bridge->bus); pci_stop_root_bus(bridge->bus);
pci_remove_root_bus(bridge->bus); pci_remove_root_bus(bridge->bus);
pci_unlock_rescan_remove(); pci_unlock_rescan_remove();
/* Disable Root Bridge I/O space, memory space and bus mastering */
val = advk_readl(pcie, PCIE_CORE_CMD_STATUS_REG);
val &= ~(PCI_COMMAND_IO | PCI_COMMAND_MEMORY | PCI_COMMAND_MASTER);
advk_writel(pcie, val, PCIE_CORE_CMD_STATUS_REG);
/* Disable MSI */
val = advk_readl(pcie, PCIE_CORE_CTRL2_REG);
val &= ~PCIE_CORE_CTRL2_MSI_ENABLE;
advk_writel(pcie, val, PCIE_CORE_CTRL2_REG);
/* Clear MSI address */
advk_writel(pcie, 0, PCIE_MSI_ADDR_LOW_REG);
advk_writel(pcie, 0, PCIE_MSI_ADDR_HIGH_REG);
/* Mask all interrupts */
advk_writel(pcie, PCIE_MSI_ALL_MASK, PCIE_MSI_MASK_REG);
advk_writel(pcie, PCIE_ISR0_ALL_MASK, PCIE_ISR0_MASK_REG);
advk_writel(pcie, PCIE_ISR1_ALL_MASK, PCIE_ISR1_MASK_REG);
advk_writel(pcie, PCIE_IRQ_ALL_MASK, HOST_CTRL_INT_MASK_REG);
/* Clear all interrupts */
advk_writel(pcie, PCIE_MSI_ALL_MASK, PCIE_MSI_STATUS_REG);
advk_writel(pcie, PCIE_ISR0_ALL_MASK, PCIE_ISR0_REG);
advk_writel(pcie, PCIE_ISR1_ALL_MASK, PCIE_ISR1_REG);
advk_writel(pcie, PCIE_IRQ_ALL_MASK, HOST_CTRL_INT_STATUS_REG);
/* Remove IRQ domains */
advk_pcie_remove_msi_irq_domain(pcie); advk_pcie_remove_msi_irq_domain(pcie);
advk_pcie_remove_irq_domain(pcie); advk_pcie_remove_irq_domain(pcie);
/* Free config space for emulated root bridge */
pci_bridge_emul_cleanup(&pcie->bridge);
/* Assert PERST# signal which prepares PCIe card for power down */
if (pcie->reset_gpio)
gpiod_set_value_cansleep(pcie->reset_gpio, 1);
/* Disable link training */
val = advk_readl(pcie, PCIE_CORE_CTRL0_REG);
val &= ~LINK_TRAINING_EN;
advk_writel(pcie, val, PCIE_CORE_CTRL0_REG);
/* Disable outbound address windows mapping */ /* Disable outbound address windows mapping */
for (i = 0; i < OB_WIN_COUNT; i++) for (i = 0; i < OB_WIN_COUNT; i++)
advk_pcie_disable_ob_win(pcie, i); advk_pcie_disable_ob_win(pcie, i);
/* Disable phy */
advk_pcie_disable_phy(pcie);
return 0; return 0;
} }
......
This diff is collapsed.
This diff is collapsed.
...@@ -93,7 +93,7 @@ ...@@ -93,7 +93,7 @@
#define RCAR_PCI_UNIT_REV_REG (RCAR_AHBPCI_PCICOM_OFFSET + 0x48) #define RCAR_PCI_UNIT_REV_REG (RCAR_AHBPCI_PCICOM_OFFSET + 0x48)
struct rcar_pci_priv { struct rcar_pci {
struct device *dev; struct device *dev;
void __iomem *reg; void __iomem *reg;
struct resource mem_res; struct resource mem_res;
...@@ -105,7 +105,7 @@ struct rcar_pci_priv { ...@@ -105,7 +105,7 @@ struct rcar_pci_priv {
static void __iomem *rcar_pci_cfg_base(struct pci_bus *bus, unsigned int devfn, static void __iomem *rcar_pci_cfg_base(struct pci_bus *bus, unsigned int devfn,
int where) int where)
{ {
struct rcar_pci_priv *priv = bus->sysdata; struct rcar_pci *priv = bus->sysdata;
int slot, val; int slot, val;
if (!pci_is_root_bus(bus) || PCI_FUNC(devfn)) if (!pci_is_root_bus(bus) || PCI_FUNC(devfn))
...@@ -132,7 +132,7 @@ static void __iomem *rcar_pci_cfg_base(struct pci_bus *bus, unsigned int devfn, ...@@ -132,7 +132,7 @@ static void __iomem *rcar_pci_cfg_base(struct pci_bus *bus, unsigned int devfn,
static irqreturn_t rcar_pci_err_irq(int irq, void *pw) static irqreturn_t rcar_pci_err_irq(int irq, void *pw)
{ {
struct rcar_pci_priv *priv = pw; struct rcar_pci *priv = pw;
struct device *dev = priv->dev; struct device *dev = priv->dev;
u32 status = ioread32(priv->reg + RCAR_PCI_INT_STATUS_REG); u32 status = ioread32(priv->reg + RCAR_PCI_INT_STATUS_REG);
...@@ -148,7 +148,7 @@ static irqreturn_t rcar_pci_err_irq(int irq, void *pw) ...@@ -148,7 +148,7 @@ static irqreturn_t rcar_pci_err_irq(int irq, void *pw)
return IRQ_NONE; return IRQ_NONE;
} }
static void rcar_pci_setup_errirq(struct rcar_pci_priv *priv) static void rcar_pci_setup_errirq(struct rcar_pci *priv)
{ {
struct device *dev = priv->dev; struct device *dev = priv->dev;
int ret; int ret;
...@@ -166,11 +166,11 @@ static void rcar_pci_setup_errirq(struct rcar_pci_priv *priv) ...@@ -166,11 +166,11 @@ static void rcar_pci_setup_errirq(struct rcar_pci_priv *priv)
iowrite32(val, priv->reg + RCAR_PCI_INT_ENABLE_REG); iowrite32(val, priv->reg + RCAR_PCI_INT_ENABLE_REG);
} }
#else #else
static inline void rcar_pci_setup_errirq(struct rcar_pci_priv *priv) { } static inline void rcar_pci_setup_errirq(struct rcar_pci *priv) { }
#endif #endif
/* PCI host controller setup */ /* PCI host controller setup */
static void rcar_pci_setup(struct rcar_pci_priv *priv) static void rcar_pci_setup(struct rcar_pci *priv)
{ {
struct pci_host_bridge *bridge = pci_host_bridge_from_priv(priv); struct pci_host_bridge *bridge = pci_host_bridge_from_priv(priv);
struct device *dev = priv->dev; struct device *dev = priv->dev;
...@@ -279,7 +279,7 @@ static int rcar_pci_probe(struct platform_device *pdev) ...@@ -279,7 +279,7 @@ static int rcar_pci_probe(struct platform_device *pdev)
{ {
struct device *dev = &pdev->dev; struct device *dev = &pdev->dev;
struct resource *cfg_res, *mem_res; struct resource *cfg_res, *mem_res;
struct rcar_pci_priv *priv; struct rcar_pci *priv;
struct pci_host_bridge *bridge; struct pci_host_bridge *bridge;
void __iomem *reg; void __iomem *reg;
......
...@@ -41,10 +41,9 @@ static int handle_ea_bar(u32 e0, int bar, struct pci_bus *bus, ...@@ -41,10 +41,9 @@ static int handle_ea_bar(u32 e0, int bar, struct pci_bus *bus,
} }
if (where_a == 0x4) { if (where_a == 0x4) {
addr = bus->ops->map_bus(bus, devfn, bar); /* BAR 0 */ addr = bus->ops->map_bus(bus, devfn, bar); /* BAR 0 */
if (!addr) { if (!addr)
*val = ~0;
return PCIBIOS_DEVICE_NOT_FOUND; return PCIBIOS_DEVICE_NOT_FOUND;
}
v = readl(addr); v = readl(addr);
v &= ~0xf; v &= ~0xf;
v |= 2; /* EA entry-1. Base-L */ v |= 2; /* EA entry-1. Base-L */
...@@ -56,10 +55,9 @@ static int handle_ea_bar(u32 e0, int bar, struct pci_bus *bus, ...@@ -56,10 +55,9 @@ static int handle_ea_bar(u32 e0, int bar, struct pci_bus *bus,
u32 barl_rb; u32 barl_rb;
addr = bus->ops->map_bus(bus, devfn, bar); /* BAR 0 */ addr = bus->ops->map_bus(bus, devfn, bar); /* BAR 0 */
if (!addr) { if (!addr)
*val = ~0;
return PCIBIOS_DEVICE_NOT_FOUND; return PCIBIOS_DEVICE_NOT_FOUND;
}
barl_orig = readl(addr + 0); barl_orig = readl(addr + 0);
writel(0xffffffff, addr + 0); writel(0xffffffff, addr + 0);
barl_rb = readl(addr + 0); barl_rb = readl(addr + 0);
...@@ -72,10 +70,9 @@ static int handle_ea_bar(u32 e0, int bar, struct pci_bus *bus, ...@@ -72,10 +70,9 @@ static int handle_ea_bar(u32 e0, int bar, struct pci_bus *bus,
} }
if (where_a == 0xc) { if (where_a == 0xc) {
addr = bus->ops->map_bus(bus, devfn, bar + 4); /* BAR 1 */ addr = bus->ops->map_bus(bus, devfn, bar + 4); /* BAR 1 */
if (!addr) { if (!addr)
*val = ~0;
return PCIBIOS_DEVICE_NOT_FOUND; return PCIBIOS_DEVICE_NOT_FOUND;
}
v = readl(addr); /* EA entry-3. Base-H */ v = readl(addr); /* EA entry-3. Base-H */
set_val(v, where, size, val); set_val(v, where, size, val);
return PCIBIOS_SUCCESSFUL; return PCIBIOS_SUCCESSFUL;
...@@ -104,10 +101,8 @@ static int thunder_ecam_p2_config_read(struct pci_bus *bus, unsigned int devfn, ...@@ -104,10 +101,8 @@ static int thunder_ecam_p2_config_read(struct pci_bus *bus, unsigned int devfn,
} }
addr = bus->ops->map_bus(bus, devfn, where_a); addr = bus->ops->map_bus(bus, devfn, where_a);
if (!addr) { if (!addr)
*val = ~0;
return PCIBIOS_DEVICE_NOT_FOUND; return PCIBIOS_DEVICE_NOT_FOUND;
}
v = readl(addr); v = readl(addr);
...@@ -135,10 +130,8 @@ static int thunder_ecam_config_read(struct pci_bus *bus, unsigned int devfn, ...@@ -135,10 +130,8 @@ static int thunder_ecam_config_read(struct pci_bus *bus, unsigned int devfn,
int where_a = where & ~3; int where_a = where & ~3;
addr = bus->ops->map_bus(bus, devfn, 0xc); addr = bus->ops->map_bus(bus, devfn, 0xc);
if (!addr) { if (!addr)
*val = ~0;
return PCIBIOS_DEVICE_NOT_FOUND; return PCIBIOS_DEVICE_NOT_FOUND;
}
v = readl(addr); v = readl(addr);
...@@ -146,10 +139,8 @@ static int thunder_ecam_config_read(struct pci_bus *bus, unsigned int devfn, ...@@ -146,10 +139,8 @@ static int thunder_ecam_config_read(struct pci_bus *bus, unsigned int devfn,
cfg_type = (v >> 16) & 0x7f; cfg_type = (v >> 16) & 0x7f;
addr = bus->ops->map_bus(bus, devfn, 8); addr = bus->ops->map_bus(bus, devfn, 8);
if (!addr) { if (!addr)
*val = ~0;
return PCIBIOS_DEVICE_NOT_FOUND; return PCIBIOS_DEVICE_NOT_FOUND;
}
class_rev = readl(addr); class_rev = readl(addr);
if (class_rev == 0xffffffff) if (class_rev == 0xffffffff)
...@@ -176,10 +167,8 @@ static int thunder_ecam_config_read(struct pci_bus *bus, unsigned int devfn, ...@@ -176,10 +167,8 @@ static int thunder_ecam_config_read(struct pci_bus *bus, unsigned int devfn,
} }
addr = bus->ops->map_bus(bus, devfn, 0); addr = bus->ops->map_bus(bus, devfn, 0);
if (!addr) { if (!addr)
*val = ~0;
return PCIBIOS_DEVICE_NOT_FOUND; return PCIBIOS_DEVICE_NOT_FOUND;
}
vendor_device = readl(addr); vendor_device = readl(addr);
if (vendor_device == 0xffffffff) if (vendor_device == 0xffffffff)
...@@ -196,10 +185,9 @@ static int thunder_ecam_config_read(struct pci_bus *bus, unsigned int devfn, ...@@ -196,10 +185,9 @@ static int thunder_ecam_config_read(struct pci_bus *bus, unsigned int devfn,
bool is_tns = (vendor_device == 0xa01f177d); bool is_tns = (vendor_device == 0xa01f177d);
addr = bus->ops->map_bus(bus, devfn, 0x70); addr = bus->ops->map_bus(bus, devfn, 0x70);
if (!addr) { if (!addr)
*val = ~0;
return PCIBIOS_DEVICE_NOT_FOUND; return PCIBIOS_DEVICE_NOT_FOUND;
}
/* E_CAP */ /* E_CAP */
v = readl(addr); v = readl(addr);
has_msix = (v & 0xff00) != 0; has_msix = (v & 0xff00) != 0;
...@@ -211,10 +199,9 @@ static int thunder_ecam_config_read(struct pci_bus *bus, unsigned int devfn, ...@@ -211,10 +199,9 @@ static int thunder_ecam_config_read(struct pci_bus *bus, unsigned int devfn,
} }
if (where_a == 0xb0) { if (where_a == 0xb0) {
addr = bus->ops->map_bus(bus, devfn, where_a); addr = bus->ops->map_bus(bus, devfn, where_a);
if (!addr) { if (!addr)
*val = ~0;
return PCIBIOS_DEVICE_NOT_FOUND; return PCIBIOS_DEVICE_NOT_FOUND;
}
v = readl(addr); v = readl(addr);
if (v & 0xff00) if (v & 0xff00)
pr_err("Bad MSIX cap header: %08x\n", v); pr_err("Bad MSIX cap header: %08x\n", v);
...@@ -268,10 +255,9 @@ static int thunder_ecam_config_read(struct pci_bus *bus, unsigned int devfn, ...@@ -268,10 +255,9 @@ static int thunder_ecam_config_read(struct pci_bus *bus, unsigned int devfn,
if (where_a == 0x70) { if (where_a == 0x70) {
addr = bus->ops->map_bus(bus, devfn, where_a); addr = bus->ops->map_bus(bus, devfn, where_a);
if (!addr) { if (!addr)
*val = ~0;
return PCIBIOS_DEVICE_NOT_FOUND; return PCIBIOS_DEVICE_NOT_FOUND;
}
v = readl(addr); v = readl(addr);
if (v & 0xff00) if (v & 0xff00)
pr_err("Bad PCIe cap header: %08x\n", v); pr_err("Bad PCIe cap header: %08x\n", v);
......
...@@ -41,10 +41,8 @@ static int thunder_pem_bridge_read(struct pci_bus *bus, unsigned int devfn, ...@@ -41,10 +41,8 @@ static int thunder_pem_bridge_read(struct pci_bus *bus, unsigned int devfn,
struct pci_config_window *cfg = bus->sysdata; struct pci_config_window *cfg = bus->sysdata;
struct thunder_pem_pci *pem_pci = (struct thunder_pem_pci *)cfg->priv; struct thunder_pem_pci *pem_pci = (struct thunder_pem_pci *)cfg->priv;
if (devfn != 0 || where >= 2048) { if (devfn != 0 || where >= 2048)
*val = ~0;
return PCIBIOS_DEVICE_NOT_FOUND; return PCIBIOS_DEVICE_NOT_FOUND;
}
/* /*
* 32-bit accesses only. Write the address to the low order * 32-bit accesses only. Write the address to the low order
......
...@@ -269,9 +269,7 @@ static void xgene_free_domains(struct xgene_msi *msi) ...@@ -269,9 +269,7 @@ static void xgene_free_domains(struct xgene_msi *msi)
static int xgene_msi_init_allocator(struct xgene_msi *xgene_msi) static int xgene_msi_init_allocator(struct xgene_msi *xgene_msi)
{ {
int size = BITS_TO_LONGS(NR_MSI_VEC) * sizeof(long); xgene_msi->bitmap = bitmap_zalloc(NR_MSI_VEC, GFP_KERNEL);
xgene_msi->bitmap = kzalloc(size, GFP_KERNEL);
if (!xgene_msi->bitmap) if (!xgene_msi->bitmap)
return -ENOMEM; return -ENOMEM;
...@@ -360,7 +358,7 @@ static int xgene_msi_remove(struct platform_device *pdev) ...@@ -360,7 +358,7 @@ static int xgene_msi_remove(struct platform_device *pdev)
kfree(msi->msi_groups); kfree(msi->msi_groups);
kfree(msi->bitmap); bitmap_free(msi->bitmap);
msi->bitmap = NULL; msi->bitmap = NULL;
xgene_free_domains(msi); xgene_free_domains(msi);
......
This diff is collapsed.
...@@ -510,10 +510,8 @@ static int altera_pcie_cfg_read(struct pci_bus *bus, unsigned int devfn, ...@@ -510,10 +510,8 @@ static int altera_pcie_cfg_read(struct pci_bus *bus, unsigned int devfn,
if (altera_pcie_hide_rc_bar(bus, devfn, where)) if (altera_pcie_hide_rc_bar(bus, devfn, where))
return PCIBIOS_BAD_REGISTER_NUMBER; return PCIBIOS_BAD_REGISTER_NUMBER;
if (!altera_pcie_valid_device(pcie, bus, PCI_SLOT(devfn))) { if (!altera_pcie_valid_device(pcie, bus, PCI_SLOT(devfn)))
*value = 0xffffffff;
return PCIBIOS_DEVICE_NOT_FOUND; return PCIBIOS_DEVICE_NOT_FOUND;
}
return _altera_pcie_cfg_read(pcie, bus->number, devfn, where, size, return _altera_pcie_cfg_read(pcie, bus->number, devfn, where, size,
value); value);
...@@ -767,7 +765,7 @@ static int altera_pcie_probe(struct platform_device *pdev) ...@@ -767,7 +765,7 @@ static int altera_pcie_probe(struct platform_device *pdev)
struct altera_pcie *pcie; struct altera_pcie *pcie;
struct pci_host_bridge *bridge; struct pci_host_bridge *bridge;
int ret; int ret;
const struct of_device_id *match; const struct altera_pcie_data *data;
bridge = devm_pci_alloc_host_bridge(dev, sizeof(*pcie)); bridge = devm_pci_alloc_host_bridge(dev, sizeof(*pcie));
if (!bridge) if (!bridge)
...@@ -777,11 +775,11 @@ static int altera_pcie_probe(struct platform_device *pdev) ...@@ -777,11 +775,11 @@ static int altera_pcie_probe(struct platform_device *pdev)
pcie->pdev = pdev; pcie->pdev = pdev;
platform_set_drvdata(pdev, pcie); platform_set_drvdata(pdev, pcie);
match = of_match_device(altera_pcie_of_match, &pdev->dev); data = of_device_get_match_data(&pdev->dev);
if (!match) if (!data)
return -ENODEV; return -ENODEV;
pcie->pcie_data = match->data; pcie->pcie_data = data;
ret = altera_pcie_parse_dt(pcie); ret = altera_pcie_parse_dt(pcie);
if (ret) { if (ret) {
......
...@@ -42,8 +42,9 @@ ...@@ -42,8 +42,9 @@
#define CORE_FABRIC_STAT_MASK 0x001F001F #define CORE_FABRIC_STAT_MASK 0x001F001F
#define CORE_LANE_CFG(port) (0x84000 + 0x4000 * (port)) #define CORE_LANE_CFG(port) (0x84000 + 0x4000 * (port))
#define CORE_LANE_CFG_REFCLK0REQ BIT(0) #define CORE_LANE_CFG_REFCLK0REQ BIT(0)
#define CORE_LANE_CFG_REFCLK1 BIT(1) #define CORE_LANE_CFG_REFCLK1REQ BIT(1)
#define CORE_LANE_CFG_REFCLK0ACK BIT(2) #define CORE_LANE_CFG_REFCLK0ACK BIT(2)
#define CORE_LANE_CFG_REFCLK1ACK BIT(3)
#define CORE_LANE_CFG_REFCLKEN (BIT(9) | BIT(10)) #define CORE_LANE_CFG_REFCLKEN (BIT(9) | BIT(10))
#define CORE_LANE_CTL(port) (0x84004 + 0x4000 * (port)) #define CORE_LANE_CTL(port) (0x84004 + 0x4000 * (port))
#define CORE_LANE_CTL_CFGACC BIT(15) #define CORE_LANE_CTL_CFGACC BIT(15)
...@@ -482,9 +483,9 @@ static int apple_pcie_setup_refclk(struct apple_pcie *pcie, ...@@ -482,9 +483,9 @@ static int apple_pcie_setup_refclk(struct apple_pcie *pcie,
if (res < 0) if (res < 0)
return res; return res;
rmw_set(CORE_LANE_CFG_REFCLK1, pcie->base + CORE_LANE_CFG(port->idx)); rmw_set(CORE_LANE_CFG_REFCLK1REQ, pcie->base + CORE_LANE_CFG(port->idx));
res = readl_relaxed_poll_timeout(pcie->base + CORE_LANE_CFG(port->idx), res = readl_relaxed_poll_timeout(pcie->base + CORE_LANE_CFG(port->idx),
stat, stat & CORE_LANE_CFG_REFCLK1, stat, stat & CORE_LANE_CFG_REFCLK1ACK,
100, 50000); 100, 50000);
if (res < 0) if (res < 0)
...@@ -563,6 +564,9 @@ static int apple_pcie_setup_port(struct apple_pcie *pcie, ...@@ -563,6 +564,9 @@ static int apple_pcie_setup_port(struct apple_pcie *pcie,
return ret; return ret;
} }
rmw_clear(PORT_REFCLK_CGDIS, port->base + PORT_REFCLK);
rmw_clear(PORT_APPCLK_CGDIS, port->base + PORT_APPCLK);
ret = apple_pcie_port_setup_irq(port); ret = apple_pcie_port_setup_irq(port);
if (ret) if (ret)
return ret; return ret;
......
This diff is collapsed.
...@@ -23,7 +23,7 @@ static void bcma_pcie2_fixup_class(struct pci_dev *dev) ...@@ -23,7 +23,7 @@ static void bcma_pcie2_fixup_class(struct pci_dev *dev)
DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_BROADCOM, 0x8011, bcma_pcie2_fixup_class); DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_BROADCOM, 0x8011, bcma_pcie2_fixup_class);
DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_BROADCOM, 0x8012, bcma_pcie2_fixup_class); DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_BROADCOM, 0x8012, bcma_pcie2_fixup_class);
static int iproc_pcie_bcma_map_irq(const struct pci_dev *dev, u8 slot, u8 pin) static int iproc_bcma_pcie_map_irq(const struct pci_dev *dev, u8 slot, u8 pin)
{ {
struct iproc_pcie *pcie = dev->sysdata; struct iproc_pcie *pcie = dev->sysdata;
struct bcma_device *bdev = container_of(pcie->dev, struct bcma_device, dev); struct bcma_device *bdev = container_of(pcie->dev, struct bcma_device, dev);
...@@ -31,7 +31,7 @@ static int iproc_pcie_bcma_map_irq(const struct pci_dev *dev, u8 slot, u8 pin) ...@@ -31,7 +31,7 @@ static int iproc_pcie_bcma_map_irq(const struct pci_dev *dev, u8 slot, u8 pin)
return bcma_core_irq(bdev, 5); return bcma_core_irq(bdev, 5);
} }
static int iproc_pcie_bcma_probe(struct bcma_device *bdev) static int iproc_bcma_pcie_probe(struct bcma_device *bdev)
{ {
struct device *dev = &bdev->dev; struct device *dev = &bdev->dev;
struct iproc_pcie *pcie; struct iproc_pcie *pcie;
...@@ -64,33 +64,33 @@ static int iproc_pcie_bcma_probe(struct bcma_device *bdev) ...@@ -64,33 +64,33 @@ static int iproc_pcie_bcma_probe(struct bcma_device *bdev)
if (ret) if (ret)
return ret; return ret;
pcie->map_irq = iproc_pcie_bcma_map_irq; pcie->map_irq = iproc_bcma_pcie_map_irq;
bcma_set_drvdata(bdev, pcie); bcma_set_drvdata(bdev, pcie);
return iproc_pcie_setup(pcie, &bridge->windows); return iproc_pcie_setup(pcie, &bridge->windows);
} }
static void iproc_pcie_bcma_remove(struct bcma_device *bdev) static void iproc_bcma_pcie_remove(struct bcma_device *bdev)
{ {
struct iproc_pcie *pcie = bcma_get_drvdata(bdev); struct iproc_pcie *pcie = bcma_get_drvdata(bdev);
iproc_pcie_remove(pcie); iproc_pcie_remove(pcie);
} }
static const struct bcma_device_id iproc_pcie_bcma_table[] = { static const struct bcma_device_id iproc_bcma_pcie_table[] = {
BCMA_CORE(BCMA_MANUF_BCM, BCMA_CORE_NS_PCIEG2, BCMA_ANY_REV, BCMA_ANY_CLASS), BCMA_CORE(BCMA_MANUF_BCM, BCMA_CORE_NS_PCIEG2, BCMA_ANY_REV, BCMA_ANY_CLASS),
{}, {},
}; };
MODULE_DEVICE_TABLE(bcma, iproc_pcie_bcma_table); MODULE_DEVICE_TABLE(bcma, iproc_bcma_pcie_table);
static struct bcma_driver iproc_pcie_bcma_driver = { static struct bcma_driver iproc_bcma_pcie_driver = {
.name = KBUILD_MODNAME, .name = KBUILD_MODNAME,
.id_table = iproc_pcie_bcma_table, .id_table = iproc_bcma_pcie_table,
.probe = iproc_pcie_bcma_probe, .probe = iproc_bcma_pcie_probe,
.remove = iproc_pcie_bcma_remove, .remove = iproc_bcma_pcie_remove,
}; };
module_bcma_driver(iproc_pcie_bcma_driver); module_bcma_driver(iproc_bcma_pcie_driver);
MODULE_AUTHOR("Hauke Mehrtens"); MODULE_AUTHOR("Hauke Mehrtens");
MODULE_DESCRIPTION("Broadcom iProc PCIe BCMA driver"); MODULE_DESCRIPTION("Broadcom iProc PCIe BCMA driver");
......
...@@ -37,7 +37,7 @@ static const struct of_device_id iproc_pcie_of_match_table[] = { ...@@ -37,7 +37,7 @@ static const struct of_device_id iproc_pcie_of_match_table[] = {
}; };
MODULE_DEVICE_TABLE(of, iproc_pcie_of_match_table); MODULE_DEVICE_TABLE(of, iproc_pcie_of_match_table);
static int iproc_pcie_pltfm_probe(struct platform_device *pdev) static int iproc_pltfm_pcie_probe(struct platform_device *pdev)
{ {
struct device *dev = &pdev->dev; struct device *dev = &pdev->dev;
struct iproc_pcie *pcie; struct iproc_pcie *pcie;
...@@ -115,30 +115,30 @@ static int iproc_pcie_pltfm_probe(struct platform_device *pdev) ...@@ -115,30 +115,30 @@ static int iproc_pcie_pltfm_probe(struct platform_device *pdev)
return 0; return 0;
} }
static int iproc_pcie_pltfm_remove(struct platform_device *pdev) static int iproc_pltfm_pcie_remove(struct platform_device *pdev)
{ {
struct iproc_pcie *pcie = platform_get_drvdata(pdev); struct iproc_pcie *pcie = platform_get_drvdata(pdev);
return iproc_pcie_remove(pcie); return iproc_pcie_remove(pcie);
} }
static void iproc_pcie_pltfm_shutdown(struct platform_device *pdev) static void iproc_pltfm_pcie_shutdown(struct platform_device *pdev)
{ {
struct iproc_pcie *pcie = platform_get_drvdata(pdev); struct iproc_pcie *pcie = platform_get_drvdata(pdev);
iproc_pcie_shutdown(pcie); iproc_pcie_shutdown(pcie);
} }
static struct platform_driver iproc_pcie_pltfm_driver = { static struct platform_driver iproc_pltfm_pcie_driver = {
.driver = { .driver = {
.name = "iproc-pcie", .name = "iproc-pcie",
.of_match_table = of_match_ptr(iproc_pcie_of_match_table), .of_match_table = of_match_ptr(iproc_pcie_of_match_table),
}, },
.probe = iproc_pcie_pltfm_probe, .probe = iproc_pltfm_pcie_probe,
.remove = iproc_pcie_pltfm_remove, .remove = iproc_pltfm_pcie_remove,
.shutdown = iproc_pcie_pltfm_shutdown, .shutdown = iproc_pltfm_pcie_shutdown,
}; };
module_platform_driver(iproc_pcie_pltfm_driver); module_platform_driver(iproc_pltfm_pcie_driver);
MODULE_AUTHOR("Ray Jui <rjui@broadcom.com>"); MODULE_AUTHOR("Ray Jui <rjui@broadcom.com>");
MODULE_DESCRIPTION("Broadcom iPROC PCIe platform driver"); MODULE_DESCRIPTION("Broadcom iPROC PCIe platform driver");
......
...@@ -659,10 +659,8 @@ static int iproc_pci_raw_config_read32(struct iproc_pcie *pcie, ...@@ -659,10 +659,8 @@ static int iproc_pci_raw_config_read32(struct iproc_pcie *pcie,
void __iomem *addr; void __iomem *addr;
addr = iproc_pcie_map_cfg_bus(pcie, 0, devfn, where & ~0x3); addr = iproc_pcie_map_cfg_bus(pcie, 0, devfn, where & ~0x3);
if (!addr) { if (!addr)
*val = ~0;
return PCIBIOS_DEVICE_NOT_FOUND; return PCIBIOS_DEVICE_NOT_FOUND;
}
*val = readl(addr); *val = readl(addr);
......
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment