1 /* Broadcom NetXtreme-C/E network driver.
3 * Copyright (c) 2016-2018 Broadcom Limited
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation.
10 #include <linux/module.h>
12 #include <linux/kernel.h>
13 #include <linux/errno.h>
14 #include <linux/interrupt.h>
15 #include <linux/pci.h>
16 #include <linux/netdevice.h>
17 #include <linux/rtnetlink.h>
18 #include <linux/bitops.h>
19 #include <linux/irq.h>
20 #include <asm/byteorder.h>
21 #include <linux/bitmap.h>
27 static int bnxt_register_dev(struct bnxt_en_dev
*edev
, int ulp_id
,
28 struct bnxt_ulp_ops
*ulp_ops
, void *handle
)
30 struct net_device
*dev
= edev
->net
;
31 struct bnxt
*bp
= netdev_priv(dev
);
35 if (ulp_id
>= BNXT_MAX_ULP
)
38 ulp
= &edev
->ulp_tbl
[ulp_id
];
39 if (rcu_access_pointer(ulp
->ulp_ops
)) {
40 netdev_err(bp
->dev
, "ulp id %d already registered\n", ulp_id
);
43 if (ulp_id
== BNXT_ROCE_ULP
) {
44 unsigned int max_stat_ctxs
;
46 max_stat_ctxs
= bnxt_get_max_func_stat_ctxs(bp
);
47 if (max_stat_ctxs
<= BNXT_MIN_ROCE_STAT_CTXS
||
48 bp
->cp_nr_rings
== max_stat_ctxs
)
52 atomic_set(&ulp
->ref_count
, 0);
54 rcu_assign_pointer(ulp
->ulp_ops
, ulp_ops
);
56 if (ulp_id
== BNXT_ROCE_ULP
) {
57 if (test_bit(BNXT_STATE_OPEN
, &bp
->state
))
58 bnxt_hwrm_vnic_cfg(bp
, 0);
64 static int bnxt_unregister_dev(struct bnxt_en_dev
*edev
, int ulp_id
)
66 struct net_device
*dev
= edev
->net
;
67 struct bnxt
*bp
= netdev_priv(dev
);
72 if (ulp_id
>= BNXT_MAX_ULP
)
75 ulp
= &edev
->ulp_tbl
[ulp_id
];
76 if (!rcu_access_pointer(ulp
->ulp_ops
)) {
77 netdev_err(bp
->dev
, "ulp id %d not registered\n", ulp_id
);
80 if (ulp_id
== BNXT_ROCE_ULP
&& ulp
->msix_requested
)
81 edev
->en_ops
->bnxt_free_msix(edev
, ulp_id
);
83 if (ulp
->max_async_event_id
)
84 bnxt_hwrm_func_drv_rgtr(bp
, NULL
, 0, true);
86 RCU_INIT_POINTER(ulp
->ulp_ops
, NULL
);
88 ulp
->max_async_event_id
= 0;
89 ulp
->async_events_bmap
= NULL
;
90 while (atomic_read(&ulp
->ref_count
) != 0 && i
< 10) {
97 static void bnxt_fill_msix_vecs(struct bnxt
*bp
, struct bnxt_msix_entry
*ent
)
99 struct bnxt_en_dev
*edev
= bp
->edev
;
100 int num_msix
, idx
, i
;
102 num_msix
= edev
->ulp_tbl
[BNXT_ROCE_ULP
].msix_requested
;
103 idx
= edev
->ulp_tbl
[BNXT_ROCE_ULP
].msix_base
;
104 for (i
= 0; i
< num_msix
; i
++) {
105 ent
[i
].vector
= bp
->irq_tbl
[idx
+ i
].vector
;
106 ent
[i
].ring_idx
= idx
+ i
;
107 if (bp
->flags
& BNXT_FLAG_CHIP_P5
) {
108 ent
[i
].db_offset
= DB_PF_OFFSET_P5
;
110 ent
[i
].db_offset
= DB_VF_OFFSET_P5
;
112 ent
[i
].db_offset
= (idx
+ i
) * 0x80;
117 static int bnxt_req_msix_vecs(struct bnxt_en_dev
*edev
, int ulp_id
,
118 struct bnxt_msix_entry
*ent
, int num_msix
)
120 struct net_device
*dev
= edev
->net
;
121 struct bnxt
*bp
= netdev_priv(dev
);
122 struct bnxt_hw_resc
*hw_resc
;
123 int max_idx
, max_cp_rings
;
129 if (ulp_id
!= BNXT_ROCE_ULP
)
132 if (!(bp
->flags
& BNXT_FLAG_USING_MSIX
))
135 if (edev
->ulp_tbl
[ulp_id
].msix_requested
)
138 max_cp_rings
= bnxt_get_max_func_cp_rings(bp
);
139 avail_msix
= bnxt_get_avail_msix(bp
, num_msix
);
142 if (avail_msix
> num_msix
)
143 avail_msix
= num_msix
;
145 if (BNXT_NEW_RM(bp
)) {
146 idx
= bp
->cp_nr_rings
;
148 max_idx
= min_t(int, bp
->total_irqs
, max_cp_rings
);
149 idx
= max_idx
- avail_msix
;
151 edev
->ulp_tbl
[ulp_id
].msix_base
= idx
;
152 edev
->ulp_tbl
[ulp_id
].msix_requested
= avail_msix
;
153 hw_resc
= &bp
->hw_resc
;
154 total_vecs
= idx
+ avail_msix
;
155 if (bp
->total_irqs
< total_vecs
||
156 (BNXT_NEW_RM(bp
) && hw_resc
->resv_irqs
< total_vecs
)) {
157 if (netif_running(dev
)) {
158 bnxt_close_nic(bp
, true, false);
159 rc
= bnxt_open_nic(bp
, true, false);
161 rc
= bnxt_reserve_rings(bp
, true);
165 edev
->ulp_tbl
[ulp_id
].msix_requested
= 0;
169 if (BNXT_NEW_RM(bp
)) {
172 resv_msix
= hw_resc
->resv_irqs
- bp
->cp_nr_rings
;
173 avail_msix
= min_t(int, resv_msix
, avail_msix
);
174 edev
->ulp_tbl
[ulp_id
].msix_requested
= avail_msix
;
176 bnxt_fill_msix_vecs(bp
, ent
);
177 edev
->flags
|= BNXT_EN_FLAG_MSIX_REQUESTED
;
181 static int bnxt_free_msix_vecs(struct bnxt_en_dev
*edev
, int ulp_id
)
183 struct net_device
*dev
= edev
->net
;
184 struct bnxt
*bp
= netdev_priv(dev
);
187 if (ulp_id
!= BNXT_ROCE_ULP
)
190 if (!(edev
->flags
& BNXT_EN_FLAG_MSIX_REQUESTED
))
193 edev
->ulp_tbl
[ulp_id
].msix_requested
= 0;
194 edev
->flags
&= ~BNXT_EN_FLAG_MSIX_REQUESTED
;
195 if (netif_running(dev
) && !(edev
->flags
& BNXT_EN_FLAG_ULP_STOPPED
)) {
196 bnxt_close_nic(bp
, true, false);
197 bnxt_open_nic(bp
, true, false);
202 int bnxt_get_ulp_msix_num(struct bnxt
*bp
)
204 if (bnxt_ulp_registered(bp
->edev
, BNXT_ROCE_ULP
)) {
205 struct bnxt_en_dev
*edev
= bp
->edev
;
207 return edev
->ulp_tbl
[BNXT_ROCE_ULP
].msix_requested
;
212 int bnxt_get_ulp_msix_base(struct bnxt
*bp
)
214 if (bnxt_ulp_registered(bp
->edev
, BNXT_ROCE_ULP
)) {
215 struct bnxt_en_dev
*edev
= bp
->edev
;
217 if (edev
->ulp_tbl
[BNXT_ROCE_ULP
].msix_requested
)
218 return edev
->ulp_tbl
[BNXT_ROCE_ULP
].msix_base
;
223 int bnxt_get_ulp_stat_ctxs(struct bnxt
*bp
)
225 if (bnxt_ulp_registered(bp
->edev
, BNXT_ROCE_ULP
))
226 return BNXT_MIN_ROCE_STAT_CTXS
;
231 static int bnxt_send_msg(struct bnxt_en_dev
*edev
, int ulp_id
,
232 struct bnxt_fw_msg
*fw_msg
)
234 struct net_device
*dev
= edev
->net
;
235 struct bnxt
*bp
= netdev_priv(dev
);
239 if (ulp_id
!= BNXT_ROCE_ULP
&& bp
->fw_reset_state
)
242 mutex_lock(&bp
->hwrm_cmd_lock
);
244 req
->resp_addr
= cpu_to_le64(bp
->hwrm_cmd_resp_dma_addr
);
245 rc
= _hwrm_send_message(bp
, fw_msg
->msg
, fw_msg
->msg_len
,
248 struct output
*resp
= bp
->hwrm_cmd_resp_addr
;
249 u32 len
= le16_to_cpu(resp
->resp_len
);
251 if (fw_msg
->resp_max_len
< len
)
252 len
= fw_msg
->resp_max_len
;
254 memcpy(fw_msg
->resp
, resp
, len
);
256 mutex_unlock(&bp
->hwrm_cmd_lock
);
260 static void bnxt_ulp_get(struct bnxt_ulp
*ulp
)
262 atomic_inc(&ulp
->ref_count
);
265 static void bnxt_ulp_put(struct bnxt_ulp
*ulp
)
267 atomic_dec(&ulp
->ref_count
);
270 void bnxt_ulp_stop(struct bnxt
*bp
)
272 struct bnxt_en_dev
*edev
= bp
->edev
;
273 struct bnxt_ulp_ops
*ops
;
279 edev
->flags
|= BNXT_EN_FLAG_ULP_STOPPED
;
280 for (i
= 0; i
< BNXT_MAX_ULP
; i
++) {
281 struct bnxt_ulp
*ulp
= &edev
->ulp_tbl
[i
];
283 ops
= rtnl_dereference(ulp
->ulp_ops
);
284 if (!ops
|| !ops
->ulp_stop
)
286 ops
->ulp_stop(ulp
->handle
);
290 void bnxt_ulp_start(struct bnxt
*bp
, int err
)
292 struct bnxt_en_dev
*edev
= bp
->edev
;
293 struct bnxt_ulp_ops
*ops
;
299 edev
->flags
&= ~BNXT_EN_FLAG_ULP_STOPPED
;
304 for (i
= 0; i
< BNXT_MAX_ULP
; i
++) {
305 struct bnxt_ulp
*ulp
= &edev
->ulp_tbl
[i
];
307 ops
= rtnl_dereference(ulp
->ulp_ops
);
308 if (!ops
|| !ops
->ulp_start
)
310 ops
->ulp_start(ulp
->handle
);
314 void bnxt_ulp_sriov_cfg(struct bnxt
*bp
, int num_vfs
)
316 struct bnxt_en_dev
*edev
= bp
->edev
;
317 struct bnxt_ulp_ops
*ops
;
323 for (i
= 0; i
< BNXT_MAX_ULP
; i
++) {
324 struct bnxt_ulp
*ulp
= &edev
->ulp_tbl
[i
];
327 ops
= rcu_dereference(ulp
->ulp_ops
);
328 if (!ops
|| !ops
->ulp_sriov_config
) {
334 ops
->ulp_sriov_config(ulp
->handle
, num_vfs
);
339 void bnxt_ulp_shutdown(struct bnxt
*bp
)
341 struct bnxt_en_dev
*edev
= bp
->edev
;
342 struct bnxt_ulp_ops
*ops
;
348 for (i
= 0; i
< BNXT_MAX_ULP
; i
++) {
349 struct bnxt_ulp
*ulp
= &edev
->ulp_tbl
[i
];
351 ops
= rtnl_dereference(ulp
->ulp_ops
);
352 if (!ops
|| !ops
->ulp_shutdown
)
354 ops
->ulp_shutdown(ulp
->handle
);
358 void bnxt_ulp_irq_stop(struct bnxt
*bp
)
360 struct bnxt_en_dev
*edev
= bp
->edev
;
361 struct bnxt_ulp_ops
*ops
;
363 if (!edev
|| !(edev
->flags
& BNXT_EN_FLAG_MSIX_REQUESTED
))
366 if (bnxt_ulp_registered(bp
->edev
, BNXT_ROCE_ULP
)) {
367 struct bnxt_ulp
*ulp
= &edev
->ulp_tbl
[BNXT_ROCE_ULP
];
369 if (!ulp
->msix_requested
)
372 ops
= rtnl_dereference(ulp
->ulp_ops
);
373 if (!ops
|| !ops
->ulp_irq_stop
)
375 ops
->ulp_irq_stop(ulp
->handle
);
379 void bnxt_ulp_irq_restart(struct bnxt
*bp
, int err
)
381 struct bnxt_en_dev
*edev
= bp
->edev
;
382 struct bnxt_ulp_ops
*ops
;
384 if (!edev
|| !(edev
->flags
& BNXT_EN_FLAG_MSIX_REQUESTED
))
387 if (bnxt_ulp_registered(bp
->edev
, BNXT_ROCE_ULP
)) {
388 struct bnxt_ulp
*ulp
= &edev
->ulp_tbl
[BNXT_ROCE_ULP
];
389 struct bnxt_msix_entry
*ent
= NULL
;
391 if (!ulp
->msix_requested
)
394 ops
= rtnl_dereference(ulp
->ulp_ops
);
395 if (!ops
|| !ops
->ulp_irq_restart
)
399 ent
= kcalloc(ulp
->msix_requested
, sizeof(*ent
),
403 bnxt_fill_msix_vecs(bp
, ent
);
405 ops
->ulp_irq_restart(ulp
->handle
, ent
);
410 void bnxt_ulp_async_events(struct bnxt
*bp
, struct hwrm_async_event_cmpl
*cmpl
)
412 u16 event_id
= le16_to_cpu(cmpl
->event_id
);
413 struct bnxt_en_dev
*edev
= bp
->edev
;
414 struct bnxt_ulp_ops
*ops
;
421 for (i
= 0; i
< BNXT_MAX_ULP
; i
++) {
422 struct bnxt_ulp
*ulp
= &edev
->ulp_tbl
[i
];
424 ops
= rcu_dereference(ulp
->ulp_ops
);
425 if (!ops
|| !ops
->ulp_async_notifier
)
427 if (!ulp
->async_events_bmap
||
428 event_id
> ulp
->max_async_event_id
)
431 /* Read max_async_event_id first before testing the bitmap. */
433 if (test_bit(event_id
, ulp
->async_events_bmap
))
434 ops
->ulp_async_notifier(ulp
->handle
, cmpl
);
439 static int bnxt_register_async_events(struct bnxt_en_dev
*edev
, int ulp_id
,
440 unsigned long *events_bmap
, u16 max_id
)
442 struct net_device
*dev
= edev
->net
;
443 struct bnxt
*bp
= netdev_priv(dev
);
444 struct bnxt_ulp
*ulp
;
446 if (ulp_id
>= BNXT_MAX_ULP
)
449 ulp
= &edev
->ulp_tbl
[ulp_id
];
450 ulp
->async_events_bmap
= events_bmap
;
451 /* Make sure bnxt_ulp_async_events() sees this order */
453 ulp
->max_async_event_id
= max_id
;
454 bnxt_hwrm_func_drv_rgtr(bp
, events_bmap
, max_id
+ 1, true);
458 static const struct bnxt_en_ops bnxt_en_ops_tbl
= {
459 .bnxt_register_device
= bnxt_register_dev
,
460 .bnxt_unregister_device
= bnxt_unregister_dev
,
461 .bnxt_request_msix
= bnxt_req_msix_vecs
,
462 .bnxt_free_msix
= bnxt_free_msix_vecs
,
463 .bnxt_send_fw_msg
= bnxt_send_msg
,
464 .bnxt_register_fw_async_events
= bnxt_register_async_events
,
467 struct bnxt_en_dev
*bnxt_ulp_probe(struct net_device
*dev
)
469 struct bnxt
*bp
= netdev_priv(dev
);
470 struct bnxt_en_dev
*edev
;
474 edev
= kzalloc(sizeof(*edev
), GFP_KERNEL
);
476 return ERR_PTR(-ENOMEM
);
477 edev
->en_ops
= &bnxt_en_ops_tbl
;
478 if (bp
->flags
& BNXT_FLAG_ROCEV1_CAP
)
479 edev
->flags
|= BNXT_EN_FLAG_ROCEV1_CAP
;
480 if (bp
->flags
& BNXT_FLAG_ROCEV2_CAP
)
481 edev
->flags
|= BNXT_EN_FLAG_ROCEV2_CAP
;
483 edev
->pdev
= bp
->pdev
;
484 edev
->l2_db_size
= bp
->db_size
;
485 edev
->l2_db_size_nc
= bp
->db_size
;