-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
PCI: hisi: Add HiSilicon SoC Hip05 PCIe driver
Add PCIe host support for HiSilicon SoC Hip05, related DT binding documentation, and maintainer update. [bhelgaas: changelog, 32-bit only config write warning text] Signed-off-by: Zhou Wang <wangzhou1@hisilicon.com> Signed-off-by: Gabriele Paoloni <gabriele.paoloni@huawei.com> Signed-off-by: liudongdong <liudongdong3@huawei.com> Signed-off-by: Bjorn Helgaas <bhelgaas@google.com> Acked-by: Rob Herring <robh@kernel.org> (DT binding)
- Loading branch information
Zhou Wang
authored and
Bjorn Helgaas
committed
Nov 2, 2015
1 parent
df77016
commit 500a1d9
Showing
6 changed files
with
275 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,44 @@ | ||
HiSilicon PCIe host bridge DT description | ||
|
||
HiSilicon PCIe host controller is based on Designware PCI core. | ||
It shares common functions with PCIe Designware core driver and inherits | ||
common properties defined in | ||
Documentation/devicetree/bindings/pci/designware-pci.txt. | ||
|
||
Additional properties are described here: | ||
|
||
Required properties: | ||
- compatible: Should contain "hisilicon,hip05-pcie". | ||
- reg: Should contain rc_dbi, config registers location and length. | ||
- reg-names: Must include the following entries: | ||
"rc_dbi": controller configuration registers; | ||
"config": PCIe configuration space registers. | ||
- msi-parent: Should be its_pcie which is an ITS receiving MSI interrupts. | ||
- port-id: Should be 0, 1, 2 or 3. | ||
|
||
Optional properties: | ||
- status: Either "ok" or "disabled". | ||
- dma-coherent: Present if DMA operations are coherent. | ||
|
||
Example: | ||
pcie@0xb0080000 { | ||
compatible = "hisilicon,hip05-pcie", "snps,dw-pcie"; | ||
reg = <0 0xb0080000 0 0x10000>, <0x220 0x00000000 0 0x2000>; | ||
reg-names = "rc_dbi", "config"; | ||
bus-range = <0 15>; | ||
msi-parent = <&its_pcie>; | ||
#address-cells = <3>; | ||
#size-cells = <2>; | ||
device_type = "pci"; | ||
dma-coherent; | ||
ranges = <0x82000000 0 0x00000000 0x220 0x00000000 0 0x10000000>; | ||
num-lanes = <8>; | ||
port-id = <1>; | ||
#interrupts-cells = <1>; | ||
interrupts-map-mask = <0xf800 0 0 7>; | ||
interrupts-map = <0x0 0 0 1 &mbigen_pcie 1 10 | ||
0x0 0 0 2 &mbigen_pcie 2 11 | ||
0x0 0 0 3 &mbigen_pcie 3 12 | ||
0x0 0 0 4 &mbigen_pcie 4 13>; | ||
status = "ok"; | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,198 @@ | ||
/* | ||
* PCIe host controller driver for HiSilicon Hip05 SoC | ||
* | ||
* Copyright (C) 2015 HiSilicon Co., Ltd. http://www.hisilicon.com | ||
* | ||
* Author: Zhou Wang <wangzhou1@hisilicon.com> | ||
* Dacai Zhu <zhudacai@hisilicon.com> | ||
* | ||
* This program is free software; you can redistribute it and/or modify | ||
* it under the terms of the GNU General Public License version 2 as | ||
* published by the Free Software Foundation. | ||
*/ | ||
#include <linux/interrupt.h> | ||
#include <linux/module.h> | ||
#include <linux/mfd/syscon.h> | ||
#include <linux/of_address.h> | ||
#include <linux/of_pci.h> | ||
#include <linux/platform_device.h> | ||
#include <linux/regmap.h> | ||
|
||
#include "pcie-designware.h" | ||
|
||
#define PCIE_SUBCTRL_SYS_STATE4_REG 0x6818 | ||
#define PCIE_LTSSM_LINKUP_STATE 0x11 | ||
#define PCIE_LTSSM_STATE_MASK 0x3F | ||
|
||
#define to_hisi_pcie(x) container_of(x, struct hisi_pcie, pp) | ||
|
||
struct hisi_pcie { | ||
struct regmap *subctrl; | ||
void __iomem *reg_base; | ||
u32 port_id; | ||
struct pcie_port pp; | ||
}; | ||
|
||
static inline void hisi_pcie_apb_writel(struct hisi_pcie *pcie, | ||
u32 val, u32 reg) | ||
{ | ||
writel(val, pcie->reg_base + reg); | ||
} | ||
|
||
static inline u32 hisi_pcie_apb_readl(struct hisi_pcie *pcie, u32 reg) | ||
{ | ||
return readl(pcie->reg_base + reg); | ||
} | ||
|
||
/* Hip05 PCIe host only supports 32-bit config access */ | ||
static int hisi_pcie_cfg_read(struct pcie_port *pp, int where, int size, | ||
u32 *val) | ||
{ | ||
u32 reg; | ||
u32 reg_val; | ||
struct hisi_pcie *pcie = to_hisi_pcie(pp); | ||
void *walker = ®_val; | ||
|
||
walker += (where & 0x3); | ||
reg = where & ~0x3; | ||
reg_val = hisi_pcie_apb_readl(pcie, reg); | ||
|
||
if (size == 1) | ||
*val = *(u8 __force *) walker; | ||
else if (size == 2) | ||
*val = *(u16 __force *) walker; | ||
else if (size != 4) | ||
return PCIBIOS_BAD_REGISTER_NUMBER; | ||
|
||
return PCIBIOS_SUCCESSFUL; | ||
} | ||
|
||
/* Hip05 PCIe host only supports 32-bit config access */ | ||
static int hisi_pcie_cfg_write(struct pcie_port *pp, int where, int size, | ||
u32 val) | ||
{ | ||
u32 reg_val; | ||
u32 reg; | ||
struct hisi_pcie *pcie = to_hisi_pcie(pp); | ||
void *walker = ®_val; | ||
|
||
walker += (where & 0x3); | ||
reg = where & ~0x3; | ||
if (size == 4) | ||
hisi_pcie_apb_writel(pcie, val, reg); | ||
else if (size == 2) { | ||
reg_val = hisi_pcie_apb_readl(pcie, reg); | ||
*(u16 __force *) walker = val; | ||
hisi_pcie_apb_writel(pcie, reg_val, reg); | ||
} else if (size == 1) { | ||
reg_val = hisi_pcie_apb_readl(pcie, reg); | ||
*(u8 __force *) walker = val; | ||
hisi_pcie_apb_writel(pcie, reg_val, reg); | ||
} else | ||
return PCIBIOS_BAD_REGISTER_NUMBER; | ||
|
||
return PCIBIOS_SUCCESSFUL; | ||
} | ||
|
||
static int hisi_pcie_link_up(struct pcie_port *pp) | ||
{ | ||
u32 val; | ||
struct hisi_pcie *hisi_pcie = to_hisi_pcie(pp); | ||
|
||
regmap_read(hisi_pcie->subctrl, PCIE_SUBCTRL_SYS_STATE4_REG + | ||
0x100 * hisi_pcie->port_id, &val); | ||
|
||
return ((val & PCIE_LTSSM_STATE_MASK) == PCIE_LTSSM_LINKUP_STATE); | ||
} | ||
|
||
static struct pcie_host_ops hisi_pcie_host_ops = { | ||
.rd_own_conf = hisi_pcie_cfg_read, | ||
.wr_own_conf = hisi_pcie_cfg_write, | ||
.link_up = hisi_pcie_link_up, | ||
}; | ||
|
||
static int __init hisi_add_pcie_port(struct pcie_port *pp, | ||
struct platform_device *pdev) | ||
{ | ||
int ret; | ||
u32 port_id; | ||
struct hisi_pcie *hisi_pcie = to_hisi_pcie(pp); | ||
|
||
if (of_property_read_u32(pdev->dev.of_node, "port-id", &port_id)) { | ||
dev_err(&pdev->dev, "failed to read port-id\n"); | ||
return -EINVAL; | ||
} | ||
if (port_id > 3) { | ||
dev_err(&pdev->dev, "Invalid port-id: %d\n", port_id); | ||
return -EINVAL; | ||
} | ||
hisi_pcie->port_id = port_id; | ||
|
||
pp->ops = &hisi_pcie_host_ops; | ||
|
||
ret = dw_pcie_host_init(pp); | ||
if (ret) { | ||
dev_err(&pdev->dev, "failed to initialize host\n"); | ||
return ret; | ||
} | ||
|
||
return 0; | ||
} | ||
|
||
static int __init hisi_pcie_probe(struct platform_device *pdev) | ||
{ | ||
struct hisi_pcie *hisi_pcie; | ||
struct pcie_port *pp; | ||
struct resource *reg; | ||
int ret; | ||
|
||
hisi_pcie = devm_kzalloc(&pdev->dev, sizeof(*hisi_pcie), GFP_KERNEL); | ||
if (!hisi_pcie) | ||
return -ENOMEM; | ||
|
||
pp = &hisi_pcie->pp; | ||
pp->dev = &pdev->dev; | ||
|
||
hisi_pcie->subctrl = | ||
syscon_regmap_lookup_by_compatible("hisilicon,pcie-sas-subctrl"); | ||
if (IS_ERR(hisi_pcie->subctrl)) { | ||
dev_err(pp->dev, "cannot get subctrl base\n"); | ||
return PTR_ERR(hisi_pcie->subctrl); | ||
} | ||
|
||
reg = platform_get_resource_byname(pdev, IORESOURCE_MEM, "rc_dbi"); | ||
hisi_pcie->reg_base = devm_ioremap_resource(&pdev->dev, reg); | ||
if (IS_ERR(hisi_pcie->reg_base)) { | ||
dev_err(pp->dev, "cannot get rc_dbi base\n"); | ||
return PTR_ERR(hisi_pcie->reg_base); | ||
} | ||
|
||
hisi_pcie->pp.dbi_base = hisi_pcie->reg_base; | ||
|
||
ret = hisi_add_pcie_port(pp, pdev); | ||
if (ret) | ||
return ret; | ||
|
||
platform_set_drvdata(pdev, hisi_pcie); | ||
|
||
dev_warn(pp->dev, "only 32-bit config accesses supported; smaller writes may corrupt adjacent RW1C fields\n"); | ||
|
||
return 0; | ||
} | ||
|
||
static const struct of_device_id hisi_pcie_of_match[] = { | ||
{.compatible = "hisilicon,hip05-pcie",}, | ||
{}, | ||
}; | ||
|
||
MODULE_DEVICE_TABLE(of, hisi_pcie_of_match); | ||
|
||
static struct platform_driver hisi_pcie_driver = { | ||
.probe = hisi_pcie_probe, | ||
.driver = { | ||
.name = "hisi-pcie", | ||
.of_match_table = hisi_pcie_of_match, | ||
}, | ||
}; | ||
|
||
module_platform_driver(hisi_pcie_driver); |