cregit-Linux how code gets into the kernel

Release 4.16 drivers/pci/dwc/pcie-designware.c

Directory: drivers/pci/dwc
// SPDX-License-Identifier: GPL-2.0
/*
 * Synopsys DesignWare PCIe host controller driver
 *
 * Copyright (C) 2013 Samsung Electronics Co., Ltd.
 *              http://www.samsung.com
 *
 * Author: Jingoo Han <jg1.han@samsung.com>
 */

#include <linux/delay.h>
#include <linux/of.h>
#include <linux/types.h>

#include "pcie-designware.h"

/* PCIe Port Logic registers */

#define PLR_OFFSET			0x700

#define PCIE_PHY_DEBUG_R1		(PLR_OFFSET + 0x2c)

#define PCIE_PHY_DEBUG_R1_LINK_UP	(0x1 << 4)

#define PCIE_PHY_DEBUG_R1_LINK_IN_TRAINING	(0x1 << 29)


int dw_pcie_read(void __iomem *addr, int size, u32 *val) { if ((uintptr_t)addr & (size - 1)) { *val = 0; return PCIBIOS_BAD_REGISTER_NUMBER; } if (size == 4) { *val = readl(addr); } else if (size == 2) { *val = readw(addr); } else if (size == 1) { *val = readb(addr); } else { *val = 0; return PCIBIOS_BAD_REGISTER_NUMBER; } return PCIBIOS_SUCCESSFUL; }

Contributors

PersonTokensPropCommitsCommitProp
Jingoo Han4947.12%233.33%
Gabriele Paoloni4846.15%233.33%
Kishon Vijay Abraham I76.73%233.33%
Total104100.00%6100.00%


int dw_pcie_write(void __iomem *addr, int size, u32 val) { if ((uintptr_t)addr & (size - 1)) return PCIBIOS_BAD_REGISTER_NUMBER; if (size == 4) writel(val, addr); else if (size == 2) writew(val, addr); else if (size == 1) writeb(val, addr); else return PCIBIOS_BAD_REGISTER_NUMBER; return PCIBIOS_SUCCESSFUL; }

Contributors

PersonTokensPropCommitsCommitProp
Jingoo Han6378.75%250.00%
Gabriele Paoloni1620.00%125.00%
Kishon Vijay Abraham I11.25%125.00%
Total80100.00%4100.00%


u32 __dw_pcie_read_dbi(struct dw_pcie *pci, void __iomem *base, u32 reg, size_t size) { int ret; u32 val; if (pci->ops->read_dbi) return pci->ops->read_dbi(pci, base, reg, size); ret = dw_pcie_read(base + reg, size, &val); if (ret) dev_err(pci->dev, "read DBI address failed\n"); return val; }

Contributors

PersonTokensPropCommitsCommitProp
Kishon Vijay Abraham I5162.96%342.86%
Jingoo Han2125.93%228.57%
Seungwon Jeon56.17%114.29%
Björn Helgaas44.94%114.29%
Total81100.00%7100.00%


void __dw_pcie_write_dbi(struct dw_pcie *pci, void __iomem *base, u32 reg, size_t size, u32 val) { int ret; if (pci->ops->write_dbi) { pci->ops->write_dbi(pci, base, reg, size, val); return; } ret = dw_pcie_write(base + reg, size, val); if (ret) dev_err(pci->dev, "write DBI address failed\n"); }

Contributors

PersonTokensPropCommitsCommitProp
Kishon Vijay Abraham I4758.02%342.86%
Jingoo Han2632.10%228.57%
Seungwon Jeon44.94%114.29%
Björn Helgaas44.94%114.29%
Total81100.00%7100.00%


static u32 dw_pcie_readl_ob_unroll(struct dw_pcie *pci, u32 index, u32 reg) { u32 offset = PCIE_GET_ATU_OUTB_UNR_REG_OFFSET(index); return dw_pcie_readl_dbi(pci, offset + reg); }

Contributors

PersonTokensPropCommitsCommitProp
Joao Pinto2777.14%120.00%
Kishon Vijay Abraham I514.29%240.00%
Jingoo Han25.71%120.00%
Seungwon Jeon12.86%120.00%
Total35100.00%5100.00%


static void dw_pcie_writel_ob_unroll(struct dw_pcie *pci, u32 index, u32 reg, u32 val) { u32 offset = PCIE_GET_ATU_OUTB_UNR_REG_OFFSET(index); dw_pcie_writel_dbi(pci, offset + reg, val); }

Contributors

PersonTokensPropCommitsCommitProp
Jingoo Han1538.46%114.29%
Joao Pinto1333.33%114.29%
Kishon Vijay Abraham I512.82%228.57%
Björn Helgaas410.26%228.57%
Seungwon Jeon25.13%114.29%
Total39100.00%7100.00%


static void dw_pcie_prog_outbound_atu_unroll(struct dw_pcie *pci, int index, int type, u64 cpu_addr, u64 pci_addr, u32 size) { u32 retries, val; dw_pcie_writel_ob_unroll(pci, index, PCIE_ATU_UNR_LOWER_BASE, lower_32_bits(cpu_addr)); dw_pcie_writel_ob_unroll(pci, index, PCIE_ATU_UNR_UPPER_BASE, upper_32_bits(cpu_addr)); dw_pcie_writel_ob_unroll(pci, index, PCIE_ATU_UNR_LIMIT, lower_32_bits(cpu_addr + size - 1)); dw_pcie_writel_ob_unroll(pci, index, PCIE_ATU_UNR_LOWER_TARGET, lower_32_bits(pci_addr)); dw_pcie_writel_ob_unroll(pci, index, PCIE_ATU_UNR_UPPER_TARGET, upper_32_bits(pci_addr)); dw_pcie_writel_ob_unroll(pci, index, PCIE_ATU_UNR_REGION_CTRL1, type); dw_pcie_writel_ob_unroll(pci, index, PCIE_ATU_UNR_REGION_CTRL2, PCIE_ATU_ENABLE); /* * Make sure ATU enable takes effect before any subsequent config * and I/O accesses. */ for (retries = 0; retries < LINK_WAIT_MAX_IATU_RETRIES; retries++) { val = dw_pcie_readl_ob_unroll(pci, index, PCIE_ATU_UNR_REGION_CTRL2); if (val & PCIE_ATU_ENABLE) return; usleep_range(LINK_WAIT_IATU_MIN, LINK_WAIT_IATU_MAX); } dev_err(pci->dev, "outbound iATU is not being enabled\n"); }

Contributors

PersonTokensPropCommitsCommitProp
Kishon Vijay Abraham I6737.85%222.22%
Joao Pinto6536.72%222.22%
JiSheng Zhang2111.86%111.11%
Björn Helgaas147.91%111.11%
Jingoo Han63.39%111.11%
Stanimir Varbanov31.69%111.11%
Carlos Palminha10.56%111.11%
Total177100.00%9100.00%


void dw_pcie_prog_outbound_atu(struct dw_pcie *pci, int index, int type, u64 cpu_addr, u64 pci_addr, u32 size) { u32 retries, val; if (pci->ops->cpu_addr_fixup) cpu_addr = pci->ops->cpu_addr_fixup(pci, cpu_addr); if (pci->iatu_unroll_enabled) { dw_pcie_prog_outbound_atu_unroll(pci, index, type, cpu_addr, pci_addr, size); return; } dw_pcie_writel_dbi(pci, PCIE_ATU_VIEWPORT, PCIE_ATU_REGION_OUTBOUND | index); dw_pcie_writel_dbi(pci, PCIE_ATU_LOWER_BASE, lower_32_bits(cpu_addr)); dw_pcie_writel_dbi(pci, PCIE_ATU_UPPER_BASE, upper_32_bits(cpu_addr)); dw_pcie_writel_dbi(pci, PCIE_ATU_LIMIT, lower_32_bits(cpu_addr + size - 1)); dw_pcie_writel_dbi(pci, PCIE_ATU_LOWER_TARGET, lower_32_bits(pci_addr)); dw_pcie_writel_dbi(pci, PCIE_ATU_UPPER_TARGET, upper_32_bits(pci_addr)); dw_pcie_writel_dbi(pci, PCIE_ATU_CR1, type); dw_pcie_writel_dbi(pci, PCIE_ATU_CR2, PCIE_ATU_ENABLE); /* * Make sure ATU enable takes effect before any subsequent config * and I/O accesses. */ for (retries = 0; retries < LINK_WAIT_MAX_IATU_RETRIES; retries++) { val = dw_pcie_readl_dbi(pci, PCIE_ATU_CR2); if (val & PCIE_ATU_ENABLE) return; usleep_range(LINK_WAIT_IATU_MIN, LINK_WAIT_IATU_MAX); } dev_err(pci->dev, "outbound iATU is not being enabled\n"); }

Contributors

PersonTokensPropCommitsCommitProp
Kishon Vijay Abraham I9142.13%222.22%
JiSheng Zhang6429.63%222.22%
Joao Pinto3214.81%111.11%
Björn Helgaas188.33%222.22%
Stanimir Varbanov94.17%111.11%
Niklas Cassel20.93%111.11%
Total216100.00%9100.00%


static u32 dw_pcie_readl_ib_unroll(struct dw_pcie *pci, u32 index, u32 reg) { u32 offset = PCIE_GET_ATU_INB_UNR_REG_OFFSET(index); return dw_pcie_readl_dbi(pci, offset + reg); }

Contributors

PersonTokensPropCommitsCommitProp
Kishon Vijay Abraham I35100.00%1100.00%
Total35100.00%1100.00%


static void dw_pcie_writel_ib_unroll(struct dw_pcie *pci, u32 index, u32 reg, u32 val) { u32 offset = PCIE_GET_ATU_INB_UNR_REG_OFFSET(index); dw_pcie_writel_dbi(pci, offset + reg, val); }

Contributors

PersonTokensPropCommitsCommitProp
Kishon Vijay Abraham I39100.00%1100.00%
Total39100.00%1100.00%


static int dw_pcie_prog_inbound_atu_unroll(struct dw_pcie *pci, int index, int bar, u64 cpu_addr, enum dw_pcie_as_type as_type) { int type; u32 retries, val; dw_pcie_writel_ib_unroll(pci, index, PCIE_ATU_UNR_LOWER_TARGET, lower_32_bits(cpu_addr)); dw_pcie_writel_ib_unroll(pci, index, PCIE_ATU_UNR_UPPER_TARGET, upper_32_bits(cpu_addr)); switch (as_type) { case DW_PCIE_AS_MEM: type = PCIE_ATU_TYPE_MEM; break; case DW_PCIE_AS_IO: type = PCIE_ATU_TYPE_IO; break; default: return -EINVAL; } dw_pcie_writel_ib_unroll(pci, index, PCIE_ATU_UNR_REGION_CTRL1, type); dw_pcie_writel_ib_unroll(pci, index, PCIE_ATU_UNR_REGION_CTRL2, PCIE_ATU_ENABLE | PCIE_ATU_BAR_MODE_ENABLE | (bar << 8)); /* * Make sure ATU enable takes effect before any subsequent config * and I/O accesses. */ for (retries = 0; retries < LINK_WAIT_MAX_IATU_RETRIES; retries++) { val = dw_pcie_readl_ib_unroll(pci, index, PCIE_ATU_UNR_REGION_CTRL2); if (val & PCIE_ATU_ENABLE) return 0; usleep_range(LINK_WAIT_IATU_MIN, LINK_WAIT_IATU_MAX); } dev_err(pci->dev, "inbound iATU is not being enabled\n"); return -EBUSY; }

Contributors

PersonTokensPropCommitsCommitProp
Kishon Vijay Abraham I17299.42%150.00%
Carlos Palminha10.58%150.00%
Total173100.00%2100.00%


int dw_pcie_prog_inbound_atu(struct dw_pcie *pci, int index, int bar, u64 cpu_addr, enum dw_pcie_as_type as_type) { int type; u32 retries, val; if (pci->iatu_unroll_enabled) return dw_pcie_prog_inbound_atu_unroll(pci, index, bar, cpu_addr, as_type); dw_pcie_writel_dbi(pci, PCIE_ATU_VIEWPORT, PCIE_ATU_REGION_INBOUND | index); dw_pcie_writel_dbi(pci, PCIE_ATU_LOWER_TARGET, lower_32_bits(cpu_addr)); dw_pcie_writel_dbi(pci, PCIE_ATU_UPPER_TARGET, upper_32_bits(cpu_addr)); switch (as_type) { case DW_PCIE_AS_MEM: type = PCIE_ATU_TYPE_MEM; break; case DW_PCIE_AS_IO: type = PCIE_ATU_TYPE_IO; break; default: return -EINVAL; } dw_pcie_writel_dbi(pci, PCIE_ATU_CR1, type); dw_pcie_writel_dbi(pci, PCIE_ATU_CR2, PCIE_ATU_ENABLE | PCIE_ATU_BAR_MODE_ENABLE | (bar << 8)); /* * Make sure ATU enable takes effect before any subsequent config * and I/O accesses. */ for (retries = 0; retries < LINK_WAIT_MAX_IATU_RETRIES; retries++) { val = dw_pcie_readl_dbi(pci, PCIE_ATU_CR2); if (val & PCIE_ATU_ENABLE) return 0; usleep_range(LINK_WAIT_IATU_MIN, LINK_WAIT_IATU_MAX); } dev_err(pci->dev, "inbound iATU is not being enabled\n"); return -EBUSY; }

Contributors

PersonTokensPropCommitsCommitProp
Kishon Vijay Abraham I193100.00%1100.00%
Total193100.00%1100.00%


void dw_pcie_disable_atu(struct dw_pcie *pci, int index, enum dw_pcie_region_type type) { int region; switch (type) { case DW_PCIE_REGION_INBOUND: region = PCIE_ATU_REGION_INBOUND; break; case DW_PCIE_REGION_OUTBOUND: region = PCIE_ATU_REGION_OUTBOUND; break; default: return; } dw_pcie_writel_dbi(pci, PCIE_ATU_VIEWPORT, region | index); dw_pcie_writel_dbi(pci, PCIE_ATU_CR2, ~PCIE_ATU_ENABLE); }

Contributors

PersonTokensPropCommitsCommitProp
Kishon Vijay Abraham I65100.00%1100.00%
Total65100.00%1100.00%


int dw_pcie_wait_for_link(struct dw_pcie *pci) { int retries; /* check if the link is up or not */ for (retries = 0; retries < LINK_WAIT_MAX_RETRIES; retries++) { if (dw_pcie_link_up(pci)) { dev_info(pci->dev, "link up\n"); return 0; } usleep_range(LINK_WAIT_USLEEP_MIN, LINK_WAIT_USLEEP_MAX); } dev_err(pci->dev, "phy link never came up\n"); return -ETIMEDOUT; }

Contributors

PersonTokensPropCommitsCommitProp
Jingoo Han3347.14%222.22%
Kishon Vijay Abraham I3042.86%222.22%
Harro Haan34.29%111.11%
Pratyush Anand22.86%222.22%
Lucas Stach11.43%111.11%
JiSheng Zhang11.43%111.11%
Total70100.00%9100.00%


int dw_pcie_link_up(struct dw_pcie *pci) { u32 val; if (pci->ops->link_up) return pci->ops->link_up(pci); val = readl(pci->dbi_base + PCIE_PHY_DEBUG_R1); return ((val & PCIE_PHY_DEBUG_R1_LINK_UP) && (!(val & PCIE_PHY_DEBUG_R1_LINK_IN_TRAINING))); }

Contributors

PersonTokensPropCommitsCommitProp
Kishon Vijay Abraham I4473.33%233.33%
Jingoo Han1220.00%233.33%
Björn Helgaas23.33%116.67%
Murali Karicheri23.33%116.67%
Total60100.00%6100.00%


void dw_pcie_setup(struct dw_pcie *pci) { int ret; u32 val; u32 lanes; struct device *dev = pci->dev; struct device_node *np = dev->of_node; ret = of_property_read_u32(np, "num-lanes", &lanes); if (ret) lanes = 0; /* set the number of lanes */ val = dw_pcie_readl_dbi(pci, PCIE_PORT_LINK_CONTROL); val &= ~PORT_LINK_MODE_MASK; switch (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; case 8: val |= PORT_LINK_MODE_8_LANES; break; default: dev_err(pci->dev, "num-lanes %u: invalid value\n", lanes); return; } dw_pcie_writel_dbi(pci, PCIE_PORT_LINK_CONTROL, val); /* set link width speed control register */ val = dw_pcie_readl_dbi(pci, PCIE_LINK_WIDTH_SPEED_CONTROL); val &= ~PORT_LOGIC_LINK_WIDTH_MASK; switch (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; case 8: val |= PORT_LOGIC_LINK_WIDTH_8_LANES; break; } dw_pcie_writel_dbi(pci, PCIE_LINK_WIDTH_SPEED_CONTROL, val); }

Contributors

PersonTokensPropCommitsCommitProp
Jingoo Han10051.55%220.00%
Kishon Vijay Abraham I5729.38%330.00%
Zhou Wang168.25%110.00%
Gabriele Paoloni126.19%110.00%
Björn Helgaas84.12%220.00%
Mohit Kumar10.52%110.00%
Total194100.00%10100.00%


Overall Contributors

PersonTokensPropCommitsCommitProp
Kishon Vijay Abraham I91354.57%1026.32%
Jingoo Han33520.02%37.89%
Joao Pinto1498.91%37.89%
JiSheng Zhang915.44%37.89%
Gabriele Paoloni764.54%37.89%
Björn Helgaas563.35%513.16%
Zhou Wang160.96%12.63%
Stanimir Varbanov120.72%12.63%
Seungwon Jeon120.72%12.63%
Harro Haan30.18%12.63%
Pratyush Anand20.12%25.26%
Niklas Cassel20.12%12.63%
Carlos Palminha20.12%12.63%
Murali Karicheri20.12%12.63%
Mohit Kumar10.06%12.63%
Lucas Stach10.06%12.63%
Total1673100.00%38100.00%
Directory: drivers/pci/dwc
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.