diff options
Diffstat (limited to 'drivers/pci/controller/dwc/pcie-designware-host.c')
| -rw-r--r-- | drivers/pci/controller/dwc/pcie-designware-host.c | 83 | 
1 files changed, 81 insertions, 2 deletions
| diff --git a/drivers/pci/controller/dwc/pcie-designware-host.c b/drivers/pci/controller/dwc/pcie-designware-host.c index ecc33f6789e3..906277f9ffaf 100644 --- a/drivers/pci/controller/dwc/pcie-designware-host.c +++ b/drivers/pci/controller/dwc/pcie-designware-host.c @@ -227,7 +227,7 @@ static const struct irq_domain_ops dw_pcie_msi_domain_ops = {  int dw_pcie_allocate_domains(struct dw_pcie_rp *pp)  {  	struct dw_pcie *pci = to_dw_pcie_from_pp(pp); -	struct fwnode_handle *fwnode = of_node_to_fwnode(pci->dev->of_node); +	struct fwnode_handle *fwnode = of_fwnode_handle(pci->dev->of_node);  	pp->irq_domain = irq_domain_create_linear(fwnode, pp->num_vectors,  					       &dw_pcie_msi_domain_ops, pp); @@ -523,6 +523,13 @@ int dw_pcie_host_init(struct dw_pcie_rp *pp)  	dw_pcie_iatu_detect(pci); +	if (pci->num_lanes < 1) +		pci->num_lanes = dw_pcie_link_get_max_link_width(pci); + +	ret = of_pci_get_equalization_presets(dev, &pp->presets, pci->num_lanes); +	if (ret) +		goto err_free_msi; +  	/*  	 * Allocate the resource for MSG TLP before programming the iATU  	 * outbound window in dw_pcie_setup_rc(). Since the allocation depends @@ -567,7 +574,7 @@ int dw_pcie_host_init(struct dw_pcie_rp *pp)  	if (pp->ops->post_init)  		pp->ops->post_init(pp); -	dwc_pcie_debugfs_init(pci); +	dwc_pcie_debugfs_init(pci, DW_PCIE_RC_TYPE);  	return 0; @@ -828,6 +835,77 @@ static int dw_pcie_iatu_setup(struct dw_pcie_rp *pp)  	return 0;  } +static void dw_pcie_program_presets(struct dw_pcie_rp *pp, enum pci_bus_speed speed) +{ +	struct dw_pcie *pci = to_dw_pcie_from_pp(pp); +	u8 lane_eq_offset, lane_reg_size, cap_id; +	u8 *presets; +	u32 cap; +	int i; + +	if (speed == PCIE_SPEED_8_0GT) { +		presets = (u8 *)pp->presets.eq_presets_8gts; +		lane_eq_offset =  PCI_SECPCI_LE_CTRL; +		cap_id = PCI_EXT_CAP_ID_SECPCI; +		/* For data rate of 8 GT/S each lane equalization control is 16bits wide*/ +		lane_reg_size = 0x2; +	} else if (speed == PCIE_SPEED_16_0GT) { +		presets = pp->presets.eq_presets_Ngts[EQ_PRESET_TYPE_16GTS - 1]; +		lane_eq_offset = PCI_PL_16GT_LE_CTRL; +		cap_id = PCI_EXT_CAP_ID_PL_16GT; +		lane_reg_size = 0x1; +	} else if (speed == PCIE_SPEED_32_0GT) { +		presets =  pp->presets.eq_presets_Ngts[EQ_PRESET_TYPE_32GTS - 1]; +		lane_eq_offset = PCI_PL_32GT_LE_CTRL; +		cap_id = PCI_EXT_CAP_ID_PL_32GT; +		lane_reg_size = 0x1; +	} else if (speed == PCIE_SPEED_64_0GT) { +		presets =  pp->presets.eq_presets_Ngts[EQ_PRESET_TYPE_64GTS - 1]; +		lane_eq_offset = PCI_PL_64GT_LE_CTRL; +		cap_id = PCI_EXT_CAP_ID_PL_64GT; +		lane_reg_size = 0x1; +	} else { +		return; +	} + +	if (presets[0] == PCI_EQ_RESV) +		return; + +	cap = dw_pcie_find_ext_capability(pci, cap_id); +	if (!cap) +		return; + +	/* +	 * Write preset values to the registers byte-by-byte for the given +	 * number of lanes and register size. +	 */ +	for (i = 0; i < pci->num_lanes * lane_reg_size; i++) +		dw_pcie_writeb_dbi(pci, cap + lane_eq_offset + i, presets[i]); +} + +static void dw_pcie_config_presets(struct dw_pcie_rp *pp) +{ +	struct dw_pcie *pci = to_dw_pcie_from_pp(pp); +	enum pci_bus_speed speed = pcie_link_speed[pci->max_link_speed]; + +	/* +	 * Lane equalization settings need to be applied for all data rates the +	 * controller supports and for all supported lanes. +	 */ + +	if (speed >= PCIE_SPEED_8_0GT) +		dw_pcie_program_presets(pp, PCIE_SPEED_8_0GT); + +	if (speed >= PCIE_SPEED_16_0GT) +		dw_pcie_program_presets(pp, PCIE_SPEED_16_0GT); + +	if (speed >= PCIE_SPEED_32_0GT) +		dw_pcie_program_presets(pp, PCIE_SPEED_32_0GT); + +	if (speed >= PCIE_SPEED_64_0GT) +		dw_pcie_program_presets(pp, PCIE_SPEED_64_0GT); +} +  int dw_pcie_setup_rc(struct dw_pcie_rp *pp)  {  	struct dw_pcie *pci = to_dw_pcie_from_pp(pp); @@ -881,6 +959,7 @@ int dw_pcie_setup_rc(struct dw_pcie_rp *pp)  		PCI_COMMAND_MASTER | PCI_COMMAND_SERR;  	dw_pcie_writel_dbi(pci, PCI_COMMAND, val); +	dw_pcie_config_presets(pp);  	/*  	 * If the platform provides its own child bus config accesses, it means  	 * the platform uses its own address translation component rather than | 
