aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorJack Morgenstein <jackm@dev.mellanox.co.il>2012-08-03 04:40:41 -0400
committerRoland Dreier <roland@purestorage.com>2012-09-30 23:33:31 -0400
commite2c76824ca16a3e8443cc7b26abcb21af7c27b10 (patch)
tree41d39111862e2e6faecb9ab356198e985c9bfe7d
parent1ffeb2eb8be9936e9dc1f9af2d5f4c14d69a0d36 (diff)
mlx4_core: Add proxy and tunnel QPs to the reserved QP area
In addition, pass the proxy and tunnel QP numbers to slaves so the driver can perform special QP paravirtualization. Signed-off-by: Jack Morgenstein <jackm@dev.mellanox.co.il> Signed-off-by: Roland Dreier <roland@purestorage.com>
-rw-r--r--drivers/net/ethernet/mellanox/mlx4/fw.c14
-rw-r--r--drivers/net/ethernet/mellanox/mlx4/fw.h3
-rw-r--r--drivers/net/ethernet/mellanox/mlx4/main.c5
-rw-r--r--drivers/net/ethernet/mellanox/mlx4/qp.c29
-rw-r--r--drivers/net/ethernet/mellanox/mlx4/resource_tracker.c3
-rw-r--r--include/linux/mlx4/device.h13
6 files changed, 62 insertions, 5 deletions
diff --git a/drivers/net/ethernet/mellanox/mlx4/fw.c b/drivers/net/ethernet/mellanox/mlx4/fw.c
index c69648487321..35be90178bef 100644
--- a/drivers/net/ethernet/mellanox/mlx4/fw.c
+++ b/drivers/net/ethernet/mellanox/mlx4/fw.c
@@ -184,6 +184,8 @@ int mlx4_QUERY_FUNC_CAP_wrapper(struct mlx4_dev *dev, int slave,
184#define QUERY_FUNC_CAP_MCG_QUOTA_OFFSET 0x28 184#define QUERY_FUNC_CAP_MCG_QUOTA_OFFSET 0x28
185#define QUERY_FUNC_CAP_MAX_EQ_OFFSET 0x2c 185#define QUERY_FUNC_CAP_MAX_EQ_OFFSET 0x2c
186#define QUERY_FUNC_CAP_RESERVED_EQ_OFFSET 0X30 186#define QUERY_FUNC_CAP_RESERVED_EQ_OFFSET 0X30
187#define QUERY_FUNC_CAP_BASE_TUNNEL_QPN_OFFSET 0x44
188#define QUERY_FUNC_CAP_BASE_PROXY_QPN_OFFSET 0x48
187 189
188#define QUERY_FUNC_CAP_FMR_FLAG 0x80 190#define QUERY_FUNC_CAP_FMR_FLAG 0x80
189#define QUERY_FUNC_CAP_FLAG_RDMA 0x40 191#define QUERY_FUNC_CAP_FLAG_RDMA 0x40
@@ -247,6 +249,12 @@ int mlx4_QUERY_FUNC_CAP_wrapper(struct mlx4_dev *dev, int slave,
247 size = dev->caps.num_mgms + dev->caps.num_amgms; 249 size = dev->caps.num_mgms + dev->caps.num_amgms;
248 MLX4_PUT(outbox->buf, size, QUERY_FUNC_CAP_MCG_QUOTA_OFFSET); 250 MLX4_PUT(outbox->buf, size, QUERY_FUNC_CAP_MCG_QUOTA_OFFSET);
249 251
252 size = dev->caps.base_tunnel_sqpn + 8 * slave;
253 MLX4_PUT(outbox->buf, size, QUERY_FUNC_CAP_BASE_TUNNEL_QPN_OFFSET);
254
255 size = dev->caps.sqp_start + 8 * slave;
256 MLX4_PUT(outbox->buf, size, QUERY_FUNC_CAP_BASE_PROXY_QPN_OFFSET);
257
250 } else 258 } else
251 err = -EINVAL; 259 err = -EINVAL;
252 260
@@ -312,6 +320,12 @@ int mlx4_QUERY_FUNC_CAP(struct mlx4_dev *dev, struct mlx4_func_cap *func_cap)
312 MLX4_GET(size, outbox, QUERY_FUNC_CAP_MCG_QUOTA_OFFSET); 320 MLX4_GET(size, outbox, QUERY_FUNC_CAP_MCG_QUOTA_OFFSET);
313 func_cap->mcg_quota = size & 0xFFFFFF; 321 func_cap->mcg_quota = size & 0xFFFFFF;
314 322
323 MLX4_GET(size, outbox, QUERY_FUNC_CAP_BASE_TUNNEL_QPN_OFFSET);
324 func_cap->base_tunnel_qpn = size & 0xFFFFFF;
325
326 MLX4_GET(size, outbox, QUERY_FUNC_CAP_BASE_PROXY_QPN_OFFSET);
327 func_cap->base_proxy_qpn = size & 0xFFFFFF;
328
315 for (i = 1; i <= func_cap->num_ports; ++i) { 329 for (i = 1; i <= func_cap->num_ports; ++i) {
316 err = mlx4_cmd_box(dev, 0, mailbox->dma, i, 1, 330 err = mlx4_cmd_box(dev, 0, mailbox->dma, i, 1,
317 MLX4_CMD_QUERY_FUNC_CAP, 331 MLX4_CMD_QUERY_FUNC_CAP,
diff --git a/drivers/net/ethernet/mellanox/mlx4/fw.h b/drivers/net/ethernet/mellanox/mlx4/fw.h
index 83fcbbf1b169..ced1de57c818 100644
--- a/drivers/net/ethernet/mellanox/mlx4/fw.h
+++ b/drivers/net/ethernet/mellanox/mlx4/fw.h
@@ -134,6 +134,9 @@ struct mlx4_func_cap {
134 int max_eq; 134 int max_eq;
135 int reserved_eq; 135 int reserved_eq;
136 int mcg_quota; 136 int mcg_quota;
137 u32 base_qpn;
138 u32 base_tunnel_qpn;
139 u32 base_proxy_qpn;
137 u8 physical_port[MLX4_MAX_PORTS + 1]; 140 u8 physical_port[MLX4_MAX_PORTS + 1];
138 u8 port_flags[MLX4_MAX_PORTS + 1]; 141 u8 port_flags[MLX4_MAX_PORTS + 1];
139}; 142};
diff --git a/drivers/net/ethernet/mellanox/mlx4/main.c b/drivers/net/ethernet/mellanox/mlx4/main.c
index 2f816c6aed72..06ef3afbc49a 100644
--- a/drivers/net/ethernet/mellanox/mlx4/main.c
+++ b/drivers/net/ethernet/mellanox/mlx4/main.c
@@ -384,6 +384,7 @@ static int mlx4_dev_cap(struct mlx4_dev *dev, struct mlx4_dev_cap *dev_cap)
384 dev->caps.reserved_qps_cnt[MLX4_QP_REGION_FC_ADDR] + 384 dev->caps.reserved_qps_cnt[MLX4_QP_REGION_FC_ADDR] +
385 dev->caps.reserved_qps_cnt[MLX4_QP_REGION_FC_EXCH]; 385 dev->caps.reserved_qps_cnt[MLX4_QP_REGION_FC_EXCH];
386 386
387 dev->caps.sqp_demux = (mlx4_is_master(dev)) ? MLX4_MAX_NUM_SLAVES : 0;
387 return 0; 388 return 0;
388} 389}
389/*The function checks if there are live vf, return the num of them*/ 390/*The function checks if there are live vf, return the num of them*/
@@ -541,6 +542,10 @@ static int mlx4_slave_cap(struct mlx4_dev *dev)
541 return -ENODEV; 542 return -ENODEV;
542 } 543 }
543 544
545 /* Calculate our sqp_start */
546 dev->caps.sqp_start = func_cap.base_proxy_qpn;
547 dev->caps.base_tunnel_sqpn = func_cap.base_tunnel_qpn;
548
544 return 0; 549 return 0;
545} 550}
546 551
diff --git a/drivers/net/ethernet/mellanox/mlx4/qp.c b/drivers/net/ethernet/mellanox/mlx4/qp.c
index fb2b36759cbf..b8da72b29cf7 100644
--- a/drivers/net/ethernet/mellanox/mlx4/qp.c
+++ b/drivers/net/ethernet/mellanox/mlx4/qp.c
@@ -406,7 +406,7 @@ int mlx4_init_qp_table(struct mlx4_dev *dev)
406 * We also reserve the MSB of the 24-bit QP number to indicate 406 * We also reserve the MSB of the 24-bit QP number to indicate
407 * that a QP is an XRC QP. 407 * that a QP is an XRC QP.
408 */ 408 */
409 dev->caps.sqp_start = 409 dev->caps.base_sqpn =
410 ALIGN(dev->caps.reserved_qps_cnt[MLX4_QP_REGION_FW], 8); 410 ALIGN(dev->caps.reserved_qps_cnt[MLX4_QP_REGION_FW], 8);
411 411
412 { 412 {
@@ -437,13 +437,36 @@ int mlx4_init_qp_table(struct mlx4_dev *dev)
437 437
438 } 438 }
439 439
440 /* Reserve 8 real SQPs in both native and SRIOV modes.
441 * In addition, in SRIOV mode, reserve 8 proxy SQPs per function
442 * (for all PFs and VFs), and 8 corresponding tunnel QPs.
443 * Each proxy SQP works opposite its own tunnel QP.
444 *
445 * The QPs are arranged as follows:
446 * a. 8 real SQPs
447 * b. All the proxy SQPs (8 per function)
448 * c. All the tunnel QPs (8 per function)
449 */
450
440 err = mlx4_bitmap_init(&qp_table->bitmap, dev->caps.num_qps, 451 err = mlx4_bitmap_init(&qp_table->bitmap, dev->caps.num_qps,
441 (1 << 23) - 1, dev->caps.sqp_start + 8, 452 (1 << 23) - 1, dev->caps.base_sqpn + 8 +
453 16 * MLX4_MFUNC_MAX * !!mlx4_is_master(dev),
442 reserved_from_top); 454 reserved_from_top);
455
456 /* In mfunc, sqp_start is the base of the proxy SQPs, since the PF also
457 * uses paravirtualized SQPs.
458 * In native mode, sqp_start is the base of the real SQPs. */
459 if (mlx4_is_mfunc(dev)) {
460 dev->caps.sqp_start = dev->caps.base_sqpn +
461 8 * (mlx4_master_func_num(dev) + 1);
462 dev->caps.base_tunnel_sqpn = dev->caps.sqp_start + 8 * MLX4_MFUNC_MAX;
463 } else
464 dev->caps.sqp_start = dev->caps.base_sqpn;
465
443 if (err) 466 if (err)
444 return err; 467 return err;
445 468
446 return mlx4_CONF_SPECIAL_QP(dev, dev->caps.sqp_start); 469 return mlx4_CONF_SPECIAL_QP(dev, dev->caps.base_sqpn);
447} 470}
448 471
449void mlx4_cleanup_qp_table(struct mlx4_dev *dev) 472void mlx4_cleanup_qp_table(struct mlx4_dev *dev)
diff --git a/drivers/net/ethernet/mellanox/mlx4/resource_tracker.c b/drivers/net/ethernet/mellanox/mlx4/resource_tracker.c
index 293c9e820c49..3c57a83e6287 100644
--- a/drivers/net/ethernet/mellanox/mlx4/resource_tracker.c
+++ b/drivers/net/ethernet/mellanox/mlx4/resource_tracker.c
@@ -1105,7 +1105,8 @@ static void res_end_move(struct mlx4_dev *dev, int slave,
1105 1105
1106static int valid_reserved(struct mlx4_dev *dev, int slave, int qpn) 1106static int valid_reserved(struct mlx4_dev *dev, int slave, int qpn)
1107{ 1107{
1108 return mlx4_is_qp_reserved(dev, qpn); 1108 return mlx4_is_qp_reserved(dev, qpn) &&
1109 (mlx4_is_master(dev) || mlx4_is_guest_proxy(dev, slave, qpn));
1109} 1110}
1110 1111
1111static int qp_alloc_res(struct mlx4_dev *dev, int slave, int op, int cmd, 1112static int qp_alloc_res(struct mlx4_dev *dev, int slave, int op, int cmd,
diff --git a/include/linux/mlx4/device.h b/include/linux/mlx4/device.h
index 07aa8232e631..d5c82b7216de 100644
--- a/include/linux/mlx4/device.h
+++ b/include/linux/mlx4/device.h
@@ -693,7 +693,18 @@ static inline int mlx4_is_master(struct mlx4_dev *dev)
693 693
694static inline int mlx4_is_qp_reserved(struct mlx4_dev *dev, u32 qpn) 694static inline int mlx4_is_qp_reserved(struct mlx4_dev *dev, u32 qpn)
695{ 695{
696 return (qpn < dev->caps.sqp_start + 8); 696 return (qpn < dev->caps.base_sqpn + 8 +
697 16 * MLX4_MFUNC_MAX * !!mlx4_is_master(dev));
698}
699
700static inline int mlx4_is_guest_proxy(struct mlx4_dev *dev, int slave, u32 qpn)
701{
702 int base = dev->caps.sqp_start + slave * 8;
703
704 if (qpn >= base && qpn < base + 8)
705 return 1;
706
707 return 0;
697} 708}
698 709
699static inline int mlx4_is_mfunc(struct mlx4_dev *dev) 710static inline int mlx4_is_mfunc(struct mlx4_dev *dev)