sonic-buildimage/platform/broadcom/saibcm-modules/systems/bde/shared/shbde_pci.c

394 lines
11 KiB
C
Raw Normal View History

/*
* Unless you and Broadcom execute a separate written software license
* agreement governing use of this software, this software is licensed to
* you under the terms of the GNU General Public License version 2 (the
* "GPL"), available at http://www.broadcom.com/licenses/GPLv2.php,
* with the following added to such license:
*
* As a special exception, the copyright holders of this software give
* you permission to link this software with independent modules, and to
* copy and distribute the resulting executable under terms of your
* choice, provided that you also meet, for each linked independent
* module, the terms and conditions of the license of that module. An
* independent module is a module which is not derived from this
* software. The special exception does not apply to any modifications
* of the software.
*/
/*
* $Id: $
* $Copyright: (c) 2014 Broadcom Corp.
* All Rights Reserved.$
*
*/
#include <shbde_pci.h>
/* PCIe capabilities */
#ifndef PCI_CAPABILITY_LIST
#define PCI_CAPABILITY_LIST 0x34
#endif
#ifndef PCI_CAP_ID_EXP
#define PCI_CAP_ID_EXP 0x10
#endif
#ifndef PCI_EXP_DEVCAP
#define PCI_EXP_DEVCAP 4
#endif
#ifndef PCI_EXP_DEVCTL
#define PCI_EXP_DEVCTL 8
#endif
#ifndef PCI_EXT_CAP_START
#define PCI_EXT_CAP_START 0x100
#endif
#ifndef PCI_EXT_CAP_ID
#define PCI_EXT_CAP_ID(_hdr) (_hdr & 0x0000ffff)
#endif
#ifndef PCI_EXT_CAP_VER
#define PCI_EXT_CAP_VER(_hdr) ((_hdr >> 16) & 0xf)
#endif
#ifndef PCI_EXT_CAP_NEXT
#define PCI_EXT_CAP_NEXT(_hdr) ((_hdr >> 20) & 0xffc)
#endif
#ifndef PCI_EXT_CAP_ID_VNDR
#define PCI_EXT_CAP_ID_VNDR 0x0b
#endif
#define LOG_OUT(_shbde, _lvl, _str, _prm) \
if ((_shbde)->log_func) { \
(_shbde)->log_func(_lvl, _str, _prm); \
}
#define LOG_ERR(_shbde, _str, _prm) LOG_OUT(_shbde, SHBDE_ERR, _str, _prm)
#define LOG_WARN(_shbde, _str, _prm) LOG_OUT(_shbde, SHBDE_WARN, _str, _prm)
#define LOG_DBG(_shbde, _str, _prm) LOG_OUT(_shbde, SHBDE_DBG, _str, _prm)
#ifndef NULL
#define NULL (void *)0
#endif
/*
* Warpper functions with null-pointer checks.
*/
static unsigned int
pcic16_read(shbde_hal_t *shbde, void *pci_dev,
unsigned int addr)
{
if (!shbde || !shbde->pcic16_read) {
return 0;
}
return shbde->pcic16_read(pci_dev, addr);
}
static void
pcic16_write(shbde_hal_t *shbde, void *pci_dev,
unsigned int addr, unsigned int data)
{
if (!shbde || !shbde->pcic16_write) {
return;
}
shbde->pcic16_write(pci_dev, addr, data);
}
static unsigned int
pcic32_read(shbde_hal_t *shbde, void *pci_dev,
unsigned int addr)
{
if (!shbde || !shbde->pcic32_read) {
return 0;
}
return shbde->pcic32_read(pci_dev, addr);
}
static void *
pci_parent_device_get(shbde_hal_t *shbde, void *pci_dev)
{
if (!shbde || !shbde->pci_parent_device_get) {
return NULL;
}
return shbde->pci_parent_device_get(pci_dev);
}
/*
* Function:
* shbde_pci_pcie_cap
* Purpose:
* Return offset of PCIe capabilities in PCI configuration space
* Parameters:
* shbde - pointer to initialized hardware abstraction module
* dev - PCI device handle (passed back to PCI HAL functions)
* Returns:
* PCI_CAP_ID_EXP offset in PCI configuration space if PCIe, otherwise 0
*/
unsigned int
shbde_pci_pcie_cap(shbde_hal_t *shbde, void *pci_dev)
{
unsigned int cap_base, rval;
cap_base = pcic16_read(shbde, pci_dev, PCI_CAPABILITY_LIST);
while (cap_base) {
rval = pcic16_read(shbde, pci_dev, cap_base);
if ((rval & 0xff) == PCI_CAP_ID_EXP) {
break;
}
cap_base = (rval >> 8) & 0xff;
}
return cap_base;
}
/*
* Function:
* shbde_pci_is_pcie
* Purpose:
* Check if PCI device is PCIe device
* Parameters:
* shbde - pointer to initialized hardware abstraction module
* dev - PCI device handle (passed back to PCI HAL functions)
* Returns:
* 1 if PCIe, otherwise 0
*/
int
shbde_pci_is_pcie(shbde_hal_t *shbde, void *pci_dev)
{
return shbde_pci_pcie_cap(shbde, pci_dev) ? 1 : 0;
}
/*
* Function:
* shbde_pci_is_iproc
* Purpose:
* Check if PCI device is iProc-based
* Parameters:
* shbde - pointer to initialized hardware abstraction module
* dev - PCI device handle (passed back to PCI HAL functions)
* cmic_bar - (OUT) PCI BAR which contains switch CMIC registers
* Returns:
* 1 if iProc-based, otherwise 0
*/
int
shbde_pci_is_iproc(shbde_hal_t *shbde, void *pci_dev, int *cmic_bar)
{
unsigned int cap_base, rval;
if (!shbde_pci_is_pcie(shbde, pci_dev)) {
return 0;
}
/* Look for PCIe vendor-specific extended capability (VSEC) */
cap_base = PCI_EXT_CAP_START;
while (cap_base) {
rval = pcic32_read(shbde, pci_dev, cap_base);
if (rval == 0xffffffff) {
/* Assume PCI HW read error */
return 0;
}
if (PCI_EXT_CAP_ID(rval) == PCI_EXT_CAP_ID_VNDR) {
break;
}
cap_base = PCI_EXT_CAP_NEXT(rval);
}
if (cap_base) {
/*
* VSEC layout:
*
* 0x00: PCI Express Extended Capability Header
* 0x04: Vendor-Specific Header
* 0x08: Vendor-Specific Register 1
* 0x0c: Vendor-Specific Register 2
* ...
* 0x24: Vendor-Specific Register 8
*/
/* 32'b // 31:12=0 Reserved; 11:08=CMIC BAR; 07:00=iProc Configuration ID */
rval = pcic32_read(shbde, pci_dev, cap_base + 8);
LOG_DBG(shbde, "Found VSEC", rval);
/* Determine PCI BAR of CMIC */
*cmic_bar = 0;
if ((rval & 0x100) == 0x100) {
*cmic_bar = 2;
}
/* Assume iProc device */
return 1;
}
return 0;
}
/*
* Function:
* shbde_pci_iproc_version_get
* Purpose:
* Get iproc, cmic versions and revisions
* Parameters:
* shbde - pointer to initialized hardware abstraction module
* dev - PCI device handle (passed back to PCI HAL functions)
* iproc_ver - (OUT) iProc version
* cmic_ver - (OUT) CMIC version
* cmic_rev - (OUT) CMIC revision
* Returns:
* 1 for no error, otherwise 0
*/
int
shbde_pci_iproc_version_get(shbde_hal_t *shbde, void *pci_dev,
unsigned int *iproc_ver,
unsigned int *cmic_ver,
unsigned int *cmic_rev)
{
unsigned int cap_base, rval;
if (!shbde_pci_is_pcie(shbde, pci_dev)) {
return 0;
}
/* Look for PCIe vendor-specific extended capability (VSEC) */
cap_base = PCI_EXT_CAP_START;
while (cap_base) {
rval = pcic32_read(shbde, pci_dev, cap_base);
if (rval == 0xffffffff) {
/* Assume PCI HW read error */
return 0;
}
if (PCI_EXT_CAP_ID(rval) == PCI_EXT_CAP_ID_VNDR) {
break;
}
cap_base = PCI_EXT_CAP_NEXT(rval);
}
if (cap_base) {
/*
* VSEC layout:
*
* 0x00: PCI Express Extended Capability Header
* 0x04: Vendor-Specific Header
* 0x08: Vendor-Specific Register 1
* 0x0c: Vendor-Specific Register 2
* ...
* 0x24: Vendor-Specific Register 8
*/
/* Read PCIe Vendor Specific Register 1 */
/* VENODR REG FORMAT
* [7:0] iProc Rev = 8'h0E (for P14)
* [11:8] CMIC BAR = 4'h1 (BAR64-1)
* [15:12] CMIC Version = 4'h4
* [19:16] CMIC Rev = 4'h1
* [22:20] SBUS Version = 4'h4
*/
rval = pcic32_read(shbde, pci_dev, cap_base + 8);
LOG_DBG(shbde, "Found VSEC", rval);
/* Determine PCI BAR of CMIC */
*iproc_ver = rval & 0xff;
*cmic_ver = (rval >> 12) & 0xf;
*cmic_rev = (rval >> 16) & 0xf;
return 1;
}
return 0;
}
/*
* Function:
* shbde_pci_max_payload_set
* Purpose:
* Set PCIe maximum payload
* Parameters:
* shbde - pointer to initialized hardware abstraction module
* dev - PCI device handle (passed back to PCI HAL functions)
* maxpayload - maximum payload (in byte)
* Returns:
* -1 if error, otherwise 0
* Notes:
* If not PCIe device, set the PCI retry count to infinte instead.
*/
int
shbde_pci_max_payload_set(shbde_hal_t *shbde, void *pci_dev, int maxpayload)
{
unsigned int cap_base, parent_cap_base;
unsigned int devcap, devctl, parent_devctl;
int max_val, max_cap, parent_max_val;
void *parent_pci_dev;
cap_base = shbde_pci_pcie_cap(shbde, pci_dev);
if (cap_base == 0) {
/* Not PCIe */
return 0;
}
/* Get current device control settings */
devctl = pcic16_read(shbde, pci_dev, cap_base + PCI_EXP_DEVCTL);
/* Get current max payload setting */
max_val = (devctl >> 5) & 0x7;
if (maxpayload) {
/* Get encoding from byte value */
max_val = 0;
while ((1 << (max_val + 7)) < maxpayload) {
max_val++;
}
LOG_DBG(shbde, "Set max payload size", maxpayload);
LOG_DBG(shbde, "Set max payload val", max_val);
/* Get max supported payload size */
devcap = pcic16_read(shbde, pci_dev, cap_base + PCI_EXP_DEVCAP);
max_cap = (devcap & 0x7);
/* Do not exceed device capabilities */
if (max_val > max_cap) {
max_val = max_cap;
LOG_DBG(shbde,
"Payload size exceeds device capability",
maxpayload);
}
/* Get currently set max payload size for the parent device
* in the PCI tree (if it exists).
*/
parent_pci_dev = pci_parent_device_get(shbde, pci_dev);
if (parent_pci_dev != NULL) {
parent_cap_base = shbde_pci_pcie_cap(shbde, parent_pci_dev);
parent_devctl = pcic16_read(shbde,
parent_pci_dev,
parent_cap_base + PCI_EXP_DEVCTL);
parent_max_val = (parent_devctl >> 5) & 0x7;
/* Do not exceed current parent max payload setting (our device
* should have an MPS setting <= current parent MPS setting in
* the tree of PCIe devices).
*/
if (max_val > parent_max_val) {
max_val = parent_max_val;
LOG_DBG(shbde,
"Payload size exceeds current parent device setting",
maxpayload);
}
}
/* Update max payload size */
devctl &= ~(0x7 << 5);
devctl |= (max_val) << 5;
/* Update max request size */
devctl &= ~(0x7 << 12);
devctl |= (max_val << 12);
}
/* Always disable relaxed ordering */
devctl &= ~(1 << 4);
/* Update device control settings */
pcic16_write(shbde, pci_dev, cap_base + PCI_EXP_DEVCTL, devctl);
/* Warn if non-default setting is used */
if (max_val > 0) {
LOG_WARN(shbde,
"Selected payload size may not be supported by all "
"PCIe bridges by default.",
(1 << (max_val + 7)));
}
return 0;
}