2013-06-21 15:24:54 +08:00
|
|
|
/*
|
2013-07-31 16:14:10 +08:00
|
|
|
* Synopsys Designware PCIe host controller driver
|
2013-06-21 15:24:54 +08:00
|
|
|
*
|
|
|
|
* Copyright (C) 2013 Samsung Electronics Co., Ltd.
|
|
|
|
* http://www.samsung.com
|
|
|
|
*
|
|
|
|
* Author: Jingoo Han <jg1.han@samsung.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.
|
|
|
|
*/
|
|
|
|
|
2013-09-06 14:54:59 +08:00
|
|
|
#include <linux/irq.h>
|
|
|
|
#include <linux/irqdomain.h>
|
2013-06-21 15:24:54 +08:00
|
|
|
#include <linux/kernel.h>
|
2013-09-06 14:54:59 +08:00
|
|
|
#include <linux/msi.h>
|
2013-06-21 15:24:54 +08:00
|
|
|
#include <linux/of_address.h>
|
2014-03-05 21:25:51 +08:00
|
|
|
#include <linux/of_pci.h>
|
2013-06-21 15:24:54 +08:00
|
|
|
#include <linux/pci.h>
|
|
|
|
#include <linux/pci_regs.h>
|
2014-07-17 17:00:40 +08:00
|
|
|
#include <linux/platform_device.h>
|
2013-06-21 15:24:54 +08:00
|
|
|
#include <linux/types.h>
|
PCI: designware: Add generic dw_pcie_wait_for_link()
Several DesignWare-based drivers (dra7xx, exynos, imx6, keystone, qcom, and
spear13xx) had similar loops waiting for the link to come up.
Add a generic dw_pcie_wait_for_link() for use by all these drivers so the
waiting is done consistently, e.g., always using usleep_range() rather than
mdelay() and using similar timeouts and retry counts.
Note that this changes the Keystone link training/wait for link strategy,
so we initiate link training, then wait longer for the link to come up
before re-initiating link training.
[bhelgaas: changelog, split into its own patch, update pci-keystone.c, pcie-qcom.c]
Signed-off-by: Joao Pinto <jpinto@synopsys.com>
Signed-off-by: Bjorn Helgaas <bhelgaas@google.com>
Acked-by: Pratyush Anand <pratyush.anand@gmail.com>
2016-03-11 04:44:35 +08:00
|
|
|
#include <linux/delay.h>
|
2013-06-21 15:24:54 +08:00
|
|
|
|
2013-07-31 16:14:10 +08:00
|
|
|
#include "pcie-designware.h"
|
2013-06-21 15:24:54 +08:00
|
|
|
|
2016-08-10 18:02:38 +08:00
|
|
|
/* Parameters for the waiting for link up routine */
|
|
|
|
#define LINK_WAIT_MAX_RETRIES 10
|
|
|
|
#define LINK_WAIT_USLEEP_MIN 90000
|
|
|
|
#define LINK_WAIT_USLEEP_MAX 100000
|
|
|
|
|
2016-08-18 02:26:07 +08:00
|
|
|
/* Parameters for the waiting for iATU enabled routine */
|
|
|
|
#define LINK_WAIT_MAX_IATU_RETRIES 5
|
|
|
|
#define LINK_WAIT_IATU_MIN 9000
|
|
|
|
#define LINK_WAIT_IATU_MAX 10000
|
|
|
|
|
|
|
|
/* Synopsys-specific PCIe configuration registers */
|
2013-06-21 15:24:54 +08:00
|
|
|
#define PCIE_PORT_LINK_CONTROL 0x710
|
|
|
|
#define PORT_LINK_MODE_MASK (0x3f << 16)
|
2013-07-31 16:14:10 +08:00
|
|
|
#define PORT_LINK_MODE_1_LANES (0x1 << 16)
|
|
|
|
#define PORT_LINK_MODE_2_LANES (0x3 << 16)
|
2013-06-21 15:24:54 +08:00
|
|
|
#define PORT_LINK_MODE_4_LANES (0x7 << 16)
|
2015-05-13 14:44:34 +08:00
|
|
|
#define PORT_LINK_MODE_8_LANES (0xf << 16)
|
2013-06-21 15:24:54 +08:00
|
|
|
|
|
|
|
#define PCIE_LINK_WIDTH_SPEED_CONTROL 0x80C
|
|
|
|
#define PORT_LOGIC_SPEED_CHANGE (0x1 << 17)
|
2015-08-26 11:17:34 +08:00
|
|
|
#define PORT_LOGIC_LINK_WIDTH_MASK (0x1f << 8)
|
2013-07-31 16:14:10 +08:00
|
|
|
#define PORT_LOGIC_LINK_WIDTH_1_LANES (0x1 << 8)
|
|
|
|
#define PORT_LOGIC_LINK_WIDTH_2_LANES (0x2 << 8)
|
|
|
|
#define PORT_LOGIC_LINK_WIDTH_4_LANES (0x4 << 8)
|
2015-05-13 14:44:34 +08:00
|
|
|
#define PORT_LOGIC_LINK_WIDTH_8_LANES (0x8 << 8)
|
2013-06-21 15:24:54 +08:00
|
|
|
|
|
|
|
#define PCIE_MSI_ADDR_LO 0x820
|
|
|
|
#define PCIE_MSI_ADDR_HI 0x824
|
|
|
|
#define PCIE_MSI_INTR0_ENABLE 0x828
|
|
|
|
#define PCIE_MSI_INTR0_MASK 0x82C
|
|
|
|
#define PCIE_MSI_INTR0_STATUS 0x830
|
|
|
|
|
|
|
|
#define PCIE_ATU_VIEWPORT 0x900
|
|
|
|
#define PCIE_ATU_REGION_INBOUND (0x1 << 31)
|
|
|
|
#define PCIE_ATU_REGION_OUTBOUND (0x0 << 31)
|
2016-07-05 00:14:42 +08:00
|
|
|
#define PCIE_ATU_REGION_INDEX2 (0x2 << 0)
|
2013-06-21 15:24:54 +08:00
|
|
|
#define PCIE_ATU_REGION_INDEX1 (0x1 << 0)
|
|
|
|
#define PCIE_ATU_REGION_INDEX0 (0x0 << 0)
|
|
|
|
#define PCIE_ATU_CR1 0x904
|
|
|
|
#define PCIE_ATU_TYPE_MEM (0x0 << 0)
|
|
|
|
#define PCIE_ATU_TYPE_IO (0x2 << 0)
|
|
|
|
#define PCIE_ATU_TYPE_CFG0 (0x4 << 0)
|
|
|
|
#define PCIE_ATU_TYPE_CFG1 (0x5 << 0)
|
|
|
|
#define PCIE_ATU_CR2 0x908
|
|
|
|
#define PCIE_ATU_ENABLE (0x1 << 31)
|
|
|
|
#define PCIE_ATU_BAR_MODE_ENABLE (0x1 << 30)
|
|
|
|
#define PCIE_ATU_LOWER_BASE 0x90C
|
|
|
|
#define PCIE_ATU_UPPER_BASE 0x910
|
|
|
|
#define PCIE_ATU_LIMIT 0x914
|
|
|
|
#define PCIE_ATU_LOWER_TARGET 0x918
|
|
|
|
#define PCIE_ATU_BUS(x) (((x) & 0xff) << 24)
|
|
|
|
#define PCIE_ATU_DEV(x) (((x) & 0x1f) << 19)
|
|
|
|
#define PCIE_ATU_FUNC(x) (((x) & 0x7) << 16)
|
|
|
|
#define PCIE_ATU_UPPER_TARGET 0x91C
|
|
|
|
|
2016-08-10 18:02:39 +08:00
|
|
|
/*
|
|
|
|
* iATU Unroll-specific register definitions
|
|
|
|
* From 4.80 core version the address translation will be made by unroll
|
|
|
|
*/
|
|
|
|
#define PCIE_ATU_UNR_REGION_CTRL1 0x00
|
|
|
|
#define PCIE_ATU_UNR_REGION_CTRL2 0x04
|
|
|
|
#define PCIE_ATU_UNR_LOWER_BASE 0x08
|
|
|
|
#define PCIE_ATU_UNR_UPPER_BASE 0x0C
|
|
|
|
#define PCIE_ATU_UNR_LIMIT 0x10
|
|
|
|
#define PCIE_ATU_UNR_LOWER_TARGET 0x14
|
|
|
|
#define PCIE_ATU_UNR_UPPER_TARGET 0x18
|
|
|
|
|
|
|
|
/* Register address builder */
|
|
|
|
#define PCIE_GET_ATU_OUTB_UNR_REG_OFFSET(region) ((0x3 << 20) | (region << 9))
|
|
|
|
|
2016-03-11 04:44:44 +08:00
|
|
|
/* PCIe Port Logic registers */
|
|
|
|
#define PLR_OFFSET 0x700
|
|
|
|
#define PCIE_PHY_DEBUG_R1 (PLR_OFFSET + 0x2c)
|
2016-08-18 04:57:37 +08:00
|
|
|
#define PCIE_PHY_DEBUG_R1_LINK_UP (0x1 << 4)
|
|
|
|
#define PCIE_PHY_DEBUG_R1_LINK_IN_TRAINING (0x1 << 29)
|
2016-03-11 04:44:44 +08:00
|
|
|
|
2015-10-30 08:57:21 +08:00
|
|
|
static struct pci_ops dw_pcie_ops;
|
2013-06-21 15:24:54 +08:00
|
|
|
|
2015-10-09 03:27:48 +08:00
|
|
|
int dw_pcie_cfg_read(void __iomem *addr, int size, u32 *val)
|
2013-06-21 15:24:54 +08:00
|
|
|
{
|
2015-10-09 03:27:53 +08:00
|
|
|
if ((uintptr_t)addr & (size - 1)) {
|
|
|
|
*val = 0;
|
|
|
|
return PCIBIOS_BAD_REGISTER_NUMBER;
|
|
|
|
}
|
|
|
|
|
2015-10-09 03:27:43 +08:00
|
|
|
if (size == 4)
|
|
|
|
*val = readl(addr);
|
2013-06-21 15:24:54 +08:00
|
|
|
else if (size == 2)
|
2015-10-09 03:27:48 +08:00
|
|
|
*val = readw(addr);
|
2015-10-09 03:27:43 +08:00
|
|
|
else if (size == 1)
|
2015-10-09 03:27:48 +08:00
|
|
|
*val = readb(addr);
|
2015-10-09 03:27:43 +08:00
|
|
|
else {
|
|
|
|
*val = 0;
|
2013-06-21 15:24:54 +08:00
|
|
|
return PCIBIOS_BAD_REGISTER_NUMBER;
|
2015-10-09 03:27:43 +08:00
|
|
|
}
|
2013-06-21 15:24:54 +08:00
|
|
|
|
|
|
|
return PCIBIOS_SUCCESSFUL;
|
|
|
|
}
|
|
|
|
|
2015-10-09 03:27:48 +08:00
|
|
|
int dw_pcie_cfg_write(void __iomem *addr, int size, u32 val)
|
2013-06-21 15:24:54 +08:00
|
|
|
{
|
2015-10-09 03:27:53 +08:00
|
|
|
if ((uintptr_t)addr & (size - 1))
|
|
|
|
return PCIBIOS_BAD_REGISTER_NUMBER;
|
|
|
|
|
2013-06-21 15:24:54 +08:00
|
|
|
if (size == 4)
|
|
|
|
writel(val, addr);
|
|
|
|
else if (size == 2)
|
2015-10-09 03:27:48 +08:00
|
|
|
writew(val, addr);
|
2013-06-21 15:24:54 +08:00
|
|
|
else if (size == 1)
|
2015-10-09 03:27:48 +08:00
|
|
|
writeb(val, addr);
|
2013-06-21 15:24:54 +08:00
|
|
|
else
|
|
|
|
return PCIBIOS_BAD_REGISTER_NUMBER;
|
|
|
|
|
|
|
|
return PCIBIOS_SUCCESSFUL;
|
|
|
|
}
|
|
|
|
|
2016-10-07 02:25:47 +08:00
|
|
|
u32 dw_pcie_readl_rc(struct pcie_port *pp, u32 reg)
|
2013-06-21 15:24:54 +08:00
|
|
|
{
|
2013-07-31 16:14:10 +08:00
|
|
|
if (pp->ops->readl_rc)
|
2016-10-07 02:25:46 +08:00
|
|
|
return pp->ops->readl_rc(pp, reg);
|
2016-08-18 03:17:58 +08:00
|
|
|
|
|
|
|
return readl(pp->dbi_base + reg);
|
2013-06-21 15:24:54 +08:00
|
|
|
}
|
|
|
|
|
2016-10-07 02:25:47 +08:00
|
|
|
void dw_pcie_writel_rc(struct pcie_port *pp, u32 reg, u32 val)
|
2013-06-21 15:24:54 +08:00
|
|
|
{
|
2013-07-31 16:14:10 +08:00
|
|
|
if (pp->ops->writel_rc)
|
2016-10-07 02:25:46 +08:00
|
|
|
pp->ops->writel_rc(pp, reg, val);
|
2013-07-31 16:14:10 +08:00
|
|
|
else
|
2013-08-28 19:53:30 +08:00
|
|
|
writel(val, pp->dbi_base + reg);
|
2013-06-21 15:24:54 +08:00
|
|
|
}
|
|
|
|
|
2016-10-11 21:33:33 +08:00
|
|
|
static u32 dw_pcie_readl_unroll(struct pcie_port *pp, u32 index, u32 reg)
|
2016-08-10 18:02:39 +08:00
|
|
|
{
|
|
|
|
u32 offset = PCIE_GET_ATU_OUTB_UNR_REG_OFFSET(index);
|
|
|
|
|
2016-10-11 21:26:21 +08:00
|
|
|
return dw_pcie_readl_rc(pp, offset + reg);
|
2016-08-10 18:02:39 +08:00
|
|
|
}
|
|
|
|
|
2016-10-11 21:33:00 +08:00
|
|
|
static void dw_pcie_writel_unroll(struct pcie_port *pp, u32 index, u32 reg,
|
|
|
|
u32 val)
|
2016-08-10 18:02:39 +08:00
|
|
|
{
|
|
|
|
u32 offset = PCIE_GET_ATU_OUTB_UNR_REG_OFFSET(index);
|
|
|
|
|
2016-10-07 02:25:46 +08:00
|
|
|
dw_pcie_writel_rc(pp, offset + reg, val);
|
2016-08-10 18:02:39 +08:00
|
|
|
}
|
|
|
|
|
2013-10-09 23:12:37 +08:00
|
|
|
static int dw_pcie_rd_own_conf(struct pcie_port *pp, int where, int size,
|
|
|
|
u32 *val)
|
2013-06-21 15:24:54 +08:00
|
|
|
{
|
2013-07-31 16:14:10 +08:00
|
|
|
if (pp->ops->rd_own_conf)
|
2016-01-06 05:48:11 +08:00
|
|
|
return pp->ops->rd_own_conf(pp, where, size, val);
|
2013-07-31 16:14:10 +08:00
|
|
|
|
2016-01-06 05:48:11 +08:00
|
|
|
return dw_pcie_cfg_read(pp->dbi_base + where, size, val);
|
2013-06-21 15:24:54 +08:00
|
|
|
}
|
|
|
|
|
2013-10-09 23:12:37 +08:00
|
|
|
static int dw_pcie_wr_own_conf(struct pcie_port *pp, int where, int size,
|
|
|
|
u32 val)
|
2013-06-21 15:24:54 +08:00
|
|
|
{
|
2013-07-31 16:14:10 +08:00
|
|
|
if (pp->ops->wr_own_conf)
|
2016-01-06 05:48:11 +08:00
|
|
|
return pp->ops->wr_own_conf(pp, where, size, val);
|
2013-07-31 16:14:10 +08:00
|
|
|
|
2016-01-06 05:48:11 +08:00
|
|
|
return dw_pcie_cfg_write(pp->dbi_base + where, size, val);
|
2013-06-21 15:24:54 +08:00
|
|
|
}
|
|
|
|
|
2015-04-30 16:22:28 +08:00
|
|
|
static void dw_pcie_prog_outbound_atu(struct pcie_port *pp, int index,
|
|
|
|
int type, u64 cpu_addr, u64 pci_addr, u32 size)
|
|
|
|
{
|
2016-08-18 02:26:07 +08:00
|
|
|
u32 retries, val;
|
2015-12-18 20:38:55 +08:00
|
|
|
|
2016-08-10 18:02:39 +08:00
|
|
|
if (pp->iatu_unroll_enabled) {
|
2016-10-11 21:33:00 +08:00
|
|
|
dw_pcie_writel_unroll(pp, index, PCIE_ATU_UNR_LOWER_BASE,
|
|
|
|
lower_32_bits(cpu_addr));
|
|
|
|
dw_pcie_writel_unroll(pp, index, PCIE_ATU_UNR_UPPER_BASE,
|
|
|
|
upper_32_bits(cpu_addr));
|
|
|
|
dw_pcie_writel_unroll(pp, index, PCIE_ATU_UNR_LIMIT,
|
|
|
|
lower_32_bits(cpu_addr + size - 1));
|
|
|
|
dw_pcie_writel_unroll(pp, index, PCIE_ATU_UNR_LOWER_TARGET,
|
|
|
|
lower_32_bits(pci_addr));
|
|
|
|
dw_pcie_writel_unroll(pp, index, PCIE_ATU_UNR_UPPER_TARGET,
|
|
|
|
upper_32_bits(pci_addr));
|
|
|
|
dw_pcie_writel_unroll(pp, index, PCIE_ATU_UNR_REGION_CTRL1,
|
|
|
|
type);
|
|
|
|
dw_pcie_writel_unroll(pp, index, PCIE_ATU_UNR_REGION_CTRL2,
|
|
|
|
PCIE_ATU_ENABLE);
|
2016-08-10 18:02:39 +08:00
|
|
|
} else {
|
2016-10-07 02:25:46 +08:00
|
|
|
dw_pcie_writel_rc(pp, PCIE_ATU_VIEWPORT,
|
|
|
|
PCIE_ATU_REGION_OUTBOUND | index);
|
|
|
|
dw_pcie_writel_rc(pp, PCIE_ATU_LOWER_BASE,
|
|
|
|
lower_32_bits(cpu_addr));
|
|
|
|
dw_pcie_writel_rc(pp, PCIE_ATU_UPPER_BASE,
|
|
|
|
upper_32_bits(cpu_addr));
|
|
|
|
dw_pcie_writel_rc(pp, PCIE_ATU_LIMIT,
|
|
|
|
lower_32_bits(cpu_addr + size - 1));
|
|
|
|
dw_pcie_writel_rc(pp, PCIE_ATU_LOWER_TARGET,
|
|
|
|
lower_32_bits(pci_addr));
|
|
|
|
dw_pcie_writel_rc(pp, PCIE_ATU_UPPER_TARGET,
|
|
|
|
upper_32_bits(pci_addr));
|
|
|
|
dw_pcie_writel_rc(pp, PCIE_ATU_CR1, type);
|
|
|
|
dw_pcie_writel_rc(pp, PCIE_ATU_CR2, PCIE_ATU_ENABLE);
|
2016-08-10 18:02:39 +08:00
|
|
|
}
|
2015-12-18 20:38:55 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Make sure ATU enable takes effect before any subsequent config
|
|
|
|
* and I/O accesses.
|
|
|
|
*/
|
2016-08-18 02:26:07 +08:00
|
|
|
for (retries = 0; retries < LINK_WAIT_MAX_IATU_RETRIES; retries++) {
|
2016-08-10 18:02:39 +08:00
|
|
|
if (pp->iatu_unroll_enabled)
|
|
|
|
val = dw_pcie_readl_unroll(pp, index,
|
|
|
|
PCIE_ATU_UNR_REGION_CTRL2);
|
|
|
|
else
|
|
|
|
val = dw_pcie_readl_rc(pp, PCIE_ATU_CR2);
|
|
|
|
|
2016-08-18 02:26:07 +08:00
|
|
|
if (val == PCIE_ATU_ENABLE)
|
|
|
|
return;
|
|
|
|
|
|
|
|
usleep_range(LINK_WAIT_IATU_MIN, LINK_WAIT_IATU_MAX);
|
|
|
|
}
|
|
|
|
dev_err(pp->dev, "iATU is not being enabled\n");
|
2015-04-30 16:22:28 +08:00
|
|
|
}
|
|
|
|
|
2013-09-06 14:54:59 +08:00
|
|
|
static struct irq_chip dw_msi_irq_chip = {
|
|
|
|
.name = "PCI-MSI",
|
2014-11-23 19:23:20 +08:00
|
|
|
.irq_enable = pci_msi_unmask_irq,
|
|
|
|
.irq_disable = pci_msi_mask_irq,
|
|
|
|
.irq_mask = pci_msi_mask_irq,
|
|
|
|
.irq_unmask = pci_msi_unmask_irq,
|
2013-09-06 14:54:59 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
/* MSI int handler */
|
2014-03-29 00:52:58 +08:00
|
|
|
irqreturn_t dw_handle_msi_irq(struct pcie_port *pp)
|
2013-09-06 14:54:59 +08:00
|
|
|
{
|
|
|
|
unsigned long val;
|
2013-10-09 20:32:12 +08:00
|
|
|
int i, pos, irq;
|
2014-03-29 00:52:58 +08:00
|
|
|
irqreturn_t ret = IRQ_NONE;
|
2013-09-06 14:54:59 +08:00
|
|
|
|
|
|
|
for (i = 0; i < MAX_MSI_CTRLS; i++) {
|
|
|
|
dw_pcie_rd_own_conf(pp, PCIE_MSI_INTR0_STATUS + i * 12, 4,
|
|
|
|
(u32 *)&val);
|
|
|
|
if (val) {
|
2014-03-29 00:52:58 +08:00
|
|
|
ret = IRQ_HANDLED;
|
2013-09-06 14:54:59 +08:00
|
|
|
pos = 0;
|
|
|
|
while ((pos = find_next_bit(&val, 32, pos)) != 32) {
|
2013-10-09 20:32:12 +08:00
|
|
|
irq = irq_find_mapping(pp->irq_domain,
|
|
|
|
i * 32 + pos);
|
2013-12-13 02:29:03 +08:00
|
|
|
dw_pcie_wr_own_conf(pp,
|
|
|
|
PCIE_MSI_INTR0_STATUS + i * 12,
|
|
|
|
4, 1 << pos);
|
2013-10-09 20:32:12 +08:00
|
|
|
generic_handle_irq(irq);
|
2013-09-06 14:54:59 +08:00
|
|
|
pos++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-03-29 00:52:58 +08:00
|
|
|
|
|
|
|
return ret;
|
2013-09-06 14:54:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void dw_pcie_msi_init(struct pcie_port *pp)
|
|
|
|
{
|
2015-09-19 02:58:35 +08:00
|
|
|
u64 msi_target;
|
|
|
|
|
2013-09-06 14:54:59 +08:00
|
|
|
pp->msi_data = __get_free_pages(GFP_KERNEL, 0);
|
2015-09-19 02:58:35 +08:00
|
|
|
msi_target = virt_to_phys((void *)pp->msi_data);
|
2013-09-06 14:54:59 +08:00
|
|
|
|
|
|
|
/* program the msi_data */
|
|
|
|
dw_pcie_wr_own_conf(pp, PCIE_MSI_ADDR_LO, 4,
|
2015-09-19 02:58:35 +08:00
|
|
|
(u32)(msi_target & 0xffffffff));
|
|
|
|
dw_pcie_wr_own_conf(pp, PCIE_MSI_ADDR_HI, 4,
|
|
|
|
(u32)(msi_target >> 32 & 0xffffffff));
|
2013-09-06 14:54:59 +08:00
|
|
|
}
|
|
|
|
|
2014-07-22 00:58:42 +08:00
|
|
|
static void dw_pcie_msi_clear_irq(struct pcie_port *pp, int irq)
|
|
|
|
{
|
|
|
|
unsigned int res, bit, val;
|
|
|
|
|
|
|
|
res = (irq / 32) * 12;
|
|
|
|
bit = irq % 32;
|
|
|
|
dw_pcie_rd_own_conf(pp, PCIE_MSI_INTR0_ENABLE + res, 4, &val);
|
|
|
|
val &= ~(1 << bit);
|
|
|
|
dw_pcie_wr_own_conf(pp, PCIE_MSI_INTR0_ENABLE + res, 4, val);
|
|
|
|
}
|
|
|
|
|
2013-11-29 21:35:24 +08:00
|
|
|
static void clear_irq_range(struct pcie_port *pp, unsigned int irq_base,
|
2013-12-27 08:30:25 +08:00
|
|
|
unsigned int nvec, unsigned int pos)
|
2013-11-29 21:35:24 +08:00
|
|
|
{
|
2014-07-22 00:58:42 +08:00
|
|
|
unsigned int i;
|
2013-11-29 21:35:24 +08:00
|
|
|
|
2013-12-10 06:11:25 +08:00
|
|
|
for (i = 0; i < nvec; i++) {
|
2013-11-29 21:35:24 +08:00
|
|
|
irq_set_msi_desc_off(irq_base, i, NULL);
|
2013-12-27 08:30:25 +08:00
|
|
|
/* Disable corresponding interrupt on MSI controller */
|
2014-07-22 00:58:42 +08:00
|
|
|
if (pp->ops->msi_clear_irq)
|
|
|
|
pp->ops->msi_clear_irq(pp, pos + i);
|
|
|
|
else
|
|
|
|
dw_pcie_msi_clear_irq(pp, pos + i);
|
2013-11-29 21:35:24 +08:00
|
|
|
}
|
2014-10-01 00:36:27 +08:00
|
|
|
|
|
|
|
bitmap_release_region(pp->msi_irq_in_use, pos, order_base_2(nvec));
|
2013-11-29 21:35:24 +08:00
|
|
|
}
|
|
|
|
|
2014-07-22 00:58:42 +08:00
|
|
|
static void dw_pcie_msi_set_irq(struct pcie_port *pp, int irq)
|
|
|
|
{
|
|
|
|
unsigned int res, bit, val;
|
|
|
|
|
|
|
|
res = (irq / 32) * 12;
|
|
|
|
bit = irq % 32;
|
|
|
|
dw_pcie_rd_own_conf(pp, PCIE_MSI_INTR0_ENABLE + res, 4, &val);
|
|
|
|
val |= 1 << bit;
|
|
|
|
dw_pcie_wr_own_conf(pp, PCIE_MSI_INTR0_ENABLE + res, 4, val);
|
|
|
|
}
|
|
|
|
|
2013-09-06 14:54:59 +08:00
|
|
|
static int assign_irq(int no_irqs, struct msi_desc *desc, int *pos)
|
|
|
|
{
|
2014-10-01 00:36:27 +08:00
|
|
|
int irq, pos0, i;
|
2015-10-30 08:57:21 +08:00
|
|
|
struct pcie_port *pp = (struct pcie_port *) msi_desc_to_pci_sysdata(desc);
|
2013-09-06 14:54:59 +08:00
|
|
|
|
2014-10-01 00:36:27 +08:00
|
|
|
pos0 = bitmap_find_free_region(pp->msi_irq_in_use, MAX_MSI_IRQS,
|
|
|
|
order_base_2(no_irqs));
|
|
|
|
if (pos0 < 0)
|
|
|
|
goto no_valid_irq;
|
2013-09-06 14:54:59 +08:00
|
|
|
|
2013-10-09 20:32:12 +08:00
|
|
|
irq = irq_find_mapping(pp->irq_domain, pos0);
|
|
|
|
if (!irq)
|
2013-09-06 14:54:59 +08:00
|
|
|
goto no_valid_irq;
|
|
|
|
|
2013-11-29 21:35:24 +08:00
|
|
|
/*
|
|
|
|
* irq_create_mapping (called from dw_pcie_host_init) pre-allocates
|
|
|
|
* descs so there is no need to allocate descs here. We can therefore
|
|
|
|
* assume that if irq_find_mapping above returns non-zero, then the
|
|
|
|
* descs are also successfully allocated.
|
|
|
|
*/
|
|
|
|
|
2013-12-10 06:11:25 +08:00
|
|
|
for (i = 0; i < no_irqs; i++) {
|
2013-11-29 21:35:24 +08:00
|
|
|
if (irq_set_msi_desc_off(irq, i, desc) != 0) {
|
|
|
|
clear_irq_range(pp, irq, i, pos0);
|
|
|
|
goto no_valid_irq;
|
|
|
|
}
|
2013-09-06 14:54:59 +08:00
|
|
|
/*Enable corresponding interrupt in MSI interrupt controller */
|
2014-07-22 00:58:42 +08:00
|
|
|
if (pp->ops->msi_set_irq)
|
|
|
|
pp->ops->msi_set_irq(pp, pos0 + i);
|
|
|
|
else
|
|
|
|
dw_pcie_msi_set_irq(pp, pos0 + i);
|
2013-09-06 14:54:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
*pos = pos0;
|
2015-09-19 02:58:35 +08:00
|
|
|
desc->nvec_used = no_irqs;
|
|
|
|
desc->msi_attrib.multiple = order_base_2(no_irqs);
|
|
|
|
|
2013-09-06 14:54:59 +08:00
|
|
|
return irq;
|
|
|
|
|
|
|
|
no_valid_irq:
|
|
|
|
*pos = pos0;
|
|
|
|
return -ENOSPC;
|
|
|
|
}
|
|
|
|
|
2015-09-19 02:58:35 +08:00
|
|
|
static void dw_msi_setup_msg(struct pcie_port *pp, unsigned int irq, u32 pos)
|
2013-09-06 14:54:59 +08:00
|
|
|
{
|
|
|
|
struct msi_msg msg;
|
2015-09-19 02:58:35 +08:00
|
|
|
u64 msi_target;
|
2013-09-06 14:54:59 +08:00
|
|
|
|
2014-09-23 22:28:58 +08:00
|
|
|
if (pp->ops->get_msi_addr)
|
2015-09-19 02:58:35 +08:00
|
|
|
msi_target = pp->ops->get_msi_addr(pp);
|
2014-07-22 00:58:42 +08:00
|
|
|
else
|
2015-09-19 02:58:35 +08:00
|
|
|
msi_target = virt_to_phys((void *)pp->msi_data);
|
|
|
|
|
|
|
|
msg.address_lo = (u32)(msi_target & 0xffffffff);
|
|
|
|
msg.address_hi = (u32)(msi_target >> 32 & 0xffffffff);
|
2014-09-23 22:28:59 +08:00
|
|
|
|
|
|
|
if (pp->ops->get_msi_data)
|
|
|
|
msg.data = pp->ops->get_msi_data(pp, pos);
|
|
|
|
else
|
|
|
|
msg.data = pos;
|
|
|
|
|
2014-11-09 23:10:34 +08:00
|
|
|
pci_write_msi_msg(irq, &msg);
|
2015-09-19 02:58:35 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int dw_msi_setup_irq(struct msi_controller *chip, struct pci_dev *pdev,
|
|
|
|
struct msi_desc *desc)
|
|
|
|
{
|
|
|
|
int irq, pos;
|
2015-10-30 08:57:21 +08:00
|
|
|
struct pcie_port *pp = pdev->bus->sysdata;
|
2015-09-19 02:58:35 +08:00
|
|
|
|
|
|
|
if (desc->msi_attrib.is_msix)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
irq = assign_irq(1, desc, &pos);
|
|
|
|
if (irq < 0)
|
|
|
|
return irq;
|
|
|
|
|
|
|
|
dw_msi_setup_msg(pp, irq, pos);
|
2013-09-06 14:54:59 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-09-19 02:58:35 +08:00
|
|
|
static int dw_msi_setup_irqs(struct msi_controller *chip, struct pci_dev *pdev,
|
|
|
|
int nvec, int type)
|
|
|
|
{
|
|
|
|
#ifdef CONFIG_PCI_MSI
|
|
|
|
int irq, pos;
|
|
|
|
struct msi_desc *desc;
|
2015-10-30 08:57:21 +08:00
|
|
|
struct pcie_port *pp = pdev->bus->sysdata;
|
2015-09-19 02:58:35 +08:00
|
|
|
|
|
|
|
/* MSI-X interrupts are not supported */
|
|
|
|
if (type == PCI_CAP_ID_MSIX)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
WARN_ON(!list_is_singular(&pdev->dev.msi_list));
|
|
|
|
desc = list_entry(pdev->dev.msi_list.next, struct msi_desc, list);
|
|
|
|
|
|
|
|
irq = assign_irq(nvec, desc, &pos);
|
|
|
|
if (irq < 0)
|
|
|
|
return irq;
|
|
|
|
|
|
|
|
dw_msi_setup_msg(pp, irq, pos);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
#else
|
|
|
|
return -EINVAL;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2014-11-12 08:45:45 +08:00
|
|
|
static void dw_msi_teardown_irq(struct msi_controller *chip, unsigned int irq)
|
2013-09-06 14:54:59 +08:00
|
|
|
{
|
2014-10-01 00:36:26 +08:00
|
|
|
struct irq_data *data = irq_get_irq_data(irq);
|
2015-06-01 16:05:41 +08:00
|
|
|
struct msi_desc *msi = irq_data_get_msi_desc(data);
|
2015-10-30 08:57:21 +08:00
|
|
|
struct pcie_port *pp = (struct pcie_port *) msi_desc_to_pci_sysdata(msi);
|
2014-10-01 00:36:26 +08:00
|
|
|
|
|
|
|
clear_irq_range(pp, irq, 1, data->hwirq);
|
2013-09-06 14:54:59 +08:00
|
|
|
}
|
|
|
|
|
2014-11-12 08:45:45 +08:00
|
|
|
static struct msi_controller dw_pcie_msi_chip = {
|
2013-09-06 14:54:59 +08:00
|
|
|
.setup_irq = dw_msi_setup_irq,
|
2015-09-19 02:58:35 +08:00
|
|
|
.setup_irqs = dw_msi_setup_irqs,
|
2013-09-06 14:54:59 +08:00
|
|
|
.teardown_irq = dw_msi_teardown_irq,
|
|
|
|
};
|
|
|
|
|
PCI: designware: Add generic dw_pcie_wait_for_link()
Several DesignWare-based drivers (dra7xx, exynos, imx6, keystone, qcom, and
spear13xx) had similar loops waiting for the link to come up.
Add a generic dw_pcie_wait_for_link() for use by all these drivers so the
waiting is done consistently, e.g., always using usleep_range() rather than
mdelay() and using similar timeouts and retry counts.
Note that this changes the Keystone link training/wait for link strategy,
so we initiate link training, then wait longer for the link to come up
before re-initiating link training.
[bhelgaas: changelog, split into its own patch, update pci-keystone.c, pcie-qcom.c]
Signed-off-by: Joao Pinto <jpinto@synopsys.com>
Signed-off-by: Bjorn Helgaas <bhelgaas@google.com>
Acked-by: Pratyush Anand <pratyush.anand@gmail.com>
2016-03-11 04:44:35 +08:00
|
|
|
int dw_pcie_wait_for_link(struct pcie_port *pp)
|
|
|
|
{
|
|
|
|
int retries;
|
|
|
|
|
|
|
|
/* check if the link is up or not */
|
|
|
|
for (retries = 0; retries < LINK_WAIT_MAX_RETRIES; retries++) {
|
|
|
|
if (dw_pcie_link_up(pp)) {
|
|
|
|
dev_info(pp->dev, "link up\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
usleep_range(LINK_WAIT_USLEEP_MIN, LINK_WAIT_USLEEP_MAX);
|
|
|
|
}
|
|
|
|
|
|
|
|
dev_err(pp->dev, "phy link never came up\n");
|
|
|
|
|
|
|
|
return -ETIMEDOUT;
|
|
|
|
}
|
|
|
|
|
2013-07-31 16:14:10 +08:00
|
|
|
int dw_pcie_link_up(struct pcie_port *pp)
|
|
|
|
{
|
2016-03-11 04:44:44 +08:00
|
|
|
u32 val;
|
|
|
|
|
2013-07-31 16:14:10 +08:00
|
|
|
if (pp->ops->link_up)
|
|
|
|
return pp->ops->link_up(pp);
|
2016-01-06 05:48:11 +08:00
|
|
|
|
2016-03-11 04:44:44 +08:00
|
|
|
val = readl(pp->dbi_base + PCIE_PHY_DEBUG_R1);
|
2016-08-18 04:57:37 +08:00
|
|
|
return ((val & PCIE_PHY_DEBUG_R1_LINK_UP) &&
|
|
|
|
(!(val & PCIE_PHY_DEBUG_R1_LINK_IN_TRAINING)));
|
2013-07-31 16:14:10 +08:00
|
|
|
}
|
|
|
|
|
2013-09-06 14:54:59 +08:00
|
|
|
static int dw_pcie_msi_map(struct irq_domain *domain, unsigned int irq,
|
|
|
|
irq_hw_number_t hwirq)
|
|
|
|
{
|
|
|
|
irq_set_chip_and_handler(irq, &dw_msi_irq_chip, handle_simple_irq);
|
|
|
|
irq_set_chip_data(irq, domain->host_data);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct irq_domain_ops msi_domain_ops = {
|
|
|
|
.map = dw_pcie_msi_map,
|
|
|
|
};
|
|
|
|
|
2016-08-10 18:02:39 +08:00
|
|
|
static u8 dw_pcie_iatu_unroll_enabled(struct pcie_port *pp)
|
|
|
|
{
|
|
|
|
u32 val;
|
|
|
|
|
|
|
|
val = dw_pcie_readl_rc(pp, PCIE_ATU_VIEWPORT);
|
|
|
|
if (val == 0xffffffff)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-02-20 01:41:48 +08:00
|
|
|
int dw_pcie_host_init(struct pcie_port *pp)
|
2013-07-31 16:14:10 +08:00
|
|
|
{
|
|
|
|
struct device_node *np = pp->dev->of_node;
|
2014-07-17 17:00:40 +08:00
|
|
|
struct platform_device *pdev = to_platform_device(pp->dev);
|
2015-10-30 08:57:21 +08:00
|
|
|
struct pci_bus *bus, *child;
|
2014-07-17 17:00:40 +08:00
|
|
|
struct resource *cfg_res;
|
2015-10-30 08:56:58 +08:00
|
|
|
int i, ret;
|
2015-10-30 08:57:06 +08:00
|
|
|
LIST_HEAD(res);
|
2016-08-16 00:50:42 +08:00
|
|
|
struct resource_entry *win, *tmp;
|
2013-09-06 14:54:59 +08:00
|
|
|
|
2014-07-17 17:00:40 +08:00
|
|
|
cfg_res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "config");
|
|
|
|
if (cfg_res) {
|
2014-09-06 07:48:54 +08:00
|
|
|
pp->cfg0_size = resource_size(cfg_res)/2;
|
|
|
|
pp->cfg1_size = resource_size(cfg_res)/2;
|
2014-07-17 17:00:40 +08:00
|
|
|
pp->cfg0_base = cfg_res->start;
|
2014-09-06 07:48:54 +08:00
|
|
|
pp->cfg1_base = cfg_res->start + pp->cfg0_size;
|
2015-07-22 05:54:11 +08:00
|
|
|
} else if (!pp->va_cfg0_base) {
|
2014-07-17 17:00:40 +08:00
|
|
|
dev_err(pp->dev, "missing *config* reg space\n");
|
|
|
|
}
|
|
|
|
|
2015-10-30 08:57:06 +08:00
|
|
|
ret = of_pci_get_host_bridge_resources(np, 0, 0xff, &res, &pp->io_base);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
2013-07-31 16:14:10 +08:00
|
|
|
|
2016-05-29 07:18:54 +08:00
|
|
|
ret = devm_request_pci_bus_resources(&pdev->dev, &res);
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
|
2013-07-31 16:14:10 +08:00
|
|
|
/* Get the I/O and memory ranges from DT */
|
2016-08-16 00:50:42 +08:00
|
|
|
resource_list_for_each_entry_safe(win, tmp, &res) {
|
2015-10-30 08:57:06 +08:00
|
|
|
switch (resource_type(win->res)) {
|
|
|
|
case IORESOURCE_IO:
|
2016-08-16 00:50:42 +08:00
|
|
|
ret = pci_remap_iospace(win->res, pp->io_base);
|
|
|
|
if (ret) {
|
2015-10-30 08:57:21 +08:00
|
|
|
dev_warn(pp->dev, "error %d: failed to map resource %pR\n",
|
2016-08-16 00:50:42 +08:00
|
|
|
ret, win->res);
|
|
|
|
resource_list_destroy_entry(win);
|
|
|
|
} else {
|
|
|
|
pp->io = win->res;
|
|
|
|
pp->io->name = "I/O";
|
|
|
|
pp->io_size = resource_size(pp->io);
|
|
|
|
pp->io_bus_addr = pp->io->start - win->offset;
|
|
|
|
}
|
2015-10-30 08:57:06 +08:00
|
|
|
break;
|
|
|
|
case IORESOURCE_MEM:
|
|
|
|
pp->mem = win->res;
|
|
|
|
pp->mem->name = "MEM";
|
|
|
|
pp->mem_size = resource_size(pp->mem);
|
|
|
|
pp->mem_bus_addr = pp->mem->start - win->offset;
|
|
|
|
break;
|
|
|
|
case 0:
|
|
|
|
pp->cfg = win->res;
|
|
|
|
pp->cfg0_size = resource_size(pp->cfg)/2;
|
|
|
|
pp->cfg1_size = resource_size(pp->cfg)/2;
|
|
|
|
pp->cfg0_base = pp->cfg->start;
|
|
|
|
pp->cfg1_base = pp->cfg->start + pp->cfg0_size;
|
|
|
|
break;
|
|
|
|
case IORESOURCE_BUS:
|
|
|
|
pp->busn = win->res;
|
|
|
|
break;
|
2013-07-31 16:14:10 +08:00
|
|
|
}
|
2014-07-24 01:52:38 +08:00
|
|
|
}
|
|
|
|
|
2013-07-31 16:14:10 +08:00
|
|
|
if (!pp->dbi_base) {
|
2015-10-30 08:57:06 +08:00
|
|
|
pp->dbi_base = devm_ioremap(pp->dev, pp->cfg->start,
|
|
|
|
resource_size(pp->cfg));
|
2013-07-31 16:14:10 +08:00
|
|
|
if (!pp->dbi_base) {
|
|
|
|
dev_err(pp->dev, "error with ioremap\n");
|
2016-06-01 00:14:08 +08:00
|
|
|
ret = -ENOMEM;
|
|
|
|
goto error;
|
2013-07-31 16:14:10 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-10-30 08:57:06 +08:00
|
|
|
pp->mem_base = pp->mem->start;
|
2013-07-31 16:14:10 +08:00
|
|
|
|
|
|
|
if (!pp->va_cfg0_base) {
|
2014-07-24 02:54:51 +08:00
|
|
|
pp->va_cfg0_base = devm_ioremap(pp->dev, pp->cfg0_base,
|
2014-09-06 07:48:54 +08:00
|
|
|
pp->cfg0_size);
|
2014-07-24 02:54:51 +08:00
|
|
|
if (!pp->va_cfg0_base) {
|
|
|
|
dev_err(pp->dev, "error with ioremap in function\n");
|
2016-06-01 00:14:08 +08:00
|
|
|
ret = -ENOMEM;
|
|
|
|
goto error;
|
2014-07-24 02:54:51 +08:00
|
|
|
}
|
2013-07-31 16:14:10 +08:00
|
|
|
}
|
2014-07-24 02:54:51 +08:00
|
|
|
|
2013-07-31 16:14:10 +08:00
|
|
|
if (!pp->va_cfg1_base) {
|
2014-07-24 02:54:51 +08:00
|
|
|
pp->va_cfg1_base = devm_ioremap(pp->dev, pp->cfg1_base,
|
2014-09-06 07:48:54 +08:00
|
|
|
pp->cfg1_size);
|
2014-07-24 02:54:51 +08:00
|
|
|
if (!pp->va_cfg1_base) {
|
|
|
|
dev_err(pp->dev, "error with ioremap\n");
|
2016-06-01 00:14:08 +08:00
|
|
|
ret = -ENOMEM;
|
|
|
|
goto error;
|
2014-07-24 02:54:51 +08:00
|
|
|
}
|
2013-07-31 16:14:10 +08:00
|
|
|
}
|
|
|
|
|
2015-09-29 00:03:10 +08:00
|
|
|
ret = of_property_read_u32(np, "num-lanes", &pp->lanes);
|
|
|
|
if (ret)
|
|
|
|
pp->lanes = 0;
|
2013-07-31 16:14:10 +08:00
|
|
|
|
2016-07-05 00:14:42 +08:00
|
|
|
ret = of_property_read_u32(np, "num-viewport", &pp->num_viewport);
|
|
|
|
if (ret)
|
|
|
|
pp->num_viewport = 2;
|
|
|
|
|
2013-09-06 14:54:59 +08:00
|
|
|
if (IS_ENABLED(CONFIG_PCI_MSI)) {
|
2014-07-24 02:54:51 +08:00
|
|
|
if (!pp->ops->msi_host_init) {
|
|
|
|
pp->irq_domain = irq_domain_add_linear(pp->dev->of_node,
|
|
|
|
MAX_MSI_IRQS, &msi_domain_ops,
|
|
|
|
&dw_pcie_msi_chip);
|
|
|
|
if (!pp->irq_domain) {
|
|
|
|
dev_err(pp->dev, "irq domain init failed\n");
|
2016-06-01 00:14:08 +08:00
|
|
|
ret = -ENXIO;
|
|
|
|
goto error;
|
2014-07-24 02:54:51 +08:00
|
|
|
}
|
2013-09-06 14:54:59 +08:00
|
|
|
|
2014-07-24 02:54:51 +08:00
|
|
|
for (i = 0; i < MAX_MSI_IRQS; i++)
|
|
|
|
irq_create_mapping(pp->irq_domain, i);
|
|
|
|
} else {
|
|
|
|
ret = pp->ops->msi_host_init(pp, &dw_pcie_msi_chip);
|
|
|
|
if (ret < 0)
|
2016-06-01 00:14:08 +08:00
|
|
|
goto error;
|
2014-07-24 02:54:51 +08:00
|
|
|
}
|
2013-09-06 14:54:59 +08:00
|
|
|
}
|
|
|
|
|
2013-07-31 16:14:10 +08:00
|
|
|
if (pp->ops->host_init)
|
|
|
|
pp->ops->host_init(pp);
|
|
|
|
|
2015-10-30 08:57:21 +08:00
|
|
|
pp->root_bus_nr = pp->busn->start;
|
|
|
|
if (IS_ENABLED(CONFIG_PCI_MSI)) {
|
|
|
|
bus = pci_scan_root_bus_msi(pp->dev, pp->root_bus_nr,
|
|
|
|
&dw_pcie_ops, pp, &res,
|
|
|
|
&dw_pcie_msi_chip);
|
|
|
|
dw_pcie_msi_chip.dev = pp->dev;
|
|
|
|
} else
|
|
|
|
bus = pci_scan_root_bus(pp->dev, pp->root_bus_nr, &dw_pcie_ops,
|
|
|
|
pp, &res);
|
2016-06-01 00:14:08 +08:00
|
|
|
if (!bus) {
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto error;
|
|
|
|
}
|
2015-10-30 08:57:21 +08:00
|
|
|
|
|
|
|
if (pp->ops->scan_bus)
|
|
|
|
pp->ops->scan_bus(pp);
|
|
|
|
|
|
|
|
#ifdef CONFIG_ARM
|
|
|
|
/* support old dtbs that incorrectly describe IRQs */
|
|
|
|
pci_fixup_irqs(pci_common_swizzle, of_irq_parse_and_map_pci);
|
2014-11-12 06:38:07 +08:00
|
|
|
#endif
|
|
|
|
|
2016-01-29 19:29:32 +08:00
|
|
|
pci_bus_size_bridges(bus);
|
|
|
|
pci_bus_assign_resources(bus);
|
2013-07-31 16:14:10 +08:00
|
|
|
|
2016-01-29 19:29:32 +08:00
|
|
|
list_for_each_entry(child, &bus->children, node)
|
|
|
|
pcie_bus_configure_settings(child);
|
2013-07-31 16:14:10 +08:00
|
|
|
|
2015-10-30 08:57:21 +08:00
|
|
|
pci_bus_add_devices(bus);
|
2013-07-31 16:14:10 +08:00
|
|
|
return 0;
|
2016-06-01 00:14:08 +08:00
|
|
|
|
|
|
|
error:
|
|
|
|
pci_free_resource_list(&res);
|
|
|
|
return ret;
|
2013-07-31 16:14:10 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int dw_pcie_rd_other_conf(struct pcie_port *pp, struct pci_bus *bus,
|
2013-06-21 15:24:54 +08:00
|
|
|
u32 devfn, int where, int size, u32 *val)
|
|
|
|
{
|
2015-04-30 16:22:29 +08:00
|
|
|
int ret, type;
|
2015-10-09 03:27:48 +08:00
|
|
|
u32 busdev, cfg_size;
|
2015-04-30 16:22:29 +08:00
|
|
|
u64 cpu_addr;
|
|
|
|
void __iomem *va_cfg_base;
|
2013-06-21 15:24:54 +08:00
|
|
|
|
2016-01-06 05:56:30 +08:00
|
|
|
if (pp->ops->rd_other_conf)
|
|
|
|
return pp->ops->rd_other_conf(pp, bus, devfn, where, size, val);
|
|
|
|
|
2013-06-21 15:24:54 +08:00
|
|
|
busdev = PCIE_ATU_BUS(bus->number) | PCIE_ATU_DEV(PCI_SLOT(devfn)) |
|
|
|
|
PCIE_ATU_FUNC(PCI_FUNC(devfn));
|
|
|
|
|
|
|
|
if (bus->parent->number == pp->root_bus_nr) {
|
2015-04-30 16:22:29 +08:00
|
|
|
type = PCIE_ATU_TYPE_CFG0;
|
2015-10-30 08:56:58 +08:00
|
|
|
cpu_addr = pp->cfg0_base;
|
2015-04-30 16:22:29 +08:00
|
|
|
cfg_size = pp->cfg0_size;
|
|
|
|
va_cfg_base = pp->va_cfg0_base;
|
2013-06-21 15:24:54 +08:00
|
|
|
} else {
|
2015-04-30 16:22:29 +08:00
|
|
|
type = PCIE_ATU_TYPE_CFG1;
|
2015-10-30 08:56:58 +08:00
|
|
|
cpu_addr = pp->cfg1_base;
|
2015-04-30 16:22:29 +08:00
|
|
|
cfg_size = pp->cfg1_size;
|
|
|
|
va_cfg_base = pp->va_cfg1_base;
|
2013-06-21 15:24:54 +08:00
|
|
|
}
|
|
|
|
|
PCI: designware: Exchange viewport of `MEMORYs' and `CFGs/IOs'
When we have only two view ports in a DesignWare PCIe platform, iatu0
is used for both CFG and IO accesses. When CFGs are sent to peripherals
(e.g., lspci), iatu0 frequently switches between CFG and IO.
For such scenarios, a MEMORY might be sent as an IOs by mistake.
Considering the following configurations:
MEMORY -> BASE_ADDR: 0xb4100000, LIMIT: 0xb4100FFF, TYPE=mem
CFG -> BASE_ADDR: 0xb4000000, LIMIT: 0xb4000FFF, TYPE=cfg
IO -> BASE_ADDR: 0xFFFFFFFF, LIMIT: 0xFFFFFFFE, TYPE=io
Suppose PCIe has just completed a CFG access. To switch back to IO, it
sets the BASE_ADDR to 0xFFFFFFFF, LIMIT 0xFFFFFFFE and TYPE to IO. When
another CFG comes, the BASE_ADDR is set to 0xb4000000 to switch to CFG. At
this moment, a MEMORY access shows up, since it matches with iatu0 (due to
0xb4000000 <= MEMORY BASE_ADDR <= MEMORY LIMIT <= 0xFFFFFFF), it is treated
as an IO access by mistake, then sent to perpheral.
This patch fixes the problem by exchanging the assignments of `MEMORYs' and
`CFGs/IOs', which assigning MEMORYs to iatu0, CFGs and IOs to iatu1.
We can still have issues with IO transfer, however memory transfer is used
predominantly therefore we are just minimizing the risk of failure.
Actually, we can not do much when we have only two viewports. We can
either not allow the less frequent IO transfers at all, or can live with a
remote possibility of getting it corrupted.
Signed-off-by: Dong Bo <dongbo4@huawei.com>
[pratyush.anand@gmail.com: Modified commit log to capture remote risk]
Signed-off-by: Pratyush Anand <pratyush.anand@gmail.com>
Signed-off-by: Bjorn Helgaas <bhelgaas@google.com>
2016-07-05 00:14:43 +08:00
|
|
|
dw_pcie_prog_outbound_atu(pp, PCIE_ATU_REGION_INDEX1,
|
2015-04-30 16:22:29 +08:00
|
|
|
type, cpu_addr,
|
|
|
|
busdev, cfg_size);
|
2015-10-09 03:27:48 +08:00
|
|
|
ret = dw_pcie_cfg_read(va_cfg_base + where, size, val);
|
2016-07-05 00:14:42 +08:00
|
|
|
if (pp->num_viewport <= 2)
|
PCI: designware: Exchange viewport of `MEMORYs' and `CFGs/IOs'
When we have only two view ports in a DesignWare PCIe platform, iatu0
is used for both CFG and IO accesses. When CFGs are sent to peripherals
(e.g., lspci), iatu0 frequently switches between CFG and IO.
For such scenarios, a MEMORY might be sent as an IOs by mistake.
Considering the following configurations:
MEMORY -> BASE_ADDR: 0xb4100000, LIMIT: 0xb4100FFF, TYPE=mem
CFG -> BASE_ADDR: 0xb4000000, LIMIT: 0xb4000FFF, TYPE=cfg
IO -> BASE_ADDR: 0xFFFFFFFF, LIMIT: 0xFFFFFFFE, TYPE=io
Suppose PCIe has just completed a CFG access. To switch back to IO, it
sets the BASE_ADDR to 0xFFFFFFFF, LIMIT 0xFFFFFFFE and TYPE to IO. When
another CFG comes, the BASE_ADDR is set to 0xb4000000 to switch to CFG. At
this moment, a MEMORY access shows up, since it matches with iatu0 (due to
0xb4000000 <= MEMORY BASE_ADDR <= MEMORY LIMIT <= 0xFFFFFFF), it is treated
as an IO access by mistake, then sent to perpheral.
This patch fixes the problem by exchanging the assignments of `MEMORYs' and
`CFGs/IOs', which assigning MEMORYs to iatu0, CFGs and IOs to iatu1.
We can still have issues with IO transfer, however memory transfer is used
predominantly therefore we are just minimizing the risk of failure.
Actually, we can not do much when we have only two viewports. We can
either not allow the less frequent IO transfers at all, or can live with a
remote possibility of getting it corrupted.
Signed-off-by: Dong Bo <dongbo4@huawei.com>
[pratyush.anand@gmail.com: Modified commit log to capture remote risk]
Signed-off-by: Pratyush Anand <pratyush.anand@gmail.com>
Signed-off-by: Bjorn Helgaas <bhelgaas@google.com>
2016-07-05 00:14:43 +08:00
|
|
|
dw_pcie_prog_outbound_atu(pp, PCIE_ATU_REGION_INDEX1,
|
2016-07-05 00:14:42 +08:00
|
|
|
PCIE_ATU_TYPE_IO, pp->io_base,
|
|
|
|
pp->io_bus_addr, pp->io_size);
|
2015-04-30 16:22:29 +08:00
|
|
|
|
2013-06-21 15:24:54 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2013-07-31 16:14:10 +08:00
|
|
|
static int dw_pcie_wr_other_conf(struct pcie_port *pp, struct pci_bus *bus,
|
2013-06-21 15:24:54 +08:00
|
|
|
u32 devfn, int where, int size, u32 val)
|
|
|
|
{
|
2015-04-30 16:22:29 +08:00
|
|
|
int ret, type;
|
2015-10-09 03:27:48 +08:00
|
|
|
u32 busdev, cfg_size;
|
2015-04-30 16:22:29 +08:00
|
|
|
u64 cpu_addr;
|
|
|
|
void __iomem *va_cfg_base;
|
2013-06-21 15:24:54 +08:00
|
|
|
|
2016-01-06 05:56:30 +08:00
|
|
|
if (pp->ops->wr_other_conf)
|
|
|
|
return pp->ops->wr_other_conf(pp, bus, devfn, where, size, val);
|
|
|
|
|
2013-06-21 15:24:54 +08:00
|
|
|
busdev = PCIE_ATU_BUS(bus->number) | PCIE_ATU_DEV(PCI_SLOT(devfn)) |
|
|
|
|
PCIE_ATU_FUNC(PCI_FUNC(devfn));
|
|
|
|
|
|
|
|
if (bus->parent->number == pp->root_bus_nr) {
|
2015-04-30 16:22:29 +08:00
|
|
|
type = PCIE_ATU_TYPE_CFG0;
|
2015-10-30 08:56:58 +08:00
|
|
|
cpu_addr = pp->cfg0_base;
|
2015-04-30 16:22:29 +08:00
|
|
|
cfg_size = pp->cfg0_size;
|
|
|
|
va_cfg_base = pp->va_cfg0_base;
|
2013-06-21 15:24:54 +08:00
|
|
|
} else {
|
2015-04-30 16:22:29 +08:00
|
|
|
type = PCIE_ATU_TYPE_CFG1;
|
2015-10-30 08:56:58 +08:00
|
|
|
cpu_addr = pp->cfg1_base;
|
2015-04-30 16:22:29 +08:00
|
|
|
cfg_size = pp->cfg1_size;
|
|
|
|
va_cfg_base = pp->va_cfg1_base;
|
2013-06-21 15:24:54 +08:00
|
|
|
}
|
|
|
|
|
PCI: designware: Exchange viewport of `MEMORYs' and `CFGs/IOs'
When we have only two view ports in a DesignWare PCIe platform, iatu0
is used for both CFG and IO accesses. When CFGs are sent to peripherals
(e.g., lspci), iatu0 frequently switches between CFG and IO.
For such scenarios, a MEMORY might be sent as an IOs by mistake.
Considering the following configurations:
MEMORY -> BASE_ADDR: 0xb4100000, LIMIT: 0xb4100FFF, TYPE=mem
CFG -> BASE_ADDR: 0xb4000000, LIMIT: 0xb4000FFF, TYPE=cfg
IO -> BASE_ADDR: 0xFFFFFFFF, LIMIT: 0xFFFFFFFE, TYPE=io
Suppose PCIe has just completed a CFG access. To switch back to IO, it
sets the BASE_ADDR to 0xFFFFFFFF, LIMIT 0xFFFFFFFE and TYPE to IO. When
another CFG comes, the BASE_ADDR is set to 0xb4000000 to switch to CFG. At
this moment, a MEMORY access shows up, since it matches with iatu0 (due to
0xb4000000 <= MEMORY BASE_ADDR <= MEMORY LIMIT <= 0xFFFFFFF), it is treated
as an IO access by mistake, then sent to perpheral.
This patch fixes the problem by exchanging the assignments of `MEMORYs' and
`CFGs/IOs', which assigning MEMORYs to iatu0, CFGs and IOs to iatu1.
We can still have issues with IO transfer, however memory transfer is used
predominantly therefore we are just minimizing the risk of failure.
Actually, we can not do much when we have only two viewports. We can
either not allow the less frequent IO transfers at all, or can live with a
remote possibility of getting it corrupted.
Signed-off-by: Dong Bo <dongbo4@huawei.com>
[pratyush.anand@gmail.com: Modified commit log to capture remote risk]
Signed-off-by: Pratyush Anand <pratyush.anand@gmail.com>
Signed-off-by: Bjorn Helgaas <bhelgaas@google.com>
2016-07-05 00:14:43 +08:00
|
|
|
dw_pcie_prog_outbound_atu(pp, PCIE_ATU_REGION_INDEX1,
|
2015-04-30 16:22:29 +08:00
|
|
|
type, cpu_addr,
|
|
|
|
busdev, cfg_size);
|
2015-10-09 03:27:48 +08:00
|
|
|
ret = dw_pcie_cfg_write(va_cfg_base + where, size, val);
|
2016-07-05 00:14:42 +08:00
|
|
|
if (pp->num_viewport <= 2)
|
PCI: designware: Exchange viewport of `MEMORYs' and `CFGs/IOs'
When we have only two view ports in a DesignWare PCIe platform, iatu0
is used for both CFG and IO accesses. When CFGs are sent to peripherals
(e.g., lspci), iatu0 frequently switches between CFG and IO.
For such scenarios, a MEMORY might be sent as an IOs by mistake.
Considering the following configurations:
MEMORY -> BASE_ADDR: 0xb4100000, LIMIT: 0xb4100FFF, TYPE=mem
CFG -> BASE_ADDR: 0xb4000000, LIMIT: 0xb4000FFF, TYPE=cfg
IO -> BASE_ADDR: 0xFFFFFFFF, LIMIT: 0xFFFFFFFE, TYPE=io
Suppose PCIe has just completed a CFG access. To switch back to IO, it
sets the BASE_ADDR to 0xFFFFFFFF, LIMIT 0xFFFFFFFE and TYPE to IO. When
another CFG comes, the BASE_ADDR is set to 0xb4000000 to switch to CFG. At
this moment, a MEMORY access shows up, since it matches with iatu0 (due to
0xb4000000 <= MEMORY BASE_ADDR <= MEMORY LIMIT <= 0xFFFFFFF), it is treated
as an IO access by mistake, then sent to perpheral.
This patch fixes the problem by exchanging the assignments of `MEMORYs' and
`CFGs/IOs', which assigning MEMORYs to iatu0, CFGs and IOs to iatu1.
We can still have issues with IO transfer, however memory transfer is used
predominantly therefore we are just minimizing the risk of failure.
Actually, we can not do much when we have only two viewports. We can
either not allow the less frequent IO transfers at all, or can live with a
remote possibility of getting it corrupted.
Signed-off-by: Dong Bo <dongbo4@huawei.com>
[pratyush.anand@gmail.com: Modified commit log to capture remote risk]
Signed-off-by: Pratyush Anand <pratyush.anand@gmail.com>
Signed-off-by: Bjorn Helgaas <bhelgaas@google.com>
2016-07-05 00:14:43 +08:00
|
|
|
dw_pcie_prog_outbound_atu(pp, PCIE_ATU_REGION_INDEX1,
|
2016-07-05 00:14:42 +08:00
|
|
|
PCIE_ATU_TYPE_IO, pp->io_base,
|
|
|
|
pp->io_bus_addr, pp->io_size);
|
2015-04-30 16:22:29 +08:00
|
|
|
|
2013-06-21 15:24:54 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2016-10-07 02:25:46 +08:00
|
|
|
static int dw_pcie_valid_device(struct pcie_port *pp, struct pci_bus *bus,
|
|
|
|
int dev)
|
2013-06-21 15:24:54 +08:00
|
|
|
{
|
|
|
|
/* If there is no link, then there is no device */
|
|
|
|
if (bus->number != pp->root_bus_nr) {
|
2013-07-31 16:14:10 +08:00
|
|
|
if (!dw_pcie_link_up(pp))
|
2013-06-21 15:24:54 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* access only one slot on each root port */
|
|
|
|
if (bus->number == pp->root_bus_nr && dev > 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2013-07-31 16:14:10 +08:00
|
|
|
static int dw_pcie_rd_conf(struct pci_bus *bus, u32 devfn, int where,
|
2013-06-21 15:24:54 +08:00
|
|
|
int size, u32 *val)
|
|
|
|
{
|
2015-10-30 08:57:21 +08:00
|
|
|
struct pcie_port *pp = bus->sysdata;
|
2013-06-21 15:24:54 +08:00
|
|
|
|
2016-10-07 02:25:46 +08:00
|
|
|
if (!dw_pcie_valid_device(pp, bus, PCI_SLOT(devfn))) {
|
2013-06-21 15:24:54 +08:00
|
|
|
*val = 0xffffffff;
|
|
|
|
return PCIBIOS_DEVICE_NOT_FOUND;
|
|
|
|
}
|
|
|
|
|
2016-01-06 05:48:11 +08:00
|
|
|
if (bus->number == pp->root_bus_nr)
|
|
|
|
return dw_pcie_rd_own_conf(pp, where, size, val);
|
2013-06-21 15:24:54 +08:00
|
|
|
|
2016-01-06 05:48:11 +08:00
|
|
|
return dw_pcie_rd_other_conf(pp, bus, devfn, where, size, val);
|
2013-06-21 15:24:54 +08:00
|
|
|
}
|
|
|
|
|
2013-07-31 16:14:10 +08:00
|
|
|
static int dw_pcie_wr_conf(struct pci_bus *bus, u32 devfn,
|
2013-06-21 15:24:54 +08:00
|
|
|
int where, int size, u32 val)
|
|
|
|
{
|
2015-10-30 08:57:21 +08:00
|
|
|
struct pcie_port *pp = bus->sysdata;
|
2013-06-21 15:24:54 +08:00
|
|
|
|
2016-10-07 02:25:46 +08:00
|
|
|
if (!dw_pcie_valid_device(pp, bus, PCI_SLOT(devfn)))
|
2013-06-21 15:24:54 +08:00
|
|
|
return PCIBIOS_DEVICE_NOT_FOUND;
|
|
|
|
|
2016-01-06 05:48:11 +08:00
|
|
|
if (bus->number == pp->root_bus_nr)
|
|
|
|
return dw_pcie_wr_own_conf(pp, where, size, val);
|
2013-06-21 15:24:54 +08:00
|
|
|
|
2016-01-06 05:48:11 +08:00
|
|
|
return dw_pcie_wr_other_conf(pp, bus, devfn, where, size, val);
|
2013-06-21 15:24:54 +08:00
|
|
|
}
|
|
|
|
|
2013-07-31 16:14:10 +08:00
|
|
|
static struct pci_ops dw_pcie_ops = {
|
|
|
|
.read = dw_pcie_rd_conf,
|
|
|
|
.write = dw_pcie_wr_conf,
|
2013-06-21 15:24:54 +08:00
|
|
|
};
|
|
|
|
|
2013-07-31 16:14:10 +08:00
|
|
|
void dw_pcie_setup_rc(struct pcie_port *pp)
|
2013-06-21 15:24:54 +08:00
|
|
|
{
|
|
|
|
u32 val;
|
|
|
|
|
2016-10-15 05:54:55 +08:00
|
|
|
/* get iATU unroll support */
|
|
|
|
pp->iatu_unroll_enabled = dw_pcie_iatu_unroll_enabled(pp);
|
|
|
|
dev_dbg(pp->dev, "iATU unroll: %s\n",
|
|
|
|
pp->iatu_unroll_enabled ? "enabled" : "disabled");
|
|
|
|
|
2014-04-15 04:22:54 +08:00
|
|
|
/* set the number of lanes */
|
2016-08-18 03:17:58 +08:00
|
|
|
val = dw_pcie_readl_rc(pp, PCIE_PORT_LINK_CONTROL);
|
2013-06-21 15:24:54 +08:00
|
|
|
val &= ~PORT_LINK_MODE_MASK;
|
2013-07-31 16:14:10 +08:00
|
|
|
switch (pp->lanes) {
|
|
|
|
case 1:
|
|
|
|
val |= PORT_LINK_MODE_1_LANES;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
val |= PORT_LINK_MODE_2_LANES;
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
val |= PORT_LINK_MODE_4_LANES;
|
|
|
|
break;
|
2015-05-13 14:44:34 +08:00
|
|
|
case 8:
|
|
|
|
val |= PORT_LINK_MODE_8_LANES;
|
|
|
|
break;
|
2015-09-29 00:03:10 +08:00
|
|
|
default:
|
|
|
|
dev_err(pp->dev, "num-lanes %u: invalid value\n", pp->lanes);
|
|
|
|
return;
|
2013-07-31 16:14:10 +08:00
|
|
|
}
|
2016-10-07 02:25:46 +08:00
|
|
|
dw_pcie_writel_rc(pp, PCIE_PORT_LINK_CONTROL, val);
|
2013-06-21 15:24:54 +08:00
|
|
|
|
|
|
|
/* set link width speed control register */
|
2016-08-18 03:17:58 +08:00
|
|
|
val = dw_pcie_readl_rc(pp, PCIE_LINK_WIDTH_SPEED_CONTROL);
|
2013-06-21 15:24:54 +08:00
|
|
|
val &= ~PORT_LOGIC_LINK_WIDTH_MASK;
|
2013-07-31 16:14:10 +08:00
|
|
|
switch (pp->lanes) {
|
|
|
|
case 1:
|
|
|
|
val |= PORT_LOGIC_LINK_WIDTH_1_LANES;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
val |= PORT_LOGIC_LINK_WIDTH_2_LANES;
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
val |= PORT_LOGIC_LINK_WIDTH_4_LANES;
|
|
|
|
break;
|
2015-05-13 14:44:34 +08:00
|
|
|
case 8:
|
|
|
|
val |= PORT_LOGIC_LINK_WIDTH_8_LANES;
|
|
|
|
break;
|
2013-07-31 16:14:10 +08:00
|
|
|
}
|
2016-10-07 02:25:46 +08:00
|
|
|
dw_pcie_writel_rc(pp, PCIE_LINK_WIDTH_SPEED_CONTROL, val);
|
2013-06-21 15:24:54 +08:00
|
|
|
|
|
|
|
/* setup RC BARs */
|
2016-10-07 02:25:46 +08:00
|
|
|
dw_pcie_writel_rc(pp, PCI_BASE_ADDRESS_0, 0x00000004);
|
|
|
|
dw_pcie_writel_rc(pp, PCI_BASE_ADDRESS_1, 0x00000000);
|
2013-06-21 15:24:54 +08:00
|
|
|
|
|
|
|
/* setup interrupt pins */
|
2016-08-18 03:17:58 +08:00
|
|
|
val = dw_pcie_readl_rc(pp, PCI_INTERRUPT_LINE);
|
2013-06-21 15:24:54 +08:00
|
|
|
val &= 0xffff00ff;
|
|
|
|
val |= 0x00000100;
|
2016-10-07 02:25:46 +08:00
|
|
|
dw_pcie_writel_rc(pp, PCI_INTERRUPT_LINE, val);
|
2013-06-21 15:24:54 +08:00
|
|
|
|
|
|
|
/* setup bus numbers */
|
2016-08-18 03:17:58 +08:00
|
|
|
val = dw_pcie_readl_rc(pp, PCI_PRIMARY_BUS);
|
2013-06-21 15:24:54 +08:00
|
|
|
val &= 0xff000000;
|
|
|
|
val |= 0x00010100;
|
2016-10-07 02:25:46 +08:00
|
|
|
dw_pcie_writel_rc(pp, PCI_PRIMARY_BUS, val);
|
2013-06-21 15:24:54 +08:00
|
|
|
|
|
|
|
/* setup command register */
|
2016-08-18 03:17:58 +08:00
|
|
|
val = dw_pcie_readl_rc(pp, PCI_COMMAND);
|
2013-06-21 15:24:54 +08:00
|
|
|
val &= 0xffff0000;
|
|
|
|
val |= PCI_COMMAND_IO | PCI_COMMAND_MEMORY |
|
|
|
|
PCI_COMMAND_MASTER | PCI_COMMAND_SERR;
|
2016-10-07 02:25:46 +08:00
|
|
|
dw_pcie_writel_rc(pp, PCI_COMMAND, val);
|
2016-03-16 19:40:33 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* If the platform provides ->rd_other_conf, it means the platform
|
|
|
|
* uses its own address translation component rather than ATU, so
|
|
|
|
* we should not program the ATU here.
|
|
|
|
*/
|
2016-07-05 00:14:42 +08:00
|
|
|
if (!pp->ops->rd_other_conf) {
|
PCI: designware: Exchange viewport of `MEMORYs' and `CFGs/IOs'
When we have only two view ports in a DesignWare PCIe platform, iatu0
is used for both CFG and IO accesses. When CFGs are sent to peripherals
(e.g., lspci), iatu0 frequently switches between CFG and IO.
For such scenarios, a MEMORY might be sent as an IOs by mistake.
Considering the following configurations:
MEMORY -> BASE_ADDR: 0xb4100000, LIMIT: 0xb4100FFF, TYPE=mem
CFG -> BASE_ADDR: 0xb4000000, LIMIT: 0xb4000FFF, TYPE=cfg
IO -> BASE_ADDR: 0xFFFFFFFF, LIMIT: 0xFFFFFFFE, TYPE=io
Suppose PCIe has just completed a CFG access. To switch back to IO, it
sets the BASE_ADDR to 0xFFFFFFFF, LIMIT 0xFFFFFFFE and TYPE to IO. When
another CFG comes, the BASE_ADDR is set to 0xb4000000 to switch to CFG. At
this moment, a MEMORY access shows up, since it matches with iatu0 (due to
0xb4000000 <= MEMORY BASE_ADDR <= MEMORY LIMIT <= 0xFFFFFFF), it is treated
as an IO access by mistake, then sent to perpheral.
This patch fixes the problem by exchanging the assignments of `MEMORYs' and
`CFGs/IOs', which assigning MEMORYs to iatu0, CFGs and IOs to iatu1.
We can still have issues with IO transfer, however memory transfer is used
predominantly therefore we are just minimizing the risk of failure.
Actually, we can not do much when we have only two viewports. We can
either not allow the less frequent IO transfers at all, or can live with a
remote possibility of getting it corrupted.
Signed-off-by: Dong Bo <dongbo4@huawei.com>
[pratyush.anand@gmail.com: Modified commit log to capture remote risk]
Signed-off-by: Pratyush Anand <pratyush.anand@gmail.com>
Signed-off-by: Bjorn Helgaas <bhelgaas@google.com>
2016-07-05 00:14:43 +08:00
|
|
|
dw_pcie_prog_outbound_atu(pp, PCIE_ATU_REGION_INDEX0,
|
2016-03-16 19:40:33 +08:00
|
|
|
PCIE_ATU_TYPE_MEM, pp->mem_base,
|
|
|
|
pp->mem_bus_addr, pp->mem_size);
|
2016-07-05 00:14:42 +08:00
|
|
|
if (pp->num_viewport > 2)
|
|
|
|
dw_pcie_prog_outbound_atu(pp, PCIE_ATU_REGION_INDEX2,
|
|
|
|
PCIE_ATU_TYPE_IO, pp->io_base,
|
|
|
|
pp->io_bus_addr, pp->io_size);
|
|
|
|
}
|
2016-03-16 19:40:33 +08:00
|
|
|
|
|
|
|
dw_pcie_wr_own_conf(pp, PCI_BASE_ADDRESS_0, 4, 0);
|
|
|
|
|
|
|
|
/* program correct class for RC */
|
|
|
|
dw_pcie_wr_own_conf(pp, PCI_CLASS_DEVICE, 2, PCI_CLASS_BRIDGE_PCI);
|
|
|
|
|
|
|
|
dw_pcie_rd_own_conf(pp, PCIE_LINK_WIDTH_SPEED_CONTROL, 4, &val);
|
|
|
|
val |= PORT_LOGIC_SPEED_CHANGE;
|
|
|
|
dw_pcie_wr_own_conf(pp, PCIE_LINK_WIDTH_SPEED_CONTROL, 4, val);
|
2013-06-21 15:24:54 +08:00
|
|
|
}
|