blob: 578c7f8f11bf23add2ac4d3c2263e371b4509136 [file] [log] [blame]
Sunil Goutham64658592016-08-12 16:51:33 +05301/*
2 * Copyright (C) 2016 Cavium, Inc.
3 *
4 * This program is free software; you can redistribute it and/or modify it
5 * under the terms of version 2 of the GNU General Public License
6 * as published by the Free Software Foundation.
7 */
8
9#include <linux/acpi.h>
10#include <linux/module.h>
11#include <linux/interrupt.h>
12#include <linux/pci.h>
13#include <linux/netdevice.h>
14#include <linux/etherdevice.h>
15#include <linux/phy.h>
16#include <linux/of.h>
17#include <linux/of_mdio.h>
18#include <linux/of_net.h>
19
20#include "nic.h"
21#include "thunder_bgx.h"
22
23#define DRV_NAME "thunder-xcv"
24#define DRV_VERSION "1.0"
25
26/* Register offsets */
27#define XCV_RESET 0x00
28#define PORT_EN BIT_ULL(63)
29#define CLK_RESET BIT_ULL(15)
30#define DLL_RESET BIT_ULL(11)
31#define COMP_EN BIT_ULL(7)
32#define TX_PKT_RESET BIT_ULL(3)
33#define TX_DATA_RESET BIT_ULL(2)
34#define RX_PKT_RESET BIT_ULL(1)
35#define RX_DATA_RESET BIT_ULL(0)
36#define XCV_DLL_CTL 0x10
37#define CLKRX_BYP BIT_ULL(23)
38#define CLKTX_BYP BIT_ULL(15)
39#define XCV_COMP_CTL 0x20
40#define DRV_BYP BIT_ULL(63)
41#define XCV_CTL 0x30
42#define XCV_INT 0x40
43#define XCV_INT_W1S 0x48
44#define XCV_INT_ENA_W1C 0x50
45#define XCV_INT_ENA_W1S 0x58
46#define XCV_INBND_STATUS 0x80
47#define XCV_BATCH_CRD_RET 0x100
48
49struct xcv {
50 void __iomem *reg_base;
51 struct pci_dev *pdev;
52};
53
54static struct xcv *xcv;
55
56/* Supported devices */
57static const struct pci_device_id xcv_id_table[] = {
58 { PCI_DEVICE(PCI_VENDOR_ID_CAVIUM, 0xA056) },
59 { 0, } /* end of table */
60};
61
62MODULE_AUTHOR("Cavium Inc");
63MODULE_DESCRIPTION("Cavium Thunder RGX/XCV Driver");
64MODULE_LICENSE("GPL v2");
65MODULE_VERSION(DRV_VERSION);
66MODULE_DEVICE_TABLE(pci, xcv_id_table);
67
68void xcv_init_hw(void)
69{
70 u64 cfg;
71
72 /* Take DLL out of reset */
73 cfg = readq_relaxed(xcv->reg_base + XCV_RESET);
74 cfg &= ~DLL_RESET;
75 writeq_relaxed(cfg, xcv->reg_base + XCV_RESET);
76
77 /* Take clock tree out of reset */
78 cfg = readq_relaxed(xcv->reg_base + XCV_RESET);
79 cfg &= ~CLK_RESET;
80 writeq_relaxed(cfg, xcv->reg_base + XCV_RESET);
81 /* Wait for DLL to lock */
82 msleep(1);
83
84 /* Configure DLL - enable or bypass
85 * TX no bypass, RX bypass
86 */
87 cfg = readq_relaxed(xcv->reg_base + XCV_DLL_CTL);
88 cfg &= ~0xFF03;
89 cfg |= CLKRX_BYP;
90 writeq_relaxed(cfg, xcv->reg_base + XCV_DLL_CTL);
91
92 /* Enable compensation controller and force the
93 * write to be visible to HW by readig back.
94 */
95 cfg = readq_relaxed(xcv->reg_base + XCV_RESET);
96 cfg |= COMP_EN;
97 writeq_relaxed(cfg, xcv->reg_base + XCV_RESET);
98 readq_relaxed(xcv->reg_base + XCV_RESET);
99 /* Wait for compensation state machine to lock */
100 msleep(10);
101
102 /* enable the XCV block */
103 cfg = readq_relaxed(xcv->reg_base + XCV_RESET);
104 cfg |= PORT_EN;
105 writeq_relaxed(cfg, xcv->reg_base + XCV_RESET);
106
107 cfg = readq_relaxed(xcv->reg_base + XCV_RESET);
108 cfg |= CLK_RESET;
109 writeq_relaxed(cfg, xcv->reg_base + XCV_RESET);
110}
111EXPORT_SYMBOL(xcv_init_hw);
112
113void xcv_setup_link(bool link_up, int link_speed)
114{
115 u64 cfg;
116 int speed = 2;
117
118 if (!xcv) {
Vincentc73e4422017-01-30 15:06:43 +0100119 pr_err("XCV init not done, probe may have failed\n");
Sunil Goutham64658592016-08-12 16:51:33 +0530120 return;
121 }
122
123 if (link_speed == 100)
124 speed = 1;
125 else if (link_speed == 10)
126 speed = 0;
127
128 if (link_up) {
129 /* set operating speed */
130 cfg = readq_relaxed(xcv->reg_base + XCV_CTL);
131 cfg &= ~0x03;
132 cfg |= speed;
133 writeq_relaxed(cfg, xcv->reg_base + XCV_CTL);
134
135 /* Reset datapaths */
136 cfg = readq_relaxed(xcv->reg_base + XCV_RESET);
137 cfg |= TX_DATA_RESET | RX_DATA_RESET;
138 writeq_relaxed(cfg, xcv->reg_base + XCV_RESET);
139
140 /* Enable the packet flow */
141 cfg = readq_relaxed(xcv->reg_base + XCV_RESET);
142 cfg |= TX_PKT_RESET | RX_PKT_RESET;
143 writeq_relaxed(cfg, xcv->reg_base + XCV_RESET);
144
145 /* Return credits to RGX */
146 writeq_relaxed(0x01, xcv->reg_base + XCV_BATCH_CRD_RET);
147 } else {
148 /* Disable packet flow */
149 cfg = readq_relaxed(xcv->reg_base + XCV_RESET);
150 cfg &= ~(TX_PKT_RESET | RX_PKT_RESET);
151 writeq_relaxed(cfg, xcv->reg_base + XCV_RESET);
152 readq_relaxed(xcv->reg_base + XCV_RESET);
153 }
154}
155EXPORT_SYMBOL(xcv_setup_link);
156
157static int xcv_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
158{
159 int err;
160 struct device *dev = &pdev->dev;
161
162 xcv = devm_kzalloc(dev, sizeof(struct xcv), GFP_KERNEL);
163 if (!xcv)
164 return -ENOMEM;
165 xcv->pdev = pdev;
166
167 pci_set_drvdata(pdev, xcv);
168
169 err = pci_enable_device(pdev);
170 if (err) {
171 dev_err(dev, "Failed to enable PCI device\n");
172 goto err_kfree;
173 }
174
175 err = pci_request_regions(pdev, DRV_NAME);
176 if (err) {
177 dev_err(dev, "PCI request regions failed 0x%x\n", err);
178 goto err_disable_device;
179 }
180
181 /* MAP configuration registers */
182 xcv->reg_base = pcim_iomap(pdev, PCI_CFG_REG_BAR_NUM, 0);
183 if (!xcv->reg_base) {
184 dev_err(dev, "XCV: Cannot map CSR memory space, aborting\n");
185 err = -ENOMEM;
186 goto err_release_regions;
187 }
188
189 return 0;
190
191err_release_regions:
192 pci_release_regions(pdev);
193err_disable_device:
194 pci_disable_device(pdev);
195err_kfree:
Sunil Goutham64658592016-08-12 16:51:33 +0530196 devm_kfree(dev, xcv);
197 xcv = NULL;
198 return err;
199}
200
201static void xcv_remove(struct pci_dev *pdev)
202{
203 struct device *dev = &pdev->dev;
204
205 if (xcv) {
206 devm_kfree(dev, xcv);
207 xcv = NULL;
208 }
209
210 pci_release_regions(pdev);
211 pci_disable_device(pdev);
Sunil Goutham64658592016-08-12 16:51:33 +0530212}
213
214static struct pci_driver xcv_driver = {
215 .name = DRV_NAME,
216 .id_table = xcv_id_table,
217 .probe = xcv_probe,
218 .remove = xcv_remove,
219};
220
221static int __init xcv_init_module(void)
222{
223 pr_info("%s, ver %s\n", DRV_NAME, DRV_VERSION);
224
225 return pci_register_driver(&xcv_driver);
226}
227
228static void __exit xcv_cleanup_module(void)
229{
230 pci_unregister_driver(&xcv_driver);
231}
232
233module_init(xcv_init_module);
234module_exit(xcv_cleanup_module);