mlx5-update-2017-05-23
First patch from Leon, came to remove the redundant usage of mlx5_vzalloc, and directly use kvzalloc across all mlx5 drivers. 2nd patch from Noa, adds new device IDs into the supported devices list. 3rd and 4th patches from Ilan are adding the basic infrastructure and support for Mellanox's mlx5 FPGA. Last two patches from Tariq came to modify the outdated driver version reported in ethtool and in mlx5_ib to more reflect the current driver state and remove the redundant date string reported in the version. Thanks, Saeed. -----BEGIN PGP SIGNATURE----- Version: GnuPG v1 iQEcBAABAgAGBQJZJBu5AAoJEEg/ir3gV/o+bCAH/Av6h8TcoR6p+NzfJBqfY+6G dXfXbxxpAGGZQeC7PDU+3BG/sw8bvQxJ4P23zs4BPw40X78s4+zUknzdnw6ZxvFh Kk9C8MbsEG60k1hw9mzSx3cYyYjol/6UdLPxVVJC4dJbYAXJl5ZZfiUf+2J85mic 3YZpvE4HymyOo2pGRDyi0i3ailr5njhQQ4bBxYLmyY0mbAm/hf/O2oKcQPuwvzTI Gwkaj4S207rA1xDO6PCp884P5mhstmL3Cs9JnsQirISn5n/5L51Ezt7/63C1ysab 2t/JM+MiEMUXk+plnGwtkPUo5/am41e044GmJfMT6Kerq+ZCwdO5/rQ7wc4qJXU= =fYgO -----END PGP SIGNATURE----- Merge tag 'mlx5-update-2017-05-23' of git://git.kernel.org/pub/scm/linux/kernel/git/mellanox/linux Saeed Mahameed says: ==================== mlx5-update-2017-05-23 First patch from Leon, came to remove the redundant usage of mlx5_vzalloc, and directly use kvzalloc across all mlx5 drivers. 2nd patch from Noa, adds new device IDs into the supported devices list. 3rd and 4th patches from Ilan are adding the basic infrastructure and support for Mellanox's mlx5 FPGA. Last two patches from Tariq came to modify the outdated driver version reported in ethtool and in mlx5_ib to more reflect the current driver state and remove the redundant date string reported in the version. ==================== Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
commit
abc7a4ef84
10
MAINTAINERS
10
MAINTAINERS
@ -8311,6 +8311,16 @@ W: http://www.mellanox.com
|
|||||||
Q: http://patchwork.ozlabs.org/project/netdev/list/
|
Q: http://patchwork.ozlabs.org/project/netdev/list/
|
||||||
F: drivers/net/ethernet/mellanox/mlx5/core/en_*
|
F: drivers/net/ethernet/mellanox/mlx5/core/en_*
|
||||||
|
|
||||||
|
MELLANOX ETHERNET INNOVA DRIVER
|
||||||
|
M: Ilan Tayari <ilant@mellanox.com>
|
||||||
|
R: Boris Pismenny <borisp@mellanox.com>
|
||||||
|
L: netdev@vger.kernel.org
|
||||||
|
S: Supported
|
||||||
|
W: http://www.mellanox.com
|
||||||
|
Q: http://patchwork.ozlabs.org/project/netdev/list/
|
||||||
|
F: drivers/net/ethernet/mellanox/mlx5/core/fpga/*
|
||||||
|
F: include/linux/mlx5/mlx5_ifc_fpga.h
|
||||||
|
|
||||||
MELLANOX ETHERNET SWITCH DRIVERS
|
MELLANOX ETHERNET SWITCH DRIVERS
|
||||||
M: Jiri Pirko <jiri@mellanox.com>
|
M: Jiri Pirko <jiri@mellanox.com>
|
||||||
M: Ido Schimmel <idosch@mellanox.com>
|
M: Ido Schimmel <idosch@mellanox.com>
|
||||||
|
@ -788,7 +788,7 @@ static int create_cq_user(struct mlx5_ib_dev *dev, struct ib_udata *udata,
|
|||||||
|
|
||||||
*inlen = MLX5_ST_SZ_BYTES(create_cq_in) +
|
*inlen = MLX5_ST_SZ_BYTES(create_cq_in) +
|
||||||
MLX5_FLD_SZ_BYTES(create_cq_in, pas[0]) * ncont;
|
MLX5_FLD_SZ_BYTES(create_cq_in, pas[0]) * ncont;
|
||||||
*cqb = mlx5_vzalloc(*inlen);
|
*cqb = kvzalloc(*inlen, GFP_KERNEL);
|
||||||
if (!*cqb) {
|
if (!*cqb) {
|
||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
goto err_db;
|
goto err_db;
|
||||||
@ -884,7 +884,7 @@ static int create_cq_kernel(struct mlx5_ib_dev *dev, struct mlx5_ib_cq *cq,
|
|||||||
|
|
||||||
*inlen = MLX5_ST_SZ_BYTES(create_cq_in) +
|
*inlen = MLX5_ST_SZ_BYTES(create_cq_in) +
|
||||||
MLX5_FLD_SZ_BYTES(create_cq_in, pas[0]) * cq->buf.buf.npages;
|
MLX5_FLD_SZ_BYTES(create_cq_in, pas[0]) * cq->buf.buf.npages;
|
||||||
*cqb = mlx5_vzalloc(*inlen);
|
*cqb = kvzalloc(*inlen, GFP_KERNEL);
|
||||||
if (!*cqb) {
|
if (!*cqb) {
|
||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
goto err_buf;
|
goto err_buf;
|
||||||
@ -1314,7 +1314,7 @@ int mlx5_ib_resize_cq(struct ib_cq *ibcq, int entries, struct ib_udata *udata)
|
|||||||
inlen = MLX5_ST_SZ_BYTES(modify_cq_in) +
|
inlen = MLX5_ST_SZ_BYTES(modify_cq_in) +
|
||||||
MLX5_FLD_SZ_BYTES(modify_cq_in, pas[0]) * npas;
|
MLX5_FLD_SZ_BYTES(modify_cq_in, pas[0]) * npas;
|
||||||
|
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in) {
|
if (!in) {
|
||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
goto ex_resize;
|
goto ex_resize;
|
||||||
|
@ -218,7 +218,7 @@ static int process_pma_cmd(struct ib_device *ibdev, u8 port_num,
|
|||||||
(struct ib_pma_portcounters_ext *)(out_mad->data + 40);
|
(struct ib_pma_portcounters_ext *)(out_mad->data + 40);
|
||||||
int sz = MLX5_ST_SZ_BYTES(query_vport_counter_out);
|
int sz = MLX5_ST_SZ_BYTES(query_vport_counter_out);
|
||||||
|
|
||||||
out_cnt = mlx5_vzalloc(sz);
|
out_cnt = kvzalloc(sz, GFP_KERNEL);
|
||||||
if (!out_cnt)
|
if (!out_cnt)
|
||||||
return IB_MAD_RESULT_FAILURE;
|
return IB_MAD_RESULT_FAILURE;
|
||||||
|
|
||||||
@ -231,7 +231,7 @@ static int process_pma_cmd(struct ib_device *ibdev, u8 port_num,
|
|||||||
(struct ib_pma_portcounters *)(out_mad->data + 40);
|
(struct ib_pma_portcounters *)(out_mad->data + 40);
|
||||||
int sz = MLX5_ST_SZ_BYTES(ppcnt_reg);
|
int sz = MLX5_ST_SZ_BYTES(ppcnt_reg);
|
||||||
|
|
||||||
out_cnt = mlx5_vzalloc(sz);
|
out_cnt = kvzalloc(sz, GFP_KERNEL);
|
||||||
if (!out_cnt)
|
if (!out_cnt)
|
||||||
return IB_MAD_RESULT_FAILURE;
|
return IB_MAD_RESULT_FAILURE;
|
||||||
|
|
||||||
|
@ -60,8 +60,7 @@
|
|||||||
#include "cmd.h"
|
#include "cmd.h"
|
||||||
|
|
||||||
#define DRIVER_NAME "mlx5_ib"
|
#define DRIVER_NAME "mlx5_ib"
|
||||||
#define DRIVER_VERSION "2.2-1"
|
#define DRIVER_VERSION "5.0-0"
|
||||||
#define DRIVER_RELDATE "Feb 2014"
|
|
||||||
|
|
||||||
MODULE_AUTHOR("Eli Cohen <eli@mellanox.com>");
|
MODULE_AUTHOR("Eli Cohen <eli@mellanox.com>");
|
||||||
MODULE_DESCRIPTION("Mellanox Connect-IB HCA IB driver");
|
MODULE_DESCRIPTION("Mellanox Connect-IB HCA IB driver");
|
||||||
@ -70,7 +69,7 @@ MODULE_VERSION(DRIVER_VERSION);
|
|||||||
|
|
||||||
static char mlx5_version[] =
|
static char mlx5_version[] =
|
||||||
DRIVER_NAME ": Mellanox Connect-IB Infiniband driver v"
|
DRIVER_NAME ": Mellanox Connect-IB Infiniband driver v"
|
||||||
DRIVER_VERSION " (" DRIVER_RELDATE ")\n";
|
DRIVER_VERSION "\n";
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
MLX5_ATOMIC_SIZE_QP_8BYTES = 1 << 3,
|
MLX5_ATOMIC_SIZE_QP_8BYTES = 1 << 3,
|
||||||
@ -2263,7 +2262,7 @@ static struct mlx5_ib_flow_handler *create_flow_rule(struct mlx5_ib_dev *dev,
|
|||||||
if (!is_valid_attr(dev->mdev, flow_attr))
|
if (!is_valid_attr(dev->mdev, flow_attr))
|
||||||
return ERR_PTR(-EINVAL);
|
return ERR_PTR(-EINVAL);
|
||||||
|
|
||||||
spec = mlx5_vzalloc(sizeof(*spec));
|
spec = kvzalloc(sizeof(*spec), GFP_KERNEL);
|
||||||
handler = kzalloc(sizeof(*handler), GFP_KERNEL);
|
handler = kzalloc(sizeof(*handler), GFP_KERNEL);
|
||||||
if (!handler || !spec) {
|
if (!handler || !spec) {
|
||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
@ -3456,7 +3455,7 @@ static int mlx5_ib_query_q_counters(struct mlx5_ib_dev *dev,
|
|||||||
__be32 val;
|
__be32 val;
|
||||||
int ret, i;
|
int ret, i;
|
||||||
|
|
||||||
out = mlx5_vzalloc(outlen);
|
out = kvzalloc(outlen, GFP_KERNEL);
|
||||||
if (!out)
|
if (!out)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -3485,7 +3484,7 @@ static int mlx5_ib_query_cong_counters(struct mlx5_ib_dev *dev,
|
|||||||
int ret, i;
|
int ret, i;
|
||||||
int offset = port->cnts.num_q_counters;
|
int offset = port->cnts.num_q_counters;
|
||||||
|
|
||||||
out = mlx5_vzalloc(outlen);
|
out = kvzalloc(outlen, GFP_KERNEL);
|
||||||
if (!out)
|
if (!out)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
|
@ -1110,7 +1110,7 @@ static struct mlx5_ib_mr *reg_create(struct ib_mr *ibmr, struct ib_pd *pd,
|
|||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(create_mkey_in) +
|
inlen = MLX5_ST_SZ_BYTES(create_mkey_in) +
|
||||||
sizeof(*pas) * ((npages + 1) / 2) * 2;
|
sizeof(*pas) * ((npages + 1) / 2) * 2;
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in) {
|
if (!in) {
|
||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
goto err_1;
|
goto err_1;
|
||||||
|
@ -823,7 +823,7 @@ static int create_user_qp(struct mlx5_ib_dev *dev, struct ib_pd *pd,
|
|||||||
|
|
||||||
*inlen = MLX5_ST_SZ_BYTES(create_qp_in) +
|
*inlen = MLX5_ST_SZ_BYTES(create_qp_in) +
|
||||||
MLX5_FLD_SZ_BYTES(create_qp_in, pas[0]) * ncont;
|
MLX5_FLD_SZ_BYTES(create_qp_in, pas[0]) * ncont;
|
||||||
*in = mlx5_vzalloc(*inlen);
|
*in = kvzalloc(*inlen, GFP_KERNEL);
|
||||||
if (!*in) {
|
if (!*in) {
|
||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
goto err_umem;
|
goto err_umem;
|
||||||
@ -931,7 +931,7 @@ static int create_kernel_qp(struct mlx5_ib_dev *dev,
|
|||||||
qp->sq.qend = mlx5_get_send_wqe(qp, qp->sq.wqe_cnt);
|
qp->sq.qend = mlx5_get_send_wqe(qp, qp->sq.wqe_cnt);
|
||||||
*inlen = MLX5_ST_SZ_BYTES(create_qp_in) +
|
*inlen = MLX5_ST_SZ_BYTES(create_qp_in) +
|
||||||
MLX5_FLD_SZ_BYTES(create_qp_in, pas[0]) * qp->buf.npages;
|
MLX5_FLD_SZ_BYTES(create_qp_in, pas[0]) * qp->buf.npages;
|
||||||
*in = mlx5_vzalloc(*inlen);
|
*in = kvzalloc(*inlen, GFP_KERNEL);
|
||||||
if (!*in) {
|
if (!*in) {
|
||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
goto err_buf;
|
goto err_buf;
|
||||||
@ -1060,7 +1060,7 @@ static int create_raw_packet_qp_sq(struct mlx5_ib_dev *dev,
|
|||||||
return err;
|
return err;
|
||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(create_sq_in) + sizeof(u64) * ncont;
|
inlen = MLX5_ST_SZ_BYTES(create_sq_in) + sizeof(u64) * ncont;
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in) {
|
if (!in) {
|
||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
goto err_umem;
|
goto err_umem;
|
||||||
@ -1140,7 +1140,7 @@ static int create_raw_packet_qp_rq(struct mlx5_ib_dev *dev,
|
|||||||
u32 rq_pas_size = get_rq_pas_size(qpc);
|
u32 rq_pas_size = get_rq_pas_size(qpc);
|
||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(create_rq_in) + rq_pas_size;
|
inlen = MLX5_ST_SZ_BYTES(create_rq_in) + rq_pas_size;
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -1193,7 +1193,7 @@ static int create_raw_packet_qp_tir(struct mlx5_ib_dev *dev,
|
|||||||
int err;
|
int err;
|
||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(create_tir_in);
|
inlen = MLX5_ST_SZ_BYTES(create_tir_in);
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -1372,7 +1372,7 @@ static int create_rss_raw_qp_tir(struct mlx5_ib_dev *dev, struct mlx5_ib_qp *qp,
|
|||||||
}
|
}
|
||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(create_tir_in);
|
inlen = MLX5_ST_SZ_BYTES(create_tir_in);
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -1633,7 +1633,7 @@ static int create_qp_common(struct mlx5_ib_dev *dev, struct ib_pd *pd,
|
|||||||
if (err)
|
if (err)
|
||||||
return err;
|
return err;
|
||||||
} else {
|
} else {
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -2164,7 +2164,7 @@ static int modify_raw_packet_eth_prio(struct mlx5_core_dev *dev,
|
|||||||
int err;
|
int err;
|
||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(modify_tis_in);
|
inlen = MLX5_ST_SZ_BYTES(modify_tis_in);
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -2189,7 +2189,7 @@ static int modify_raw_packet_tx_affinity(struct mlx5_core_dev *dev,
|
|||||||
int err;
|
int err;
|
||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(modify_tis_in);
|
inlen = MLX5_ST_SZ_BYTES(modify_tis_in);
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -2434,7 +2434,7 @@ static int modify_raw_packet_qp_rq(struct mlx5_ib_dev *dev,
|
|||||||
int err;
|
int err;
|
||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(modify_rq_in);
|
inlen = MLX5_ST_SZ_BYTES(modify_rq_in);
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -2479,7 +2479,7 @@ static int modify_raw_packet_qp_sq(struct mlx5_core_dev *dev,
|
|||||||
int err;
|
int err;
|
||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(modify_sq_in);
|
inlen = MLX5_ST_SZ_BYTES(modify_sq_in);
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -4294,7 +4294,7 @@ static int query_raw_packet_qp_sq_state(struct mlx5_ib_dev *dev,
|
|||||||
int err;
|
int err;
|
||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(query_sq_out);
|
inlen = MLX5_ST_SZ_BYTES(query_sq_out);
|
||||||
out = mlx5_vzalloc(inlen);
|
out = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!out)
|
if (!out)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -4321,7 +4321,7 @@ static int query_raw_packet_qp_rq_state(struct mlx5_ib_dev *dev,
|
|||||||
int err;
|
int err;
|
||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(query_rq_out);
|
inlen = MLX5_ST_SZ_BYTES(query_rq_out);
|
||||||
out = mlx5_vzalloc(inlen);
|
out = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!out)
|
if (!out)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -4625,7 +4625,7 @@ static int create_rq(struct mlx5_ib_rwq *rwq, struct ib_pd *pd,
|
|||||||
dev = to_mdev(pd->device);
|
dev = to_mdev(pd->device);
|
||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(create_rq_in) + sizeof(u64) * rwq->rq_num_pas;
|
inlen = MLX5_ST_SZ_BYTES(create_rq_in) + sizeof(u64) * rwq->rq_num_pas;
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -4855,7 +4855,7 @@ struct ib_rwq_ind_table *mlx5_ib_create_rwq_ind_table(struct ib_device *device,
|
|||||||
return ERR_PTR(-ENOMEM);
|
return ERR_PTR(-ENOMEM);
|
||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(create_rqt_in) + sizeof(u32) * sz;
|
inlen = MLX5_ST_SZ_BYTES(create_rqt_in) + sizeof(u32) * sz;
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in) {
|
if (!in) {
|
||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
goto err;
|
goto err;
|
||||||
@ -4934,7 +4934,7 @@ int mlx5_ib_modify_wq(struct ib_wq *wq, struct ib_wq_attr *wq_attr,
|
|||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(modify_rq_in);
|
inlen = MLX5_ST_SZ_BYTES(modify_rq_in);
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
|
@ -127,7 +127,7 @@ static int create_srq_user(struct ib_pd *pd, struct mlx5_ib_srq *srq,
|
|||||||
goto err_umem;
|
goto err_umem;
|
||||||
}
|
}
|
||||||
|
|
||||||
in->pas = mlx5_vzalloc(sizeof(*in->pas) * ncont);
|
in->pas = kvzalloc(sizeof(*in->pas) * ncont, GFP_KERNEL);
|
||||||
if (!in->pas) {
|
if (!in->pas) {
|
||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
goto err_umem;
|
goto err_umem;
|
||||||
@ -189,7 +189,7 @@ static int create_srq_kernel(struct mlx5_ib_dev *dev, struct mlx5_ib_srq *srq,
|
|||||||
}
|
}
|
||||||
|
|
||||||
mlx5_ib_dbg(dev, "srq->buf.page_shift = %d\n", srq->buf.page_shift);
|
mlx5_ib_dbg(dev, "srq->buf.page_shift = %d\n", srq->buf.page_shift);
|
||||||
in->pas = mlx5_vzalloc(sizeof(*in->pas) * srq->buf.npages);
|
in->pas = kvzalloc(sizeof(*in->pas) * srq->buf.npages, GFP_KERNEL);
|
||||||
if (!in->pas) {
|
if (!in->pas) {
|
||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
goto err_buf;
|
goto err_buf;
|
||||||
|
@ -11,6 +11,16 @@ config MLX5_CORE
|
|||||||
Core driver for low level functionality of the ConnectX-4 and
|
Core driver for low level functionality of the ConnectX-4 and
|
||||||
Connect-IB cards by Mellanox Technologies.
|
Connect-IB cards by Mellanox Technologies.
|
||||||
|
|
||||||
|
config MLX5_FPGA
|
||||||
|
bool "Mellanox Technologies Innova support"
|
||||||
|
depends on MLX5_CORE
|
||||||
|
---help---
|
||||||
|
Build support for the Innova family of network cards by Mellanox
|
||||||
|
Technologies. Innova network cards are comprised of a ConnectX chip
|
||||||
|
and an FPGA chip on one board. If you select this option, the
|
||||||
|
mlx5_core driver will include the Innova FPGA core and allow building
|
||||||
|
sandbox-specific client drivers.
|
||||||
|
|
||||||
config MLX5_CORE_EN
|
config MLX5_CORE_EN
|
||||||
bool "Mellanox Technologies ConnectX-4 Ethernet support"
|
bool "Mellanox Technologies ConnectX-4 Ethernet support"
|
||||||
depends on NETDEVICES && ETHERNET && INET && PCI && MLX5_CORE
|
depends on NETDEVICES && ETHERNET && INET && PCI && MLX5_CORE
|
||||||
|
@ -1,10 +1,13 @@
|
|||||||
obj-$(CONFIG_MLX5_CORE) += mlx5_core.o
|
obj-$(CONFIG_MLX5_CORE) += mlx5_core.o
|
||||||
|
subdir-ccflags-y += -I$(src)
|
||||||
|
|
||||||
mlx5_core-y := main.o cmd.o debugfs.o fw.o eq.o uar.o pagealloc.o \
|
mlx5_core-y := main.o cmd.o debugfs.o fw.o eq.o uar.o pagealloc.o \
|
||||||
health.o mcg.o cq.o srq.o alloc.o qp.o port.o mr.o pd.o \
|
health.o mcg.o cq.o srq.o alloc.o qp.o port.o mr.o pd.o \
|
||||||
mad.o transobj.o vport.o sriov.o fs_cmd.o fs_core.o \
|
mad.o transobj.o vport.o sriov.o fs_cmd.o fs_core.o \
|
||||||
fs_counters.o rl.o lag.o dev.o
|
fs_counters.o rl.o lag.o dev.o
|
||||||
|
|
||||||
|
mlx5_core-$(CONFIG_MLX5_FPGA) += fpga/cmd.o fpga/core.o
|
||||||
|
|
||||||
mlx5_core-$(CONFIG_MLX5_CORE_EN) += wq.o eswitch.o eswitch_offloads.o \
|
mlx5_core-$(CONFIG_MLX5_CORE_EN) += wq.o eswitch.o eswitch_offloads.o \
|
||||||
en_main.o en_common.o en_fs.o en_ethtool.o en_tx.o \
|
en_main.o en_common.o en_fs.o en_ethtool.o en_tx.o \
|
||||||
en_rx.o en_rx_am.o en_txrx.o en_clock.o vxlan.o \
|
en_rx.o en_rx_am.o en_txrx.o en_clock.o vxlan.o \
|
||||||
|
@ -405,7 +405,7 @@ static u64 cq_read_field(struct mlx5_core_dev *dev, struct mlx5_core_cq *cq,
|
|||||||
u32 *out;
|
u32 *out;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
out = mlx5_vzalloc(outlen);
|
out = kvzalloc(outlen, GFP_KERNEL);
|
||||||
if (!out)
|
if (!out)
|
||||||
return param;
|
return param;
|
||||||
|
|
||||||
|
@ -180,9 +180,8 @@ static int arfs_add_default_rule(struct mlx5e_priv *priv,
|
|||||||
struct mlx5_flow_spec *spec;
|
struct mlx5_flow_spec *spec;
|
||||||
int err = 0;
|
int err = 0;
|
||||||
|
|
||||||
spec = mlx5_vzalloc(sizeof(*spec));
|
spec = kvzalloc(sizeof(*spec), GFP_KERNEL);
|
||||||
if (!spec) {
|
if (!spec) {
|
||||||
netdev_err(priv->netdev, "%s: alloc failed\n", __func__);
|
|
||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
@ -237,7 +236,7 @@ static int arfs_create_groups(struct mlx5e_flow_table *ft,
|
|||||||
|
|
||||||
ft->g = kcalloc(MLX5E_ARFS_NUM_GROUPS,
|
ft->g = kcalloc(MLX5E_ARFS_NUM_GROUPS,
|
||||||
sizeof(*ft->g), GFP_KERNEL);
|
sizeof(*ft->g), GFP_KERNEL);
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in || !ft->g) {
|
if (!in || !ft->g) {
|
||||||
kvfree(ft->g);
|
kvfree(ft->g);
|
||||||
kvfree(in);
|
kvfree(in);
|
||||||
@ -481,9 +480,8 @@ static struct mlx5_flow_handle *arfs_add_rule(struct mlx5e_priv *priv,
|
|||||||
struct mlx5_flow_table *ft;
|
struct mlx5_flow_table *ft;
|
||||||
int err = 0;
|
int err = 0;
|
||||||
|
|
||||||
spec = mlx5_vzalloc(sizeof(*spec));
|
spec = kvzalloc(sizeof(*spec), GFP_KERNEL);
|
||||||
if (!spec) {
|
if (!spec) {
|
||||||
netdev_err(priv->netdev, "%s: alloc failed\n", __func__);
|
|
||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
@ -65,7 +65,7 @@ static int mlx5e_create_mkey(struct mlx5_core_dev *mdev, u32 pdn,
|
|||||||
u32 *in;
|
u32 *in;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -147,7 +147,7 @@ int mlx5e_refresh_tirs(struct mlx5e_priv *priv, bool enable_uc_lb)
|
|||||||
|
|
||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(modify_tir_in);
|
inlen = MLX5_ST_SZ_BYTES(modify_tir_in);
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
|
@ -39,7 +39,7 @@ static void mlx5e_get_drvinfo(struct net_device *dev,
|
|||||||
struct mlx5_core_dev *mdev = priv->mdev;
|
struct mlx5_core_dev *mdev = priv->mdev;
|
||||||
|
|
||||||
strlcpy(drvinfo->driver, DRIVER_NAME, sizeof(drvinfo->driver));
|
strlcpy(drvinfo->driver, DRIVER_NAME, sizeof(drvinfo->driver));
|
||||||
strlcpy(drvinfo->version, DRIVER_VERSION " (" DRIVER_RELDATE ")",
|
strlcpy(drvinfo->version, DRIVER_VERSION,
|
||||||
sizeof(drvinfo->version));
|
sizeof(drvinfo->version));
|
||||||
snprintf(drvinfo->fw_version, sizeof(drvinfo->fw_version),
|
snprintf(drvinfo->fw_version, sizeof(drvinfo->fw_version),
|
||||||
"%d.%d.%04d (%.16s)",
|
"%d.%d.%04d (%.16s)",
|
||||||
@ -1048,7 +1048,7 @@ static int mlx5e_set_rxfh(struct net_device *dev, const u32 *indir,
|
|||||||
(hfunc != ETH_RSS_HASH_TOP))
|
(hfunc != ETH_RSS_HASH_TOP))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
|
@ -218,11 +218,9 @@ static int mlx5e_add_vlan_rule(struct mlx5e_priv *priv,
|
|||||||
struct mlx5_flow_spec *spec;
|
struct mlx5_flow_spec *spec;
|
||||||
int err = 0;
|
int err = 0;
|
||||||
|
|
||||||
spec = mlx5_vzalloc(sizeof(*spec));
|
spec = kvzalloc(sizeof(*spec), GFP_KERNEL);
|
||||||
if (!spec) {
|
if (!spec)
|
||||||
netdev_err(priv->netdev, "%s: alloc failed\n", __func__);
|
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
|
||||||
|
|
||||||
if (rule_type == MLX5E_VLAN_RULE_TYPE_MATCH_VID)
|
if (rule_type == MLX5E_VLAN_RULE_TYPE_MATCH_VID)
|
||||||
mlx5e_vport_context_update_vlans(priv);
|
mlx5e_vport_context_update_vlans(priv);
|
||||||
@ -660,11 +658,9 @@ mlx5e_generate_ttc_rule(struct mlx5e_priv *priv,
|
|||||||
struct mlx5_flow_spec *spec;
|
struct mlx5_flow_spec *spec;
|
||||||
int err = 0;
|
int err = 0;
|
||||||
|
|
||||||
spec = mlx5_vzalloc(sizeof(*spec));
|
spec = kvzalloc(sizeof(*spec), GFP_KERNEL);
|
||||||
if (!spec) {
|
if (!spec)
|
||||||
netdev_err(priv->netdev, "%s: alloc failed\n", __func__);
|
|
||||||
return ERR_PTR(-ENOMEM);
|
return ERR_PTR(-ENOMEM);
|
||||||
}
|
|
||||||
|
|
||||||
if (proto) {
|
if (proto) {
|
||||||
spec->match_criteria_enable = MLX5_MATCH_OUTER_HEADERS;
|
spec->match_criteria_enable = MLX5_MATCH_OUTER_HEADERS;
|
||||||
@ -742,7 +738,7 @@ static int mlx5e_create_ttc_table_groups(struct mlx5e_ttc_table *ttc)
|
|||||||
sizeof(*ft->g), GFP_KERNEL);
|
sizeof(*ft->g), GFP_KERNEL);
|
||||||
if (!ft->g)
|
if (!ft->g)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in) {
|
if (!in) {
|
||||||
kfree(ft->g);
|
kfree(ft->g);
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
@ -852,11 +848,9 @@ static int mlx5e_add_l2_flow_rule(struct mlx5e_priv *priv,
|
|||||||
u8 *mc_dmac;
|
u8 *mc_dmac;
|
||||||
u8 *mv_dmac;
|
u8 *mv_dmac;
|
||||||
|
|
||||||
spec = mlx5_vzalloc(sizeof(*spec));
|
spec = kvzalloc(sizeof(*spec), GFP_KERNEL);
|
||||||
if (!spec) {
|
if (!spec)
|
||||||
netdev_err(priv->netdev, "%s: alloc failed\n", __func__);
|
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
|
||||||
|
|
||||||
mc_dmac = MLX5_ADDR_OF(fte_match_param, spec->match_criteria,
|
mc_dmac = MLX5_ADDR_OF(fte_match_param, spec->match_criteria,
|
||||||
outer_headers.dmac_47_16);
|
outer_headers.dmac_47_16);
|
||||||
@ -916,7 +910,7 @@ static int mlx5e_create_l2_table_groups(struct mlx5e_l2_table *l2_table)
|
|||||||
ft->g = kcalloc(MLX5E_NUM_L2_GROUPS, sizeof(*ft->g), GFP_KERNEL);
|
ft->g = kcalloc(MLX5E_NUM_L2_GROUPS, sizeof(*ft->g), GFP_KERNEL);
|
||||||
if (!ft->g)
|
if (!ft->g)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in) {
|
if (!in) {
|
||||||
kfree(ft->g);
|
kfree(ft->g);
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
@ -1071,7 +1065,7 @@ static int mlx5e_create_vlan_table_groups(struct mlx5e_flow_table *ft)
|
|||||||
int inlen = MLX5_ST_SZ_BYTES(create_flow_group_in);
|
int inlen = MLX5_ST_SZ_BYTES(create_flow_group_in);
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
|
@ -296,7 +296,7 @@ add_ethtool_flow_rule(struct mlx5e_priv *priv,
|
|||||||
struct mlx5_flow_handle *rule;
|
struct mlx5_flow_handle *rule;
|
||||||
int err = 0;
|
int err = 0;
|
||||||
|
|
||||||
spec = mlx5_vzalloc(sizeof(*spec));
|
spec = kvzalloc(sizeof(*spec), GFP_KERNEL);
|
||||||
if (!spec)
|
if (!spec)
|
||||||
return ERR_PTR(-ENOMEM);
|
return ERR_PTR(-ENOMEM);
|
||||||
err = set_flow_attrs(spec->match_criteria, spec->match_value,
|
err = set_flow_attrs(spec->match_criteria, spec->match_value,
|
||||||
|
@ -252,9 +252,9 @@ static void mlx5e_update_pport_counters(struct mlx5e_priv *priv)
|
|||||||
void *out;
|
void *out;
|
||||||
u32 *in;
|
u32 *in;
|
||||||
|
|
||||||
in = mlx5_vzalloc(sz);
|
in = kvzalloc(sz, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
goto free_out;
|
return;
|
||||||
|
|
||||||
MLX5_SET(ppcnt_reg, in, local_port, 1);
|
MLX5_SET(ppcnt_reg, in, local_port, 1);
|
||||||
|
|
||||||
@ -288,7 +288,6 @@ static void mlx5e_update_pport_counters(struct mlx5e_priv *priv)
|
|||||||
MLX5_REG_PPCNT, 0, 0);
|
MLX5_REG_PPCNT, 0, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
free_out:
|
|
||||||
kvfree(in);
|
kvfree(in);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -314,7 +313,7 @@ static void mlx5e_update_pcie_counters(struct mlx5e_priv *priv)
|
|||||||
if (!MLX5_CAP_MCAM_FEATURE(mdev, pcie_performance_group))
|
if (!MLX5_CAP_MCAM_FEATURE(mdev, pcie_performance_group))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
in = mlx5_vzalloc(sz);
|
in = kvzalloc(sz, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
@ -503,7 +502,7 @@ static int mlx5e_create_umr_mkey(struct mlx5_core_dev *mdev,
|
|||||||
if (!MLX5E_VALID_NUM_MTTS(npages))
|
if (!MLX5E_VALID_NUM_MTTS(npages))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -711,7 +710,7 @@ static int mlx5e_create_rq(struct mlx5e_rq *rq,
|
|||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(create_rq_in) +
|
inlen = MLX5_ST_SZ_BYTES(create_rq_in) +
|
||||||
sizeof(u64) * rq->wq_ctrl.buf.npages;
|
sizeof(u64) * rq->wq_ctrl.buf.npages;
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -748,7 +747,7 @@ static int mlx5e_modify_rq_state(struct mlx5e_rq *rq, int curr_state,
|
|||||||
int err;
|
int err;
|
||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(modify_rq_in);
|
inlen = MLX5_ST_SZ_BYTES(modify_rq_in);
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -776,7 +775,7 @@ static int mlx5e_modify_rq_scatter_fcs(struct mlx5e_rq *rq, bool enable)
|
|||||||
int err;
|
int err;
|
||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(modify_rq_in);
|
inlen = MLX5_ST_SZ_BYTES(modify_rq_in);
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -805,7 +804,7 @@ static int mlx5e_modify_rq_vsd(struct mlx5e_rq *rq, bool vsd)
|
|||||||
int err;
|
int err;
|
||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(modify_rq_in);
|
inlen = MLX5_ST_SZ_BYTES(modify_rq_in);
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -1134,7 +1133,7 @@ static int mlx5e_create_sq(struct mlx5_core_dev *mdev,
|
|||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(create_sq_in) +
|
inlen = MLX5_ST_SZ_BYTES(create_sq_in) +
|
||||||
sizeof(u64) * csp->wq_ctrl->buf.npages;
|
sizeof(u64) * csp->wq_ctrl->buf.npages;
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -1182,7 +1181,7 @@ static int mlx5e_modify_sq(struct mlx5_core_dev *mdev, u32 sqn,
|
|||||||
int err;
|
int err;
|
||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(modify_sq_in);
|
inlen = MLX5_ST_SZ_BYTES(modify_sq_in);
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -1496,7 +1495,7 @@ static int mlx5e_create_cq(struct mlx5e_cq *cq, struct mlx5e_cq_param *param)
|
|||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(create_cq_in) +
|
inlen = MLX5_ST_SZ_BYTES(create_cq_in) +
|
||||||
sizeof(u64) * cq->wq_ctrl.frag_buf.npages;
|
sizeof(u64) * cq->wq_ctrl.frag_buf.npages;
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -2091,7 +2090,7 @@ mlx5e_create_rqt(struct mlx5e_priv *priv, int sz, struct mlx5e_rqt *rqt)
|
|||||||
int i;
|
int i;
|
||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(create_rqt_in) + sizeof(u32) * sz;
|
inlen = MLX5_ST_SZ_BYTES(create_rqt_in) + sizeof(u32) * sz;
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -2210,7 +2209,7 @@ int mlx5e_redirect_rqt(struct mlx5e_priv *priv, u32 rqtn, int sz,
|
|||||||
int err;
|
int err;
|
||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(modify_rqt_in) + sizeof(u32) * sz;
|
inlen = MLX5_ST_SZ_BYTES(modify_rqt_in) + sizeof(u32) * sz;
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -2433,7 +2432,7 @@ static int mlx5e_modify_tirs_lro(struct mlx5e_priv *priv)
|
|||||||
int ix;
|
int ix;
|
||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(modify_tir_in);
|
inlen = MLX5_ST_SZ_BYTES(modify_tir_in);
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -2850,7 +2849,7 @@ int mlx5e_create_indirect_tirs(struct mlx5e_priv *priv)
|
|||||||
int tt;
|
int tt;
|
||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(create_tir_in);
|
inlen = MLX5_ST_SZ_BYTES(create_tir_in);
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -2889,7 +2888,7 @@ int mlx5e_create_direct_tirs(struct mlx5e_priv *priv)
|
|||||||
int ix;
|
int ix;
|
||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(create_tir_in);
|
inlen = MLX5_ST_SZ_BYTES(create_tir_in);
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
|
@ -1738,7 +1738,7 @@ int mlx5e_configure_flower(struct mlx5e_priv *priv, __be16 protocol,
|
|||||||
}
|
}
|
||||||
|
|
||||||
flow = kzalloc(sizeof(*flow) + attr_size, GFP_KERNEL);
|
flow = kzalloc(sizeof(*flow) + attr_size, GFP_KERNEL);
|
||||||
parse_attr = mlx5_vzalloc(sizeof(*parse_attr));
|
parse_attr = kvzalloc(sizeof(*parse_attr), GFP_KERNEL);
|
||||||
if (!parse_attr || !flow) {
|
if (!parse_attr || !flow) {
|
||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
goto err_free;
|
goto err_free;
|
||||||
|
@ -35,6 +35,7 @@
|
|||||||
#include <linux/mlx5/driver.h>
|
#include <linux/mlx5/driver.h>
|
||||||
#include <linux/mlx5/cmd.h>
|
#include <linux/mlx5/cmd.h>
|
||||||
#include "mlx5_core.h"
|
#include "mlx5_core.h"
|
||||||
|
#include "fpga/core.h"
|
||||||
#ifdef CONFIG_MLX5_CORE_EN
|
#ifdef CONFIG_MLX5_CORE_EN
|
||||||
#include "eswitch.h"
|
#include "eswitch.h"
|
||||||
#endif
|
#endif
|
||||||
@ -156,6 +157,8 @@ static const char *eqe_type_str(u8 type)
|
|||||||
return "MLX5_EVENT_TYPE_PAGE_FAULT";
|
return "MLX5_EVENT_TYPE_PAGE_FAULT";
|
||||||
case MLX5_EVENT_TYPE_PPS_EVENT:
|
case MLX5_EVENT_TYPE_PPS_EVENT:
|
||||||
return "MLX5_EVENT_TYPE_PPS_EVENT";
|
return "MLX5_EVENT_TYPE_PPS_EVENT";
|
||||||
|
case MLX5_EVENT_TYPE_FPGA_ERROR:
|
||||||
|
return "MLX5_EVENT_TYPE_FPGA_ERROR";
|
||||||
default:
|
default:
|
||||||
return "Unrecognized event";
|
return "Unrecognized event";
|
||||||
}
|
}
|
||||||
@ -476,6 +479,11 @@ static irqreturn_t mlx5_eq_int(int irq, void *eq_ptr)
|
|||||||
if (dev->event)
|
if (dev->event)
|
||||||
dev->event(dev, MLX5_DEV_EVENT_PPS, (unsigned long)eqe);
|
dev->event(dev, MLX5_DEV_EVENT_PPS, (unsigned long)eqe);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case MLX5_EVENT_TYPE_FPGA_ERROR:
|
||||||
|
mlx5_fpga_event(dev, eqe->type, &eqe->data.raw);
|
||||||
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
mlx5_core_warn(dev, "Unhandled event 0x%x on EQ 0x%x\n",
|
mlx5_core_warn(dev, "Unhandled event 0x%x on EQ 0x%x\n",
|
||||||
eqe->type, eq->eqn);
|
eqe->type, eq->eqn);
|
||||||
@ -548,7 +556,7 @@ int mlx5_create_map_eq(struct mlx5_core_dev *dev, struct mlx5_eq *eq, u8 vecidx,
|
|||||||
inlen = MLX5_ST_SZ_BYTES(create_eq_in) +
|
inlen = MLX5_ST_SZ_BYTES(create_eq_in) +
|
||||||
MLX5_FLD_SZ_BYTES(create_eq_in, pas[0]) * eq->buf.npages;
|
MLX5_FLD_SZ_BYTES(create_eq_in, pas[0]) * eq->buf.npages;
|
||||||
|
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in) {
|
if (!in) {
|
||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
goto err_buf;
|
goto err_buf;
|
||||||
@ -693,6 +701,9 @@ int mlx5_start_eqs(struct mlx5_core_dev *dev)
|
|||||||
if (MLX5_CAP_GEN(dev, pps))
|
if (MLX5_CAP_GEN(dev, pps))
|
||||||
async_event_mask |= (1ull << MLX5_EVENT_TYPE_PPS_EVENT);
|
async_event_mask |= (1ull << MLX5_EVENT_TYPE_PPS_EVENT);
|
||||||
|
|
||||||
|
if (MLX5_CAP_GEN(dev, fpga))
|
||||||
|
async_event_mask |= (1ull << MLX5_EVENT_TYPE_FPGA_ERROR);
|
||||||
|
|
||||||
err = mlx5_create_map_eq(dev, &table->cmd_eq, MLX5_EQ_VEC_CMD,
|
err = mlx5_create_map_eq(dev, &table->cmd_eq, MLX5_EQ_VEC_CMD,
|
||||||
MLX5_NUM_CMD_EQE, 1ull << MLX5_EVENT_TYPE_CMD,
|
MLX5_NUM_CMD_EQE, 1ull << MLX5_EVENT_TYPE_CMD,
|
||||||
"mlx5_cmd_eq", MLX5_EQ_TYPE_ASYNC);
|
"mlx5_cmd_eq", MLX5_EQ_TYPE_ASYNC);
|
||||||
|
@ -248,11 +248,10 @@ __esw_fdb_set_vport_rule(struct mlx5_eswitch *esw, u32 vport, bool rx_rule,
|
|||||||
if (rx_rule)
|
if (rx_rule)
|
||||||
match_header |= MLX5_MATCH_MISC_PARAMETERS;
|
match_header |= MLX5_MATCH_MISC_PARAMETERS;
|
||||||
|
|
||||||
spec = mlx5_vzalloc(sizeof(*spec));
|
spec = kvzalloc(sizeof(*spec), GFP_KERNEL);
|
||||||
if (!spec) {
|
if (!spec)
|
||||||
esw_warn(esw->dev, "FDB: Failed to alloc match parameters\n");
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
|
||||||
dmac_v = MLX5_ADDR_OF(fte_match_param, spec->match_value,
|
dmac_v = MLX5_ADDR_OF(fte_match_param, spec->match_value,
|
||||||
outer_headers.dmac_47_16);
|
outer_headers.dmac_47_16);
|
||||||
dmac_c = MLX5_ADDR_OF(fte_match_param, spec->match_criteria,
|
dmac_c = MLX5_ADDR_OF(fte_match_param, spec->match_criteria,
|
||||||
@ -350,10 +349,9 @@ static int esw_create_legacy_fdb_table(struct mlx5_eswitch *esw, int nvports)
|
|||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
}
|
}
|
||||||
|
|
||||||
flow_group_in = mlx5_vzalloc(inlen);
|
flow_group_in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!flow_group_in)
|
if (!flow_group_in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
memset(flow_group_in, 0, inlen);
|
|
||||||
|
|
||||||
table_size = BIT(MLX5_CAP_ESW_FLOWTABLE_FDB(dev, log_max_ft_size));
|
table_size = BIT(MLX5_CAP_ESW_FLOWTABLE_FDB(dev, log_max_ft_size));
|
||||||
|
|
||||||
@ -961,7 +959,7 @@ static int esw_vport_enable_egress_acl(struct mlx5_eswitch *esw,
|
|||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
}
|
}
|
||||||
|
|
||||||
flow_group_in = mlx5_vzalloc(inlen);
|
flow_group_in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!flow_group_in)
|
if (!flow_group_in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -1078,7 +1076,7 @@ static int esw_vport_enable_ingress_acl(struct mlx5_eswitch *esw,
|
|||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
}
|
}
|
||||||
|
|
||||||
flow_group_in = mlx5_vzalloc(inlen);
|
flow_group_in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!flow_group_in)
|
if (!flow_group_in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -1241,11 +1239,9 @@ static int esw_vport_ingress_config(struct mlx5_eswitch *esw,
|
|||||||
"vport[%d] configure ingress rules, vlan(%d) qos(%d)\n",
|
"vport[%d] configure ingress rules, vlan(%d) qos(%d)\n",
|
||||||
vport->vport, vport->info.vlan, vport->info.qos);
|
vport->vport, vport->info.vlan, vport->info.qos);
|
||||||
|
|
||||||
spec = mlx5_vzalloc(sizeof(*spec));
|
spec = kvzalloc(sizeof(*spec), GFP_KERNEL);
|
||||||
if (!spec) {
|
if (!spec) {
|
||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
esw_warn(esw->dev, "vport[%d] configure ingress rules failed, err(%d)\n",
|
|
||||||
vport->vport, err);
|
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1322,11 +1318,9 @@ static int esw_vport_egress_config(struct mlx5_eswitch *esw,
|
|||||||
"vport[%d] configure egress rules, vlan(%d) qos(%d)\n",
|
"vport[%d] configure egress rules, vlan(%d) qos(%d)\n",
|
||||||
vport->vport, vport->info.vlan, vport->info.qos);
|
vport->vport, vport->info.vlan, vport->info.qos);
|
||||||
|
|
||||||
spec = mlx5_vzalloc(sizeof(*spec));
|
spec = kvzalloc(sizeof(*spec), GFP_KERNEL);
|
||||||
if (!spec) {
|
if (!spec) {
|
||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
esw_warn(esw->dev, "vport[%d] configure egress rules failed, err(%d)\n",
|
|
||||||
vport->vport, err);
|
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2158,7 +2152,7 @@ int mlx5_eswitch_get_vport_stats(struct mlx5_eswitch *esw,
|
|||||||
if (!LEGAL_VPORT(esw, vport))
|
if (!LEGAL_VPORT(esw, vport))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
out = mlx5_vzalloc(outlen);
|
out = kvzalloc(outlen, GFP_KERNEL);
|
||||||
if (!out)
|
if (!out)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
|
@ -311,9 +311,8 @@ mlx5_eswitch_add_send_to_vport_rule(struct mlx5_eswitch *esw, int vport, u32 sqn
|
|||||||
struct mlx5_flow_spec *spec;
|
struct mlx5_flow_spec *spec;
|
||||||
void *misc;
|
void *misc;
|
||||||
|
|
||||||
spec = mlx5_vzalloc(sizeof(*spec));
|
spec = kvzalloc(sizeof(*spec), GFP_KERNEL);
|
||||||
if (!spec) {
|
if (!spec) {
|
||||||
esw_warn(esw->dev, "FDB: Failed to alloc match parameters\n");
|
|
||||||
flow_rule = ERR_PTR(-ENOMEM);
|
flow_rule = ERR_PTR(-ENOMEM);
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
@ -401,9 +400,8 @@ static int esw_add_fdb_miss_rule(struct mlx5_eswitch *esw)
|
|||||||
struct mlx5_flow_spec *spec;
|
struct mlx5_flow_spec *spec;
|
||||||
int err = 0;
|
int err = 0;
|
||||||
|
|
||||||
spec = mlx5_vzalloc(sizeof(*spec));
|
spec = kvzalloc(sizeof(*spec), GFP_KERNEL);
|
||||||
if (!spec) {
|
if (!spec) {
|
||||||
esw_warn(esw->dev, "FDB: Failed to alloc match parameters\n");
|
|
||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
@ -488,7 +486,7 @@ static int esw_create_offloads_fdb_tables(struct mlx5_eswitch *esw, int nvports)
|
|||||||
u32 *flow_group_in;
|
u32 *flow_group_in;
|
||||||
|
|
||||||
esw_debug(esw->dev, "Create offloads FDB Tables\n");
|
esw_debug(esw->dev, "Create offloads FDB Tables\n");
|
||||||
flow_group_in = mlx5_vzalloc(inlen);
|
flow_group_in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!flow_group_in)
|
if (!flow_group_in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -631,7 +629,7 @@ static int esw_create_vport_rx_group(struct mlx5_eswitch *esw)
|
|||||||
int err = 0;
|
int err = 0;
|
||||||
int nvports = priv->sriov.num_vfs + 2;
|
int nvports = priv->sriov.num_vfs + 2;
|
||||||
|
|
||||||
flow_group_in = mlx5_vzalloc(inlen);
|
flow_group_in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!flow_group_in)
|
if (!flow_group_in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -675,9 +673,8 @@ mlx5_eswitch_create_vport_rx_rule(struct mlx5_eswitch *esw, int vport, u32 tirn)
|
|||||||
struct mlx5_flow_spec *spec;
|
struct mlx5_flow_spec *spec;
|
||||||
void *misc;
|
void *misc;
|
||||||
|
|
||||||
spec = mlx5_vzalloc(sizeof(*spec));
|
spec = kvzalloc(sizeof(*spec), GFP_KERNEL);
|
||||||
if (!spec) {
|
if (!spec) {
|
||||||
esw_warn(esw->dev, "Failed to alloc match parameters\n");
|
|
||||||
flow_rule = ERR_PTR(-ENOMEM);
|
flow_rule = ERR_PTR(-ENOMEM);
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
64
drivers/net/ethernet/mellanox/mlx5/core/fpga/cmd.c
Normal file
64
drivers/net/ethernet/mellanox/mlx5/core/fpga/cmd.c
Normal file
@ -0,0 +1,64 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2017, Mellanox Technologies. All rights reserved.
|
||||||
|
*
|
||||||
|
* This software is available to you under a choice of one of two
|
||||||
|
* licenses. You may choose to be licensed under the terms of the GNU
|
||||||
|
* General Public License (GPL) Version 2, available from the file
|
||||||
|
* COPYING in the main directory of this source tree, or the
|
||||||
|
* OpenIB.org BSD license below:
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or
|
||||||
|
* without modification, are permitted provided that the following
|
||||||
|
* conditions are met:
|
||||||
|
*
|
||||||
|
* - Redistributions of source code must retain the above
|
||||||
|
* copyright notice, this list of conditions and the following
|
||||||
|
* disclaimer.
|
||||||
|
*
|
||||||
|
* - Redistributions in binary form must reproduce the above
|
||||||
|
* copyright notice, this list of conditions and the following
|
||||||
|
* disclaimer in the documentation and/or other materials
|
||||||
|
* provided with the distribution.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
|
||||||
|
* BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
|
||||||
|
* ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
||||||
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
* SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <linux/etherdevice.h>
|
||||||
|
#include <linux/mlx5/cmd.h>
|
||||||
|
#include <linux/mlx5/driver.h>
|
||||||
|
|
||||||
|
#include "mlx5_core.h"
|
||||||
|
#include "fpga/cmd.h"
|
||||||
|
|
||||||
|
int mlx5_fpga_caps(struct mlx5_core_dev *dev, u32 *caps)
|
||||||
|
{
|
||||||
|
u32 in[MLX5_ST_SZ_DW(fpga_cap)] = {0};
|
||||||
|
|
||||||
|
return mlx5_core_access_reg(dev, in, sizeof(in), caps,
|
||||||
|
MLX5_ST_SZ_BYTES(fpga_cap),
|
||||||
|
MLX5_REG_FPGA_CAP, 0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
int mlx5_fpga_query(struct mlx5_core_dev *dev, struct mlx5_fpga_query *query)
|
||||||
|
{
|
||||||
|
u32 in[MLX5_ST_SZ_DW(fpga_ctrl)] = {0};
|
||||||
|
u32 out[MLX5_ST_SZ_DW(fpga_ctrl)];
|
||||||
|
int err;
|
||||||
|
|
||||||
|
err = mlx5_core_access_reg(dev, in, sizeof(in), out, sizeof(out),
|
||||||
|
MLX5_REG_FPGA_CTRL, 0, false);
|
||||||
|
if (err)
|
||||||
|
return err;
|
||||||
|
|
||||||
|
query->status = MLX5_GET(fpga_ctrl, out, status);
|
||||||
|
query->admin_image = MLX5_GET(fpga_ctrl, out, flash_select_admin);
|
||||||
|
query->oper_image = MLX5_GET(fpga_ctrl, out, flash_select_oper);
|
||||||
|
return 0;
|
||||||
|
}
|
59
drivers/net/ethernet/mellanox/mlx5/core/fpga/cmd.h
Normal file
59
drivers/net/ethernet/mellanox/mlx5/core/fpga/cmd.h
Normal file
@ -0,0 +1,59 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2017, Mellanox Technologies, Ltd. All rights reserved.
|
||||||
|
*
|
||||||
|
* This software is available to you under a choice of one of two
|
||||||
|
* licenses. You may choose to be licensed under the terms of the GNU
|
||||||
|
* General Public License (GPL) Version 2, available from the file
|
||||||
|
* COPYING in the main directory of this source tree, or the
|
||||||
|
* OpenIB.org BSD license below:
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or
|
||||||
|
* without modification, are permitted provided that the following
|
||||||
|
* conditions are met:
|
||||||
|
*
|
||||||
|
* - Redistributions of source code must retain the above
|
||||||
|
* copyright notice, this list of conditions and the following
|
||||||
|
* disclaimer.
|
||||||
|
*
|
||||||
|
* - Redistributions in binary form must reproduce the above
|
||||||
|
* copyright notice, this list of conditions and the following
|
||||||
|
* disclaimer in the documentation and/or other materials
|
||||||
|
* provided with the distribution.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
|
||||||
|
* BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
|
||||||
|
* ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
||||||
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
* SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __MLX5_FPGA_H__
|
||||||
|
#define __MLX5_FPGA_H__
|
||||||
|
|
||||||
|
#include <linux/mlx5/driver.h>
|
||||||
|
|
||||||
|
enum mlx5_fpga_image {
|
||||||
|
MLX5_FPGA_IMAGE_USER = 0,
|
||||||
|
MLX5_FPGA_IMAGE_FACTORY,
|
||||||
|
};
|
||||||
|
|
||||||
|
enum mlx5_fpga_status {
|
||||||
|
MLX5_FPGA_STATUS_SUCCESS = 0,
|
||||||
|
MLX5_FPGA_STATUS_FAILURE = 1,
|
||||||
|
MLX5_FPGA_STATUS_IN_PROGRESS = 2,
|
||||||
|
MLX5_FPGA_STATUS_NONE = 0xFFFF,
|
||||||
|
};
|
||||||
|
|
||||||
|
struct mlx5_fpga_query {
|
||||||
|
enum mlx5_fpga_image admin_image;
|
||||||
|
enum mlx5_fpga_image oper_image;
|
||||||
|
enum mlx5_fpga_status status;
|
||||||
|
};
|
||||||
|
|
||||||
|
int mlx5_fpga_caps(struct mlx5_core_dev *dev, u32 *caps);
|
||||||
|
int mlx5_fpga_query(struct mlx5_core_dev *dev, struct mlx5_fpga_query *query);
|
||||||
|
|
||||||
|
#endif /* __MLX5_FPGA_H__ */
|
202
drivers/net/ethernet/mellanox/mlx5/core/fpga/core.c
Normal file
202
drivers/net/ethernet/mellanox/mlx5/core/fpga/core.c
Normal file
@ -0,0 +1,202 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2017, Mellanox Technologies. All rights reserved.
|
||||||
|
*
|
||||||
|
* This software is available to you under a choice of one of two
|
||||||
|
* licenses. You may choose to be licensed under the terms of the GNU
|
||||||
|
* General Public License (GPL) Version 2, available from the file
|
||||||
|
* COPYING in the main directory of this source tree, or the
|
||||||
|
* OpenIB.org BSD license below:
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or
|
||||||
|
* without modification, are permitted provided that the following
|
||||||
|
* conditions are met:
|
||||||
|
*
|
||||||
|
* - Redistributions of source code must retain the above
|
||||||
|
* copyright notice, this list of conditions and the following
|
||||||
|
* disclaimer.
|
||||||
|
*
|
||||||
|
* - Redistributions in binary form must reproduce the above
|
||||||
|
* copyright notice, this list of conditions and the following
|
||||||
|
* disclaimer in the documentation and/or other materials
|
||||||
|
* provided with the distribution.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
|
||||||
|
* BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
|
||||||
|
* ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
||||||
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
* SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <linux/module.h>
|
||||||
|
#include <linux/etherdevice.h>
|
||||||
|
#include <linux/mlx5/driver.h>
|
||||||
|
|
||||||
|
#include "mlx5_core.h"
|
||||||
|
#include "fpga/core.h"
|
||||||
|
|
||||||
|
static const char *const mlx5_fpga_error_strings[] = {
|
||||||
|
"Null Syndrome",
|
||||||
|
"Corrupted DDR",
|
||||||
|
"Flash Timeout",
|
||||||
|
"Internal Link Error",
|
||||||
|
"Watchdog HW Failure",
|
||||||
|
"I2C Failure",
|
||||||
|
"Image Changed",
|
||||||
|
"Temperature Critical",
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct mlx5_fpga_device *mlx5_fpga_device_alloc(void)
|
||||||
|
{
|
||||||
|
struct mlx5_fpga_device *fdev = NULL;
|
||||||
|
|
||||||
|
fdev = kzalloc(sizeof(*fdev), GFP_KERNEL);
|
||||||
|
if (!fdev)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
spin_lock_init(&fdev->state_lock);
|
||||||
|
fdev->state = MLX5_FPGA_STATUS_NONE;
|
||||||
|
return fdev;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const char *mlx5_fpga_image_name(enum mlx5_fpga_image image)
|
||||||
|
{
|
||||||
|
switch (image) {
|
||||||
|
case MLX5_FPGA_IMAGE_USER:
|
||||||
|
return "user";
|
||||||
|
case MLX5_FPGA_IMAGE_FACTORY:
|
||||||
|
return "factory";
|
||||||
|
default:
|
||||||
|
return "unknown";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int mlx5_fpga_device_load_check(struct mlx5_fpga_device *fdev)
|
||||||
|
{
|
||||||
|
struct mlx5_fpga_query query;
|
||||||
|
int err;
|
||||||
|
|
||||||
|
err = mlx5_fpga_query(fdev->mdev, &query);
|
||||||
|
if (err) {
|
||||||
|
mlx5_fpga_err(fdev, "Failed to query status: %d\n", err);
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
fdev->last_admin_image = query.admin_image;
|
||||||
|
fdev->last_oper_image = query.oper_image;
|
||||||
|
|
||||||
|
mlx5_fpga_dbg(fdev, "Status %u; Admin image %u; Oper image %u\n",
|
||||||
|
query.status, query.admin_image, query.oper_image);
|
||||||
|
|
||||||
|
if (query.status != MLX5_FPGA_STATUS_SUCCESS) {
|
||||||
|
mlx5_fpga_err(fdev, "%s image failed to load; status %u\n",
|
||||||
|
mlx5_fpga_image_name(fdev->last_oper_image),
|
||||||
|
query.status);
|
||||||
|
return -EIO;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int mlx5_fpga_device_start(struct mlx5_core_dev *mdev)
|
||||||
|
{
|
||||||
|
struct mlx5_fpga_device *fdev = mdev->fpga;
|
||||||
|
unsigned long flags;
|
||||||
|
int err;
|
||||||
|
|
||||||
|
if (!fdev)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
err = mlx5_fpga_device_load_check(fdev);
|
||||||
|
if (err)
|
||||||
|
goto out;
|
||||||
|
|
||||||
|
err = mlx5_fpga_caps(fdev->mdev,
|
||||||
|
fdev->mdev->caps.hca_cur[MLX5_CAP_FPGA]);
|
||||||
|
if (err)
|
||||||
|
goto out;
|
||||||
|
|
||||||
|
mlx5_fpga_info(fdev, "device %u; %s image, version %u\n",
|
||||||
|
MLX5_CAP_FPGA(fdev->mdev, fpga_device),
|
||||||
|
mlx5_fpga_image_name(fdev->last_oper_image),
|
||||||
|
MLX5_CAP_FPGA(fdev->mdev, image_version));
|
||||||
|
|
||||||
|
out:
|
||||||
|
spin_lock_irqsave(&fdev->state_lock, flags);
|
||||||
|
fdev->state = err ? MLX5_FPGA_STATUS_FAILURE : MLX5_FPGA_STATUS_SUCCESS;
|
||||||
|
spin_unlock_irqrestore(&fdev->state_lock, flags);
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
int mlx5_fpga_device_init(struct mlx5_core_dev *mdev)
|
||||||
|
{
|
||||||
|
struct mlx5_fpga_device *fdev = NULL;
|
||||||
|
|
||||||
|
if (!MLX5_CAP_GEN(mdev, fpga)) {
|
||||||
|
mlx5_core_dbg(mdev, "FPGA capability not present\n");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
mlx5_core_dbg(mdev, "Initializing FPGA\n");
|
||||||
|
|
||||||
|
fdev = mlx5_fpga_device_alloc();
|
||||||
|
if (!fdev)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
fdev->mdev = mdev;
|
||||||
|
mdev->fpga = fdev;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void mlx5_fpga_device_cleanup(struct mlx5_core_dev *mdev)
|
||||||
|
{
|
||||||
|
kfree(mdev->fpga);
|
||||||
|
mdev->fpga = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const char *mlx5_fpga_syndrome_to_string(u8 syndrome)
|
||||||
|
{
|
||||||
|
if (syndrome < ARRAY_SIZE(mlx5_fpga_error_strings))
|
||||||
|
return mlx5_fpga_error_strings[syndrome];
|
||||||
|
return "Unknown";
|
||||||
|
}
|
||||||
|
|
||||||
|
void mlx5_fpga_event(struct mlx5_core_dev *mdev, u8 event, void *data)
|
||||||
|
{
|
||||||
|
struct mlx5_fpga_device *fdev = mdev->fpga;
|
||||||
|
const char *event_name;
|
||||||
|
bool teardown = false;
|
||||||
|
unsigned long flags;
|
||||||
|
u8 syndrome;
|
||||||
|
|
||||||
|
if (event != MLX5_EVENT_TYPE_FPGA_ERROR) {
|
||||||
|
mlx5_fpga_warn_ratelimited(fdev, "Unexpected event %u\n",
|
||||||
|
event);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
syndrome = MLX5_GET(fpga_error_event, data, syndrome);
|
||||||
|
event_name = mlx5_fpga_syndrome_to_string(syndrome);
|
||||||
|
|
||||||
|
spin_lock_irqsave(&fdev->state_lock, flags);
|
||||||
|
switch (fdev->state) {
|
||||||
|
case MLX5_FPGA_STATUS_SUCCESS:
|
||||||
|
mlx5_fpga_warn(fdev, "Error %u: %s\n", syndrome, event_name);
|
||||||
|
teardown = true;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
mlx5_fpga_warn_ratelimited(fdev, "Unexpected error event %u: %s\n",
|
||||||
|
syndrome, event_name);
|
||||||
|
}
|
||||||
|
spin_unlock_irqrestore(&fdev->state_lock, flags);
|
||||||
|
/* We tear-down the card's interfaces and functionality because
|
||||||
|
* the FPGA bump-on-the-wire is misbehaving and we lose ability
|
||||||
|
* to communicate with the network. User may still be able to
|
||||||
|
* recover by re-programming or debugging the FPGA
|
||||||
|
*/
|
||||||
|
if (teardown)
|
||||||
|
mlx5_trigger_health_work(fdev->mdev);
|
||||||
|
}
|
99
drivers/net/ethernet/mellanox/mlx5/core/fpga/core.h
Normal file
99
drivers/net/ethernet/mellanox/mlx5/core/fpga/core.h
Normal file
@ -0,0 +1,99 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2017, Mellanox Technologies, Ltd. All rights reserved.
|
||||||
|
*
|
||||||
|
* This software is available to you under a choice of one of two
|
||||||
|
* licenses. You may choose to be licensed under the terms of the GNU
|
||||||
|
* General Public License (GPL) Version 2, available from the file
|
||||||
|
* COPYING in the main directory of this source tree, or the
|
||||||
|
* OpenIB.org BSD license below:
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or
|
||||||
|
* without modification, are permitted provided that the following
|
||||||
|
* conditions are met:
|
||||||
|
*
|
||||||
|
* - Redistributions of source code must retain the above
|
||||||
|
* copyright notice, this list of conditions and the following
|
||||||
|
* disclaimer.
|
||||||
|
*
|
||||||
|
* - Redistributions in binary form must reproduce the above
|
||||||
|
* copyright notice, this list of conditions and the following
|
||||||
|
* disclaimer in the documentation and/or other materials
|
||||||
|
* provided with the distribution.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
|
||||||
|
* BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
|
||||||
|
* ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
||||||
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
* SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __MLX5_FPGA_CORE_H__
|
||||||
|
#define __MLX5_FPGA_CORE_H__
|
||||||
|
|
||||||
|
#ifdef CONFIG_MLX5_FPGA
|
||||||
|
|
||||||
|
#include "fpga/cmd.h"
|
||||||
|
|
||||||
|
/* Represents an Innova device */
|
||||||
|
struct mlx5_fpga_device {
|
||||||
|
struct mlx5_core_dev *mdev;
|
||||||
|
spinlock_t state_lock; /* Protects state transitions */
|
||||||
|
enum mlx5_fpga_status state;
|
||||||
|
enum mlx5_fpga_image last_admin_image;
|
||||||
|
enum mlx5_fpga_image last_oper_image;
|
||||||
|
};
|
||||||
|
|
||||||
|
#define mlx5_fpga_dbg(__adev, format, ...) \
|
||||||
|
dev_dbg(&(__adev)->mdev->pdev->dev, "FPGA: %s:%d:(pid %d): " format, \
|
||||||
|
__func__, __LINE__, current->pid, ##__VA_ARGS__)
|
||||||
|
|
||||||
|
#define mlx5_fpga_err(__adev, format, ...) \
|
||||||
|
dev_err(&(__adev)->mdev->pdev->dev, "FPGA: %s:%d:(pid %d): " format, \
|
||||||
|
__func__, __LINE__, current->pid, ##__VA_ARGS__)
|
||||||
|
|
||||||
|
#define mlx5_fpga_warn(__adev, format, ...) \
|
||||||
|
dev_warn(&(__adev)->mdev->pdev->dev, "FPGA: %s:%d:(pid %d): " format, \
|
||||||
|
__func__, __LINE__, current->pid, ##__VA_ARGS__)
|
||||||
|
|
||||||
|
#define mlx5_fpga_warn_ratelimited(__adev, format, ...) \
|
||||||
|
dev_warn_ratelimited(&(__adev)->mdev->pdev->dev, "FPGA: %s:%d: " \
|
||||||
|
format, __func__, __LINE__, ##__VA_ARGS__)
|
||||||
|
|
||||||
|
#define mlx5_fpga_notice(__adev, format, ...) \
|
||||||
|
dev_notice(&(__adev)->mdev->pdev->dev, "FPGA: " format, ##__VA_ARGS__)
|
||||||
|
|
||||||
|
#define mlx5_fpga_info(__adev, format, ...) \
|
||||||
|
dev_info(&(__adev)->mdev->pdev->dev, "FPGA: " format, ##__VA_ARGS__)
|
||||||
|
|
||||||
|
int mlx5_fpga_device_init(struct mlx5_core_dev *mdev);
|
||||||
|
void mlx5_fpga_device_cleanup(struct mlx5_core_dev *mdev);
|
||||||
|
int mlx5_fpga_device_start(struct mlx5_core_dev *mdev);
|
||||||
|
void mlx5_fpga_event(struct mlx5_core_dev *mdev, u8 event, void *data);
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
static inline int mlx5_fpga_device_init(struct mlx5_core_dev *mdev)
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void mlx5_fpga_device_cleanup(struct mlx5_core_dev *mdev)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline int mlx5_fpga_device_start(struct mlx5_core_dev *mdev)
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void mlx5_fpga_event(struct mlx5_core_dev *mdev, u8 event,
|
||||||
|
void *data)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif /* __MLX5_FPGA_CORE_H__ */
|
@ -232,11 +232,9 @@ static int mlx5_cmd_set_fte(struct mlx5_core_dev *dev,
|
|||||||
u32 *in;
|
u32 *in;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in) {
|
if (!in)
|
||||||
mlx5_core_warn(dev, "failed to allocate inbox\n");
|
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
|
||||||
|
|
||||||
MLX5_SET(set_fte_in, in, opcode, MLX5_CMD_OP_SET_FLOW_TABLE_ENTRY);
|
MLX5_SET(set_fte_in, in, opcode, MLX5_CMD_OP_SET_FLOW_TABLE_ENTRY);
|
||||||
MLX5_SET(set_fte_in, in, op_mod, opmod);
|
MLX5_SET(set_fte_in, in, op_mod, opmod);
|
||||||
|
@ -376,11 +376,9 @@ static void del_rule(struct fs_node *node)
|
|||||||
int err;
|
int err;
|
||||||
bool update_fte = false;
|
bool update_fte = false;
|
||||||
|
|
||||||
match_value = mlx5_vzalloc(match_len);
|
match_value = kvzalloc(match_len, GFP_KERNEL);
|
||||||
if (!match_value) {
|
if (!match_value)
|
||||||
mlx5_core_warn(dev, "failed to allocate inbox\n");
|
|
||||||
return;
|
return;
|
||||||
}
|
|
||||||
|
|
||||||
fs_get_obj(rule, node);
|
fs_get_obj(rule, node);
|
||||||
fs_get_obj(fte, rule->node.parent);
|
fs_get_obj(fte, rule->node.parent);
|
||||||
@ -1157,7 +1155,7 @@ static struct mlx5_flow_group *create_autogroup(struct mlx5_flow_table *ft,
|
|||||||
if (!ft->autogroup.active)
|
if (!ft->autogroup.active)
|
||||||
return ERR_PTR(-ENOENT);
|
return ERR_PTR(-ENOENT);
|
||||||
|
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return ERR_PTR(-ENOMEM);
|
return ERR_PTR(-ENOMEM);
|
||||||
|
|
||||||
@ -1777,7 +1775,7 @@ static struct mlx5_flow_root_namespace *create_root_ns(struct mlx5_flow_steering
|
|||||||
struct mlx5_flow_namespace *ns;
|
struct mlx5_flow_namespace *ns;
|
||||||
|
|
||||||
/* Create the root namespace */
|
/* Create the root namespace */
|
||||||
root_ns = mlx5_vzalloc(sizeof(*root_ns));
|
root_ns = kvzalloc(sizeof(*root_ns), GFP_KERNEL);
|
||||||
if (!root_ns)
|
if (!root_ns)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
|
@ -185,6 +185,7 @@ static void health_care(struct work_struct *work)
|
|||||||
struct mlx5_core_health *health;
|
struct mlx5_core_health *health;
|
||||||
struct mlx5_core_dev *dev;
|
struct mlx5_core_dev *dev;
|
||||||
struct mlx5_priv *priv;
|
struct mlx5_priv *priv;
|
||||||
|
unsigned long flags;
|
||||||
|
|
||||||
health = container_of(work, struct mlx5_core_health, work);
|
health = container_of(work, struct mlx5_core_health, work);
|
||||||
priv = container_of(health, struct mlx5_priv, health);
|
priv = container_of(health, struct mlx5_priv, health);
|
||||||
@ -192,13 +193,13 @@ static void health_care(struct work_struct *work)
|
|||||||
mlx5_core_warn(dev, "handling bad device here\n");
|
mlx5_core_warn(dev, "handling bad device here\n");
|
||||||
mlx5_handle_bad_state(dev);
|
mlx5_handle_bad_state(dev);
|
||||||
|
|
||||||
spin_lock(&health->wq_lock);
|
spin_lock_irqsave(&health->wq_lock, flags);
|
||||||
if (!test_bit(MLX5_DROP_NEW_HEALTH_WORK, &health->flags))
|
if (!test_bit(MLX5_DROP_NEW_HEALTH_WORK, &health->flags))
|
||||||
schedule_delayed_work(&health->recover_work, recover_delay);
|
schedule_delayed_work(&health->recover_work, recover_delay);
|
||||||
else
|
else
|
||||||
dev_err(&dev->pdev->dev,
|
dev_err(&dev->pdev->dev,
|
||||||
"new health works are not permitted at this stage\n");
|
"new health works are not permitted at this stage\n");
|
||||||
spin_unlock(&health->wq_lock);
|
spin_unlock_irqrestore(&health->wq_lock, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
static const char *hsynd_str(u8 synd)
|
static const char *hsynd_str(u8 synd)
|
||||||
@ -269,6 +270,20 @@ static unsigned long get_next_poll_jiffies(void)
|
|||||||
return next;
|
return next;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void mlx5_trigger_health_work(struct mlx5_core_dev *dev)
|
||||||
|
{
|
||||||
|
struct mlx5_core_health *health = &dev->priv.health;
|
||||||
|
unsigned long flags;
|
||||||
|
|
||||||
|
spin_lock_irqsave(&health->wq_lock, flags);
|
||||||
|
if (!test_bit(MLX5_DROP_NEW_HEALTH_WORK, &health->flags))
|
||||||
|
queue_work(health->wq, &health->work);
|
||||||
|
else
|
||||||
|
dev_err(&dev->pdev->dev,
|
||||||
|
"new health works are not permitted at this stage\n");
|
||||||
|
spin_unlock_irqrestore(&health->wq_lock, flags);
|
||||||
|
}
|
||||||
|
|
||||||
static void poll_health(unsigned long data)
|
static void poll_health(unsigned long data)
|
||||||
{
|
{
|
||||||
struct mlx5_core_dev *dev = (struct mlx5_core_dev *)data;
|
struct mlx5_core_dev *dev = (struct mlx5_core_dev *)data;
|
||||||
@ -297,13 +312,7 @@ static void poll_health(unsigned long data)
|
|||||||
if (in_fatal(dev) && !health->sick) {
|
if (in_fatal(dev) && !health->sick) {
|
||||||
health->sick = true;
|
health->sick = true;
|
||||||
print_health_info(dev);
|
print_health_info(dev);
|
||||||
spin_lock(&health->wq_lock);
|
mlx5_trigger_health_work(dev);
|
||||||
if (!test_bit(MLX5_DROP_NEW_HEALTH_WORK, &health->flags))
|
|
||||||
queue_work(health->wq, &health->work);
|
|
||||||
else
|
|
||||||
dev_err(&dev->pdev->dev,
|
|
||||||
"new health works are not permitted at this stage\n");
|
|
||||||
spin_unlock(&health->wq_lock);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -333,10 +342,11 @@ void mlx5_stop_health_poll(struct mlx5_core_dev *dev)
|
|||||||
void mlx5_drain_health_wq(struct mlx5_core_dev *dev)
|
void mlx5_drain_health_wq(struct mlx5_core_dev *dev)
|
||||||
{
|
{
|
||||||
struct mlx5_core_health *health = &dev->priv.health;
|
struct mlx5_core_health *health = &dev->priv.health;
|
||||||
|
unsigned long flags;
|
||||||
|
|
||||||
spin_lock(&health->wq_lock);
|
spin_lock_irqsave(&health->wq_lock, flags);
|
||||||
set_bit(MLX5_DROP_NEW_HEALTH_WORK, &health->flags);
|
set_bit(MLX5_DROP_NEW_HEALTH_WORK, &health->flags);
|
||||||
spin_unlock(&health->wq_lock);
|
spin_unlock_irqrestore(&health->wq_lock, flags);
|
||||||
cancel_delayed_work_sync(&health->recover_work);
|
cancel_delayed_work_sync(&health->recover_work);
|
||||||
cancel_work_sync(&health->work);
|
cancel_work_sync(&health->work);
|
||||||
}
|
}
|
||||||
|
@ -102,7 +102,7 @@ static int mlx5i_create_underlay_qp(struct mlx5_core_dev *mdev, struct mlx5_core
|
|||||||
void *qpc;
|
void *qpc;
|
||||||
|
|
||||||
inlen = MLX5_ST_SZ_BYTES(create_qp_in);
|
inlen = MLX5_ST_SZ_BYTES(create_qp_in);
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
|
@ -56,6 +56,7 @@
|
|||||||
#ifdef CONFIG_MLX5_CORE_EN
|
#ifdef CONFIG_MLX5_CORE_EN
|
||||||
#include "eswitch.h"
|
#include "eswitch.h"
|
||||||
#endif
|
#endif
|
||||||
|
#include "fpga/core.h"
|
||||||
|
|
||||||
MODULE_AUTHOR("Eli Cohen <eli@mellanox.com>");
|
MODULE_AUTHOR("Eli Cohen <eli@mellanox.com>");
|
||||||
MODULE_DESCRIPTION("Mellanox Connect-IB, ConnectX-4 core driver");
|
MODULE_DESCRIPTION("Mellanox Connect-IB, ConnectX-4 core driver");
|
||||||
@ -1113,10 +1114,16 @@ static int mlx5_load_one(struct mlx5_core_dev *dev, struct mlx5_priv *priv,
|
|||||||
goto err_disable_msix;
|
goto err_disable_msix;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
err = mlx5_fpga_device_init(dev);
|
||||||
|
if (err) {
|
||||||
|
dev_err(&pdev->dev, "fpga device init failed %d\n", err);
|
||||||
|
goto err_put_uars;
|
||||||
|
}
|
||||||
|
|
||||||
err = mlx5_start_eqs(dev);
|
err = mlx5_start_eqs(dev);
|
||||||
if (err) {
|
if (err) {
|
||||||
dev_err(&pdev->dev, "Failed to start pages and async EQs\n");
|
dev_err(&pdev->dev, "Failed to start pages and async EQs\n");
|
||||||
goto err_put_uars;
|
goto err_fpga_init;
|
||||||
}
|
}
|
||||||
|
|
||||||
err = alloc_comp_eqs(dev);
|
err = alloc_comp_eqs(dev);
|
||||||
@ -1147,6 +1154,12 @@ static int mlx5_load_one(struct mlx5_core_dev *dev, struct mlx5_priv *priv,
|
|||||||
goto err_sriov;
|
goto err_sriov;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
err = mlx5_fpga_device_start(dev);
|
||||||
|
if (err) {
|
||||||
|
dev_err(&pdev->dev, "fpga device start failed %d\n", err);
|
||||||
|
goto err_reg_dev;
|
||||||
|
}
|
||||||
|
|
||||||
if (mlx5_device_registered(dev)) {
|
if (mlx5_device_registered(dev)) {
|
||||||
mlx5_attach_device(dev);
|
mlx5_attach_device(dev);
|
||||||
} else {
|
} else {
|
||||||
@ -1182,6 +1195,9 @@ err_affinity_hints:
|
|||||||
err_stop_eqs:
|
err_stop_eqs:
|
||||||
mlx5_stop_eqs(dev);
|
mlx5_stop_eqs(dev);
|
||||||
|
|
||||||
|
err_fpga_init:
|
||||||
|
mlx5_fpga_device_cleanup(dev);
|
||||||
|
|
||||||
err_put_uars:
|
err_put_uars:
|
||||||
mlx5_put_uars_page(dev, priv->uar);
|
mlx5_put_uars_page(dev, priv->uar);
|
||||||
|
|
||||||
@ -1246,6 +1262,7 @@ static int mlx5_unload_one(struct mlx5_core_dev *dev, struct mlx5_priv *priv,
|
|||||||
mlx5_irq_clear_affinity_hints(dev);
|
mlx5_irq_clear_affinity_hints(dev);
|
||||||
free_comp_eqs(dev);
|
free_comp_eqs(dev);
|
||||||
mlx5_stop_eqs(dev);
|
mlx5_stop_eqs(dev);
|
||||||
|
mlx5_fpga_device_cleanup(dev);
|
||||||
mlx5_put_uars_page(dev, priv->uar);
|
mlx5_put_uars_page(dev, priv->uar);
|
||||||
mlx5_disable_msix(dev);
|
mlx5_disable_msix(dev);
|
||||||
if (cleanup)
|
if (cleanup)
|
||||||
@ -1520,6 +1537,8 @@ static const struct pci_device_id mlx5_core_pci_table[] = {
|
|||||||
{ PCI_VDEVICE(MELLANOX, 0x101a), MLX5_PCI_DEV_IS_VF}, /* ConnectX-5 Ex VF */
|
{ PCI_VDEVICE(MELLANOX, 0x101a), MLX5_PCI_DEV_IS_VF}, /* ConnectX-5 Ex VF */
|
||||||
{ PCI_VDEVICE(MELLANOX, 0x101b) }, /* ConnectX-6 */
|
{ PCI_VDEVICE(MELLANOX, 0x101b) }, /* ConnectX-6 */
|
||||||
{ PCI_VDEVICE(MELLANOX, 0x101c), MLX5_PCI_DEV_IS_VF}, /* ConnectX-6 VF */
|
{ PCI_VDEVICE(MELLANOX, 0x101c), MLX5_PCI_DEV_IS_VF}, /* ConnectX-6 VF */
|
||||||
|
{ PCI_VDEVICE(MELLANOX, 0xa2d2) }, /* BlueField integrated ConnectX-5 network controller */
|
||||||
|
{ PCI_VDEVICE(MELLANOX, 0xa2d3), MLX5_PCI_DEV_IS_VF}, /* BlueField integrated ConnectX-5 network controller VF */
|
||||||
{ 0, }
|
{ 0, }
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -39,8 +39,7 @@
|
|||||||
#include <linux/if_link.h>
|
#include <linux/if_link.h>
|
||||||
|
|
||||||
#define DRIVER_NAME "mlx5_core"
|
#define DRIVER_NAME "mlx5_core"
|
||||||
#define DRIVER_VERSION "3.0-1"
|
#define DRIVER_VERSION "5.0-0"
|
||||||
#define DRIVER_RELDATE "January 2015"
|
|
||||||
|
|
||||||
#define MLX5_TOTAL_VPORTS(mdev) (1 + pci_sriov_get_totalvfs(mdev->pdev))
|
#define MLX5_TOTAL_VPORTS(mdev) (1 + pci_sriov_get_totalvfs(mdev->pdev))
|
||||||
|
|
||||||
|
@ -279,7 +279,7 @@ static int give_pages(struct mlx5_core_dev *dev, u16 func_id, int npages,
|
|||||||
int i;
|
int i;
|
||||||
|
|
||||||
inlen += npages * MLX5_FLD_SZ_BYTES(manage_pages_in, pas[0]);
|
inlen += npages * MLX5_FLD_SZ_BYTES(manage_pages_in, pas[0]);
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in) {
|
if (!in) {
|
||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
mlx5_core_warn(dev, "vzalloc failed %d\n", inlen);
|
mlx5_core_warn(dev, "vzalloc failed %d\n", inlen);
|
||||||
@ -376,7 +376,7 @@ static int reclaim_pages(struct mlx5_core_dev *dev, u32 func_id, int npages,
|
|||||||
*nclaimed = 0;
|
*nclaimed = 0;
|
||||||
|
|
||||||
outlen += npages * MLX5_FLD_SZ_BYTES(manage_pages_out, pas[0]);
|
outlen += npages * MLX5_FLD_SZ_BYTES(manage_pages_out, pas[0]);
|
||||||
out = mlx5_vzalloc(outlen);
|
out = kvzalloc(outlen, GFP_KERNEL);
|
||||||
if (!out)
|
if (!out)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
|
@ -47,8 +47,8 @@ int mlx5_core_access_reg(struct mlx5_core_dev *dev, void *data_in,
|
|||||||
u32 *in = NULL;
|
u32 *in = NULL;
|
||||||
void *data;
|
void *data;
|
||||||
|
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
out = mlx5_vzalloc(outlen);
|
out = kvzalloc(outlen, GFP_KERNEL);
|
||||||
if (!in || !out)
|
if (!in || !out)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
@ -454,7 +454,7 @@ int mlx5_core_query_ib_ppcnt(struct mlx5_core_dev *dev,
|
|||||||
u32 *in;
|
u32 *in;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
in = mlx5_vzalloc(sz);
|
in = kvzalloc(sz, GFP_KERNEL);
|
||||||
if (!in) {
|
if (!in) {
|
||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
return err;
|
return err;
|
||||||
|
@ -527,7 +527,7 @@ int mlx5_core_query_out_of_buffer(struct mlx5_core_dev *dev, u16 counter_id,
|
|||||||
void *out;
|
void *out;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
out = mlx5_vzalloc(outlen);
|
out = kvzalloc(outlen, GFP_KERNEL);
|
||||||
if (!out)
|
if (!out)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
|
@ -162,7 +162,7 @@ static int create_srq_cmd(struct mlx5_core_dev *dev, struct mlx5_core_srq *srq,
|
|||||||
|
|
||||||
pas_size = get_pas_size(in);
|
pas_size = get_pas_size(in);
|
||||||
inlen = MLX5_ST_SZ_BYTES(create_srq_in) + pas_size;
|
inlen = MLX5_ST_SZ_BYTES(create_srq_in) + pas_size;
|
||||||
create_in = mlx5_vzalloc(inlen);
|
create_in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!create_in)
|
if (!create_in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -221,7 +221,7 @@ static int query_srq_cmd(struct mlx5_core_dev *dev, struct mlx5_core_srq *srq,
|
|||||||
void *srqc;
|
void *srqc;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
srq_out = mlx5_vzalloc(MLX5_ST_SZ_BYTES(query_srq_out));
|
srq_out = kvzalloc(MLX5_ST_SZ_BYTES(query_srq_out), GFP_KERNEL);
|
||||||
if (!srq_out)
|
if (!srq_out)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -256,7 +256,7 @@ static int create_xrc_srq_cmd(struct mlx5_core_dev *dev,
|
|||||||
|
|
||||||
pas_size = get_pas_size(in);
|
pas_size = get_pas_size(in);
|
||||||
inlen = MLX5_ST_SZ_BYTES(create_xrc_srq_in) + pas_size;
|
inlen = MLX5_ST_SZ_BYTES(create_xrc_srq_in) + pas_size;
|
||||||
create_in = mlx5_vzalloc(inlen);
|
create_in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!create_in)
|
if (!create_in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -320,7 +320,7 @@ static int query_xrc_srq_cmd(struct mlx5_core_dev *dev,
|
|||||||
void *xrc_srqc;
|
void *xrc_srqc;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
xrcsrq_out = mlx5_vzalloc(MLX5_ST_SZ_BYTES(query_xrc_srq_out));
|
xrcsrq_out = kvzalloc(MLX5_ST_SZ_BYTES(query_xrc_srq_out), GFP_KERNEL);
|
||||||
if (!xrcsrq_out)
|
if (!xrcsrq_out)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
memset(xrcsrq_in, 0, sizeof(xrcsrq_in));
|
memset(xrcsrq_in, 0, sizeof(xrcsrq_in));
|
||||||
@ -357,7 +357,7 @@ static int create_rmp_cmd(struct mlx5_core_dev *dev, struct mlx5_core_srq *srq,
|
|||||||
|
|
||||||
pas_size = get_pas_size(in);
|
pas_size = get_pas_size(in);
|
||||||
inlen = MLX5_ST_SZ_BYTES(create_rmp_in) + pas_size;
|
inlen = MLX5_ST_SZ_BYTES(create_rmp_in) + pas_size;
|
||||||
create_in = mlx5_vzalloc(inlen);
|
create_in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!create_in)
|
if (!create_in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -390,7 +390,7 @@ static int arm_rmp_cmd(struct mlx5_core_dev *dev,
|
|||||||
void *bitmask;
|
void *bitmask;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
in = mlx5_vzalloc(MLX5_ST_SZ_BYTES(modify_rmp_in));
|
in = kvzalloc(MLX5_ST_SZ_BYTES(modify_rmp_in), GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -417,7 +417,7 @@ static int query_rmp_cmd(struct mlx5_core_dev *dev, struct mlx5_core_srq *srq,
|
|||||||
void *rmpc;
|
void *rmpc;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
rmp_out = mlx5_vzalloc(MLX5_ST_SZ_BYTES(query_rmp_out));
|
rmp_out = kvzalloc(MLX5_ST_SZ_BYTES(query_rmp_out), GFP_KERNEL);
|
||||||
if (!rmp_out)
|
if (!rmp_out)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
|
@ -284,7 +284,7 @@ int mlx5_core_arm_rmp(struct mlx5_core_dev *dev, u32 rmpn, u16 lwm)
|
|||||||
void *bitmask;
|
void *bitmask;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
in = mlx5_vzalloc(MLX5_ST_SZ_BYTES(modify_rmp_in));
|
in = kvzalloc(MLX5_ST_SZ_BYTES(modify_rmp_in), GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
|
@ -172,7 +172,7 @@ int mlx5_query_nic_vport_mac_address(struct mlx5_core_dev *mdev,
|
|||||||
u8 *out_addr;
|
u8 *out_addr;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
out = mlx5_vzalloc(outlen);
|
out = kvzalloc(outlen, GFP_KERNEL);
|
||||||
if (!out)
|
if (!out)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -197,11 +197,9 @@ int mlx5_modify_nic_vport_mac_address(struct mlx5_core_dev *mdev,
|
|||||||
void *nic_vport_ctx;
|
void *nic_vport_ctx;
|
||||||
u8 *perm_mac;
|
u8 *perm_mac;
|
||||||
|
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in) {
|
if (!in)
|
||||||
mlx5_core_warn(mdev, "failed to allocate inbox\n");
|
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
|
||||||
|
|
||||||
MLX5_SET(modify_nic_vport_context_in, in,
|
MLX5_SET(modify_nic_vport_context_in, in,
|
||||||
field_select.permanent_address, 1);
|
field_select.permanent_address, 1);
|
||||||
@ -231,7 +229,7 @@ int mlx5_query_nic_vport_mtu(struct mlx5_core_dev *mdev, u16 *mtu)
|
|||||||
u32 *out;
|
u32 *out;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
out = mlx5_vzalloc(outlen);
|
out = kvzalloc(outlen, GFP_KERNEL);
|
||||||
if (!out)
|
if (!out)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -251,7 +249,7 @@ int mlx5_modify_nic_vport_mtu(struct mlx5_core_dev *mdev, u16 mtu)
|
|||||||
void *in;
|
void *in;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -501,7 +499,7 @@ int mlx5_query_nic_vport_system_image_guid(struct mlx5_core_dev *mdev,
|
|||||||
u32 *out;
|
u32 *out;
|
||||||
int outlen = MLX5_ST_SZ_BYTES(query_nic_vport_context_out);
|
int outlen = MLX5_ST_SZ_BYTES(query_nic_vport_context_out);
|
||||||
|
|
||||||
out = mlx5_vzalloc(outlen);
|
out = kvzalloc(outlen, GFP_KERNEL);
|
||||||
if (!out)
|
if (!out)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -521,7 +519,7 @@ int mlx5_query_nic_vport_node_guid(struct mlx5_core_dev *mdev, u64 *node_guid)
|
|||||||
u32 *out;
|
u32 *out;
|
||||||
int outlen = MLX5_ST_SZ_BYTES(query_nic_vport_context_out);
|
int outlen = MLX5_ST_SZ_BYTES(query_nic_vport_context_out);
|
||||||
|
|
||||||
out = mlx5_vzalloc(outlen);
|
out = kvzalloc(outlen, GFP_KERNEL);
|
||||||
if (!out)
|
if (!out)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -551,7 +549,7 @@ int mlx5_modify_nic_vport_node_guid(struct mlx5_core_dev *mdev,
|
|||||||
if (!MLX5_CAP_ESW(mdev, nic_vport_node_guid_modify))
|
if (!MLX5_CAP_ESW(mdev, nic_vport_node_guid_modify))
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in)
|
if (!in)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -577,7 +575,7 @@ int mlx5_query_nic_vport_qkey_viol_cntr(struct mlx5_core_dev *mdev,
|
|||||||
u32 *out;
|
u32 *out;
|
||||||
int outlen = MLX5_ST_SZ_BYTES(query_nic_vport_context_out);
|
int outlen = MLX5_ST_SZ_BYTES(query_nic_vport_context_out);
|
||||||
|
|
||||||
out = mlx5_vzalloc(outlen);
|
out = kvzalloc(outlen, GFP_KERNEL);
|
||||||
if (!out)
|
if (!out)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -879,11 +877,9 @@ int mlx5_modify_nic_vport_promisc(struct mlx5_core_dev *mdev,
|
|||||||
int inlen = MLX5_ST_SZ_BYTES(modify_nic_vport_context_in);
|
int inlen = MLX5_ST_SZ_BYTES(modify_nic_vport_context_in);
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in) {
|
if (!in)
|
||||||
mlx5_core_err(mdev, "failed to allocate inbox\n");
|
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
|
||||||
|
|
||||||
MLX5_SET(modify_nic_vport_context_in, in, field_select.promisc, 1);
|
MLX5_SET(modify_nic_vport_context_in, in, field_select.promisc, 1);
|
||||||
MLX5_SET(modify_nic_vport_context_in, in,
|
MLX5_SET(modify_nic_vport_context_in, in,
|
||||||
@ -913,11 +909,9 @@ static int mlx5_nic_vport_update_roce_state(struct mlx5_core_dev *mdev,
|
|||||||
int inlen = MLX5_ST_SZ_BYTES(modify_nic_vport_context_in);
|
int inlen = MLX5_ST_SZ_BYTES(modify_nic_vport_context_in);
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
in = mlx5_vzalloc(inlen);
|
in = kvzalloc(inlen, GFP_KERNEL);
|
||||||
if (!in) {
|
if (!in)
|
||||||
mlx5_core_warn(mdev, "failed to allocate inbox\n");
|
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
|
||||||
|
|
||||||
MLX5_SET(modify_nic_vport_context_in, in, field_select.roce_en, 1);
|
MLX5_SET(modify_nic_vport_context_in, in, field_select.roce_en, 1);
|
||||||
MLX5_SET(modify_nic_vport_context_in, in, nic_vport_context.roce_en,
|
MLX5_SET(modify_nic_vport_context_in, in, nic_vport_context.roce_en,
|
||||||
@ -952,7 +946,7 @@ int mlx5_core_query_vport_counter(struct mlx5_core_dev *dev, u8 other_vport,
|
|||||||
int err;
|
int err;
|
||||||
|
|
||||||
is_group_manager = MLX5_CAP_GEN(dev, vport_group_manager);
|
is_group_manager = MLX5_CAP_GEN(dev, vport_group_manager);
|
||||||
in = mlx5_vzalloc(in_sz);
|
in = kvzalloc(in_sz, GFP_KERNEL);
|
||||||
if (!in) {
|
if (!in) {
|
||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
return err;
|
return err;
|
||||||
|
@ -300,6 +300,8 @@ enum mlx5_event {
|
|||||||
|
|
||||||
MLX5_EVENT_TYPE_PAGE_FAULT = 0xc,
|
MLX5_EVENT_TYPE_PAGE_FAULT = 0xc,
|
||||||
MLX5_EVENT_TYPE_NIC_VPORT_CHANGE = 0xd,
|
MLX5_EVENT_TYPE_NIC_VPORT_CHANGE = 0xd,
|
||||||
|
|
||||||
|
MLX5_EVENT_TYPE_FPGA_ERROR = 0x20,
|
||||||
};
|
};
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
@ -967,6 +969,7 @@ enum mlx5_cap_type {
|
|||||||
MLX5_CAP_RESERVED,
|
MLX5_CAP_RESERVED,
|
||||||
MLX5_CAP_VECTOR_CALC,
|
MLX5_CAP_VECTOR_CALC,
|
||||||
MLX5_CAP_QOS,
|
MLX5_CAP_QOS,
|
||||||
|
MLX5_CAP_FPGA,
|
||||||
/* NUM OF CAP Types */
|
/* NUM OF CAP Types */
|
||||||
MLX5_CAP_NUM
|
MLX5_CAP_NUM
|
||||||
};
|
};
|
||||||
@ -1088,6 +1091,9 @@ enum mlx5_mcam_feature_groups {
|
|||||||
#define MLX5_CAP_MCAM_FEATURE(mdev, fld) \
|
#define MLX5_CAP_MCAM_FEATURE(mdev, fld) \
|
||||||
MLX5_GET(mcam_reg, (mdev)->caps.mcam, mng_feature_cap_mask.enhanced_features.fld)
|
MLX5_GET(mcam_reg, (mdev)->caps.mcam, mng_feature_cap_mask.enhanced_features.fld)
|
||||||
|
|
||||||
|
#define MLX5_CAP_FPGA(mdev, cap) \
|
||||||
|
MLX5_GET(fpga_cap, (mdev)->caps.hca_cur[MLX5_CAP_FPGA], cap)
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
MLX5_CMD_STAT_OK = 0x0,
|
MLX5_CMD_STAT_OK = 0x0,
|
||||||
MLX5_CMD_STAT_INT_ERR = 0x1,
|
MLX5_CMD_STAT_INT_ERR = 0x1,
|
||||||
|
@ -108,6 +108,8 @@ enum {
|
|||||||
MLX5_REG_QTCT = 0x400a,
|
MLX5_REG_QTCT = 0x400a,
|
||||||
MLX5_REG_DCBX_PARAM = 0x4020,
|
MLX5_REG_DCBX_PARAM = 0x4020,
|
||||||
MLX5_REG_DCBX_APP = 0x4021,
|
MLX5_REG_DCBX_APP = 0x4021,
|
||||||
|
MLX5_REG_FPGA_CAP = 0x4022,
|
||||||
|
MLX5_REG_FPGA_CTRL = 0x4023,
|
||||||
MLX5_REG_PCAP = 0x5001,
|
MLX5_REG_PCAP = 0x5001,
|
||||||
MLX5_REG_PMTU = 0x5003,
|
MLX5_REG_PMTU = 0x5003,
|
||||||
MLX5_REG_PTYS = 0x5004,
|
MLX5_REG_PTYS = 0x5004,
|
||||||
@ -761,6 +763,9 @@ struct mlx5_core_dev {
|
|||||||
atomic_t num_qps;
|
atomic_t num_qps;
|
||||||
u32 issi;
|
u32 issi;
|
||||||
struct mlx5e_resources mlx5e_res;
|
struct mlx5e_resources mlx5e_res;
|
||||||
|
#ifdef CONFIG_MLX5_FPGA
|
||||||
|
struct mlx5_fpga_device *fpga;
|
||||||
|
#endif
|
||||||
#ifdef CONFIG_RFS_ACCEL
|
#ifdef CONFIG_RFS_ACCEL
|
||||||
struct cpu_rmap *rmap;
|
struct cpu_rmap *rmap;
|
||||||
#endif
|
#endif
|
||||||
@ -890,11 +895,6 @@ static inline u16 cmdif_rev(struct mlx5_core_dev *dev)
|
|||||||
return ioread32be(&dev->iseg->cmdif_rev_fw_sub) >> 16;
|
return ioread32be(&dev->iseg->cmdif_rev_fw_sub) >> 16;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void *mlx5_vzalloc(unsigned long size)
|
|
||||||
{
|
|
||||||
return kvzalloc(size, GFP_KERNEL);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline u32 mlx5_base_mkey(const u32 key)
|
static inline u32 mlx5_base_mkey(const u32 key)
|
||||||
{
|
{
|
||||||
return key & 0xffffff00u;
|
return key & 0xffffff00u;
|
||||||
@ -920,6 +920,7 @@ int mlx5_health_init(struct mlx5_core_dev *dev);
|
|||||||
void mlx5_start_health_poll(struct mlx5_core_dev *dev);
|
void mlx5_start_health_poll(struct mlx5_core_dev *dev);
|
||||||
void mlx5_stop_health_poll(struct mlx5_core_dev *dev);
|
void mlx5_stop_health_poll(struct mlx5_core_dev *dev);
|
||||||
void mlx5_drain_health_wq(struct mlx5_core_dev *dev);
|
void mlx5_drain_health_wq(struct mlx5_core_dev *dev);
|
||||||
|
void mlx5_trigger_health_work(struct mlx5_core_dev *dev);
|
||||||
int mlx5_buf_alloc_node(struct mlx5_core_dev *dev, int size,
|
int mlx5_buf_alloc_node(struct mlx5_core_dev *dev, int size,
|
||||||
struct mlx5_buf *buf, int node);
|
struct mlx5_buf *buf, int node);
|
||||||
int mlx5_buf_alloc(struct mlx5_core_dev *dev, int size, struct mlx5_buf *buf);
|
int mlx5_buf_alloc(struct mlx5_core_dev *dev, int size, struct mlx5_buf *buf);
|
||||||
|
@ -32,6 +32,8 @@
|
|||||||
#ifndef MLX5_IFC_H
|
#ifndef MLX5_IFC_H
|
||||||
#define MLX5_IFC_H
|
#define MLX5_IFC_H
|
||||||
|
|
||||||
|
#include "mlx5_ifc_fpga.h"
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
MLX5_EVENT_TYPE_CODING_COMPLETION_EVENTS = 0x0,
|
MLX5_EVENT_TYPE_CODING_COMPLETION_EVENTS = 0x0,
|
||||||
MLX5_EVENT_TYPE_CODING_PATH_MIGRATED_SUCCEEDED = 0x1,
|
MLX5_EVENT_TYPE_CODING_PATH_MIGRATED_SUCCEEDED = 0x1,
|
||||||
@ -56,7 +58,8 @@ enum {
|
|||||||
MLX5_EVENT_TYPE_CODING_STALL_VL_EVENT = 0x1b,
|
MLX5_EVENT_TYPE_CODING_STALL_VL_EVENT = 0x1b,
|
||||||
MLX5_EVENT_TYPE_CODING_DROPPED_PACKET_LOGGED_EVENT = 0x1f,
|
MLX5_EVENT_TYPE_CODING_DROPPED_PACKET_LOGGED_EVENT = 0x1f,
|
||||||
MLX5_EVENT_TYPE_CODING_COMMAND_INTERFACE_COMPLETION = 0xa,
|
MLX5_EVENT_TYPE_CODING_COMMAND_INTERFACE_COMPLETION = 0xa,
|
||||||
MLX5_EVENT_TYPE_CODING_PAGE_REQUEST = 0xb
|
MLX5_EVENT_TYPE_CODING_PAGE_REQUEST = 0xb,
|
||||||
|
MLX5_EVENT_TYPE_CODING_FPGA_ERROR = 0x20,
|
||||||
};
|
};
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
@ -854,7 +857,8 @@ struct mlx5_ifc_cmd_hca_cap_bits {
|
|||||||
u8 max_tc[0x4];
|
u8 max_tc[0x4];
|
||||||
u8 reserved_at_1d0[0x1];
|
u8 reserved_at_1d0[0x1];
|
||||||
u8 dcbx[0x1];
|
u8 dcbx[0x1];
|
||||||
u8 reserved_at_1d2[0x4];
|
u8 reserved_at_1d2[0x3];
|
||||||
|
u8 fpga[0x1];
|
||||||
u8 rol_s[0x1];
|
u8 rol_s[0x1];
|
||||||
u8 rol_g[0x1];
|
u8 rol_g[0x1];
|
||||||
u8 reserved_at_1d8[0x1];
|
u8 reserved_at_1d8[0x1];
|
||||||
@ -2186,6 +2190,7 @@ union mlx5_ifc_hca_cap_union_bits {
|
|||||||
struct mlx5_ifc_e_switch_cap_bits e_switch_cap;
|
struct mlx5_ifc_e_switch_cap_bits e_switch_cap;
|
||||||
struct mlx5_ifc_vector_calc_cap_bits vector_calc_cap;
|
struct mlx5_ifc_vector_calc_cap_bits vector_calc_cap;
|
||||||
struct mlx5_ifc_qos_cap_bits qos_cap;
|
struct mlx5_ifc_qos_cap_bits qos_cap;
|
||||||
|
struct mlx5_ifc_fpga_cap_bits fpga_cap;
|
||||||
u8 reserved_at_0[0x8000];
|
u8 reserved_at_0[0x8000];
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -8182,6 +8187,8 @@ union mlx5_ifc_ports_control_registers_document_bits {
|
|||||||
struct mlx5_ifc_sltp_reg_bits sltp_reg;
|
struct mlx5_ifc_sltp_reg_bits sltp_reg;
|
||||||
struct mlx5_ifc_mtpps_reg_bits mtpps_reg;
|
struct mlx5_ifc_mtpps_reg_bits mtpps_reg;
|
||||||
struct mlx5_ifc_mtppse_reg_bits mtppse_reg;
|
struct mlx5_ifc_mtppse_reg_bits mtppse_reg;
|
||||||
|
struct mlx5_ifc_fpga_ctrl_bits fpga_ctrl_bits;
|
||||||
|
struct mlx5_ifc_fpga_cap_bits fpga_cap_bits;
|
||||||
u8 reserved_at_0[0x60e0];
|
u8 reserved_at_0[0x60e0];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
144
include/linux/mlx5/mlx5_ifc_fpga.h
Normal file
144
include/linux/mlx5/mlx5_ifc_fpga.h
Normal file
@ -0,0 +1,144 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2017, Mellanox Technologies, Ltd. All rights reserved.
|
||||||
|
*
|
||||||
|
* This software is available to you under a choice of one of two
|
||||||
|
* licenses. You may choose to be licensed under the terms of the GNU
|
||||||
|
* General Public License (GPL) Version 2, available from the file
|
||||||
|
* COPYING in the main directory of this source tree, or the
|
||||||
|
* OpenIB.org BSD license below:
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or
|
||||||
|
* without modification, are permitted provided that the following
|
||||||
|
* conditions are met:
|
||||||
|
*
|
||||||
|
* - Redistributions of source code must retain the above
|
||||||
|
* copyright notice, this list of conditions and the following
|
||||||
|
* disclaimer.
|
||||||
|
*
|
||||||
|
* - Redistributions in binary form must reproduce the above
|
||||||
|
* copyright notice, this list of conditions and the following
|
||||||
|
* disclaimer in the documentation and/or other materials
|
||||||
|
* provided with the distribution.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
|
||||||
|
* BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
|
||||||
|
* ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
||||||
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
* SOFTWARE.
|
||||||
|
*/
|
||||||
|
#ifndef MLX5_IFC_FPGA_H
|
||||||
|
#define MLX5_IFC_FPGA_H
|
||||||
|
|
||||||
|
struct mlx5_ifc_fpga_shell_caps_bits {
|
||||||
|
u8 max_num_qps[0x10];
|
||||||
|
u8 reserved_at_10[0x8];
|
||||||
|
u8 total_rcv_credits[0x8];
|
||||||
|
|
||||||
|
u8 reserved_at_20[0xe];
|
||||||
|
u8 qp_type[0x2];
|
||||||
|
u8 reserved_at_30[0x5];
|
||||||
|
u8 rae[0x1];
|
||||||
|
u8 rwe[0x1];
|
||||||
|
u8 rre[0x1];
|
||||||
|
u8 reserved_at_38[0x4];
|
||||||
|
u8 dc[0x1];
|
||||||
|
u8 ud[0x1];
|
||||||
|
u8 uc[0x1];
|
||||||
|
u8 rc[0x1];
|
||||||
|
|
||||||
|
u8 reserved_at_40[0x1a];
|
||||||
|
u8 log_ddr_size[0x6];
|
||||||
|
|
||||||
|
u8 max_fpga_qp_msg_size[0x20];
|
||||||
|
|
||||||
|
u8 reserved_at_80[0x180];
|
||||||
|
};
|
||||||
|
|
||||||
|
struct mlx5_ifc_fpga_cap_bits {
|
||||||
|
u8 fpga_id[0x8];
|
||||||
|
u8 fpga_device[0x18];
|
||||||
|
|
||||||
|
u8 register_file_ver[0x20];
|
||||||
|
|
||||||
|
u8 fpga_ctrl_modify[0x1];
|
||||||
|
u8 reserved_at_41[0x5];
|
||||||
|
u8 access_reg_query_mode[0x2];
|
||||||
|
u8 reserved_at_48[0x6];
|
||||||
|
u8 access_reg_modify_mode[0x2];
|
||||||
|
u8 reserved_at_50[0x10];
|
||||||
|
|
||||||
|
u8 reserved_at_60[0x20];
|
||||||
|
|
||||||
|
u8 image_version[0x20];
|
||||||
|
|
||||||
|
u8 image_date[0x20];
|
||||||
|
|
||||||
|
u8 image_time[0x20];
|
||||||
|
|
||||||
|
u8 shell_version[0x20];
|
||||||
|
|
||||||
|
u8 reserved_at_100[0x80];
|
||||||
|
|
||||||
|
struct mlx5_ifc_fpga_shell_caps_bits shell_caps;
|
||||||
|
|
||||||
|
u8 reserved_at_380[0x8];
|
||||||
|
u8 ieee_vendor_id[0x18];
|
||||||
|
|
||||||
|
u8 sandbox_product_version[0x10];
|
||||||
|
u8 sandbox_product_id[0x10];
|
||||||
|
|
||||||
|
u8 sandbox_basic_caps[0x20];
|
||||||
|
|
||||||
|
u8 reserved_at_3e0[0x10];
|
||||||
|
u8 sandbox_extended_caps_len[0x10];
|
||||||
|
|
||||||
|
u8 sandbox_extended_caps_addr[0x40];
|
||||||
|
|
||||||
|
u8 fpga_ddr_start_addr[0x40];
|
||||||
|
|
||||||
|
u8 fpga_cr_space_start_addr[0x40];
|
||||||
|
|
||||||
|
u8 fpga_ddr_size[0x20];
|
||||||
|
|
||||||
|
u8 fpga_cr_space_size[0x20];
|
||||||
|
|
||||||
|
u8 reserved_at_500[0x300];
|
||||||
|
};
|
||||||
|
|
||||||
|
struct mlx5_ifc_fpga_ctrl_bits {
|
||||||
|
u8 reserved_at_0[0x8];
|
||||||
|
u8 operation[0x8];
|
||||||
|
u8 reserved_at_10[0x8];
|
||||||
|
u8 status[0x8];
|
||||||
|
|
||||||
|
u8 reserved_at_20[0x8];
|
||||||
|
u8 flash_select_admin[0x8];
|
||||||
|
u8 reserved_at_30[0x8];
|
||||||
|
u8 flash_select_oper[0x8];
|
||||||
|
|
||||||
|
u8 reserved_at_40[0x40];
|
||||||
|
};
|
||||||
|
|
||||||
|
enum {
|
||||||
|
MLX5_FPGA_ERROR_EVENT_SYNDROME_CORRUPTED_DDR = 0x1,
|
||||||
|
MLX5_FPGA_ERROR_EVENT_SYNDROME_FLASH_TIMEOUT = 0x2,
|
||||||
|
MLX5_FPGA_ERROR_EVENT_SYNDROME_INTERNAL_LINK_ERROR = 0x3,
|
||||||
|
MLX5_FPGA_ERROR_EVENT_SYNDROME_WATCHDOG_FAILURE = 0x4,
|
||||||
|
MLX5_FPGA_ERROR_EVENT_SYNDROME_I2C_FAILURE = 0x5,
|
||||||
|
MLX5_FPGA_ERROR_EVENT_SYNDROME_IMAGE_CHANGED = 0x6,
|
||||||
|
MLX5_FPGA_ERROR_EVENT_SYNDROME_TEMPERATURE_CRITICAL = 0x7,
|
||||||
|
};
|
||||||
|
|
||||||
|
struct mlx5_ifc_fpga_error_event_bits {
|
||||||
|
u8 reserved_at_0[0x40];
|
||||||
|
|
||||||
|
u8 reserved_at_40[0x18];
|
||||||
|
u8 syndrome[0x8];
|
||||||
|
|
||||||
|
u8 reserved_at_60[0x80];
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif /* MLX5_IFC_FPGA_H */
|
Loading…
Reference in New Issue
Block a user