commit
0675f992db
@ -0,0 +1,95 @@ |
||||
/*
|
||||
* Copyright 2016 Freescale Semiconductor, Inc. |
||||
* Hou Zhiqiang <Zhiqiang.Hou@freescale.com> |
||||
* |
||||
* SPDX-License-Identifier: GPL-2.0+ |
||||
*/ |
||||
|
||||
#include <common.h> |
||||
#include <errno.h> |
||||
#include <i2c.h> |
||||
#include <power/pmic.h> |
||||
#include <power/mc34vr500_pmic.h> |
||||
|
||||
static uint8_t swxvolt_addr[4] = { MC34VR500_SW1VOLT, |
||||
MC34VR500_SW2VOLT, |
||||
MC34VR500_SW3VOLT, |
||||
MC34VR500_SW4VOLT }; |
||||
|
||||
static uint8_t swx_set_point_base[4] = { 13, 9, 9, 9 }; |
||||
|
||||
int mc34vr500_get_sw_volt(uint8_t sw) |
||||
{ |
||||
struct pmic *p; |
||||
u32 swxvolt; |
||||
uint8_t spb; |
||||
int sw_volt; |
||||
int ret; |
||||
|
||||
debug("%s: Get SW%u volt from swxvolt_addr = 0x%x\n", |
||||
__func__, sw + 1, swxvolt_addr[sw]); |
||||
if (sw > SW4) { |
||||
printf("%s: Unsupported SW(sw%d)\n", __func__, sw + 1); |
||||
return -EINVAL; |
||||
} |
||||
|
||||
p = pmic_get("MC34VR500"); |
||||
if (!p) { |
||||
printf("%s: Did NOT find PMIC MC34VR500\n", __func__); |
||||
return -ENODEV; |
||||
} |
||||
|
||||
ret = pmic_probe(p); |
||||
if (ret) |
||||
return ret; |
||||
|
||||
ret = pmic_reg_read(p, swxvolt_addr[sw], &swxvolt); |
||||
if (ret) { |
||||
printf("%s: Failed to get SW%u volt\n", __func__, sw + 1); |
||||
return ret; |
||||
} |
||||
|
||||
debug("%s: SW%d step point swxvolt = %u\n", __func__, sw + 1, swxvolt); |
||||
spb = swx_set_point_base[sw]; |
||||
/* The base of SW volt is 625mV and increase by step 25mV */ |
||||
sw_volt = 625 + (swxvolt - spb) * 25; |
||||
|
||||
debug("%s: SW%u volt = %dmV\n", __func__, sw + 1, sw_volt); |
||||
return sw_volt; |
||||
} |
||||
|
||||
int mc34vr500_set_sw_volt(uint8_t sw, int sw_volt) |
||||
{ |
||||
struct pmic *p; |
||||
u32 swxvolt; |
||||
uint8_t spb; |
||||
int ret; |
||||
|
||||
debug("%s: Set SW%u volt to %dmV\n", __func__, sw + 1, sw_volt); |
||||
/* The least SW volt is 625mV, and only 4 SW outputs */ |
||||
if (sw > SW4 || sw_volt < 625) |
||||
return -EINVAL; |
||||
|
||||
p = pmic_get("MC34VR500"); |
||||
if (!p) { |
||||
printf("%s: Did NOT find PMIC MC34VR500\n", __func__); |
||||
return -ENODEV; |
||||
} |
||||
|
||||
ret = pmic_probe(p); |
||||
if (ret) |
||||
return ret; |
||||
|
||||
spb = swx_set_point_base[sw]; |
||||
/* The base of SW volt is 625mV and increase by step 25mV */ |
||||
swxvolt = (sw_volt - 625) / 25 + spb; |
||||
debug("%s: SW%d step point swxvolt = %u\n", __func__, sw + 1, swxvolt); |
||||
if (swxvolt > 63) |
||||
return -EINVAL; |
||||
|
||||
ret = pmic_reg_write(p, swxvolt_addr[sw], swxvolt); |
||||
if (ret) |
||||
return ret; |
||||
|
||||
return 0; |
||||
} |
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,140 @@ |
||||
/*
|
||||
* Copyright 2014-2015 Freescale Semiconductor, Inc. |
||||
* Layerscape PCIe driver |
||||
* |
||||
* SPDX-License-Identifier: GPL-2.0+ |
||||
*/ |
||||
|
||||
#ifndef _PCIE_LAYERSCAPE_H_ |
||||
#define _PCIE_LAYERSCAPE_H_ |
||||
#include <pci.h> |
||||
#include <dm.h> |
||||
|
||||
#ifndef CONFIG_SYS_PCI_MEMORY_BUS |
||||
#define CONFIG_SYS_PCI_MEMORY_BUS CONFIG_SYS_SDRAM_BASE |
||||
#endif |
||||
|
||||
#ifndef CONFIG_SYS_PCI_MEMORY_PHYS |
||||
#define CONFIG_SYS_PCI_MEMORY_PHYS CONFIG_SYS_SDRAM_BASE |
||||
#endif |
||||
|
||||
#ifndef CONFIG_SYS_PCI_MEMORY_SIZE |
||||
#define CONFIG_SYS_PCI_MEMORY_SIZE (2 * 1024 * 1024 * 1024UL) /* 2G */ |
||||
#endif |
||||
|
||||
#ifndef CONFIG_SYS_PCI_EP_MEMORY_BASE |
||||
#define CONFIG_SYS_PCI_EP_MEMORY_BASE CONFIG_SYS_LOAD_ADDR |
||||
#endif |
||||
|
||||
/* iATU registers */ |
||||
#define PCIE_ATU_VIEWPORT 0x900 |
||||
#define PCIE_ATU_REGION_INBOUND (0x1 << 31) |
||||
#define PCIE_ATU_REGION_OUTBOUND (0x0 << 31) |
||||
#define PCIE_ATU_REGION_INDEX0 (0x0 << 0) |
||||
#define PCIE_ATU_REGION_INDEX1 (0x1 << 0) |
||||
#define PCIE_ATU_REGION_INDEX2 (0x2 << 0) |
||||
#define PCIE_ATU_REGION_INDEX3 (0x3 << 0) |
||||
#define PCIE_ATU_REGION_NUM 6 |
||||
#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_BAR_NUM(bar) ((bar) << 8) |
||||
#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 |
||||
|
||||
/* DBI registers */ |
||||
#define PCIE_SRIOV 0x178 |
||||
#define PCIE_STRFMR1 0x71c /* Symbol Timer & Filter Mask Register1 */ |
||||
#define PCIE_DBI_RO_WR_EN 0x8bc |
||||
|
||||
#define PCIE_LINK_CAP 0x7c |
||||
#define PCIE_LINK_SPEED_MASK 0xf |
||||
#define PCIE_LINK_WIDTH_MASK 0x3f0 |
||||
#define PCIE_LINK_STA 0x82 |
||||
|
||||
#define LTSSM_STATE_MASK 0x3f |
||||
#define LTSSM_PCIE_L0 0x11 /* L0 state */ |
||||
|
||||
#define PCIE_DBI_SIZE 0x100000 /* 1M */ |
||||
|
||||
#define PCIE_LCTRL0_CFG2_ENABLE (1 << 31) |
||||
#define PCIE_LCTRL0_VF(vf) ((vf) << 22) |
||||
#define PCIE_LCTRL0_PF(pf) ((pf) << 16) |
||||
#define PCIE_LCTRL0_VF_ACTIVE (1 << 21) |
||||
#define PCIE_LCTRL0_VAL(pf, vf) (PCIE_LCTRL0_PF(pf) | \ |
||||
PCIE_LCTRL0_VF(vf) | \
|
||||
((vf) == 0 ? 0 : PCIE_LCTRL0_VF_ACTIVE) | \
|
||||
PCIE_LCTRL0_CFG2_ENABLE) |
||||
|
||||
#define PCIE_NO_SRIOV_BAR_BASE 0x1000 |
||||
|
||||
#define PCIE_PF_NUM 2 |
||||
#define PCIE_VF_NUM 64 |
||||
|
||||
#define PCIE_BAR0_SIZE (4 * 1024) /* 4K */ |
||||
#define PCIE_BAR1_SIZE (8 * 1024) /* 8K for MSIX */ |
||||
#define PCIE_BAR2_SIZE (4 * 1024) /* 4K */ |
||||
#define PCIE_BAR4_SIZE (1 * 1024 * 1024) /* 1M */ |
||||
|
||||
/* LUT registers */ |
||||
#define PCIE_LUT_UDR(n) (0x800 + (n) * 8) |
||||
#define PCIE_LUT_LDR(n) (0x804 + (n) * 8) |
||||
#define PCIE_LUT_ENABLE (1 << 31) |
||||
#define PCIE_LUT_ENTRY_COUNT 32 |
||||
|
||||
/* PF Controll registers */ |
||||
#define PCIE_PF_VF_CTRL 0x7F8 |
||||
#define PCIE_PF_DBG 0x7FC |
||||
|
||||
#define PCIE_SRDS_PRTCL(idx) (PCIE1 + (idx)) |
||||
#define PCIE_SYS_BASE_ADDR 0x3400000 |
||||
#define PCIE_CCSR_SIZE 0x0100000 |
||||
|
||||
/* CS2 */ |
||||
#define PCIE_CS2_OFFSET 0x1000 /* For PCIe without SR-IOV */ |
||||
|
||||
#define SVR_LS102XA 0 |
||||
#define SVR_VAR_PER_SHIFT 8 |
||||
#define SVR_LS102XA_MASK 0x700 |
||||
|
||||
/* LS1021a PCIE space */ |
||||
#define LS1021_PCIE_SPACE_OFFSET 0x4000000000ULL |
||||
#define LS1021_PCIE_SPACE_SIZE 0x0800000000ULL |
||||
|
||||
/* LS1021a PEX1/2 Misc Ports Status Register */ |
||||
#define LS1021_PEXMSCPORTSR(pex_idx) (0x94 + (pex_idx) * 4) |
||||
#define LS1021_LTSSM_STATE_SHIFT 20 |
||||
|
||||
struct ls_pcie { |
||||
int idx; |
||||
struct list_head list; |
||||
struct udevice *bus; |
||||
struct fdt_resource dbi_res; |
||||
struct fdt_resource lut_res; |
||||
struct fdt_resource ctrl_res; |
||||
struct fdt_resource cfg_res; |
||||
void __iomem *dbi; |
||||
void __iomem *lut; |
||||
void __iomem *ctrl; |
||||
void __iomem *cfg0; |
||||
void __iomem *cfg1; |
||||
bool big_endian; |
||||
bool enabled; |
||||
int next_lut_index; |
||||
struct pci_controller hose; |
||||
}; |
||||
|
||||
extern struct list_head ls_pcie_list; |
||||
|
||||
#endif /* _PCIE_LAYERSCAPE_H_ */ |
@ -0,0 +1,187 @@ |
||||
/*
|
||||
* Copyright 2014-2015 Freescale Semiconductor, Inc. |
||||
* Layerscape PCIe driver |
||||
* |
||||
* SPDX-License-Identifier: GPL-2.0+ |
||||
*/ |
||||
|
||||
#include <common.h> |
||||
#include <pci.h> |
||||
#include <asm/arch/fsl_serdes.h> |
||||
#include <asm/io.h> |
||||
#include <errno.h> |
||||
#ifdef CONFIG_OF_BOARD_SETUP |
||||
#include <libfdt.h> |
||||
#include <fdt_support.h> |
||||
#include "pcie_layerscape.h" |
||||
|
||||
#ifdef CONFIG_FSL_LSCH3 |
||||
/*
|
||||
* Return next available LUT index. |
||||
*/ |
||||
static int ls_pcie_next_lut_index(struct ls_pcie *pcie) |
||||
{ |
||||
if (pcie->next_lut_index < PCIE_LUT_ENTRY_COUNT) |
||||
return pcie->next_lut_index++; |
||||
else |
||||
return -ENOSPC; /* LUT is full */ |
||||
} |
||||
|
||||
/* returns the next available streamid for pcie, -errno if failed */ |
||||
static int ls_pcie_next_streamid(void) |
||||
{ |
||||
static int next_stream_id = FSL_PEX_STREAM_ID_START; |
||||
|
||||
if (next_stream_id > FSL_PEX_STREAM_ID_END) |
||||
return -EINVAL; |
||||
|
||||
return next_stream_id++; |
||||
} |
||||
|
||||
static void lut_writel(struct ls_pcie *pcie, unsigned int value, |
||||
unsigned int offset) |
||||
{ |
||||
if (pcie->big_endian) |
||||
out_be32(pcie->lut + offset, value); |
||||
else |
||||
out_le32(pcie->lut + offset, value); |
||||
} |
||||
|
||||
/*
|
||||
* Program a single LUT entry |
||||
*/ |
||||
static void ls_pcie_lut_set_mapping(struct ls_pcie *pcie, int index, u32 devid, |
||||
u32 streamid) |
||||
{ |
||||
/* leave mask as all zeroes, want to match all bits */ |
||||
lut_writel(pcie, devid << 16, PCIE_LUT_UDR(index)); |
||||
lut_writel(pcie, streamid | PCIE_LUT_ENABLE, PCIE_LUT_LDR(index)); |
||||
} |
||||
|
||||
/*
|
||||
* An msi-map is a property to be added to the pci controller |
||||
* node. It is a table, where each entry consists of 4 fields |
||||
* e.g.: |
||||
* |
||||
* msi-map = <[devid] [phandle-to-msi-ctrl] [stream-id] [count] |
||||
* [devid] [phandle-to-msi-ctrl] [stream-id] [count]>; |
||||
*/ |
||||
static void fdt_pcie_set_msi_map_entry(void *blob, struct ls_pcie *pcie, |
||||
u32 devid, u32 streamid) |
||||
{ |
||||
u32 *prop; |
||||
u32 phandle; |
||||
int nodeoffset; |
||||
|
||||
/* find pci controller node */ |
||||
nodeoffset = fdt_node_offset_by_compat_reg(blob, "fsl,ls-pcie", |
||||
pcie->dbi_res.start); |
||||
if (nodeoffset < 0) { |
||||
#ifdef CONFIG_FSL_PCIE_COMPAT /* Compatible with older version of dts node */ |
||||
nodeoffset = fdt_node_offset_by_compat_reg(blob, |
||||
CONFIG_FSL_PCIE_COMPAT, pcie->dbi_res.start); |
||||
if (nodeoffset < 0) |
||||
return; |
||||
#else |
||||
return; |
||||
#endif |
||||
} |
||||
|
||||
/* get phandle to MSI controller */ |
||||
prop = (u32 *)fdt_getprop(blob, nodeoffset, "msi-parent", 0); |
||||
if (prop == NULL) { |
||||
debug("\n%s: ERROR: missing msi-parent: PCIe%d\n", |
||||
__func__, pcie->idx); |
||||
return; |
||||
} |
||||
phandle = fdt32_to_cpu(*prop); |
||||
|
||||
/* set one msi-map row */ |
||||
fdt_appendprop_u32(blob, nodeoffset, "msi-map", devid); |
||||
fdt_appendprop_u32(blob, nodeoffset, "msi-map", phandle); |
||||
fdt_appendprop_u32(blob, nodeoffset, "msi-map", streamid); |
||||
fdt_appendprop_u32(blob, nodeoffset, "msi-map", 1); |
||||
} |
||||
|
||||
static void fdt_fixup_pcie(void *blob) |
||||
{ |
||||
struct udevice *dev, *bus; |
||||
struct ls_pcie *pcie; |
||||
int streamid; |
||||
int index; |
||||
pci_dev_t bdf; |
||||
|
||||
/* Scan all known buses */ |
||||
for (pci_find_first_device(&dev); |
||||
dev; |
||||
pci_find_next_device(&dev)) { |
||||
for (bus = dev; device_is_on_pci_bus(bus);) |
||||
bus = bus->parent; |
||||
pcie = dev_get_priv(bus); |
||||
|
||||
streamid = ls_pcie_next_streamid(); |
||||
if (streamid < 0) { |
||||
debug("ERROR: no stream ids free\n"); |
||||
continue; |
||||
} |
||||
|
||||
index = ls_pcie_next_lut_index(pcie); |
||||
if (index < 0) { |
||||
debug("ERROR: no LUT indexes free\n"); |
||||
continue; |
||||
} |
||||
|
||||
/* the DT fixup must be relative to the hose first_busno */ |
||||
bdf = dm_pci_get_bdf(dev) - PCI_BDF(bus->seq, 0, 0); |
||||
/* map PCI b.d.f to streamID in LUT */ |
||||
ls_pcie_lut_set_mapping(pcie, index, bdf >> 8, |
||||
streamid); |
||||
/* update msi-map in device tree */ |
||||
fdt_pcie_set_msi_map_entry(blob, pcie, bdf >> 8, |
||||
streamid); |
||||
} |
||||
} |
||||
#endif |
||||
|
||||
static void ft_pcie_ls_setup(void *blob, struct ls_pcie *pcie) |
||||
{ |
||||
int off; |
||||
|
||||
off = fdt_node_offset_by_compat_reg(blob, "fsl,ls-pcie", |
||||
pcie->dbi_res.start); |
||||
if (off < 0) { |
||||
#ifdef CONFIG_FSL_PCIE_COMPAT /* Compatible with older version of dts node */ |
||||
off = fdt_node_offset_by_compat_reg(blob, |
||||
CONFIG_FSL_PCIE_COMPAT, |
||||
pcie->dbi_res.start); |
||||
if (off < 0) |
||||
return; |
||||
#else |
||||
return; |
||||
#endif |
||||
} |
||||
|
||||
if (pcie->enabled) |
||||
fdt_set_node_status(blob, off, FDT_STATUS_OKAY, 0); |
||||
else |
||||
fdt_set_node_status(blob, off, FDT_STATUS_DISABLED, 0); |
||||
} |
||||
|
||||
/* Fixup Kernel DT for PCIe */ |
||||
void ft_pci_setup(void *blob, bd_t *bd) |
||||
{ |
||||
struct ls_pcie *pcie; |
||||
|
||||
list_for_each_entry(pcie, &ls_pcie_list, list) |
||||
ft_pcie_ls_setup(blob, pcie); |
||||
|
||||
#ifdef CONFIG_FSL_LSCH3 |
||||
fdt_fixup_pcie(blob); |
||||
#endif |
||||
} |
||||
|
||||
#else /* !CONFIG_OF_BOARD_SETUP */ |
||||
void ft_pci_setup(void *blob, bd_t *bd) |
||||
{ |
||||
} |
||||
#endif |
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in new issue