Skip to content

Commit c27a02c

Browse files
Yevgeny PetrilinRoland Dreier
authored andcommitted
mlx4_en: Add driver for Mellanox ConnectX 10GbE NIC
The Mellanox ConnectX can operate as an InfiniBand adapter, as an Ethernet NIC, or as a Fibre Channel (FC) HBA. The kernel has a low-level driver, mlx4_core, which handles multiplexing access to the device, and there is also already an InfiniBad driver, mlx4_ib. This patch adds a new driver, mlx4_en, which implements a standard Ethernet NIC driver. Signed-off-by: Liran Liss <[email protected]> Signed-off-by: Yevgeny Petrilin <[email protected]> Signed-off-by: Roland Dreier <[email protected]>
1 parent 7ff93f8 commit c27a02c

File tree

12 files changed

+5370
-0
lines changed

12 files changed

+5370
-0
lines changed

drivers/net/Kconfig

Lines changed: 9 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -2465,6 +2465,15 @@ config PASEMI_MAC
24652465
This driver supports the on-chip 1/10Gbit Ethernet controller on
24662466
PA Semi's PWRficient line of chips.
24672467

2468+
config MLX4_EN
2469+
tristate "Mellanox Technologies 10Gbit Ethernet support"
2470+
depends on PCI && INET
2471+
select MLX4_CORE
2472+
select INET_LRO
2473+
help
2474+
This driver supports Mellanox Technologies ConnectX Ethernet
2475+
devices.
2476+
24682477
config MLX4_CORE
24692478
tristate
24702479
depends on PCI

drivers/net/mlx4/Makefile

Lines changed: 5 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -2,3 +2,8 @@ obj-$(CONFIG_MLX4_CORE) += mlx4_core.o
22

33
mlx4_core-y := alloc.o catas.o cmd.o cq.o eq.o fw.o icm.o intf.o main.o mcg.o \
44
mr.o pd.o port.o profile.o qp.o reset.o srq.o
5+
6+
obj-$(CONFIG_MLX4_EN) += mlx4_en.o
7+
8+
mlx4_en-y := en_main.o en_tx.o en_rx.o en_params.o en_port.o en_cq.o \
9+
en_resources.o en_netdev.o

drivers/net/mlx4/en_cq.c

Lines changed: 146 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,146 @@
1+
/*
2+
* Copyright (c) 2007 Mellanox Technologies. All rights reserved.
3+
*
4+
* This software is available to you under a choice of one of two
5+
* licenses. You may choose to be licensed under the terms of the GNU
6+
* General Public License (GPL) Version 2, available from the file
7+
* COPYING in the main directory of this source tree, or the
8+
* OpenIB.org BSD license below:
9+
*
10+
* Redistribution and use in source and binary forms, with or
11+
* without modification, are permitted provided that the following
12+
* conditions are met:
13+
*
14+
* - Redistributions of source code must retain the above
15+
* copyright notice, this list of conditions and the following
16+
* disclaimer.
17+
*
18+
* - Redistributions in binary form must reproduce the above
19+
* copyright notice, this list of conditions and the following
20+
* disclaimer in the documentation and/or other materials
21+
* provided with the distribution.
22+
*
23+
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
24+
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
25+
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
26+
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
27+
* BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
28+
* ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
29+
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
30+
* SOFTWARE.
31+
*
32+
*/
33+
34+
#include <linux/mlx4/cq.h>
35+
#include <linux/mlx4/qp.h>
36+
#include <linux/mlx4/cmd.h>
37+
38+
#include "mlx4_en.h"
39+
40+
static void mlx4_en_cq_event(struct mlx4_cq *cq, enum mlx4_event event)
41+
{
42+
return;
43+
}
44+
45+
46+
int mlx4_en_create_cq(struct mlx4_en_priv *priv,
47+
struct mlx4_en_cq *cq,
48+
int entries, int ring, enum cq_type mode)
49+
{
50+
struct mlx4_en_dev *mdev = priv->mdev;
51+
int err;
52+
53+
cq->size = entries;
54+
if (mode == RX)
55+
cq->buf_size = cq->size * sizeof(struct mlx4_cqe);
56+
else
57+
cq->buf_size = sizeof(struct mlx4_cqe);
58+
59+
cq->ring = ring;
60+
cq->is_tx = mode;
61+
spin_lock_init(&cq->lock);
62+
63+
err = mlx4_alloc_hwq_res(mdev->dev, &cq->wqres,
64+
cq->buf_size, 2 * PAGE_SIZE);
65+
if (err)
66+
return err;
67+
68+
err = mlx4_en_map_buffer(&cq->wqres.buf);
69+
if (err)
70+
mlx4_free_hwq_res(mdev->dev, &cq->wqres, cq->buf_size);
71+
72+
return err;
73+
}
74+
75+
int mlx4_en_activate_cq(struct mlx4_en_priv *priv, struct mlx4_en_cq *cq)
76+
{
77+
struct mlx4_en_dev *mdev = priv->mdev;
78+
int err;
79+
80+
cq->dev = mdev->pndev[priv->port];
81+
cq->mcq.set_ci_db = cq->wqres.db.db;
82+
cq->mcq.arm_db = cq->wqres.db.db + 1;
83+
*cq->mcq.set_ci_db = 0;
84+
*cq->mcq.arm_db = 0;
85+
cq->buf = (struct mlx4_cqe *) cq->wqres.buf.direct.buf;
86+
memset(cq->buf, 0, cq->buf_size);
87+
88+
err = mlx4_cq_alloc(mdev->dev, cq->size, &cq->wqres.mtt, &mdev->priv_uar,
89+
cq->wqres.db.dma, &cq->mcq, cq->is_tx);
90+
if (err)
91+
return err;
92+
93+
cq->mcq.comp = cq->is_tx ? mlx4_en_tx_irq : mlx4_en_rx_irq;
94+
cq->mcq.event = mlx4_en_cq_event;
95+
96+
if (cq->is_tx) {
97+
init_timer(&cq->timer);
98+
cq->timer.function = mlx4_en_poll_tx_cq;
99+
cq->timer.data = (unsigned long) cq;
100+
} else {
101+
netif_napi_add(cq->dev, &cq->napi, mlx4_en_poll_rx_cq, 64);
102+
napi_enable(&cq->napi);
103+
}
104+
105+
return 0;
106+
}
107+
108+
void mlx4_en_destroy_cq(struct mlx4_en_priv *priv, struct mlx4_en_cq *cq)
109+
{
110+
struct mlx4_en_dev *mdev = priv->mdev;
111+
112+
mlx4_en_unmap_buffer(&cq->wqres.buf);
113+
mlx4_free_hwq_res(mdev->dev, &cq->wqres, cq->buf_size);
114+
cq->buf_size = 0;
115+
cq->buf = NULL;
116+
}
117+
118+
void mlx4_en_deactivate_cq(struct mlx4_en_priv *priv, struct mlx4_en_cq *cq)
119+
{
120+
struct mlx4_en_dev *mdev = priv->mdev;
121+
122+
if (cq->is_tx)
123+
del_timer(&cq->timer);
124+
else
125+
napi_disable(&cq->napi);
126+
127+
mlx4_cq_free(mdev->dev, &cq->mcq);
128+
}
129+
130+
/* Set rx cq moderation parameters */
131+
int mlx4_en_set_cq_moder(struct mlx4_en_priv *priv, struct mlx4_en_cq *cq)
132+
{
133+
return mlx4_cq_modify(priv->mdev->dev, &cq->mcq,
134+
cq->moder_cnt, cq->moder_time);
135+
}
136+
137+
int mlx4_en_arm_cq(struct mlx4_en_priv *priv, struct mlx4_en_cq *cq)
138+
{
139+
cq->armed = 1;
140+
mlx4_cq_arm(&cq->mcq, MLX4_CQ_DB_REQ_NOT, priv->mdev->uar_map,
141+
&priv->mdev->uar_lock);
142+
143+
return 0;
144+
}
145+
146+

0 commit comments

Comments
 (0)