diff options
author | Jack Morgenstein <jackm@dev.mellanox.co.il> | 2012-08-03 04:40:41 -0400 |
---|---|---|
committer | Roland Dreier <roland@purestorage.com> | 2012-09-30 23:33:31 -0400 |
commit | e2c76824ca16a3e8443cc7b26abcb21af7c27b10 (patch) | |
tree | 41d39111862e2e6faecb9ab356198e985c9bfe7d | |
parent | 1ffeb2eb8be9936e9dc1f9af2d5f4c14d69a0d36 (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.c | 14 | ||||
-rw-r--r-- | drivers/net/ethernet/mellanox/mlx4/fw.h | 3 | ||||
-rw-r--r-- | drivers/net/ethernet/mellanox/mlx4/main.c | 5 | ||||
-rw-r--r-- | drivers/net/ethernet/mellanox/mlx4/qp.c | 29 | ||||
-rw-r--r-- | drivers/net/ethernet/mellanox/mlx4/resource_tracker.c | 3 | ||||
-rw-r--r-- | include/linux/mlx4/device.h | 13 |
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 | ||
449 | void mlx4_cleanup_qp_table(struct mlx4_dev *dev) | 472 | void 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 | ||
1106 | static int valid_reserved(struct mlx4_dev *dev, int slave, int qpn) | 1106 | static 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 | ||
1111 | static int qp_alloc_res(struct mlx4_dev *dev, int slave, int op, int cmd, | 1112 | static 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 | ||
694 | static inline int mlx4_is_qp_reserved(struct mlx4_dev *dev, u32 qpn) | 694 | static 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 | |||
700 | static 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 | ||
699 | static inline int mlx4_is_mfunc(struct mlx4_dev *dev) | 710 | static inline int mlx4_is_mfunc(struct mlx4_dev *dev) |