|
|
|
/*
|
|
|
|
* Copyright (C) 2016 Cavium, Inc.
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
|
|
* under the terms of version 2 of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/acpi.h>
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/interrupt.h>
|
|
|
|
#include <linux/pci.h>
|
|
|
|
#include <linux/netdevice.h>
|
|
|
|
#include <linux/etherdevice.h>
|
|
|
|
#include <linux/phy.h>
|
|
|
|
#include <linux/of.h>
|
|
|
|
#include <linux/of_mdio.h>
|
|
|
|
#include <linux/of_net.h>
|
|
|
|
|
|
|
|
#include "nic.h"
|
|
|
|
#include "thunder_bgx.h"
|
|
|
|
|
|
|
|
#define DRV_NAME "thunder-xcv"
|
|
|
|
#define DRV_VERSION "1.0"
|
|
|
|
|
|
|
|
/* Register offsets */
|
|
|
|
#define XCV_RESET 0x00
|
|
|
|
#define PORT_EN BIT_ULL(63)
|
|
|
|
#define CLK_RESET BIT_ULL(15)
|
|
|
|
#define DLL_RESET BIT_ULL(11)
|
|
|
|
#define COMP_EN BIT_ULL(7)
|
|
|
|
#define TX_PKT_RESET BIT_ULL(3)
|
|
|
|
#define TX_DATA_RESET BIT_ULL(2)
|
|
|
|
#define RX_PKT_RESET BIT_ULL(1)
|
|
|
|
#define RX_DATA_RESET BIT_ULL(0)
|
|
|
|
#define XCV_DLL_CTL 0x10
|
|
|
|
#define CLKRX_BYP BIT_ULL(23)
|
|
|
|
#define CLKTX_BYP BIT_ULL(15)
|
|
|
|
#define XCV_COMP_CTL 0x20
|
|
|
|
#define DRV_BYP BIT_ULL(63)
|
|
|
|
#define XCV_CTL 0x30
|
|
|
|
#define XCV_INT 0x40
|
|
|
|
#define XCV_INT_W1S 0x48
|
|
|
|
#define XCV_INT_ENA_W1C 0x50
|
|
|
|
#define XCV_INT_ENA_W1S 0x58
|
|
|
|
#define XCV_INBND_STATUS 0x80
|
|
|
|
#define XCV_BATCH_CRD_RET 0x100
|
|
|
|
|
|
|
|
struct xcv {
|
|
|
|
void __iomem *reg_base;
|
|
|
|
struct pci_dev *pdev;
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct xcv *xcv;
|
|
|
|
|
|
|
|
/* Supported devices */
|
|
|
|
static const struct pci_device_id xcv_id_table[] = {
|
|
|
|
{ PCI_DEVICE(PCI_VENDOR_ID_CAVIUM, 0xA056) },
|
|
|
|
{ 0, } /* end of table */
|
|
|
|
};
|
|
|
|
|
|
|
|
MODULE_AUTHOR("Cavium Inc");
|
|
|
|
MODULE_DESCRIPTION("Cavium Thunder RGX/XCV Driver");
|
|
|
|
MODULE_LICENSE("GPL v2");
|
|
|
|
MODULE_VERSION(DRV_VERSION);
|
|
|
|
MODULE_DEVICE_TABLE(pci, xcv_id_table);
|
|
|
|
|
|
|
|
void xcv_init_hw(void)
|
|
|
|
{
|
|
|
|
u64 cfg;
|
|
|
|
|
|
|
|
/* Take DLL out of reset */
|
|
|
|
cfg = readq_relaxed(xcv->reg_base + XCV_RESET);
|
|
|
|
cfg &= ~DLL_RESET;
|
|
|
|
writeq_relaxed(cfg, xcv->reg_base + XCV_RESET);
|
|
|
|
|
|
|
|
/* Take clock tree out of reset */
|
|
|
|
cfg = readq_relaxed(xcv->reg_base + XCV_RESET);
|
|
|
|
cfg &= ~CLK_RESET;
|
|
|
|
writeq_relaxed(cfg, xcv->reg_base + XCV_RESET);
|
|
|
|
/* Wait for DLL to lock */
|
|
|
|
msleep(1);
|
|
|
|
|
|
|
|
/* Configure DLL - enable or bypass
|
|
|
|
* TX no bypass, RX bypass
|
|
|
|
*/
|
|
|
|
cfg = readq_relaxed(xcv->reg_base + XCV_DLL_CTL);
|
|
|
|
cfg &= ~0xFF03;
|
|
|
|
cfg |= CLKRX_BYP;
|
|
|
|
writeq_relaxed(cfg, xcv->reg_base + XCV_DLL_CTL);
|
|
|
|
|
|
|
|
/* Enable compensation controller and force the
|
|
|
|
* write to be visible to HW by readig back.
|
|
|
|
*/
|
|
|
|
cfg = readq_relaxed(xcv->reg_base + XCV_RESET);
|
|
|
|
cfg |= COMP_EN;
|
|
|
|
writeq_relaxed(cfg, xcv->reg_base + XCV_RESET);
|
|
|
|
readq_relaxed(xcv->reg_base + XCV_RESET);
|
|
|
|
/* Wait for compensation state machine to lock */
|
|
|
|
msleep(10);
|
|
|
|
|
|
|
|
/* enable the XCV block */
|
|
|
|
cfg = readq_relaxed(xcv->reg_base + XCV_RESET);
|
|
|
|
cfg |= PORT_EN;
|
|
|
|
writeq_relaxed(cfg, xcv->reg_base + XCV_RESET);
|
|
|
|
|
|
|
|
cfg = readq_relaxed(xcv->reg_base + XCV_RESET);
|
|
|
|
cfg |= CLK_RESET;
|
|
|
|
writeq_relaxed(cfg, xcv->reg_base + XCV_RESET);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(xcv_init_hw);
|
|
|
|
|
|
|
|
void xcv_setup_link(bool link_up, int link_speed)
|
|
|
|
{
|
|
|
|
u64 cfg;
|
|
|
|
int speed = 2;
|
|
|
|
|
|
|
|
if (!xcv) {
|
|
|
|
pr_err("XCV init not done, probe may have failed\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (link_speed == 100)
|
|
|
|
speed = 1;
|
|
|
|
else if (link_speed == 10)
|
|
|
|
speed = 0;
|
|
|
|
|
|
|
|
if (link_up) {
|
|
|
|
/* set operating speed */
|
|
|
|
cfg = readq_relaxed(xcv->reg_base + XCV_CTL);
|
|
|
|
cfg &= ~0x03;
|
|
|
|
cfg |= speed;
|
|
|
|
writeq_relaxed(cfg, xcv->reg_base + XCV_CTL);
|
|
|
|
|
|
|
|
/* Reset datapaths */
|
|
|
|
cfg = readq_relaxed(xcv->reg_base + XCV_RESET);
|
|
|
|
cfg |= TX_DATA_RESET | RX_DATA_RESET;
|
|
|
|
writeq_relaxed(cfg, xcv->reg_base + XCV_RESET);
|
|
|
|
|
|
|
|
/* Enable the packet flow */
|
|
|
|
cfg = readq_relaxed(xcv->reg_base + XCV_RESET);
|
|
|
|
cfg |= TX_PKT_RESET | RX_PKT_RESET;
|
|
|
|
writeq_relaxed(cfg, xcv->reg_base + XCV_RESET);
|
|
|
|
|
|
|
|
/* Return credits to RGX */
|
|
|
|
writeq_relaxed(0x01, xcv->reg_base + XCV_BATCH_CRD_RET);
|
|
|
|
} else {
|
|
|
|
/* Disable packet flow */
|
|
|
|
cfg = readq_relaxed(xcv->reg_base + XCV_RESET);
|
|
|
|
cfg &= ~(TX_PKT_RESET | RX_PKT_RESET);
|
|
|
|
writeq_relaxed(cfg, xcv->reg_base + XCV_RESET);
|
|
|
|
readq_relaxed(xcv->reg_base + XCV_RESET);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(xcv_setup_link);
|
|
|
|
|
|
|
|
static int xcv_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
|
|
|
|
{
|
|
|
|
int err;
|
|
|
|
struct device *dev = &pdev->dev;
|
|
|
|
|
|
|
|
xcv = devm_kzalloc(dev, sizeof(struct xcv), GFP_KERNEL);
|
|
|
|
if (!xcv)
|
|
|
|
return -ENOMEM;
|
|
|
|
xcv->pdev = pdev;
|
|
|
|
|
|
|
|
pci_set_drvdata(pdev, xcv);
|
|
|
|
|
|
|
|
err = pci_enable_device(pdev);
|
|
|
|
if (err) {
|
|
|
|
dev_err(dev, "Failed to enable PCI device\n");
|
|
|
|
goto err_kfree;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = pci_request_regions(pdev, DRV_NAME);
|
|
|
|
if (err) {
|
|
|
|
dev_err(dev, "PCI request regions failed 0x%x\n", err);
|
|
|
|
goto err_disable_device;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* MAP configuration registers */
|
|
|
|
xcv->reg_base = pcim_iomap(pdev, PCI_CFG_REG_BAR_NUM, 0);
|
|
|
|
if (!xcv->reg_base) {
|
|
|
|
dev_err(dev, "XCV: Cannot map CSR memory space, aborting\n");
|
|
|
|
err = -ENOMEM;
|
|
|
|
goto err_release_regions;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
err_release_regions:
|
|
|
|
pci_release_regions(pdev);
|
|
|
|
err_disable_device:
|
|
|
|
pci_disable_device(pdev);
|
|
|
|
err_kfree:
|
|
|
|
devm_kfree(dev, xcv);
|
|
|
|
xcv = NULL;
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void xcv_remove(struct pci_dev *pdev)
|
|
|
|
{
|
|
|
|
struct device *dev = &pdev->dev;
|
|
|
|
|
|
|
|
if (xcv) {
|
|
|
|
devm_kfree(dev, xcv);
|
|
|
|
xcv = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
pci_release_regions(pdev);
|
|
|
|
pci_disable_device(pdev);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct pci_driver xcv_driver = {
|
|
|
|
.name = DRV_NAME,
|
|
|
|
.id_table = xcv_id_table,
|
|
|
|
.probe = xcv_probe,
|
|
|
|
.remove = xcv_remove,
|
|
|
|
};
|
|
|
|
|
|
|
|
static int __init xcv_init_module(void)
|
|
|
|
{
|
|
|
|
pr_info("%s, ver %s\n", DRV_NAME, DRV_VERSION);
|
|
|
|
|
|
|
|
return pci_register_driver(&xcv_driver);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void __exit xcv_cleanup_module(void)
|
|
|
|
{
|
|
|
|
pci_unregister_driver(&xcv_driver);
|
|
|
|
}
|
|
|
|
|
|
|
|
module_init(xcv_init_module);
|
|
|
|
module_exit(xcv_cleanup_module);
|