Initial roundup of 4.5 merge window patches
- Remove usage of ib_query_device and instead store attributes in ib_device struct - Move iopoll out of block and into lib, rename to irqpoll, and use in several places in the rdma stack as our new completion queue polling library mechanism. Update the other block drivers that already used iopoll to use the new mechanism too. - Replace the per-entry GID table locks with a single GID table lock - IPoIB multicast cleanup - Cleanups to the IB MR facility - Add support for 64bit extended IB counters - Fix for netlink oops while parsing RDMA nl messages - RoCEv2 support for the core IB code - mlx4 RoCEv2 support - mlx5 RoCEv2 support - Cross Channel support for mlx5 - Timestamp support for mlx5 - Atomic support for mlx5 - Raw QP support for mlx5 - MAINTAINERS update for mlx4/mlx5 - Misc ocrdma, qib, nes, usNIC, cxgb3, cxgb4, mlx4, mlx5 updates - Add support for remote invalidate to the iSER driver (pushed through the RDMA tree due to dependencies, acknowledged by nab) - Update to NFSoRDMA (pushed through the RDMA tree due to dependencies, acknowledged by Bruce) -----BEGIN PGP SIGNATURE----- Version: GnuPG v1 iQIcBAABAgAGBQJWoSygAAoJELgmozMOVy/dDjsP/2vbTda2MvQfkfkGEZBQdJSg 095RN0gQgCJdg78lAl8yuaK8r4VN/7uefpDtFdudH1I/Pei7X0wxN9R1UzFNG4KR AD53lz92IVPs15328SbPR2kvNWISR9aBFQo3rlElq3Grqlp0EMn2Ou1vtu87rekF aMllxr8Nl0uZhP+eWusOsYpJUUtwirLgRnrAyfqo2UxZh/TMIroT0TCx1KXjVcAg dhDARiZAdu3OgSc6OsWqmH+DELEq6dFVA5F+DDBGAb8bFZqlJc7cuMHWInwNsNXT so4bnEQ835alTbsdYtqs5DUNS8heJTAJP4Uz0ehkTh/uNCcvnKeUTw1c2P/lXI1k 7s33gMM+0FXj0swMBw0kKwAF2d9Hhus9UAN7NwjBuOyHcjGRd5q7SAnfWkvKx000 s9jVW19slb2I38gB58nhjOh8s+vXUArgxnV1+kTia1+bJSR5swvVoWRicRXdF0vh TvLX/BjbSIU73g1TnnLNYoBTV3ybFKQ6bVdQW7fzSTDs54dsI1vvdHXi3bYZCpnL HVwQTZRfEzkvb0AdKbcvf8p/TlaAHem3ODqtO1eHvO4if1QJBSn+SptTEeJVYYdK n4B3l/dMoBH4JXJUmEHB9jwAvYOpv/YLAFIvdL7NFwbqGNsC3nfXFcmkVORB1W3B KEMcM2we4bz+uyKMjEAD =5oO7 -----END PGP SIGNATURE----- Merge tag 'for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/dledford/rdma Pull rdma updates from Doug Ledford: "Initial roundup of 4.5 merge window patches - Remove usage of ib_query_device and instead store attributes in ib_device struct - Move iopoll out of block and into lib, rename to irqpoll, and use in several places in the rdma stack as our new completion queue polling library mechanism. Update the other block drivers that already used iopoll to use the new mechanism too. - Replace the per-entry GID table locks with a single GID table lock - IPoIB multicast cleanup - Cleanups to the IB MR facility - Add support for 64bit extended IB counters - Fix for netlink oops while parsing RDMA nl messages - RoCEv2 support for the core IB code - mlx4 RoCEv2 support - mlx5 RoCEv2 support - Cross Channel support for mlx5 - Timestamp support for mlx5 - Atomic support for mlx5 - Raw QP support for mlx5 - MAINTAINERS update for mlx4/mlx5 - Misc ocrdma, qib, nes, usNIC, cxgb3, cxgb4, mlx4, mlx5 updates - Add support for remote invalidate to the iSER driver (pushed through the RDMA tree due to dependencies, acknowledged by nab) - Update to NFSoRDMA (pushed through the RDMA tree due to dependencies, acknowledged by Bruce)" * tag 'for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/dledford/rdma: (169 commits) IB/mlx5: Unify CQ create flags check IB/mlx5: Expose Raw Packet QP to user space consumers {IB, net}/mlx5: Move the modify QP operation table to mlx5_ib IB/mlx5: Support setting Ethernet priority for Raw Packet QPs IB/mlx5: Add Raw Packet QP query functionality IB/mlx5: Add create and destroy functionality for Raw Packet QP IB/mlx5: Refactor mlx5_ib_qp to accommodate other QP types IB/mlx5: Allocate a Transport Domain for each ucontext net/mlx5_core: Warn on unsupported events of QP/RQ/SQ net/mlx5_core: Add RQ and SQ event handling net/mlx5_core: Export transport objects IB/mlx5: Expose CQE version to user-space IB/mlx5: Add CQE version 1 support to user QPs and SRQs IB/mlx5: Fix data validation in mlx5_ib_alloc_ucontext IB/sa: Fix netlink local service GFP crash IB/srpt: Remove redundant wc array IB/qib: Improve ipoib UD performance IB/mlx4: Advertise RoCE v2 support IB/mlx4: Create and use another QP1 for RoCEv2 IB/mlx4: Enable send of RoCE QP1 packets with IP/UDP headers ...
This commit is contained in:
commit
048ccca8c1
174 changed files with 7122 additions and 4681 deletions
22
Documentation/ABI/testing/configfs-rdma_cm
Normal file
22
Documentation/ABI/testing/configfs-rdma_cm
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
What: /config/rdma_cm
|
||||||
|
Date: November 29, 2015
|
||||||
|
KernelVersion: 4.4.0
|
||||||
|
Description: Interface is used to configure RDMA-cable HCAs in respect to
|
||||||
|
RDMA-CM attributes.
|
||||||
|
|
||||||
|
Attributes are visible only when configfs is mounted. To mount
|
||||||
|
configfs in /config directory use:
|
||||||
|
# mount -t configfs none /config/
|
||||||
|
|
||||||
|
In order to set parameters related to a specific HCA, a directory
|
||||||
|
for this HCA has to be created:
|
||||||
|
mkdir -p /config/rdma_cm/<hca>
|
||||||
|
|
||||||
|
|
||||||
|
What: /config/rdma_cm/<hca>/ports/<port-num>/default_roce_mode
|
||||||
|
Date: November 29, 2015
|
||||||
|
KernelVersion: 4.4.0
|
||||||
|
Description: RDMA-CM based connections from HCA <hca> at port <port-num>
|
||||||
|
will be initiated with this RoCE type as default.
|
||||||
|
The possible RoCE types are either "IB/RoCE v1" or "RoCE v2".
|
||||||
|
This parameter has RW access.
|
16
Documentation/ABI/testing/sysfs-class-infiniband
Normal file
16
Documentation/ABI/testing/sysfs-class-infiniband
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
What: /sys/class/infiniband/<hca>/ports/<port-number>/gid_attrs/ndevs/<gid-index>
|
||||||
|
Date: November 29, 2015
|
||||||
|
KernelVersion: 4.4.0
|
||||||
|
Contact: linux-rdma@vger.kernel.org
|
||||||
|
Description: The net-device's name associated with the GID resides
|
||||||
|
at index <gid-index>.
|
||||||
|
|
||||||
|
What: /sys/class/infiniband/<hca>/ports/<port-number>/gid_attrs/types/<gid-index>
|
||||||
|
Date: November 29, 2015
|
||||||
|
KernelVersion: 4.4.0
|
||||||
|
Contact: linux-rdma@vger.kernel.org
|
||||||
|
Description: The RoCE type of the associated GID resides at index <gid-index>.
|
||||||
|
This could either be "IB/RoCE v1" for IB and RoCE v1 based GODs
|
||||||
|
or "RoCE v2" for RoCE v2 based GIDs.
|
||||||
|
|
||||||
|
|
|
@ -15,7 +15,6 @@ Sleeping and interrupt context
|
||||||
modify_ah
|
modify_ah
|
||||||
query_ah
|
query_ah
|
||||||
destroy_ah
|
destroy_ah
|
||||||
bind_mw
|
|
||||||
post_send
|
post_send
|
||||||
post_recv
|
post_recv
|
||||||
poll_cq
|
poll_cq
|
||||||
|
@ -31,7 +30,6 @@ Sleeping and interrupt context
|
||||||
ib_modify_ah
|
ib_modify_ah
|
||||||
ib_query_ah
|
ib_query_ah
|
||||||
ib_destroy_ah
|
ib_destroy_ah
|
||||||
ib_bind_mw
|
|
||||||
ib_post_send
|
ib_post_send
|
||||||
ib_post_recv
|
ib_post_recv
|
||||||
ib_req_notify_cq
|
ib_req_notify_cq
|
||||||
|
|
|
@ -90,7 +90,7 @@ BLOCK_SOFTIRQ: Do all of the following:
|
||||||
from being initiated from tasks that might run on the CPU to
|
from being initiated from tasks that might run on the CPU to
|
||||||
be de-jittered. (It is OK to force this CPU offline and then
|
be de-jittered. (It is OK to force this CPU offline and then
|
||||||
bring it back online before you start your application.)
|
bring it back online before you start your application.)
|
||||||
BLOCK_IOPOLL_SOFTIRQ: Do all of the following:
|
IRQ_POLL_SOFTIRQ: Do all of the following:
|
||||||
1. Force block-device interrupts onto some other CPU.
|
1. Force block-device interrupts onto some other CPU.
|
||||||
2. Initiate any block I/O and block-I/O polling on other CPUs.
|
2. Initiate any block I/O and block-I/O polling on other CPUs.
|
||||||
3. Once your application has started, prevent CPU-hotplug operations
|
3. Once your application has started, prevent CPU-hotplug operations
|
||||||
|
|
32
MAINTAINERS
32
MAINTAINERS
|
@ -7151,27 +7151,45 @@ W: https://linuxtv.org
|
||||||
S: Odd Fixes
|
S: Odd Fixes
|
||||||
F: drivers/media/radio/radio-miropcm20*
|
F: drivers/media/radio/radio-miropcm20*
|
||||||
|
|
||||||
Mellanox MLX5 core VPI driver
|
MELLANOX MLX4 core VPI driver
|
||||||
M: Eli Cohen <eli@mellanox.com>
|
M: Yishai Hadas <yishaih@mellanox.com>
|
||||||
L: netdev@vger.kernel.org
|
L: netdev@vger.kernel.org
|
||||||
L: linux-rdma@vger.kernel.org
|
L: linux-rdma@vger.kernel.org
|
||||||
W: http://www.mellanox.com
|
W: http://www.mellanox.com
|
||||||
Q: http://patchwork.ozlabs.org/project/netdev/list/
|
Q: http://patchwork.ozlabs.org/project/netdev/list/
|
||||||
|
S: Supported
|
||||||
|
F: drivers/net/ethernet/mellanox/mlx4/
|
||||||
|
F: include/linux/mlx4/
|
||||||
|
|
||||||
|
MELLANOX MLX4 IB driver
|
||||||
|
M: Yishai Hadas <yishaih@mellanox.com>
|
||||||
|
L: linux-rdma@vger.kernel.org
|
||||||
|
W: http://www.mellanox.com
|
||||||
Q: http://patchwork.kernel.org/project/linux-rdma/list/
|
Q: http://patchwork.kernel.org/project/linux-rdma/list/
|
||||||
T: git git://openfabrics.org/~eli/connect-ib.git
|
S: Supported
|
||||||
|
F: drivers/infiniband/hw/mlx4/
|
||||||
|
F: include/linux/mlx4/
|
||||||
|
|
||||||
|
MELLANOX MLX5 core VPI driver
|
||||||
|
M: Matan Barak <matanb@mellanox.com>
|
||||||
|
M: Leon Romanovsky <leonro@mellanox.com>
|
||||||
|
L: netdev@vger.kernel.org
|
||||||
|
L: linux-rdma@vger.kernel.org
|
||||||
|
W: http://www.mellanox.com
|
||||||
|
Q: http://patchwork.ozlabs.org/project/netdev/list/
|
||||||
S: Supported
|
S: Supported
|
||||||
F: drivers/net/ethernet/mellanox/mlx5/core/
|
F: drivers/net/ethernet/mellanox/mlx5/core/
|
||||||
F: include/linux/mlx5/
|
F: include/linux/mlx5/
|
||||||
|
|
||||||
Mellanox MLX5 IB driver
|
MELLANOX MLX5 IB driver
|
||||||
M: Eli Cohen <eli@mellanox.com>
|
M: Matan Barak <matanb@mellanox.com>
|
||||||
|
M: Leon Romanovsky <leonro@mellanox.com>
|
||||||
L: linux-rdma@vger.kernel.org
|
L: linux-rdma@vger.kernel.org
|
||||||
W: http://www.mellanox.com
|
W: http://www.mellanox.com
|
||||||
Q: http://patchwork.kernel.org/project/linux-rdma/list/
|
Q: http://patchwork.kernel.org/project/linux-rdma/list/
|
||||||
T: git git://openfabrics.org/~eli/connect-ib.git
|
|
||||||
S: Supported
|
S: Supported
|
||||||
F: include/linux/mlx5/
|
|
||||||
F: drivers/infiniband/hw/mlx5/
|
F: drivers/infiniband/hw/mlx5/
|
||||||
|
F: include/linux/mlx5/
|
||||||
|
|
||||||
MELEXIS MLX90614 DRIVER
|
MELEXIS MLX90614 DRIVER
|
||||||
M: Crt Mori <cmo@melexis.com>
|
M: Crt Mori <cmo@melexis.com>
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
obj-$(CONFIG_BLOCK) := bio.o elevator.o blk-core.o blk-tag.o blk-sysfs.o \
|
obj-$(CONFIG_BLOCK) := bio.o elevator.o blk-core.o blk-tag.o blk-sysfs.o \
|
||||||
blk-flush.o blk-settings.o blk-ioc.o blk-map.o \
|
blk-flush.o blk-settings.o blk-ioc.o blk-map.o \
|
||||||
blk-exec.o blk-merge.o blk-softirq.o blk-timeout.o \
|
blk-exec.o blk-merge.o blk-softirq.o blk-timeout.o \
|
||||||
blk-iopoll.o blk-lib.o blk-mq.o blk-mq-tag.o \
|
blk-lib.o blk-mq.o blk-mq-tag.o \
|
||||||
blk-mq-sysfs.o blk-mq-cpu.o blk-mq-cpumap.o ioctl.o \
|
blk-mq-sysfs.o blk-mq-cpu.o blk-mq-cpumap.o ioctl.o \
|
||||||
genhd.o scsi_ioctl.o partition-generic.o ioprio.o \
|
genhd.o scsi_ioctl.o partition-generic.o ioprio.o \
|
||||||
badblocks.o partitions/
|
badblocks.o partitions/
|
||||||
|
|
|
@ -5,6 +5,7 @@ menuconfig INFINIBAND
|
||||||
depends on NET
|
depends on NET
|
||||||
depends on INET
|
depends on INET
|
||||||
depends on m || IPV6 != m
|
depends on m || IPV6 != m
|
||||||
|
select IRQ_POLL
|
||||||
---help---
|
---help---
|
||||||
Core support for InfiniBand (IB). Make sure to also select
|
Core support for InfiniBand (IB). Make sure to also select
|
||||||
any protocols you wish to use as well as drivers for your
|
any protocols you wish to use as well as drivers for your
|
||||||
|
@ -54,6 +55,15 @@ config INFINIBAND_ADDR_TRANS
|
||||||
depends on INFINIBAND
|
depends on INFINIBAND
|
||||||
default y
|
default y
|
||||||
|
|
||||||
|
config INFINIBAND_ADDR_TRANS_CONFIGFS
|
||||||
|
bool
|
||||||
|
depends on INFINIBAND_ADDR_TRANS && CONFIGFS_FS && !(INFINIBAND=y && CONFIGFS_FS=m)
|
||||||
|
default y
|
||||||
|
---help---
|
||||||
|
ConfigFS support for RDMA communication manager (CM).
|
||||||
|
This allows the user to config the default GID type that the CM
|
||||||
|
uses for each device, when initiaing new connections.
|
||||||
|
|
||||||
source "drivers/infiniband/hw/mthca/Kconfig"
|
source "drivers/infiniband/hw/mthca/Kconfig"
|
||||||
source "drivers/infiniband/hw/qib/Kconfig"
|
source "drivers/infiniband/hw/qib/Kconfig"
|
||||||
source "drivers/infiniband/hw/cxgb3/Kconfig"
|
source "drivers/infiniband/hw/cxgb3/Kconfig"
|
||||||
|
|
|
@ -8,7 +8,7 @@ obj-$(CONFIG_INFINIBAND_USER_MAD) += ib_umad.o
|
||||||
obj-$(CONFIG_INFINIBAND_USER_ACCESS) += ib_uverbs.o ib_ucm.o \
|
obj-$(CONFIG_INFINIBAND_USER_ACCESS) += ib_uverbs.o ib_ucm.o \
|
||||||
$(user_access-y)
|
$(user_access-y)
|
||||||
|
|
||||||
ib_core-y := packer.o ud_header.o verbs.o sysfs.o \
|
ib_core-y := packer.o ud_header.o verbs.o cq.o sysfs.o \
|
||||||
device.o fmr_pool.o cache.o netlink.o \
|
device.o fmr_pool.o cache.o netlink.o \
|
||||||
roce_gid_mgmt.o
|
roce_gid_mgmt.o
|
||||||
ib_core-$(CONFIG_INFINIBAND_USER_MEM) += umem.o
|
ib_core-$(CONFIG_INFINIBAND_USER_MEM) += umem.o
|
||||||
|
@ -24,6 +24,8 @@ iw_cm-y := iwcm.o iwpm_util.o iwpm_msg.o
|
||||||
|
|
||||||
rdma_cm-y := cma.o
|
rdma_cm-y := cma.o
|
||||||
|
|
||||||
|
rdma_cm-$(CONFIG_INFINIBAND_ADDR_TRANS_CONFIGFS) += cma_configfs.o
|
||||||
|
|
||||||
rdma_ucm-y := ucma.o
|
rdma_ucm-y := ucma.o
|
||||||
|
|
||||||
ib_addr-y := addr.o
|
ib_addr-y := addr.o
|
||||||
|
|
|
@ -121,7 +121,8 @@ int rdma_copy_addr(struct rdma_dev_addr *dev_addr, struct net_device *dev,
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(rdma_copy_addr);
|
EXPORT_SYMBOL(rdma_copy_addr);
|
||||||
|
|
||||||
int rdma_translate_ip(struct sockaddr *addr, struct rdma_dev_addr *dev_addr,
|
int rdma_translate_ip(const struct sockaddr *addr,
|
||||||
|
struct rdma_dev_addr *dev_addr,
|
||||||
u16 *vlan_id)
|
u16 *vlan_id)
|
||||||
{
|
{
|
||||||
struct net_device *dev;
|
struct net_device *dev;
|
||||||
|
@ -139,7 +140,7 @@ int rdma_translate_ip(struct sockaddr *addr, struct rdma_dev_addr *dev_addr,
|
||||||
switch (addr->sa_family) {
|
switch (addr->sa_family) {
|
||||||
case AF_INET:
|
case AF_INET:
|
||||||
dev = ip_dev_find(dev_addr->net,
|
dev = ip_dev_find(dev_addr->net,
|
||||||
((struct sockaddr_in *) addr)->sin_addr.s_addr);
|
((const struct sockaddr_in *)addr)->sin_addr.s_addr);
|
||||||
|
|
||||||
if (!dev)
|
if (!dev)
|
||||||
return ret;
|
return ret;
|
||||||
|
@ -154,7 +155,7 @@ int rdma_translate_ip(struct sockaddr *addr, struct rdma_dev_addr *dev_addr,
|
||||||
rcu_read_lock();
|
rcu_read_lock();
|
||||||
for_each_netdev_rcu(dev_addr->net, dev) {
|
for_each_netdev_rcu(dev_addr->net, dev) {
|
||||||
if (ipv6_chk_addr(dev_addr->net,
|
if (ipv6_chk_addr(dev_addr->net,
|
||||||
&((struct sockaddr_in6 *) addr)->sin6_addr,
|
&((const struct sockaddr_in6 *)addr)->sin6_addr,
|
||||||
dev, 1)) {
|
dev, 1)) {
|
||||||
ret = rdma_copy_addr(dev_addr, dev, NULL);
|
ret = rdma_copy_addr(dev_addr, dev, NULL);
|
||||||
if (vlan_id)
|
if (vlan_id)
|
||||||
|
@ -198,7 +199,8 @@ static void queue_req(struct addr_req *req)
|
||||||
mutex_unlock(&lock);
|
mutex_unlock(&lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int dst_fetch_ha(struct dst_entry *dst, struct rdma_dev_addr *dev_addr, void *daddr)
|
static int dst_fetch_ha(struct dst_entry *dst, struct rdma_dev_addr *dev_addr,
|
||||||
|
const void *daddr)
|
||||||
{
|
{
|
||||||
struct neighbour *n;
|
struct neighbour *n;
|
||||||
int ret;
|
int ret;
|
||||||
|
@ -222,8 +224,9 @@ static int dst_fetch_ha(struct dst_entry *dst, struct rdma_dev_addr *dev_addr, v
|
||||||
}
|
}
|
||||||
|
|
||||||
static int addr4_resolve(struct sockaddr_in *src_in,
|
static int addr4_resolve(struct sockaddr_in *src_in,
|
||||||
struct sockaddr_in *dst_in,
|
const struct sockaddr_in *dst_in,
|
||||||
struct rdma_dev_addr *addr)
|
struct rdma_dev_addr *addr,
|
||||||
|
struct rtable **prt)
|
||||||
{
|
{
|
||||||
__be32 src_ip = src_in->sin_addr.s_addr;
|
__be32 src_ip = src_in->sin_addr.s_addr;
|
||||||
__be32 dst_ip = dst_in->sin_addr.s_addr;
|
__be32 dst_ip = dst_in->sin_addr.s_addr;
|
||||||
|
@ -243,33 +246,29 @@ static int addr4_resolve(struct sockaddr_in *src_in,
|
||||||
src_in->sin_family = AF_INET;
|
src_in->sin_family = AF_INET;
|
||||||
src_in->sin_addr.s_addr = fl4.saddr;
|
src_in->sin_addr.s_addr = fl4.saddr;
|
||||||
|
|
||||||
if (rt->dst.dev->flags & IFF_LOOPBACK) {
|
/* If there's a gateway, we're definitely in RoCE v2 (as RoCE v1 isn't
|
||||||
ret = rdma_translate_ip((struct sockaddr *)dst_in, addr, NULL);
|
* routable) and we could set the network type accordingly.
|
||||||
if (!ret)
|
*/
|
||||||
memcpy(addr->dst_dev_addr, addr->src_dev_addr, MAX_ADDR_LEN);
|
if (rt->rt_uses_gateway)
|
||||||
goto put;
|
addr->network = RDMA_NETWORK_IPV4;
|
||||||
}
|
|
||||||
|
|
||||||
/* If the device does ARP internally, return 'done' */
|
addr->hoplimit = ip4_dst_hoplimit(&rt->dst);
|
||||||
if (rt->dst.dev->flags & IFF_NOARP) {
|
|
||||||
ret = rdma_copy_addr(addr, rt->dst.dev, NULL);
|
|
||||||
goto put;
|
|
||||||
}
|
|
||||||
|
|
||||||
ret = dst_fetch_ha(&rt->dst, addr, &fl4.daddr);
|
*prt = rt;
|
||||||
put:
|
return 0;
|
||||||
ip_rt_put(rt);
|
|
||||||
out:
|
out:
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
#if IS_ENABLED(CONFIG_IPV6)
|
#if IS_ENABLED(CONFIG_IPV6)
|
||||||
static int addr6_resolve(struct sockaddr_in6 *src_in,
|
static int addr6_resolve(struct sockaddr_in6 *src_in,
|
||||||
struct sockaddr_in6 *dst_in,
|
const struct sockaddr_in6 *dst_in,
|
||||||
struct rdma_dev_addr *addr)
|
struct rdma_dev_addr *addr,
|
||||||
|
struct dst_entry **pdst)
|
||||||
{
|
{
|
||||||
struct flowi6 fl6;
|
struct flowi6 fl6;
|
||||||
struct dst_entry *dst;
|
struct dst_entry *dst;
|
||||||
|
struct rt6_info *rt;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
memset(&fl6, 0, sizeof fl6);
|
memset(&fl6, 0, sizeof fl6);
|
||||||
|
@ -281,6 +280,7 @@ static int addr6_resolve(struct sockaddr_in6 *src_in,
|
||||||
if ((ret = dst->error))
|
if ((ret = dst->error))
|
||||||
goto put;
|
goto put;
|
||||||
|
|
||||||
|
rt = (struct rt6_info *)dst;
|
||||||
if (ipv6_addr_any(&fl6.saddr)) {
|
if (ipv6_addr_any(&fl6.saddr)) {
|
||||||
ret = ipv6_dev_get_saddr(addr->net, ip6_dst_idev(dst)->dev,
|
ret = ipv6_dev_get_saddr(addr->net, ip6_dst_idev(dst)->dev,
|
||||||
&fl6.daddr, 0, &fl6.saddr);
|
&fl6.daddr, 0, &fl6.saddr);
|
||||||
|
@ -291,43 +291,111 @@ static int addr6_resolve(struct sockaddr_in6 *src_in,
|
||||||
src_in->sin6_addr = fl6.saddr;
|
src_in->sin6_addr = fl6.saddr;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dst->dev->flags & IFF_LOOPBACK) {
|
/* If there's a gateway, we're definitely in RoCE v2 (as RoCE v1 isn't
|
||||||
ret = rdma_translate_ip((struct sockaddr *)dst_in, addr, NULL);
|
* routable) and we could set the network type accordingly.
|
||||||
if (!ret)
|
*/
|
||||||
memcpy(addr->dst_dev_addr, addr->src_dev_addr, MAX_ADDR_LEN);
|
if (rt->rt6i_flags & RTF_GATEWAY)
|
||||||
goto put;
|
addr->network = RDMA_NETWORK_IPV6;
|
||||||
}
|
|
||||||
|
|
||||||
/* If the device does ARP internally, return 'done' */
|
addr->hoplimit = ip6_dst_hoplimit(dst);
|
||||||
if (dst->dev->flags & IFF_NOARP) {
|
|
||||||
ret = rdma_copy_addr(addr, dst->dev, NULL);
|
|
||||||
goto put;
|
|
||||||
}
|
|
||||||
|
|
||||||
ret = dst_fetch_ha(dst, addr, &fl6.daddr);
|
*pdst = dst;
|
||||||
|
return 0;
|
||||||
put:
|
put:
|
||||||
dst_release(dst);
|
dst_release(dst);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
static int addr6_resolve(struct sockaddr_in6 *src_in,
|
static int addr6_resolve(struct sockaddr_in6 *src_in,
|
||||||
struct sockaddr_in6 *dst_in,
|
const struct sockaddr_in6 *dst_in,
|
||||||
struct rdma_dev_addr *addr)
|
struct rdma_dev_addr *addr,
|
||||||
|
struct dst_entry **pdst)
|
||||||
{
|
{
|
||||||
return -EADDRNOTAVAIL;
|
return -EADDRNOTAVAIL;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
static int addr_resolve(struct sockaddr *src_in,
|
static int addr_resolve_neigh(struct dst_entry *dst,
|
||||||
struct sockaddr *dst_in,
|
const struct sockaddr *dst_in,
|
||||||
struct rdma_dev_addr *addr)
|
struct rdma_dev_addr *addr)
|
||||||
{
|
{
|
||||||
|
if (dst->dev->flags & IFF_LOOPBACK) {
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
ret = rdma_translate_ip(dst_in, addr, NULL);
|
||||||
|
if (!ret)
|
||||||
|
memcpy(addr->dst_dev_addr, addr->src_dev_addr,
|
||||||
|
MAX_ADDR_LEN);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* If the device doesn't do ARP internally */
|
||||||
|
if (!(dst->dev->flags & IFF_NOARP)) {
|
||||||
|
const struct sockaddr_in *dst_in4 =
|
||||||
|
(const struct sockaddr_in *)dst_in;
|
||||||
|
const struct sockaddr_in6 *dst_in6 =
|
||||||
|
(const struct sockaddr_in6 *)dst_in;
|
||||||
|
|
||||||
|
return dst_fetch_ha(dst, addr,
|
||||||
|
dst_in->sa_family == AF_INET ?
|
||||||
|
(const void *)&dst_in4->sin_addr.s_addr :
|
||||||
|
(const void *)&dst_in6->sin6_addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
return rdma_copy_addr(addr, dst->dev, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int addr_resolve(struct sockaddr *src_in,
|
||||||
|
const struct sockaddr *dst_in,
|
||||||
|
struct rdma_dev_addr *addr,
|
||||||
|
bool resolve_neigh)
|
||||||
|
{
|
||||||
|
struct net_device *ndev;
|
||||||
|
struct dst_entry *dst;
|
||||||
|
int ret;
|
||||||
|
|
||||||
if (src_in->sa_family == AF_INET) {
|
if (src_in->sa_family == AF_INET) {
|
||||||
return addr4_resolve((struct sockaddr_in *) src_in,
|
struct rtable *rt = NULL;
|
||||||
(struct sockaddr_in *) dst_in, addr);
|
const struct sockaddr_in *dst_in4 =
|
||||||
} else
|
(const struct sockaddr_in *)dst_in;
|
||||||
return addr6_resolve((struct sockaddr_in6 *) src_in,
|
|
||||||
(struct sockaddr_in6 *) dst_in, addr);
|
ret = addr4_resolve((struct sockaddr_in *)src_in,
|
||||||
|
dst_in4, addr, &rt);
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
if (resolve_neigh)
|
||||||
|
ret = addr_resolve_neigh(&rt->dst, dst_in, addr);
|
||||||
|
|
||||||
|
ndev = rt->dst.dev;
|
||||||
|
dev_hold(ndev);
|
||||||
|
|
||||||
|
ip_rt_put(rt);
|
||||||
|
} else {
|
||||||
|
const struct sockaddr_in6 *dst_in6 =
|
||||||
|
(const struct sockaddr_in6 *)dst_in;
|
||||||
|
|
||||||
|
ret = addr6_resolve((struct sockaddr_in6 *)src_in,
|
||||||
|
dst_in6, addr,
|
||||||
|
&dst);
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
if (resolve_neigh)
|
||||||
|
ret = addr_resolve_neigh(dst, dst_in, addr);
|
||||||
|
|
||||||
|
ndev = dst->dev;
|
||||||
|
dev_hold(ndev);
|
||||||
|
|
||||||
|
dst_release(dst);
|
||||||
|
}
|
||||||
|
|
||||||
|
addr->bound_dev_if = ndev->ifindex;
|
||||||
|
addr->net = dev_net(ndev);
|
||||||
|
dev_put(ndev);
|
||||||
|
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void process_req(struct work_struct *work)
|
static void process_req(struct work_struct *work)
|
||||||
|
@ -343,7 +411,8 @@ static void process_req(struct work_struct *work)
|
||||||
if (req->status == -ENODATA) {
|
if (req->status == -ENODATA) {
|
||||||
src_in = (struct sockaddr *) &req->src_addr;
|
src_in = (struct sockaddr *) &req->src_addr;
|
||||||
dst_in = (struct sockaddr *) &req->dst_addr;
|
dst_in = (struct sockaddr *) &req->dst_addr;
|
||||||
req->status = addr_resolve(src_in, dst_in, req->addr);
|
req->status = addr_resolve(src_in, dst_in, req->addr,
|
||||||
|
true);
|
||||||
if (req->status && time_after_eq(jiffies, req->timeout))
|
if (req->status && time_after_eq(jiffies, req->timeout))
|
||||||
req->status = -ETIMEDOUT;
|
req->status = -ETIMEDOUT;
|
||||||
else if (req->status == -ENODATA)
|
else if (req->status == -ENODATA)
|
||||||
|
@ -403,7 +472,7 @@ int rdma_resolve_ip(struct rdma_addr_client *client,
|
||||||
req->client = client;
|
req->client = client;
|
||||||
atomic_inc(&client->refcount);
|
atomic_inc(&client->refcount);
|
||||||
|
|
||||||
req->status = addr_resolve(src_in, dst_in, addr);
|
req->status = addr_resolve(src_in, dst_in, addr, true);
|
||||||
switch (req->status) {
|
switch (req->status) {
|
||||||
case 0:
|
case 0:
|
||||||
req->timeout = jiffies;
|
req->timeout = jiffies;
|
||||||
|
@ -425,6 +494,26 @@ int rdma_resolve_ip(struct rdma_addr_client *client,
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(rdma_resolve_ip);
|
EXPORT_SYMBOL(rdma_resolve_ip);
|
||||||
|
|
||||||
|
int rdma_resolve_ip_route(struct sockaddr *src_addr,
|
||||||
|
const struct sockaddr *dst_addr,
|
||||||
|
struct rdma_dev_addr *addr)
|
||||||
|
{
|
||||||
|
struct sockaddr_storage ssrc_addr = {};
|
||||||
|
struct sockaddr *src_in = (struct sockaddr *)&ssrc_addr;
|
||||||
|
|
||||||
|
if (src_addr) {
|
||||||
|
if (src_addr->sa_family != dst_addr->sa_family)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
memcpy(src_in, src_addr, rdma_addr_size(src_addr));
|
||||||
|
} else {
|
||||||
|
src_in->sa_family = dst_addr->sa_family;
|
||||||
|
}
|
||||||
|
|
||||||
|
return addr_resolve(src_in, dst_addr, addr, false);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(rdma_resolve_ip_route);
|
||||||
|
|
||||||
void rdma_addr_cancel(struct rdma_dev_addr *addr)
|
void rdma_addr_cancel(struct rdma_dev_addr *addr)
|
||||||
{
|
{
|
||||||
struct addr_req *req, *temp_req;
|
struct addr_req *req, *temp_req;
|
||||||
|
@ -456,8 +545,10 @@ static void resolve_cb(int status, struct sockaddr *src_addr,
|
||||||
complete(&((struct resolve_cb_context *)context)->comp);
|
complete(&((struct resolve_cb_context *)context)->comp);
|
||||||
}
|
}
|
||||||
|
|
||||||
int rdma_addr_find_dmac_by_grh(const union ib_gid *sgid, const union ib_gid *dgid,
|
int rdma_addr_find_l2_eth_by_grh(const union ib_gid *sgid,
|
||||||
u8 *dmac, u16 *vlan_id, int if_index)
|
const union ib_gid *dgid,
|
||||||
|
u8 *dmac, u16 *vlan_id, int *if_index,
|
||||||
|
int *hoplimit)
|
||||||
{
|
{
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
struct rdma_dev_addr dev_addr;
|
struct rdma_dev_addr dev_addr;
|
||||||
|
@ -475,7 +566,8 @@ int rdma_addr_find_dmac_by_grh(const union ib_gid *sgid, const union ib_gid *dgi
|
||||||
rdma_gid2ip(&dgid_addr._sockaddr, dgid);
|
rdma_gid2ip(&dgid_addr._sockaddr, dgid);
|
||||||
|
|
||||||
memset(&dev_addr, 0, sizeof(dev_addr));
|
memset(&dev_addr, 0, sizeof(dev_addr));
|
||||||
dev_addr.bound_dev_if = if_index;
|
if (if_index)
|
||||||
|
dev_addr.bound_dev_if = *if_index;
|
||||||
dev_addr.net = &init_net;
|
dev_addr.net = &init_net;
|
||||||
|
|
||||||
ctx.addr = &dev_addr;
|
ctx.addr = &dev_addr;
|
||||||
|
@ -491,12 +583,16 @@ int rdma_addr_find_dmac_by_grh(const union ib_gid *sgid, const union ib_gid *dgi
|
||||||
dev = dev_get_by_index(&init_net, dev_addr.bound_dev_if);
|
dev = dev_get_by_index(&init_net, dev_addr.bound_dev_if);
|
||||||
if (!dev)
|
if (!dev)
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
|
if (if_index)
|
||||||
|
*if_index = dev_addr.bound_dev_if;
|
||||||
if (vlan_id)
|
if (vlan_id)
|
||||||
*vlan_id = rdma_vlan_dev_vlan_id(dev);
|
*vlan_id = rdma_vlan_dev_vlan_id(dev);
|
||||||
|
if (hoplimit)
|
||||||
|
*hoplimit = dev_addr.hoplimit;
|
||||||
dev_put(dev);
|
dev_put(dev);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(rdma_addr_find_dmac_by_grh);
|
EXPORT_SYMBOL(rdma_addr_find_l2_eth_by_grh);
|
||||||
|
|
||||||
int rdma_addr_find_smac_by_sgid(union ib_gid *sgid, u8 *smac, u16 *vlan_id)
|
int rdma_addr_find_smac_by_sgid(union ib_gid *sgid, u8 *smac, u16 *vlan_id)
|
||||||
{
|
{
|
||||||
|
|
|
@ -64,6 +64,7 @@ enum gid_attr_find_mask {
|
||||||
GID_ATTR_FIND_MASK_GID = 1UL << 0,
|
GID_ATTR_FIND_MASK_GID = 1UL << 0,
|
||||||
GID_ATTR_FIND_MASK_NETDEV = 1UL << 1,
|
GID_ATTR_FIND_MASK_NETDEV = 1UL << 1,
|
||||||
GID_ATTR_FIND_MASK_DEFAULT = 1UL << 2,
|
GID_ATTR_FIND_MASK_DEFAULT = 1UL << 2,
|
||||||
|
GID_ATTR_FIND_MASK_GID_TYPE = 1UL << 3,
|
||||||
};
|
};
|
||||||
|
|
||||||
enum gid_table_entry_props {
|
enum gid_table_entry_props {
|
||||||
|
@ -81,10 +82,6 @@ enum gid_table_write_action {
|
||||||
};
|
};
|
||||||
|
|
||||||
struct ib_gid_table_entry {
|
struct ib_gid_table_entry {
|
||||||
/* This lock protects an entry from being
|
|
||||||
* read and written simultaneously.
|
|
||||||
*/
|
|
||||||
rwlock_t lock;
|
|
||||||
unsigned long props;
|
unsigned long props;
|
||||||
union ib_gid gid;
|
union ib_gid gid;
|
||||||
struct ib_gid_attr attr;
|
struct ib_gid_attr attr;
|
||||||
|
@ -109,28 +106,86 @@ struct ib_gid_table {
|
||||||
* are locked by this lock.
|
* are locked by this lock.
|
||||||
**/
|
**/
|
||||||
struct mutex lock;
|
struct mutex lock;
|
||||||
|
/* This lock protects the table entries from being
|
||||||
|
* read and written simultaneously.
|
||||||
|
*/
|
||||||
|
rwlock_t rwlock;
|
||||||
struct ib_gid_table_entry *data_vec;
|
struct ib_gid_table_entry *data_vec;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static void dispatch_gid_change_event(struct ib_device *ib_dev, u8 port)
|
||||||
|
{
|
||||||
|
if (rdma_cap_roce_gid_table(ib_dev, port)) {
|
||||||
|
struct ib_event event;
|
||||||
|
|
||||||
|
event.device = ib_dev;
|
||||||
|
event.element.port_num = port;
|
||||||
|
event.event = IB_EVENT_GID_CHANGE;
|
||||||
|
|
||||||
|
ib_dispatch_event(&event);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static const char * const gid_type_str[] = {
|
||||||
|
[IB_GID_TYPE_IB] = "IB/RoCE v1",
|
||||||
|
[IB_GID_TYPE_ROCE_UDP_ENCAP] = "RoCE v2",
|
||||||
|
};
|
||||||
|
|
||||||
|
const char *ib_cache_gid_type_str(enum ib_gid_type gid_type)
|
||||||
|
{
|
||||||
|
if (gid_type < ARRAY_SIZE(gid_type_str) && gid_type_str[gid_type])
|
||||||
|
return gid_type_str[gid_type];
|
||||||
|
|
||||||
|
return "Invalid GID type";
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(ib_cache_gid_type_str);
|
||||||
|
|
||||||
|
int ib_cache_gid_parse_type_str(const char *buf)
|
||||||
|
{
|
||||||
|
unsigned int i;
|
||||||
|
size_t len;
|
||||||
|
int err = -EINVAL;
|
||||||
|
|
||||||
|
len = strlen(buf);
|
||||||
|
if (len == 0)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
if (buf[len - 1] == '\n')
|
||||||
|
len--;
|
||||||
|
|
||||||
|
for (i = 0; i < ARRAY_SIZE(gid_type_str); ++i)
|
||||||
|
if (gid_type_str[i] && !strncmp(buf, gid_type_str[i], len) &&
|
||||||
|
len == strlen(gid_type_str[i])) {
|
||||||
|
err = i;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(ib_cache_gid_parse_type_str);
|
||||||
|
|
||||||
|
/* This function expects that rwlock will be write locked in all
|
||||||
|
* scenarios and that lock will be locked in sleep-able (RoCE)
|
||||||
|
* scenarios.
|
||||||
|
*/
|
||||||
static int write_gid(struct ib_device *ib_dev, u8 port,
|
static int write_gid(struct ib_device *ib_dev, u8 port,
|
||||||
struct ib_gid_table *table, int ix,
|
struct ib_gid_table *table, int ix,
|
||||||
const union ib_gid *gid,
|
const union ib_gid *gid,
|
||||||
const struct ib_gid_attr *attr,
|
const struct ib_gid_attr *attr,
|
||||||
enum gid_table_write_action action,
|
enum gid_table_write_action action,
|
||||||
bool default_gid)
|
bool default_gid)
|
||||||
|
__releases(&table->rwlock) __acquires(&table->rwlock)
|
||||||
{
|
{
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
struct net_device *old_net_dev;
|
struct net_device *old_net_dev;
|
||||||
unsigned long flags;
|
|
||||||
|
|
||||||
/* in rdma_cap_roce_gid_table, this funciton should be protected by a
|
/* in rdma_cap_roce_gid_table, this funciton should be protected by a
|
||||||
* sleep-able lock.
|
* sleep-able lock.
|
||||||
*/
|
*/
|
||||||
write_lock_irqsave(&table->data_vec[ix].lock, flags);
|
|
||||||
|
|
||||||
if (rdma_cap_roce_gid_table(ib_dev, port)) {
|
if (rdma_cap_roce_gid_table(ib_dev, port)) {
|
||||||
table->data_vec[ix].props |= GID_TABLE_ENTRY_INVALID;
|
table->data_vec[ix].props |= GID_TABLE_ENTRY_INVALID;
|
||||||
write_unlock_irqrestore(&table->data_vec[ix].lock, flags);
|
write_unlock_irq(&table->rwlock);
|
||||||
/* GID_TABLE_WRITE_ACTION_MODIFY currently isn't supported by
|
/* GID_TABLE_WRITE_ACTION_MODIFY currently isn't supported by
|
||||||
* RoCE providers and thus only updates the cache.
|
* RoCE providers and thus only updates the cache.
|
||||||
*/
|
*/
|
||||||
|
@ -140,7 +195,7 @@ static int write_gid(struct ib_device *ib_dev, u8 port,
|
||||||
else if (action == GID_TABLE_WRITE_ACTION_DEL)
|
else if (action == GID_TABLE_WRITE_ACTION_DEL)
|
||||||
ret = ib_dev->del_gid(ib_dev, port, ix,
|
ret = ib_dev->del_gid(ib_dev, port, ix,
|
||||||
&table->data_vec[ix].context);
|
&table->data_vec[ix].context);
|
||||||
write_lock_irqsave(&table->data_vec[ix].lock, flags);
|
write_lock_irq(&table->rwlock);
|
||||||
}
|
}
|
||||||
|
|
||||||
old_net_dev = table->data_vec[ix].attr.ndev;
|
old_net_dev = table->data_vec[ix].attr.ndev;
|
||||||
|
@ -162,17 +217,6 @@ static int write_gid(struct ib_device *ib_dev, u8 port,
|
||||||
|
|
||||||
table->data_vec[ix].props &= ~GID_TABLE_ENTRY_INVALID;
|
table->data_vec[ix].props &= ~GID_TABLE_ENTRY_INVALID;
|
||||||
|
|
||||||
write_unlock_irqrestore(&table->data_vec[ix].lock, flags);
|
|
||||||
|
|
||||||
if (!ret && rdma_cap_roce_gid_table(ib_dev, port)) {
|
|
||||||
struct ib_event event;
|
|
||||||
|
|
||||||
event.device = ib_dev;
|
|
||||||
event.element.port_num = port;
|
|
||||||
event.event = IB_EVENT_GID_CHANGE;
|
|
||||||
|
|
||||||
ib_dispatch_event(&event);
|
|
||||||
}
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -201,41 +245,58 @@ static int del_gid(struct ib_device *ib_dev, u8 port,
|
||||||
GID_TABLE_WRITE_ACTION_DEL, default_gid);
|
GID_TABLE_WRITE_ACTION_DEL, default_gid);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* rwlock should be read locked */
|
||||||
static int find_gid(struct ib_gid_table *table, const union ib_gid *gid,
|
static int find_gid(struct ib_gid_table *table, const union ib_gid *gid,
|
||||||
const struct ib_gid_attr *val, bool default_gid,
|
const struct ib_gid_attr *val, bool default_gid,
|
||||||
unsigned long mask)
|
unsigned long mask, int *pempty)
|
||||||
{
|
{
|
||||||
int i;
|
int i = 0;
|
||||||
|
int found = -1;
|
||||||
|
int empty = pempty ? -1 : 0;
|
||||||
|
|
||||||
for (i = 0; i < table->sz; i++) {
|
while (i < table->sz && (found < 0 || empty < 0)) {
|
||||||
unsigned long flags;
|
struct ib_gid_table_entry *data = &table->data_vec[i];
|
||||||
struct ib_gid_attr *attr = &table->data_vec[i].attr;
|
struct ib_gid_attr *attr = &data->attr;
|
||||||
|
int curr_index = i;
|
||||||
|
|
||||||
read_lock_irqsave(&table->data_vec[i].lock, flags);
|
i++;
|
||||||
|
|
||||||
if (table->data_vec[i].props & GID_TABLE_ENTRY_INVALID)
|
if (data->props & GID_TABLE_ENTRY_INVALID)
|
||||||
goto next;
|
continue;
|
||||||
|
|
||||||
|
if (empty < 0)
|
||||||
|
if (!memcmp(&data->gid, &zgid, sizeof(*gid)) &&
|
||||||
|
!memcmp(attr, &zattr, sizeof(*attr)) &&
|
||||||
|
!data->props)
|
||||||
|
empty = curr_index;
|
||||||
|
|
||||||
|
if (found >= 0)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (mask & GID_ATTR_FIND_MASK_GID_TYPE &&
|
||||||
|
attr->gid_type != val->gid_type)
|
||||||
|
continue;
|
||||||
|
|
||||||
if (mask & GID_ATTR_FIND_MASK_GID &&
|
if (mask & GID_ATTR_FIND_MASK_GID &&
|
||||||
memcmp(gid, &table->data_vec[i].gid, sizeof(*gid)))
|
memcmp(gid, &data->gid, sizeof(*gid)))
|
||||||
goto next;
|
continue;
|
||||||
|
|
||||||
if (mask & GID_ATTR_FIND_MASK_NETDEV &&
|
if (mask & GID_ATTR_FIND_MASK_NETDEV &&
|
||||||
attr->ndev != val->ndev)
|
attr->ndev != val->ndev)
|
||||||
goto next;
|
continue;
|
||||||
|
|
||||||
if (mask & GID_ATTR_FIND_MASK_DEFAULT &&
|
if (mask & GID_ATTR_FIND_MASK_DEFAULT &&
|
||||||
!!(table->data_vec[i].props & GID_TABLE_ENTRY_DEFAULT) !=
|
!!(data->props & GID_TABLE_ENTRY_DEFAULT) !=
|
||||||
default_gid)
|
default_gid)
|
||||||
goto next;
|
continue;
|
||||||
|
|
||||||
read_unlock_irqrestore(&table->data_vec[i].lock, flags);
|
found = curr_index;
|
||||||
return i;
|
|
||||||
next:
|
|
||||||
read_unlock_irqrestore(&table->data_vec[i].lock, flags);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return -1;
|
if (pempty)
|
||||||
|
*pempty = empty;
|
||||||
|
|
||||||
|
return found;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void make_default_gid(struct net_device *dev, union ib_gid *gid)
|
static void make_default_gid(struct net_device *dev, union ib_gid *gid)
|
||||||
|
@ -252,6 +313,7 @@ int ib_cache_gid_add(struct ib_device *ib_dev, u8 port,
|
||||||
int ix;
|
int ix;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
struct net_device *idev;
|
struct net_device *idev;
|
||||||
|
int empty;
|
||||||
|
|
||||||
table = ports_table[port - rdma_start_port(ib_dev)];
|
table = ports_table[port - rdma_start_port(ib_dev)];
|
||||||
|
|
||||||
|
@ -275,22 +337,25 @@ int ib_cache_gid_add(struct ib_device *ib_dev, u8 port,
|
||||||
}
|
}
|
||||||
|
|
||||||
mutex_lock(&table->lock);
|
mutex_lock(&table->lock);
|
||||||
|
write_lock_irq(&table->rwlock);
|
||||||
|
|
||||||
ix = find_gid(table, gid, attr, false, GID_ATTR_FIND_MASK_GID |
|
ix = find_gid(table, gid, attr, false, GID_ATTR_FIND_MASK_GID |
|
||||||
GID_ATTR_FIND_MASK_NETDEV);
|
GID_ATTR_FIND_MASK_GID_TYPE |
|
||||||
|
GID_ATTR_FIND_MASK_NETDEV, &empty);
|
||||||
if (ix >= 0)
|
if (ix >= 0)
|
||||||
goto out_unlock;
|
goto out_unlock;
|
||||||
|
|
||||||
ix = find_gid(table, &zgid, NULL, false, GID_ATTR_FIND_MASK_GID |
|
if (empty < 0) {
|
||||||
GID_ATTR_FIND_MASK_DEFAULT);
|
|
||||||
if (ix < 0) {
|
|
||||||
ret = -ENOSPC;
|
ret = -ENOSPC;
|
||||||
goto out_unlock;
|
goto out_unlock;
|
||||||
}
|
}
|
||||||
|
|
||||||
add_gid(ib_dev, port, table, ix, gid, attr, false);
|
ret = add_gid(ib_dev, port, table, empty, gid, attr, false);
|
||||||
|
if (!ret)
|
||||||
|
dispatch_gid_change_event(ib_dev, port);
|
||||||
|
|
||||||
out_unlock:
|
out_unlock:
|
||||||
|
write_unlock_irq(&table->rwlock);
|
||||||
mutex_unlock(&table->lock);
|
mutex_unlock(&table->lock);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -305,17 +370,22 @@ int ib_cache_gid_del(struct ib_device *ib_dev, u8 port,
|
||||||
table = ports_table[port - rdma_start_port(ib_dev)];
|
table = ports_table[port - rdma_start_port(ib_dev)];
|
||||||
|
|
||||||
mutex_lock(&table->lock);
|
mutex_lock(&table->lock);
|
||||||
|
write_lock_irq(&table->rwlock);
|
||||||
|
|
||||||
ix = find_gid(table, gid, attr, false,
|
ix = find_gid(table, gid, attr, false,
|
||||||
GID_ATTR_FIND_MASK_GID |
|
GID_ATTR_FIND_MASK_GID |
|
||||||
|
GID_ATTR_FIND_MASK_GID_TYPE |
|
||||||
GID_ATTR_FIND_MASK_NETDEV |
|
GID_ATTR_FIND_MASK_NETDEV |
|
||||||
GID_ATTR_FIND_MASK_DEFAULT);
|
GID_ATTR_FIND_MASK_DEFAULT,
|
||||||
|
NULL);
|
||||||
if (ix < 0)
|
if (ix < 0)
|
||||||
goto out_unlock;
|
goto out_unlock;
|
||||||
|
|
||||||
del_gid(ib_dev, port, table, ix, false);
|
if (!del_gid(ib_dev, port, table, ix, false))
|
||||||
|
dispatch_gid_change_event(ib_dev, port);
|
||||||
|
|
||||||
out_unlock:
|
out_unlock:
|
||||||
|
write_unlock_irq(&table->rwlock);
|
||||||
mutex_unlock(&table->lock);
|
mutex_unlock(&table->lock);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -326,16 +396,24 @@ int ib_cache_gid_del_all_netdev_gids(struct ib_device *ib_dev, u8 port,
|
||||||
struct ib_gid_table **ports_table = ib_dev->cache.gid_cache;
|
struct ib_gid_table **ports_table = ib_dev->cache.gid_cache;
|
||||||
struct ib_gid_table *table;
|
struct ib_gid_table *table;
|
||||||
int ix;
|
int ix;
|
||||||
|
bool deleted = false;
|
||||||
|
|
||||||
table = ports_table[port - rdma_start_port(ib_dev)];
|
table = ports_table[port - rdma_start_port(ib_dev)];
|
||||||
|
|
||||||
mutex_lock(&table->lock);
|
mutex_lock(&table->lock);
|
||||||
|
write_lock_irq(&table->rwlock);
|
||||||
|
|
||||||
for (ix = 0; ix < table->sz; ix++)
|
for (ix = 0; ix < table->sz; ix++)
|
||||||
if (table->data_vec[ix].attr.ndev == ndev)
|
if (table->data_vec[ix].attr.ndev == ndev)
|
||||||
del_gid(ib_dev, port, table, ix, false);
|
if (!del_gid(ib_dev, port, table, ix, false))
|
||||||
|
deleted = true;
|
||||||
|
|
||||||
|
write_unlock_irq(&table->rwlock);
|
||||||
mutex_unlock(&table->lock);
|
mutex_unlock(&table->lock);
|
||||||
|
|
||||||
|
if (deleted)
|
||||||
|
dispatch_gid_change_event(ib_dev, port);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -344,18 +422,14 @@ static int __ib_cache_gid_get(struct ib_device *ib_dev, u8 port, int index,
|
||||||
{
|
{
|
||||||
struct ib_gid_table **ports_table = ib_dev->cache.gid_cache;
|
struct ib_gid_table **ports_table = ib_dev->cache.gid_cache;
|
||||||
struct ib_gid_table *table;
|
struct ib_gid_table *table;
|
||||||
unsigned long flags;
|
|
||||||
|
|
||||||
table = ports_table[port - rdma_start_port(ib_dev)];
|
table = ports_table[port - rdma_start_port(ib_dev)];
|
||||||
|
|
||||||
if (index < 0 || index >= table->sz)
|
if (index < 0 || index >= table->sz)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
read_lock_irqsave(&table->data_vec[index].lock, flags);
|
if (table->data_vec[index].props & GID_TABLE_ENTRY_INVALID)
|
||||||
if (table->data_vec[index].props & GID_TABLE_ENTRY_INVALID) {
|
|
||||||
read_unlock_irqrestore(&table->data_vec[index].lock, flags);
|
|
||||||
return -EAGAIN;
|
return -EAGAIN;
|
||||||
}
|
|
||||||
|
|
||||||
memcpy(gid, &table->data_vec[index].gid, sizeof(*gid));
|
memcpy(gid, &table->data_vec[index].gid, sizeof(*gid));
|
||||||
if (attr) {
|
if (attr) {
|
||||||
|
@ -364,7 +438,6 @@ static int __ib_cache_gid_get(struct ib_device *ib_dev, u8 port, int index,
|
||||||
dev_hold(attr->ndev);
|
dev_hold(attr->ndev);
|
||||||
}
|
}
|
||||||
|
|
||||||
read_unlock_irqrestore(&table->data_vec[index].lock, flags);
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -378,17 +451,21 @@ static int _ib_cache_gid_table_find(struct ib_device *ib_dev,
|
||||||
struct ib_gid_table *table;
|
struct ib_gid_table *table;
|
||||||
u8 p;
|
u8 p;
|
||||||
int local_index;
|
int local_index;
|
||||||
|
unsigned long flags;
|
||||||
|
|
||||||
for (p = 0; p < ib_dev->phys_port_cnt; p++) {
|
for (p = 0; p < ib_dev->phys_port_cnt; p++) {
|
||||||
table = ports_table[p];
|
table = ports_table[p];
|
||||||
local_index = find_gid(table, gid, val, false, mask);
|
read_lock_irqsave(&table->rwlock, flags);
|
||||||
|
local_index = find_gid(table, gid, val, false, mask, NULL);
|
||||||
if (local_index >= 0) {
|
if (local_index >= 0) {
|
||||||
if (index)
|
if (index)
|
||||||
*index = local_index;
|
*index = local_index;
|
||||||
if (port)
|
if (port)
|
||||||
*port = p + rdma_start_port(ib_dev);
|
*port = p + rdma_start_port(ib_dev);
|
||||||
|
read_unlock_irqrestore(&table->rwlock, flags);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
read_unlock_irqrestore(&table->rwlock, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
return -ENOENT;
|
return -ENOENT;
|
||||||
|
@ -396,11 +473,13 @@ static int _ib_cache_gid_table_find(struct ib_device *ib_dev,
|
||||||
|
|
||||||
static int ib_cache_gid_find(struct ib_device *ib_dev,
|
static int ib_cache_gid_find(struct ib_device *ib_dev,
|
||||||
const union ib_gid *gid,
|
const union ib_gid *gid,
|
||||||
|
enum ib_gid_type gid_type,
|
||||||
struct net_device *ndev, u8 *port,
|
struct net_device *ndev, u8 *port,
|
||||||
u16 *index)
|
u16 *index)
|
||||||
{
|
{
|
||||||
unsigned long mask = GID_ATTR_FIND_MASK_GID;
|
unsigned long mask = GID_ATTR_FIND_MASK_GID |
|
||||||
struct ib_gid_attr gid_attr_val = {.ndev = ndev};
|
GID_ATTR_FIND_MASK_GID_TYPE;
|
||||||
|
struct ib_gid_attr gid_attr_val = {.ndev = ndev, .gid_type = gid_type};
|
||||||
|
|
||||||
if (ndev)
|
if (ndev)
|
||||||
mask |= GID_ATTR_FIND_MASK_NETDEV;
|
mask |= GID_ATTR_FIND_MASK_NETDEV;
|
||||||
|
@ -411,14 +490,17 @@ static int ib_cache_gid_find(struct ib_device *ib_dev,
|
||||||
|
|
||||||
int ib_find_cached_gid_by_port(struct ib_device *ib_dev,
|
int ib_find_cached_gid_by_port(struct ib_device *ib_dev,
|
||||||
const union ib_gid *gid,
|
const union ib_gid *gid,
|
||||||
|
enum ib_gid_type gid_type,
|
||||||
u8 port, struct net_device *ndev,
|
u8 port, struct net_device *ndev,
|
||||||
u16 *index)
|
u16 *index)
|
||||||
{
|
{
|
||||||
int local_index;
|
int local_index;
|
||||||
struct ib_gid_table **ports_table = ib_dev->cache.gid_cache;
|
struct ib_gid_table **ports_table = ib_dev->cache.gid_cache;
|
||||||
struct ib_gid_table *table;
|
struct ib_gid_table *table;
|
||||||
unsigned long mask = GID_ATTR_FIND_MASK_GID;
|
unsigned long mask = GID_ATTR_FIND_MASK_GID |
|
||||||
struct ib_gid_attr val = {.ndev = ndev};
|
GID_ATTR_FIND_MASK_GID_TYPE;
|
||||||
|
struct ib_gid_attr val = {.ndev = ndev, .gid_type = gid_type};
|
||||||
|
unsigned long flags;
|
||||||
|
|
||||||
if (port < rdma_start_port(ib_dev) ||
|
if (port < rdma_start_port(ib_dev) ||
|
||||||
port > rdma_end_port(ib_dev))
|
port > rdma_end_port(ib_dev))
|
||||||
|
@ -429,13 +511,16 @@ int ib_find_cached_gid_by_port(struct ib_device *ib_dev,
|
||||||
if (ndev)
|
if (ndev)
|
||||||
mask |= GID_ATTR_FIND_MASK_NETDEV;
|
mask |= GID_ATTR_FIND_MASK_NETDEV;
|
||||||
|
|
||||||
local_index = find_gid(table, gid, &val, false, mask);
|
read_lock_irqsave(&table->rwlock, flags);
|
||||||
|
local_index = find_gid(table, gid, &val, false, mask, NULL);
|
||||||
if (local_index >= 0) {
|
if (local_index >= 0) {
|
||||||
if (index)
|
if (index)
|
||||||
*index = local_index;
|
*index = local_index;
|
||||||
|
read_unlock_irqrestore(&table->rwlock, flags);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
read_unlock_irqrestore(&table->rwlock, flags);
|
||||||
return -ENOENT;
|
return -ENOENT;
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(ib_find_cached_gid_by_port);
|
EXPORT_SYMBOL(ib_find_cached_gid_by_port);
|
||||||
|
@ -472,6 +557,7 @@ static int ib_cache_gid_find_by_filter(struct ib_device *ib_dev,
|
||||||
struct ib_gid_table **ports_table = ib_dev->cache.gid_cache;
|
struct ib_gid_table **ports_table = ib_dev->cache.gid_cache;
|
||||||
struct ib_gid_table *table;
|
struct ib_gid_table *table;
|
||||||
unsigned int i;
|
unsigned int i;
|
||||||
|
unsigned long flags;
|
||||||
bool found = false;
|
bool found = false;
|
||||||
|
|
||||||
if (!ports_table)
|
if (!ports_table)
|
||||||
|
@ -484,11 +570,10 @@ static int ib_cache_gid_find_by_filter(struct ib_device *ib_dev,
|
||||||
|
|
||||||
table = ports_table[port - rdma_start_port(ib_dev)];
|
table = ports_table[port - rdma_start_port(ib_dev)];
|
||||||
|
|
||||||
|
read_lock_irqsave(&table->rwlock, flags);
|
||||||
for (i = 0; i < table->sz; i++) {
|
for (i = 0; i < table->sz; i++) {
|
||||||
struct ib_gid_attr attr;
|
struct ib_gid_attr attr;
|
||||||
unsigned long flags;
|
|
||||||
|
|
||||||
read_lock_irqsave(&table->data_vec[i].lock, flags);
|
|
||||||
if (table->data_vec[i].props & GID_TABLE_ENTRY_INVALID)
|
if (table->data_vec[i].props & GID_TABLE_ENTRY_INVALID)
|
||||||
goto next;
|
goto next;
|
||||||
|
|
||||||
|
@ -501,11 +586,10 @@ static int ib_cache_gid_find_by_filter(struct ib_device *ib_dev,
|
||||||
found = true;
|
found = true;
|
||||||
|
|
||||||
next:
|
next:
|
||||||
read_unlock_irqrestore(&table->data_vec[i].lock, flags);
|
|
||||||
|
|
||||||
if (found)
|
if (found)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
read_unlock_irqrestore(&table->rwlock, flags);
|
||||||
|
|
||||||
if (!found)
|
if (!found)
|
||||||
return -ENOENT;
|
return -ENOENT;
|
||||||
|
@ -517,9 +601,9 @@ static int ib_cache_gid_find_by_filter(struct ib_device *ib_dev,
|
||||||
|
|
||||||
static struct ib_gid_table *alloc_gid_table(int sz)
|
static struct ib_gid_table *alloc_gid_table(int sz)
|
||||||
{
|
{
|
||||||
unsigned int i;
|
|
||||||
struct ib_gid_table *table =
|
struct ib_gid_table *table =
|
||||||
kzalloc(sizeof(struct ib_gid_table), GFP_KERNEL);
|
kzalloc(sizeof(struct ib_gid_table), GFP_KERNEL);
|
||||||
|
|
||||||
if (!table)
|
if (!table)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
|
@ -530,9 +614,7 @@ static struct ib_gid_table *alloc_gid_table(int sz)
|
||||||
mutex_init(&table->lock);
|
mutex_init(&table->lock);
|
||||||
|
|
||||||
table->sz = sz;
|
table->sz = sz;
|
||||||
|
rwlock_init(&table->rwlock);
|
||||||
for (i = 0; i < sz; i++)
|
|
||||||
rwlock_init(&table->data_vec[i].lock);
|
|
||||||
|
|
||||||
return table;
|
return table;
|
||||||
|
|
||||||
|
@ -553,30 +635,37 @@ static void cleanup_gid_table_port(struct ib_device *ib_dev, u8 port,
|
||||||
struct ib_gid_table *table)
|
struct ib_gid_table *table)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
|
bool deleted = false;
|
||||||
|
|
||||||
if (!table)
|
if (!table)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
write_lock_irq(&table->rwlock);
|
||||||
for (i = 0; i < table->sz; ++i) {
|
for (i = 0; i < table->sz; ++i) {
|
||||||
if (memcmp(&table->data_vec[i].gid, &zgid,
|
if (memcmp(&table->data_vec[i].gid, &zgid,
|
||||||
sizeof(table->data_vec[i].gid)))
|
sizeof(table->data_vec[i].gid)))
|
||||||
del_gid(ib_dev, port, table, i,
|
if (!del_gid(ib_dev, port, table, i,
|
||||||
table->data_vec[i].props &
|
table->data_vec[i].props &
|
||||||
GID_ATTR_FIND_MASK_DEFAULT);
|
GID_ATTR_FIND_MASK_DEFAULT))
|
||||||
|
deleted = true;
|
||||||
}
|
}
|
||||||
|
write_unlock_irq(&table->rwlock);
|
||||||
|
|
||||||
|
if (deleted)
|
||||||
|
dispatch_gid_change_event(ib_dev, port);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ib_cache_gid_set_default_gid(struct ib_device *ib_dev, u8 port,
|
void ib_cache_gid_set_default_gid(struct ib_device *ib_dev, u8 port,
|
||||||
struct net_device *ndev,
|
struct net_device *ndev,
|
||||||
|
unsigned long gid_type_mask,
|
||||||
enum ib_cache_gid_default_mode mode)
|
enum ib_cache_gid_default_mode mode)
|
||||||
{
|
{
|
||||||
struct ib_gid_table **ports_table = ib_dev->cache.gid_cache;
|
struct ib_gid_table **ports_table = ib_dev->cache.gid_cache;
|
||||||
union ib_gid gid;
|
union ib_gid gid;
|
||||||
struct ib_gid_attr gid_attr;
|
struct ib_gid_attr gid_attr;
|
||||||
|
struct ib_gid_attr zattr_type = zattr;
|
||||||
struct ib_gid_table *table;
|
struct ib_gid_table *table;
|
||||||
int ix;
|
unsigned int gid_type;
|
||||||
union ib_gid current_gid;
|
|
||||||
struct ib_gid_attr current_gid_attr = {};
|
|
||||||
|
|
||||||
table = ports_table[port - rdma_start_port(ib_dev)];
|
table = ports_table[port - rdma_start_port(ib_dev)];
|
||||||
|
|
||||||
|
@ -584,46 +673,82 @@ void ib_cache_gid_set_default_gid(struct ib_device *ib_dev, u8 port,
|
||||||
memset(&gid_attr, 0, sizeof(gid_attr));
|
memset(&gid_attr, 0, sizeof(gid_attr));
|
||||||
gid_attr.ndev = ndev;
|
gid_attr.ndev = ndev;
|
||||||
|
|
||||||
mutex_lock(&table->lock);
|
for (gid_type = 0; gid_type < IB_GID_TYPE_SIZE; ++gid_type) {
|
||||||
ix = find_gid(table, NULL, NULL, true, GID_ATTR_FIND_MASK_DEFAULT);
|
int ix;
|
||||||
|
union ib_gid current_gid;
|
||||||
|
struct ib_gid_attr current_gid_attr = {};
|
||||||
|
|
||||||
/* Coudn't find default GID location */
|
if (1UL << gid_type & ~gid_type_mask)
|
||||||
WARN_ON(ix < 0);
|
continue;
|
||||||
|
|
||||||
if (!__ib_cache_gid_get(ib_dev, port, ix,
|
gid_attr.gid_type = gid_type;
|
||||||
¤t_gid, ¤t_gid_attr) &&
|
|
||||||
mode == IB_CACHE_GID_DEFAULT_MODE_SET &&
|
|
||||||
!memcmp(&gid, ¤t_gid, sizeof(gid)) &&
|
|
||||||
!memcmp(&gid_attr, ¤t_gid_attr, sizeof(gid_attr)))
|
|
||||||
goto unlock;
|
|
||||||
|
|
||||||
if ((memcmp(¤t_gid, &zgid, sizeof(current_gid)) ||
|
mutex_lock(&table->lock);
|
||||||
memcmp(¤t_gid_attr, &zattr,
|
write_lock_irq(&table->rwlock);
|
||||||
sizeof(current_gid_attr))) &&
|
ix = find_gid(table, NULL, &gid_attr, true,
|
||||||
del_gid(ib_dev, port, table, ix, true)) {
|
GID_ATTR_FIND_MASK_GID_TYPE |
|
||||||
pr_warn("ib_cache_gid: can't delete index %d for default gid %pI6\n",
|
GID_ATTR_FIND_MASK_DEFAULT,
|
||||||
ix, gid.raw);
|
NULL);
|
||||||
goto unlock;
|
|
||||||
|
/* Coudn't find default GID location */
|
||||||
|
WARN_ON(ix < 0);
|
||||||
|
|
||||||
|
zattr_type.gid_type = gid_type;
|
||||||
|
|
||||||
|
if (!__ib_cache_gid_get(ib_dev, port, ix,
|
||||||
|
¤t_gid, ¤t_gid_attr) &&
|
||||||
|
mode == IB_CACHE_GID_DEFAULT_MODE_SET &&
|
||||||
|
!memcmp(&gid, ¤t_gid, sizeof(gid)) &&
|
||||||
|
!memcmp(&gid_attr, ¤t_gid_attr, sizeof(gid_attr)))
|
||||||
|
goto release;
|
||||||
|
|
||||||
|
if (memcmp(¤t_gid, &zgid, sizeof(current_gid)) ||
|
||||||
|
memcmp(¤t_gid_attr, &zattr_type,
|
||||||
|
sizeof(current_gid_attr))) {
|
||||||
|
if (del_gid(ib_dev, port, table, ix, true)) {
|
||||||
|
pr_warn("ib_cache_gid: can't delete index %d for default gid %pI6\n",
|
||||||
|
ix, gid.raw);
|
||||||
|
goto release;
|
||||||
|
} else {
|
||||||
|
dispatch_gid_change_event(ib_dev, port);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mode == IB_CACHE_GID_DEFAULT_MODE_SET) {
|
||||||
|
if (add_gid(ib_dev, port, table, ix, &gid, &gid_attr, true))
|
||||||
|
pr_warn("ib_cache_gid: unable to add default gid %pI6\n",
|
||||||
|
gid.raw);
|
||||||
|
else
|
||||||
|
dispatch_gid_change_event(ib_dev, port);
|
||||||
|
}
|
||||||
|
|
||||||
|
release:
|
||||||
|
if (current_gid_attr.ndev)
|
||||||
|
dev_put(current_gid_attr.ndev);
|
||||||
|
write_unlock_irq(&table->rwlock);
|
||||||
|
mutex_unlock(&table->lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mode == IB_CACHE_GID_DEFAULT_MODE_SET)
|
|
||||||
if (add_gid(ib_dev, port, table, ix, &gid, &gid_attr, true))
|
|
||||||
pr_warn("ib_cache_gid: unable to add default gid %pI6\n",
|
|
||||||
gid.raw);
|
|
||||||
|
|
||||||
unlock:
|
|
||||||
if (current_gid_attr.ndev)
|
|
||||||
dev_put(current_gid_attr.ndev);
|
|
||||||
mutex_unlock(&table->lock);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int gid_table_reserve_default(struct ib_device *ib_dev, u8 port,
|
static int gid_table_reserve_default(struct ib_device *ib_dev, u8 port,
|
||||||
struct ib_gid_table *table)
|
struct ib_gid_table *table)
|
||||||
{
|
{
|
||||||
if (rdma_protocol_roce(ib_dev, port)) {
|
unsigned int i;
|
||||||
struct ib_gid_table_entry *entry = &table->data_vec[0];
|
unsigned long roce_gid_type_mask;
|
||||||
|
unsigned int num_default_gids;
|
||||||
|
unsigned int current_gid = 0;
|
||||||
|
|
||||||
|
roce_gid_type_mask = roce_gid_type_mask_support(ib_dev, port);
|
||||||
|
num_default_gids = hweight_long(roce_gid_type_mask);
|
||||||
|
for (i = 0; i < num_default_gids && i < table->sz; i++) {
|
||||||
|
struct ib_gid_table_entry *entry =
|
||||||
|
&table->data_vec[i];
|
||||||
|
|
||||||
entry->props |= GID_TABLE_ENTRY_DEFAULT;
|
entry->props |= GID_TABLE_ENTRY_DEFAULT;
|
||||||
|
current_gid = find_next_bit(&roce_gid_type_mask,
|
||||||
|
BITS_PER_LONG,
|
||||||
|
current_gid);
|
||||||
|
entry->attr.gid_type = current_gid++;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -728,20 +853,30 @@ int ib_get_cached_gid(struct ib_device *device,
|
||||||
union ib_gid *gid,
|
union ib_gid *gid,
|
||||||
struct ib_gid_attr *gid_attr)
|
struct ib_gid_attr *gid_attr)
|
||||||
{
|
{
|
||||||
|
int res;
|
||||||
|
unsigned long flags;
|
||||||
|
struct ib_gid_table **ports_table = device->cache.gid_cache;
|
||||||
|
struct ib_gid_table *table = ports_table[port_num - rdma_start_port(device)];
|
||||||
|
|
||||||
if (port_num < rdma_start_port(device) || port_num > rdma_end_port(device))
|
if (port_num < rdma_start_port(device) || port_num > rdma_end_port(device))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
return __ib_cache_gid_get(device, port_num, index, gid, gid_attr);
|
read_lock_irqsave(&table->rwlock, flags);
|
||||||
|
res = __ib_cache_gid_get(device, port_num, index, gid, gid_attr);
|
||||||
|
read_unlock_irqrestore(&table->rwlock, flags);
|
||||||
|
|
||||||
|
return res;
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(ib_get_cached_gid);
|
EXPORT_SYMBOL(ib_get_cached_gid);
|
||||||
|
|
||||||
int ib_find_cached_gid(struct ib_device *device,
|
int ib_find_cached_gid(struct ib_device *device,
|
||||||
const union ib_gid *gid,
|
const union ib_gid *gid,
|
||||||
|
enum ib_gid_type gid_type,
|
||||||
struct net_device *ndev,
|
struct net_device *ndev,
|
||||||
u8 *port_num,
|
u8 *port_num,
|
||||||
u16 *index)
|
u16 *index)
|
||||||
{
|
{
|
||||||
return ib_cache_gid_find(device, gid, ndev, port_num, index);
|
return ib_cache_gid_find(device, gid, gid_type, ndev, port_num, index);
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(ib_find_cached_gid);
|
EXPORT_SYMBOL(ib_find_cached_gid);
|
||||||
|
|
||||||
|
@ -956,10 +1091,12 @@ static void ib_cache_update(struct ib_device *device,
|
||||||
|
|
||||||
device->cache.pkey_cache[port - rdma_start_port(device)] = pkey_cache;
|
device->cache.pkey_cache[port - rdma_start_port(device)] = pkey_cache;
|
||||||
if (!use_roce_gid_table) {
|
if (!use_roce_gid_table) {
|
||||||
|
write_lock(&table->rwlock);
|
||||||
for (i = 0; i < gid_cache->table_len; i++) {
|
for (i = 0; i < gid_cache->table_len; i++) {
|
||||||
modify_gid(device, port, table, i, gid_cache->table + i,
|
modify_gid(device, port, table, i, gid_cache->table + i,
|
||||||
&zattr, false);
|
&zattr, false);
|
||||||
}
|
}
|
||||||
|
write_unlock(&table->rwlock);
|
||||||
}
|
}
|
||||||
|
|
||||||
device->cache.lmc_cache[port - rdma_start_port(device)] = tprops->lmc;
|
device->cache.lmc_cache[port - rdma_start_port(device)] = tprops->lmc;
|
||||||
|
|
|
@ -364,7 +364,7 @@ static int cm_init_av_by_path(struct ib_sa_path_rec *path, struct cm_av *av)
|
||||||
read_lock_irqsave(&cm.device_lock, flags);
|
read_lock_irqsave(&cm.device_lock, flags);
|
||||||
list_for_each_entry(cm_dev, &cm.device_list, list) {
|
list_for_each_entry(cm_dev, &cm.device_list, list) {
|
||||||
if (!ib_find_cached_gid(cm_dev->ib_device, &path->sgid,
|
if (!ib_find_cached_gid(cm_dev->ib_device, &path->sgid,
|
||||||
ndev, &p, NULL)) {
|
path->gid_type, ndev, &p, NULL)) {
|
||||||
port = cm_dev->port[p-1];
|
port = cm_dev->port[p-1];
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -782,11 +782,11 @@ static void cm_enter_timewait(struct cm_id_private *cm_id_priv)
|
||||||
wait_time = cm_convert_to_ms(cm_id_priv->av.timeout);
|
wait_time = cm_convert_to_ms(cm_id_priv->av.timeout);
|
||||||
|
|
||||||
/* Check if the device started its remove_one */
|
/* Check if the device started its remove_one */
|
||||||
spin_lock_irq(&cm.lock);
|
spin_lock_irqsave(&cm.lock, flags);
|
||||||
if (!cm_dev->going_down)
|
if (!cm_dev->going_down)
|
||||||
queue_delayed_work(cm.wq, &cm_id_priv->timewait_info->work.work,
|
queue_delayed_work(cm.wq, &cm_id_priv->timewait_info->work.work,
|
||||||
msecs_to_jiffies(wait_time));
|
msecs_to_jiffies(wait_time));
|
||||||
spin_unlock_irq(&cm.lock);
|
spin_unlock_irqrestore(&cm.lock, flags);
|
||||||
|
|
||||||
cm_id_priv->timewait_info = NULL;
|
cm_id_priv->timewait_info = NULL;
|
||||||
}
|
}
|
||||||
|
@ -1600,6 +1600,8 @@ static int cm_req_handler(struct cm_work *work)
|
||||||
struct ib_cm_id *cm_id;
|
struct ib_cm_id *cm_id;
|
||||||
struct cm_id_private *cm_id_priv, *listen_cm_id_priv;
|
struct cm_id_private *cm_id_priv, *listen_cm_id_priv;
|
||||||
struct cm_req_msg *req_msg;
|
struct cm_req_msg *req_msg;
|
||||||
|
union ib_gid gid;
|
||||||
|
struct ib_gid_attr gid_attr;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
req_msg = (struct cm_req_msg *)work->mad_recv_wc->recv_buf.mad;
|
req_msg = (struct cm_req_msg *)work->mad_recv_wc->recv_buf.mad;
|
||||||
|
@ -1639,11 +1641,31 @@ static int cm_req_handler(struct cm_work *work)
|
||||||
cm_format_paths_from_req(req_msg, &work->path[0], &work->path[1]);
|
cm_format_paths_from_req(req_msg, &work->path[0], &work->path[1]);
|
||||||
|
|
||||||
memcpy(work->path[0].dmac, cm_id_priv->av.ah_attr.dmac, ETH_ALEN);
|
memcpy(work->path[0].dmac, cm_id_priv->av.ah_attr.dmac, ETH_ALEN);
|
||||||
ret = cm_init_av_by_path(&work->path[0], &cm_id_priv->av);
|
work->path[0].hop_limit = cm_id_priv->av.ah_attr.grh.hop_limit;
|
||||||
|
ret = ib_get_cached_gid(work->port->cm_dev->ib_device,
|
||||||
|
work->port->port_num,
|
||||||
|
cm_id_priv->av.ah_attr.grh.sgid_index,
|
||||||
|
&gid, &gid_attr);
|
||||||
|
if (!ret) {
|
||||||
|
if (gid_attr.ndev) {
|
||||||
|
work->path[0].ifindex = gid_attr.ndev->ifindex;
|
||||||
|
work->path[0].net = dev_net(gid_attr.ndev);
|
||||||
|
dev_put(gid_attr.ndev);
|
||||||
|
}
|
||||||
|
work->path[0].gid_type = gid_attr.gid_type;
|
||||||
|
ret = cm_init_av_by_path(&work->path[0], &cm_id_priv->av);
|
||||||
|
}
|
||||||
if (ret) {
|
if (ret) {
|
||||||
ib_get_cached_gid(work->port->cm_dev->ib_device,
|
int err = ib_get_cached_gid(work->port->cm_dev->ib_device,
|
||||||
work->port->port_num, 0, &work->path[0].sgid,
|
work->port->port_num, 0,
|
||||||
NULL);
|
&work->path[0].sgid,
|
||||||
|
&gid_attr);
|
||||||
|
if (!err && gid_attr.ndev) {
|
||||||
|
work->path[0].ifindex = gid_attr.ndev->ifindex;
|
||||||
|
work->path[0].net = dev_net(gid_attr.ndev);
|
||||||
|
dev_put(gid_attr.ndev);
|
||||||
|
}
|
||||||
|
work->path[0].gid_type = gid_attr.gid_type;
|
||||||
ib_send_cm_rej(cm_id, IB_CM_REJ_INVALID_GID,
|
ib_send_cm_rej(cm_id, IB_CM_REJ_INVALID_GID,
|
||||||
&work->path[0].sgid, sizeof work->path[0].sgid,
|
&work->path[0].sgid, sizeof work->path[0].sgid,
|
||||||
NULL, 0);
|
NULL, 0);
|
||||||
|
@ -3482,6 +3504,7 @@ int ib_cm_notify(struct ib_cm_id *cm_id, enum ib_event_type event)
|
||||||
EXPORT_SYMBOL(ib_cm_notify);
|
EXPORT_SYMBOL(ib_cm_notify);
|
||||||
|
|
||||||
static void cm_recv_handler(struct ib_mad_agent *mad_agent,
|
static void cm_recv_handler(struct ib_mad_agent *mad_agent,
|
||||||
|
struct ib_mad_send_buf *send_buf,
|
||||||
struct ib_mad_recv_wc *mad_recv_wc)
|
struct ib_mad_recv_wc *mad_recv_wc)
|
||||||
{
|
{
|
||||||
struct cm_port *port = mad_agent->context;
|
struct cm_port *port = mad_agent->context;
|
||||||
|
@ -3731,16 +3754,6 @@ int ib_cm_init_qp_attr(struct ib_cm_id *cm_id,
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(ib_cm_init_qp_attr);
|
EXPORT_SYMBOL(ib_cm_init_qp_attr);
|
||||||
|
|
||||||
static void cm_get_ack_delay(struct cm_device *cm_dev)
|
|
||||||
{
|
|
||||||
struct ib_device_attr attr;
|
|
||||||
|
|
||||||
if (ib_query_device(cm_dev->ib_device, &attr))
|
|
||||||
cm_dev->ack_delay = 0; /* acks will rely on packet life time */
|
|
||||||
else
|
|
||||||
cm_dev->ack_delay = attr.local_ca_ack_delay;
|
|
||||||
}
|
|
||||||
|
|
||||||
static ssize_t cm_show_counter(struct kobject *obj, struct attribute *attr,
|
static ssize_t cm_show_counter(struct kobject *obj, struct attribute *attr,
|
||||||
char *buf)
|
char *buf)
|
||||||
{
|
{
|
||||||
|
@ -3852,7 +3865,7 @@ static void cm_add_one(struct ib_device *ib_device)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
cm_dev->ib_device = ib_device;
|
cm_dev->ib_device = ib_device;
|
||||||
cm_get_ack_delay(cm_dev);
|
cm_dev->ack_delay = ib_device->attrs.local_ca_ack_delay;
|
||||||
cm_dev->going_down = 0;
|
cm_dev->going_down = 0;
|
||||||
cm_dev->device = device_create(&cm_class, &ib_device->dev,
|
cm_dev->device = device_create(&cm_class, &ib_device->dev,
|
||||||
MKDEV(0, 0), NULL,
|
MKDEV(0, 0), NULL,
|
||||||
|
|
|
@ -38,6 +38,7 @@
|
||||||
#include <linux/in6.h>
|
#include <linux/in6.h>
|
||||||
#include <linux/mutex.h>
|
#include <linux/mutex.h>
|
||||||
#include <linux/random.h>
|
#include <linux/random.h>
|
||||||
|
#include <linux/igmp.h>
|
||||||
#include <linux/idr.h>
|
#include <linux/idr.h>
|
||||||
#include <linux/inetdevice.h>
|
#include <linux/inetdevice.h>
|
||||||
#include <linux/slab.h>
|
#include <linux/slab.h>
|
||||||
|
@ -60,6 +61,8 @@
|
||||||
#include <rdma/ib_sa.h>
|
#include <rdma/ib_sa.h>
|
||||||
#include <rdma/iw_cm.h>
|
#include <rdma/iw_cm.h>
|
||||||
|
|
||||||
|
#include "core_priv.h"
|
||||||
|
|
||||||
MODULE_AUTHOR("Sean Hefty");
|
MODULE_AUTHOR("Sean Hefty");
|
||||||
MODULE_DESCRIPTION("Generic RDMA CM Agent");
|
MODULE_DESCRIPTION("Generic RDMA CM Agent");
|
||||||
MODULE_LICENSE("Dual BSD/GPL");
|
MODULE_LICENSE("Dual BSD/GPL");
|
||||||
|
@ -150,6 +153,7 @@ struct cma_device {
|
||||||
struct completion comp;
|
struct completion comp;
|
||||||
atomic_t refcount;
|
atomic_t refcount;
|
||||||
struct list_head id_list;
|
struct list_head id_list;
|
||||||
|
enum ib_gid_type *default_gid_type;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct rdma_bind_list {
|
struct rdma_bind_list {
|
||||||
|
@ -185,6 +189,67 @@ enum {
|
||||||
CMA_OPTION_AFONLY,
|
CMA_OPTION_AFONLY,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
void cma_ref_dev(struct cma_device *cma_dev)
|
||||||
|
{
|
||||||
|
atomic_inc(&cma_dev->refcount);
|
||||||
|
}
|
||||||
|
|
||||||
|
struct cma_device *cma_enum_devices_by_ibdev(cma_device_filter filter,
|
||||||
|
void *cookie)
|
||||||
|
{
|
||||||
|
struct cma_device *cma_dev;
|
||||||
|
struct cma_device *found_cma_dev = NULL;
|
||||||
|
|
||||||
|
mutex_lock(&lock);
|
||||||
|
|
||||||
|
list_for_each_entry(cma_dev, &dev_list, list)
|
||||||
|
if (filter(cma_dev->device, cookie)) {
|
||||||
|
found_cma_dev = cma_dev;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (found_cma_dev)
|
||||||
|
cma_ref_dev(found_cma_dev);
|
||||||
|
mutex_unlock(&lock);
|
||||||
|
return found_cma_dev;
|
||||||
|
}
|
||||||
|
|
||||||
|
int cma_get_default_gid_type(struct cma_device *cma_dev,
|
||||||
|
unsigned int port)
|
||||||
|
{
|
||||||
|
if (port < rdma_start_port(cma_dev->device) ||
|
||||||
|
port > rdma_end_port(cma_dev->device))
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
return cma_dev->default_gid_type[port - rdma_start_port(cma_dev->device)];
|
||||||
|
}
|
||||||
|
|
||||||
|
int cma_set_default_gid_type(struct cma_device *cma_dev,
|
||||||
|
unsigned int port,
|
||||||
|
enum ib_gid_type default_gid_type)
|
||||||
|
{
|
||||||
|
unsigned long supported_gids;
|
||||||
|
|
||||||
|
if (port < rdma_start_port(cma_dev->device) ||
|
||||||
|
port > rdma_end_port(cma_dev->device))
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
supported_gids = roce_gid_type_mask_support(cma_dev->device, port);
|
||||||
|
|
||||||
|
if (!(supported_gids & 1 << default_gid_type))
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
cma_dev->default_gid_type[port - rdma_start_port(cma_dev->device)] =
|
||||||
|
default_gid_type;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct ib_device *cma_get_ib_dev(struct cma_device *cma_dev)
|
||||||
|
{
|
||||||
|
return cma_dev->device;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Device removal can occur at anytime, so we need extra handling to
|
* Device removal can occur at anytime, so we need extra handling to
|
||||||
* serialize notifying the user of device removal with other callbacks.
|
* serialize notifying the user of device removal with other callbacks.
|
||||||
|
@ -228,6 +293,7 @@ struct rdma_id_private {
|
||||||
u8 tos;
|
u8 tos;
|
||||||
u8 reuseaddr;
|
u8 reuseaddr;
|
||||||
u8 afonly;
|
u8 afonly;
|
||||||
|
enum ib_gid_type gid_type;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct cma_multicast {
|
struct cma_multicast {
|
||||||
|
@ -239,6 +305,7 @@ struct cma_multicast {
|
||||||
void *context;
|
void *context;
|
||||||
struct sockaddr_storage addr;
|
struct sockaddr_storage addr;
|
||||||
struct kref mcref;
|
struct kref mcref;
|
||||||
|
bool igmp_joined;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct cma_work {
|
struct cma_work {
|
||||||
|
@ -335,18 +402,48 @@ static inline void cma_set_ip_ver(struct cma_hdr *hdr, u8 ip_ver)
|
||||||
hdr->ip_version = (ip_ver << 4) | (hdr->ip_version & 0xF);
|
hdr->ip_version = (ip_ver << 4) | (hdr->ip_version & 0xF);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void cma_attach_to_dev(struct rdma_id_private *id_priv,
|
static int cma_igmp_send(struct net_device *ndev, union ib_gid *mgid, bool join)
|
||||||
struct cma_device *cma_dev)
|
|
||||||
{
|
{
|
||||||
atomic_inc(&cma_dev->refcount);
|
struct in_device *in_dev = NULL;
|
||||||
|
|
||||||
|
if (ndev) {
|
||||||
|
rtnl_lock();
|
||||||
|
in_dev = __in_dev_get_rtnl(ndev);
|
||||||
|
if (in_dev) {
|
||||||
|
if (join)
|
||||||
|
ip_mc_inc_group(in_dev,
|
||||||
|
*(__be32 *)(mgid->raw + 12));
|
||||||
|
else
|
||||||
|
ip_mc_dec_group(in_dev,
|
||||||
|
*(__be32 *)(mgid->raw + 12));
|
||||||
|
}
|
||||||
|
rtnl_unlock();
|
||||||
|
}
|
||||||
|
return (in_dev) ? 0 : -ENODEV;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void _cma_attach_to_dev(struct rdma_id_private *id_priv,
|
||||||
|
struct cma_device *cma_dev)
|
||||||
|
{
|
||||||
|
cma_ref_dev(cma_dev);
|
||||||
id_priv->cma_dev = cma_dev;
|
id_priv->cma_dev = cma_dev;
|
||||||
|
id_priv->gid_type = 0;
|
||||||
id_priv->id.device = cma_dev->device;
|
id_priv->id.device = cma_dev->device;
|
||||||
id_priv->id.route.addr.dev_addr.transport =
|
id_priv->id.route.addr.dev_addr.transport =
|
||||||
rdma_node_get_transport(cma_dev->device->node_type);
|
rdma_node_get_transport(cma_dev->device->node_type);
|
||||||
list_add_tail(&id_priv->list, &cma_dev->id_list);
|
list_add_tail(&id_priv->list, &cma_dev->id_list);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void cma_deref_dev(struct cma_device *cma_dev)
|
static void cma_attach_to_dev(struct rdma_id_private *id_priv,
|
||||||
|
struct cma_device *cma_dev)
|
||||||
|
{
|
||||||
|
_cma_attach_to_dev(id_priv, cma_dev);
|
||||||
|
id_priv->gid_type =
|
||||||
|
cma_dev->default_gid_type[id_priv->id.port_num -
|
||||||
|
rdma_start_port(cma_dev->device)];
|
||||||
|
}
|
||||||
|
|
||||||
|
void cma_deref_dev(struct cma_device *cma_dev)
|
||||||
{
|
{
|
||||||
if (atomic_dec_and_test(&cma_dev->refcount))
|
if (atomic_dec_and_test(&cma_dev->refcount))
|
||||||
complete(&cma_dev->comp);
|
complete(&cma_dev->comp);
|
||||||
|
@ -441,6 +538,7 @@ static int cma_translate_addr(struct sockaddr *addr, struct rdma_dev_addr *dev_a
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int cma_validate_port(struct ib_device *device, u8 port,
|
static inline int cma_validate_port(struct ib_device *device, u8 port,
|
||||||
|
enum ib_gid_type gid_type,
|
||||||
union ib_gid *gid, int dev_type,
|
union ib_gid *gid, int dev_type,
|
||||||
int bound_if_index)
|
int bound_if_index)
|
||||||
{
|
{
|
||||||
|
@ -453,10 +551,25 @@ static inline int cma_validate_port(struct ib_device *device, u8 port,
|
||||||
if ((dev_type != ARPHRD_INFINIBAND) && rdma_protocol_ib(device, port))
|
if ((dev_type != ARPHRD_INFINIBAND) && rdma_protocol_ib(device, port))
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
if (dev_type == ARPHRD_ETHER)
|
if (dev_type == ARPHRD_ETHER && rdma_protocol_roce(device, port)) {
|
||||||
ndev = dev_get_by_index(&init_net, bound_if_index);
|
ndev = dev_get_by_index(&init_net, bound_if_index);
|
||||||
|
if (ndev && ndev->flags & IFF_LOOPBACK) {
|
||||||
|
pr_info("detected loopback device\n");
|
||||||
|
dev_put(ndev);
|
||||||
|
|
||||||
ret = ib_find_cached_gid_by_port(device, gid, port, ndev, NULL);
|
if (!device->get_netdev)
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
|
ndev = device->get_netdev(device, port);
|
||||||
|
if (!ndev)
|
||||||
|
return -ENODEV;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
gid_type = IB_GID_TYPE_IB;
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = ib_find_cached_gid_by_port(device, gid, gid_type, port,
|
||||||
|
ndev, NULL);
|
||||||
|
|
||||||
if (ndev)
|
if (ndev)
|
||||||
dev_put(ndev);
|
dev_put(ndev);
|
||||||
|
@ -490,7 +603,10 @@ static int cma_acquire_dev(struct rdma_id_private *id_priv,
|
||||||
gidp = rdma_protocol_roce(cma_dev->device, port) ?
|
gidp = rdma_protocol_roce(cma_dev->device, port) ?
|
||||||
&iboe_gid : &gid;
|
&iboe_gid : &gid;
|
||||||
|
|
||||||
ret = cma_validate_port(cma_dev->device, port, gidp,
|
ret = cma_validate_port(cma_dev->device, port,
|
||||||
|
rdma_protocol_ib(cma_dev->device, port) ?
|
||||||
|
IB_GID_TYPE_IB :
|
||||||
|
listen_id_priv->gid_type, gidp,
|
||||||
dev_addr->dev_type,
|
dev_addr->dev_type,
|
||||||
dev_addr->bound_dev_if);
|
dev_addr->bound_dev_if);
|
||||||
if (!ret) {
|
if (!ret) {
|
||||||
|
@ -509,8 +625,11 @@ static int cma_acquire_dev(struct rdma_id_private *id_priv,
|
||||||
gidp = rdma_protocol_roce(cma_dev->device, port) ?
|
gidp = rdma_protocol_roce(cma_dev->device, port) ?
|
||||||
&iboe_gid : &gid;
|
&iboe_gid : &gid;
|
||||||
|
|
||||||
ret = cma_validate_port(cma_dev->device, port, gidp,
|
ret = cma_validate_port(cma_dev->device, port,
|
||||||
dev_addr->dev_type,
|
rdma_protocol_ib(cma_dev->device, port) ?
|
||||||
|
IB_GID_TYPE_IB :
|
||||||
|
cma_dev->default_gid_type[port - 1],
|
||||||
|
gidp, dev_addr->dev_type,
|
||||||
dev_addr->bound_dev_if);
|
dev_addr->bound_dev_if);
|
||||||
if (!ret) {
|
if (!ret) {
|
||||||
id_priv->id.port_num = port;
|
id_priv->id.port_num = port;
|
||||||
|
@ -1437,8 +1556,24 @@ static void cma_leave_mc_groups(struct rdma_id_private *id_priv)
|
||||||
id_priv->id.port_num)) {
|
id_priv->id.port_num)) {
|
||||||
ib_sa_free_multicast(mc->multicast.ib);
|
ib_sa_free_multicast(mc->multicast.ib);
|
||||||
kfree(mc);
|
kfree(mc);
|
||||||
} else
|
} else {
|
||||||
|
if (mc->igmp_joined) {
|
||||||
|
struct rdma_dev_addr *dev_addr =
|
||||||
|
&id_priv->id.route.addr.dev_addr;
|
||||||
|
struct net_device *ndev = NULL;
|
||||||
|
|
||||||
|
if (dev_addr->bound_dev_if)
|
||||||
|
ndev = dev_get_by_index(&init_net,
|
||||||
|
dev_addr->bound_dev_if);
|
||||||
|
if (ndev) {
|
||||||
|
cma_igmp_send(ndev,
|
||||||
|
&mc->multicast.ib->rec.mgid,
|
||||||
|
false);
|
||||||
|
dev_put(ndev);
|
||||||
|
}
|
||||||
|
}
|
||||||
kref_put(&mc->mcref, release_mc);
|
kref_put(&mc->mcref, release_mc);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1896,7 +2031,6 @@ static int iw_conn_req_handler(struct iw_cm_id *cm_id,
|
||||||
struct rdma_id_private *listen_id, *conn_id;
|
struct rdma_id_private *listen_id, *conn_id;
|
||||||
struct rdma_cm_event event;
|
struct rdma_cm_event event;
|
||||||
int ret;
|
int ret;
|
||||||
struct ib_device_attr attr;
|
|
||||||
struct sockaddr *laddr = (struct sockaddr *)&iw_event->local_addr;
|
struct sockaddr *laddr = (struct sockaddr *)&iw_event->local_addr;
|
||||||
struct sockaddr *raddr = (struct sockaddr *)&iw_event->remote_addr;
|
struct sockaddr *raddr = (struct sockaddr *)&iw_event->remote_addr;
|
||||||
|
|
||||||
|
@ -1938,13 +2072,6 @@ static int iw_conn_req_handler(struct iw_cm_id *cm_id,
|
||||||
memcpy(cma_src_addr(conn_id), laddr, rdma_addr_size(laddr));
|
memcpy(cma_src_addr(conn_id), laddr, rdma_addr_size(laddr));
|
||||||
memcpy(cma_dst_addr(conn_id), raddr, rdma_addr_size(raddr));
|
memcpy(cma_dst_addr(conn_id), raddr, rdma_addr_size(raddr));
|
||||||
|
|
||||||
ret = ib_query_device(conn_id->id.device, &attr);
|
|
||||||
if (ret) {
|
|
||||||
mutex_unlock(&conn_id->handler_mutex);
|
|
||||||
rdma_destroy_id(new_cm_id);
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
|
|
||||||
memset(&event, 0, sizeof event);
|
memset(&event, 0, sizeof event);
|
||||||
event.event = RDMA_CM_EVENT_CONNECT_REQUEST;
|
event.event = RDMA_CM_EVENT_CONNECT_REQUEST;
|
||||||
event.param.conn.private_data = iw_event->private_data;
|
event.param.conn.private_data = iw_event->private_data;
|
||||||
|
@ -2051,7 +2178,7 @@ static void cma_listen_on_dev(struct rdma_id_private *id_priv,
|
||||||
memcpy(cma_src_addr(dev_id_priv), cma_src_addr(id_priv),
|
memcpy(cma_src_addr(dev_id_priv), cma_src_addr(id_priv),
|
||||||
rdma_addr_size(cma_src_addr(id_priv)));
|
rdma_addr_size(cma_src_addr(id_priv)));
|
||||||
|
|
||||||
cma_attach_to_dev(dev_id_priv, cma_dev);
|
_cma_attach_to_dev(dev_id_priv, cma_dev);
|
||||||
list_add_tail(&dev_id_priv->listen_list, &id_priv->listen_list);
|
list_add_tail(&dev_id_priv->listen_list, &id_priv->listen_list);
|
||||||
atomic_inc(&id_priv->refcount);
|
atomic_inc(&id_priv->refcount);
|
||||||
dev_id_priv->internal_id = 1;
|
dev_id_priv->internal_id = 1;
|
||||||
|
@ -2321,8 +2448,23 @@ static int cma_resolve_iboe_route(struct rdma_id_private *id_priv)
|
||||||
|
|
||||||
if (addr->dev_addr.bound_dev_if) {
|
if (addr->dev_addr.bound_dev_if) {
|
||||||
ndev = dev_get_by_index(&init_net, addr->dev_addr.bound_dev_if);
|
ndev = dev_get_by_index(&init_net, addr->dev_addr.bound_dev_if);
|
||||||
|
if (!ndev)
|
||||||
|
return -ENODEV;
|
||||||
|
|
||||||
|
if (ndev->flags & IFF_LOOPBACK) {
|
||||||
|
dev_put(ndev);
|
||||||
|
if (!id_priv->id.device->get_netdev)
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
|
ndev = id_priv->id.device->get_netdev(id_priv->id.device,
|
||||||
|
id_priv->id.port_num);
|
||||||
|
if (!ndev)
|
||||||
|
return -ENODEV;
|
||||||
|
}
|
||||||
|
|
||||||
route->path_rec->net = &init_net;
|
route->path_rec->net = &init_net;
|
||||||
route->path_rec->ifindex = addr->dev_addr.bound_dev_if;
|
route->path_rec->ifindex = ndev->ifindex;
|
||||||
|
route->path_rec->gid_type = id_priv->gid_type;
|
||||||
}
|
}
|
||||||
if (!ndev) {
|
if (!ndev) {
|
||||||
ret = -ENODEV;
|
ret = -ENODEV;
|
||||||
|
@ -2336,7 +2478,14 @@ static int cma_resolve_iboe_route(struct rdma_id_private *id_priv)
|
||||||
rdma_ip2gid((struct sockaddr *)&id_priv->id.route.addr.dst_addr,
|
rdma_ip2gid((struct sockaddr *)&id_priv->id.route.addr.dst_addr,
|
||||||
&route->path_rec->dgid);
|
&route->path_rec->dgid);
|
||||||
|
|
||||||
route->path_rec->hop_limit = 1;
|
/* Use the hint from IP Stack to select GID Type */
|
||||||
|
if (route->path_rec->gid_type < ib_network_to_gid_type(addr->dev_addr.network))
|
||||||
|
route->path_rec->gid_type = ib_network_to_gid_type(addr->dev_addr.network);
|
||||||
|
if (((struct sockaddr *)&id_priv->id.route.addr.dst_addr)->sa_family != AF_IB)
|
||||||
|
/* TODO: get the hoplimit from the inet/inet6 device */
|
||||||
|
route->path_rec->hop_limit = addr->dev_addr.hoplimit;
|
||||||
|
else
|
||||||
|
route->path_rec->hop_limit = 1;
|
||||||
route->path_rec->reversible = 1;
|
route->path_rec->reversible = 1;
|
||||||
route->path_rec->pkey = cpu_to_be16(0xffff);
|
route->path_rec->pkey = cpu_to_be16(0xffff);
|
||||||
route->path_rec->mtu_selector = IB_SA_EQ;
|
route->path_rec->mtu_selector = IB_SA_EQ;
|
||||||
|
@ -3534,12 +3683,23 @@ static int cma_ib_mc_handler(int status, struct ib_sa_multicast *multicast)
|
||||||
event.status = status;
|
event.status = status;
|
||||||
event.param.ud.private_data = mc->context;
|
event.param.ud.private_data = mc->context;
|
||||||
if (!status) {
|
if (!status) {
|
||||||
|
struct rdma_dev_addr *dev_addr =
|
||||||
|
&id_priv->id.route.addr.dev_addr;
|
||||||
|
struct net_device *ndev =
|
||||||
|
dev_get_by_index(&init_net, dev_addr->bound_dev_if);
|
||||||
|
enum ib_gid_type gid_type =
|
||||||
|
id_priv->cma_dev->default_gid_type[id_priv->id.port_num -
|
||||||
|
rdma_start_port(id_priv->cma_dev->device)];
|
||||||
|
|
||||||
event.event = RDMA_CM_EVENT_MULTICAST_JOIN;
|
event.event = RDMA_CM_EVENT_MULTICAST_JOIN;
|
||||||
ib_init_ah_from_mcmember(id_priv->id.device,
|
ib_init_ah_from_mcmember(id_priv->id.device,
|
||||||
id_priv->id.port_num, &multicast->rec,
|
id_priv->id.port_num, &multicast->rec,
|
||||||
|
ndev, gid_type,
|
||||||
&event.param.ud.ah_attr);
|
&event.param.ud.ah_attr);
|
||||||
event.param.ud.qp_num = 0xFFFFFF;
|
event.param.ud.qp_num = 0xFFFFFF;
|
||||||
event.param.ud.qkey = be32_to_cpu(multicast->rec.qkey);
|
event.param.ud.qkey = be32_to_cpu(multicast->rec.qkey);
|
||||||
|
if (ndev)
|
||||||
|
dev_put(ndev);
|
||||||
} else
|
} else
|
||||||
event.event = RDMA_CM_EVENT_MULTICAST_ERROR;
|
event.event = RDMA_CM_EVENT_MULTICAST_ERROR;
|
||||||
|
|
||||||
|
@ -3672,9 +3832,10 @@ static int cma_iboe_join_multicast(struct rdma_id_private *id_priv,
|
||||||
{
|
{
|
||||||
struct iboe_mcast_work *work;
|
struct iboe_mcast_work *work;
|
||||||
struct rdma_dev_addr *dev_addr = &id_priv->id.route.addr.dev_addr;
|
struct rdma_dev_addr *dev_addr = &id_priv->id.route.addr.dev_addr;
|
||||||
int err;
|
int err = 0;
|
||||||
struct sockaddr *addr = (struct sockaddr *)&mc->addr;
|
struct sockaddr *addr = (struct sockaddr *)&mc->addr;
|
||||||
struct net_device *ndev = NULL;
|
struct net_device *ndev = NULL;
|
||||||
|
enum ib_gid_type gid_type;
|
||||||
|
|
||||||
if (cma_zero_addr((struct sockaddr *)&mc->addr))
|
if (cma_zero_addr((struct sockaddr *)&mc->addr))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
@ -3704,9 +3865,25 @@ static int cma_iboe_join_multicast(struct rdma_id_private *id_priv,
|
||||||
mc->multicast.ib->rec.rate = iboe_get_rate(ndev);
|
mc->multicast.ib->rec.rate = iboe_get_rate(ndev);
|
||||||
mc->multicast.ib->rec.hop_limit = 1;
|
mc->multicast.ib->rec.hop_limit = 1;
|
||||||
mc->multicast.ib->rec.mtu = iboe_get_mtu(ndev->mtu);
|
mc->multicast.ib->rec.mtu = iboe_get_mtu(ndev->mtu);
|
||||||
|
|
||||||
|
gid_type = id_priv->cma_dev->default_gid_type[id_priv->id.port_num -
|
||||||
|
rdma_start_port(id_priv->cma_dev->device)];
|
||||||
|
if (addr->sa_family == AF_INET) {
|
||||||
|
if (gid_type == IB_GID_TYPE_ROCE_UDP_ENCAP)
|
||||||
|
err = cma_igmp_send(ndev, &mc->multicast.ib->rec.mgid,
|
||||||
|
true);
|
||||||
|
if (!err) {
|
||||||
|
mc->igmp_joined = true;
|
||||||
|
mc->multicast.ib->rec.hop_limit = IPV6_DEFAULT_HOPLIMIT;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (gid_type == IB_GID_TYPE_ROCE_UDP_ENCAP)
|
||||||
|
err = -ENOTSUPP;
|
||||||
|
}
|
||||||
dev_put(ndev);
|
dev_put(ndev);
|
||||||
if (!mc->multicast.ib->rec.mtu) {
|
if (err || !mc->multicast.ib->rec.mtu) {
|
||||||
err = -EINVAL;
|
if (!err)
|
||||||
|
err = -EINVAL;
|
||||||
goto out2;
|
goto out2;
|
||||||
}
|
}
|
||||||
rdma_ip2gid((struct sockaddr *)&id_priv->id.route.addr.src_addr,
|
rdma_ip2gid((struct sockaddr *)&id_priv->id.route.addr.src_addr,
|
||||||
|
@ -3745,7 +3922,7 @@ int rdma_join_multicast(struct rdma_cm_id *id, struct sockaddr *addr,
|
||||||
memcpy(&mc->addr, addr, rdma_addr_size(addr));
|
memcpy(&mc->addr, addr, rdma_addr_size(addr));
|
||||||
mc->context = context;
|
mc->context = context;
|
||||||
mc->id_priv = id_priv;
|
mc->id_priv = id_priv;
|
||||||
|
mc->igmp_joined = false;
|
||||||
spin_lock(&id_priv->lock);
|
spin_lock(&id_priv->lock);
|
||||||
list_add(&mc->list, &id_priv->mc_list);
|
list_add(&mc->list, &id_priv->mc_list);
|
||||||
spin_unlock(&id_priv->lock);
|
spin_unlock(&id_priv->lock);
|
||||||
|
@ -3790,9 +3967,25 @@ void rdma_leave_multicast(struct rdma_cm_id *id, struct sockaddr *addr)
|
||||||
if (rdma_cap_ib_mcast(id->device, id->port_num)) {
|
if (rdma_cap_ib_mcast(id->device, id->port_num)) {
|
||||||
ib_sa_free_multicast(mc->multicast.ib);
|
ib_sa_free_multicast(mc->multicast.ib);
|
||||||
kfree(mc);
|
kfree(mc);
|
||||||
} else if (rdma_protocol_roce(id->device, id->port_num))
|
} else if (rdma_protocol_roce(id->device, id->port_num)) {
|
||||||
kref_put(&mc->mcref, release_mc);
|
if (mc->igmp_joined) {
|
||||||
|
struct rdma_dev_addr *dev_addr =
|
||||||
|
&id->route.addr.dev_addr;
|
||||||
|
struct net_device *ndev = NULL;
|
||||||
|
|
||||||
|
if (dev_addr->bound_dev_if)
|
||||||
|
ndev = dev_get_by_index(&init_net,
|
||||||
|
dev_addr->bound_dev_if);
|
||||||
|
if (ndev) {
|
||||||
|
cma_igmp_send(ndev,
|
||||||
|
&mc->multicast.ib->rec.mgid,
|
||||||
|
false);
|
||||||
|
dev_put(ndev);
|
||||||
|
}
|
||||||
|
mc->igmp_joined = false;
|
||||||
|
}
|
||||||
|
kref_put(&mc->mcref, release_mc);
|
||||||
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3861,12 +4054,27 @@ static void cma_add_one(struct ib_device *device)
|
||||||
{
|
{
|
||||||
struct cma_device *cma_dev;
|
struct cma_device *cma_dev;
|
||||||
struct rdma_id_private *id_priv;
|
struct rdma_id_private *id_priv;
|
||||||
|
unsigned int i;
|
||||||
|
unsigned long supported_gids = 0;
|
||||||
|
|
||||||
cma_dev = kmalloc(sizeof *cma_dev, GFP_KERNEL);
|
cma_dev = kmalloc(sizeof *cma_dev, GFP_KERNEL);
|
||||||
if (!cma_dev)
|
if (!cma_dev)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
cma_dev->device = device;
|
cma_dev->device = device;
|
||||||
|
cma_dev->default_gid_type = kcalloc(device->phys_port_cnt,
|
||||||
|
sizeof(*cma_dev->default_gid_type),
|
||||||
|
GFP_KERNEL);
|
||||||
|
if (!cma_dev->default_gid_type) {
|
||||||
|
kfree(cma_dev);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
for (i = rdma_start_port(device); i <= rdma_end_port(device); i++) {
|
||||||
|
supported_gids = roce_gid_type_mask_support(device, i);
|
||||||
|
WARN_ON(!supported_gids);
|
||||||
|
cma_dev->default_gid_type[i - rdma_start_port(device)] =
|
||||||
|
find_first_bit(&supported_gids, BITS_PER_LONG);
|
||||||
|
}
|
||||||
|
|
||||||
init_completion(&cma_dev->comp);
|
init_completion(&cma_dev->comp);
|
||||||
atomic_set(&cma_dev->refcount, 1);
|
atomic_set(&cma_dev->refcount, 1);
|
||||||
|
@ -3946,6 +4154,7 @@ static void cma_remove_one(struct ib_device *device, void *client_data)
|
||||||
mutex_unlock(&lock);
|
mutex_unlock(&lock);
|
||||||
|
|
||||||
cma_process_remove(cma_dev);
|
cma_process_remove(cma_dev);
|
||||||
|
kfree(cma_dev->default_gid_type);
|
||||||
kfree(cma_dev);
|
kfree(cma_dev);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -4079,6 +4288,7 @@ static int __init cma_init(void)
|
||||||
|
|
||||||
if (ibnl_add_client(RDMA_NL_RDMA_CM, RDMA_NL_RDMA_CM_NUM_OPS, cma_cb_table))
|
if (ibnl_add_client(RDMA_NL_RDMA_CM, RDMA_NL_RDMA_CM_NUM_OPS, cma_cb_table))
|
||||||
printk(KERN_WARNING "RDMA CMA: failed to add netlink callback\n");
|
printk(KERN_WARNING "RDMA CMA: failed to add netlink callback\n");
|
||||||
|
cma_configfs_init();
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
@ -4093,6 +4303,7 @@ static int __init cma_init(void)
|
||||||
|
|
||||||
static void __exit cma_cleanup(void)
|
static void __exit cma_cleanup(void)
|
||||||
{
|
{
|
||||||
|
cma_configfs_exit();
|
||||||
ibnl_remove_client(RDMA_NL_RDMA_CM);
|
ibnl_remove_client(RDMA_NL_RDMA_CM);
|
||||||
ib_unregister_client(&cma_client);
|
ib_unregister_client(&cma_client);
|
||||||
unregister_netdevice_notifier(&cma_nb);
|
unregister_netdevice_notifier(&cma_nb);
|
||||||
|
|
321
drivers/infiniband/core/cma_configfs.c
Normal file
321
drivers/infiniband/core/cma_configfs.c
Normal file
|
@ -0,0 +1,321 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2015, Mellanox Technologies inc. 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/configfs.h>
|
||||||
|
#include <rdma/ib_verbs.h>
|
||||||
|
#include "core_priv.h"
|
||||||
|
|
||||||
|
struct cma_device;
|
||||||
|
|
||||||
|
struct cma_dev_group;
|
||||||
|
|
||||||
|
struct cma_dev_port_group {
|
||||||
|
unsigned int port_num;
|
||||||
|
struct cma_dev_group *cma_dev_group;
|
||||||
|
struct config_group group;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct cma_dev_group {
|
||||||
|
char name[IB_DEVICE_NAME_MAX];
|
||||||
|
struct config_group device_group;
|
||||||
|
struct config_group ports_group;
|
||||||
|
struct config_group *default_dev_group[2];
|
||||||
|
struct config_group **default_ports_group;
|
||||||
|
struct cma_dev_port_group *ports;
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct cma_dev_port_group *to_dev_port_group(struct config_item *item)
|
||||||
|
{
|
||||||
|
struct config_group *group;
|
||||||
|
|
||||||
|
if (!item)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
group = container_of(item, struct config_group, cg_item);
|
||||||
|
return container_of(group, struct cma_dev_port_group, group);
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool filter_by_name(struct ib_device *ib_dev, void *cookie)
|
||||||
|
{
|
||||||
|
return !strcmp(ib_dev->name, cookie);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int cma_configfs_params_get(struct config_item *item,
|
||||||
|
struct cma_device **pcma_dev,
|
||||||
|
struct cma_dev_port_group **pgroup)
|
||||||
|
{
|
||||||
|
struct cma_dev_port_group *group = to_dev_port_group(item);
|
||||||
|
struct cma_device *cma_dev;
|
||||||
|
|
||||||
|
if (!group)
|
||||||
|
return -ENODEV;
|
||||||
|
|
||||||
|
cma_dev = cma_enum_devices_by_ibdev(filter_by_name,
|
||||||
|
group->cma_dev_group->name);
|
||||||
|
if (!cma_dev)
|
||||||
|
return -ENODEV;
|
||||||
|
|
||||||
|
*pcma_dev = cma_dev;
|
||||||
|
*pgroup = group;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void cma_configfs_params_put(struct cma_device *cma_dev)
|
||||||
|
{
|
||||||
|
cma_deref_dev(cma_dev);
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t default_roce_mode_show(struct config_item *item,
|
||||||
|
char *buf)
|
||||||
|
{
|
||||||
|
struct cma_device *cma_dev;
|
||||||
|
struct cma_dev_port_group *group;
|
||||||
|
int gid_type;
|
||||||
|
ssize_t ret;
|
||||||
|
|
||||||
|
ret = cma_configfs_params_get(item, &cma_dev, &group);
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
gid_type = cma_get_default_gid_type(cma_dev, group->port_num);
|
||||||
|
cma_configfs_params_put(cma_dev);
|
||||||
|
|
||||||
|
if (gid_type < 0)
|
||||||
|
return gid_type;
|
||||||
|
|
||||||
|
return sprintf(buf, "%s\n", ib_cache_gid_type_str(gid_type));
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t default_roce_mode_store(struct config_item *item,
|
||||||
|
const char *buf, size_t count)
|
||||||
|
{
|
||||||
|
struct cma_device *cma_dev;
|
||||||
|
struct cma_dev_port_group *group;
|
||||||
|
int gid_type = ib_cache_gid_parse_type_str(buf);
|
||||||
|
ssize_t ret;
|
||||||
|
|
||||||
|
if (gid_type < 0)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
ret = cma_configfs_params_get(item, &cma_dev, &group);
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
ret = cma_set_default_gid_type(cma_dev, group->port_num, gid_type);
|
||||||
|
|
||||||
|
cma_configfs_params_put(cma_dev);
|
||||||
|
|
||||||
|
return !ret ? strnlen(buf, count) : ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
CONFIGFS_ATTR(, default_roce_mode);
|
||||||
|
|
||||||
|
static struct configfs_attribute *cma_configfs_attributes[] = {
|
||||||
|
&attr_default_roce_mode,
|
||||||
|
NULL,
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct config_item_type cma_port_group_type = {
|
||||||
|
.ct_attrs = cma_configfs_attributes,
|
||||||
|
.ct_owner = THIS_MODULE
|
||||||
|
};
|
||||||
|
|
||||||
|
static int make_cma_ports(struct cma_dev_group *cma_dev_group,
|
||||||
|
struct cma_device *cma_dev)
|
||||||
|
{
|
||||||
|
struct ib_device *ibdev;
|
||||||
|
unsigned int i;
|
||||||
|
unsigned int ports_num;
|
||||||
|
struct cma_dev_port_group *ports;
|
||||||
|
struct config_group **ports_group;
|
||||||
|
int err;
|
||||||
|
|
||||||
|
ibdev = cma_get_ib_dev(cma_dev);
|
||||||
|
|
||||||
|
if (!ibdev)
|
||||||
|
return -ENODEV;
|
||||||
|
|
||||||
|
ports_num = ibdev->phys_port_cnt;
|
||||||
|
ports = kcalloc(ports_num, sizeof(*cma_dev_group->ports),
|
||||||
|
GFP_KERNEL);
|
||||||
|
ports_group = kcalloc(ports_num + 1, sizeof(*ports_group), GFP_KERNEL);
|
||||||
|
|
||||||
|
if (!ports || !ports_group) {
|
||||||
|
err = -ENOMEM;
|
||||||
|
goto free;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i = 0; i < ports_num; i++) {
|
||||||
|
char port_str[10];
|
||||||
|
|
||||||
|
ports[i].port_num = i + 1;
|
||||||
|
snprintf(port_str, sizeof(port_str), "%u", i + 1);
|
||||||
|
ports[i].cma_dev_group = cma_dev_group;
|
||||||
|
config_group_init_type_name(&ports[i].group,
|
||||||
|
port_str,
|
||||||
|
&cma_port_group_type);
|
||||||
|
ports_group[i] = &ports[i].group;
|
||||||
|
}
|
||||||
|
ports_group[i] = NULL;
|
||||||
|
cma_dev_group->default_ports_group = ports_group;
|
||||||
|
cma_dev_group->ports = ports;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
free:
|
||||||
|
kfree(ports);
|
||||||
|
kfree(ports_group);
|
||||||
|
cma_dev_group->ports = NULL;
|
||||||
|
cma_dev_group->default_ports_group = NULL;
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void release_cma_dev(struct config_item *item)
|
||||||
|
{
|
||||||
|
struct config_group *group = container_of(item, struct config_group,
|
||||||
|
cg_item);
|
||||||
|
struct cma_dev_group *cma_dev_group = container_of(group,
|
||||||
|
struct cma_dev_group,
|
||||||
|
device_group);
|
||||||
|
|
||||||
|
kfree(cma_dev_group);
|
||||||
|
};
|
||||||
|
|
||||||
|
static void release_cma_ports_group(struct config_item *item)
|
||||||
|
{
|
||||||
|
struct config_group *group = container_of(item, struct config_group,
|
||||||
|
cg_item);
|
||||||
|
struct cma_dev_group *cma_dev_group = container_of(group,
|
||||||
|
struct cma_dev_group,
|
||||||
|
ports_group);
|
||||||
|
|
||||||
|
kfree(cma_dev_group->ports);
|
||||||
|
kfree(cma_dev_group->default_ports_group);
|
||||||
|
cma_dev_group->ports = NULL;
|
||||||
|
cma_dev_group->default_ports_group = NULL;
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct configfs_item_operations cma_ports_item_ops = {
|
||||||
|
.release = release_cma_ports_group
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct config_item_type cma_ports_group_type = {
|
||||||
|
.ct_item_ops = &cma_ports_item_ops,
|
||||||
|
.ct_owner = THIS_MODULE
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct configfs_item_operations cma_device_item_ops = {
|
||||||
|
.release = release_cma_dev
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct config_item_type cma_device_group_type = {
|
||||||
|
.ct_item_ops = &cma_device_item_ops,
|
||||||
|
.ct_owner = THIS_MODULE
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct config_group *make_cma_dev(struct config_group *group,
|
||||||
|
const char *name)
|
||||||
|
{
|
||||||
|
int err = -ENODEV;
|
||||||
|
struct cma_device *cma_dev = cma_enum_devices_by_ibdev(filter_by_name,
|
||||||
|
(void *)name);
|
||||||
|
struct cma_dev_group *cma_dev_group = NULL;
|
||||||
|
|
||||||
|
if (!cma_dev)
|
||||||
|
goto fail;
|
||||||
|
|
||||||
|
cma_dev_group = kzalloc(sizeof(*cma_dev_group), GFP_KERNEL);
|
||||||
|
|
||||||
|
if (!cma_dev_group) {
|
||||||
|
err = -ENOMEM;
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
strncpy(cma_dev_group->name, name, sizeof(cma_dev_group->name));
|
||||||
|
|
||||||
|
err = make_cma_ports(cma_dev_group, cma_dev);
|
||||||
|
if (err)
|
||||||
|
goto fail;
|
||||||
|
|
||||||
|
cma_dev_group->ports_group.default_groups =
|
||||||
|
cma_dev_group->default_ports_group;
|
||||||
|
config_group_init_type_name(&cma_dev_group->ports_group, "ports",
|
||||||
|
&cma_ports_group_type);
|
||||||
|
|
||||||
|
cma_dev_group->device_group.default_groups
|
||||||
|
= cma_dev_group->default_dev_group;
|
||||||
|
cma_dev_group->default_dev_group[0] = &cma_dev_group->ports_group;
|
||||||
|
cma_dev_group->default_dev_group[1] = NULL;
|
||||||
|
|
||||||
|
config_group_init_type_name(&cma_dev_group->device_group, name,
|
||||||
|
&cma_device_group_type);
|
||||||
|
|
||||||
|
cma_deref_dev(cma_dev);
|
||||||
|
return &cma_dev_group->device_group;
|
||||||
|
|
||||||
|
fail:
|
||||||
|
if (cma_dev)
|
||||||
|
cma_deref_dev(cma_dev);
|
||||||
|
kfree(cma_dev_group);
|
||||||
|
return ERR_PTR(err);
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct configfs_group_operations cma_subsys_group_ops = {
|
||||||
|
.make_group = make_cma_dev,
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct config_item_type cma_subsys_type = {
|
||||||
|
.ct_group_ops = &cma_subsys_group_ops,
|
||||||
|
.ct_owner = THIS_MODULE,
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct configfs_subsystem cma_subsys = {
|
||||||
|
.su_group = {
|
||||||
|
.cg_item = {
|
||||||
|
.ci_namebuf = "rdma_cm",
|
||||||
|
.ci_type = &cma_subsys_type,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
int __init cma_configfs_init(void)
|
||||||
|
{
|
||||||
|
config_group_init(&cma_subsys.su_group);
|
||||||
|
mutex_init(&cma_subsys.su_mutex);
|
||||||
|
return configfs_register_subsystem(&cma_subsys);
|
||||||
|
}
|
||||||
|
|
||||||
|
void __exit cma_configfs_exit(void)
|
||||||
|
{
|
||||||
|
configfs_unregister_subsystem(&cma_subsys);
|
||||||
|
}
|
|
@ -38,6 +38,32 @@
|
||||||
|
|
||||||
#include <rdma/ib_verbs.h>
|
#include <rdma/ib_verbs.h>
|
||||||
|
|
||||||
|
#if IS_ENABLED(CONFIG_INFINIBAND_ADDR_TRANS_CONFIGFS)
|
||||||
|
int cma_configfs_init(void);
|
||||||
|
void cma_configfs_exit(void);
|
||||||
|
#else
|
||||||
|
static inline int cma_configfs_init(void)
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void cma_configfs_exit(void)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
struct cma_device;
|
||||||
|
void cma_ref_dev(struct cma_device *cma_dev);
|
||||||
|
void cma_deref_dev(struct cma_device *cma_dev);
|
||||||
|
typedef bool (*cma_device_filter)(struct ib_device *, void *);
|
||||||
|
struct cma_device *cma_enum_devices_by_ibdev(cma_device_filter filter,
|
||||||
|
void *cookie);
|
||||||
|
int cma_get_default_gid_type(struct cma_device *cma_dev,
|
||||||
|
unsigned int port);
|
||||||
|
int cma_set_default_gid_type(struct cma_device *cma_dev,
|
||||||
|
unsigned int port,
|
||||||
|
enum ib_gid_type default_gid_type);
|
||||||
|
struct ib_device *cma_get_ib_dev(struct cma_device *cma_dev);
|
||||||
|
|
||||||
int ib_device_register_sysfs(struct ib_device *device,
|
int ib_device_register_sysfs(struct ib_device *device,
|
||||||
int (*port_callback)(struct ib_device *,
|
int (*port_callback)(struct ib_device *,
|
||||||
u8, struct kobject *));
|
u8, struct kobject *));
|
||||||
|
@ -70,8 +96,13 @@ enum ib_cache_gid_default_mode {
|
||||||
IB_CACHE_GID_DEFAULT_MODE_DELETE
|
IB_CACHE_GID_DEFAULT_MODE_DELETE
|
||||||
};
|
};
|
||||||
|
|
||||||
|
int ib_cache_gid_parse_type_str(const char *buf);
|
||||||
|
|
||||||
|
const char *ib_cache_gid_type_str(enum ib_gid_type gid_type);
|
||||||
|
|
||||||
void ib_cache_gid_set_default_gid(struct ib_device *ib_dev, u8 port,
|
void ib_cache_gid_set_default_gid(struct ib_device *ib_dev, u8 port,
|
||||||
struct net_device *ndev,
|
struct net_device *ndev,
|
||||||
|
unsigned long gid_type_mask,
|
||||||
enum ib_cache_gid_default_mode mode);
|
enum ib_cache_gid_default_mode mode);
|
||||||
|
|
||||||
int ib_cache_gid_add(struct ib_device *ib_dev, u8 port,
|
int ib_cache_gid_add(struct ib_device *ib_dev, u8 port,
|
||||||
|
@ -87,9 +118,23 @@ int roce_gid_mgmt_init(void);
|
||||||
void roce_gid_mgmt_cleanup(void);
|
void roce_gid_mgmt_cleanup(void);
|
||||||
|
|
||||||
int roce_rescan_device(struct ib_device *ib_dev);
|
int roce_rescan_device(struct ib_device *ib_dev);
|
||||||
|
unsigned long roce_gid_type_mask_support(struct ib_device *ib_dev, u8 port);
|
||||||
|
|
||||||
int ib_cache_setup_one(struct ib_device *device);
|
int ib_cache_setup_one(struct ib_device *device);
|
||||||
void ib_cache_cleanup_one(struct ib_device *device);
|
void ib_cache_cleanup_one(struct ib_device *device);
|
||||||
void ib_cache_release_one(struct ib_device *device);
|
void ib_cache_release_one(struct ib_device *device);
|
||||||
|
|
||||||
|
static inline bool rdma_is_upper_dev_rcu(struct net_device *dev,
|
||||||
|
struct net_device *upper)
|
||||||
|
{
|
||||||
|
struct net_device *_upper = NULL;
|
||||||
|
struct list_head *iter;
|
||||||
|
|
||||||
|
netdev_for_each_all_upper_dev_rcu(dev, _upper, iter)
|
||||||
|
if (_upper == upper)
|
||||||
|
break;
|
||||||
|
|
||||||
|
return _upper == upper;
|
||||||
|
}
|
||||||
|
|
||||||
#endif /* _CORE_PRIV_H */
|
#endif /* _CORE_PRIV_H */
|
||||||
|
|
209
drivers/infiniband/core/cq.c
Normal file
209
drivers/infiniband/core/cq.c
Normal file
|
@ -0,0 +1,209 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2015 HGST, a Western Digital Company.
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms and conditions of the GNU General Public License,
|
||||||
|
* version 2, as published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope it will be useful, but WITHOUT
|
||||||
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||||
|
* more details.
|
||||||
|
*/
|
||||||
|
#include <linux/module.h>
|
||||||
|
#include <linux/err.h>
|
||||||
|
#include <linux/slab.h>
|
||||||
|
#include <rdma/ib_verbs.h>
|
||||||
|
|
||||||
|
/* # of WCs to poll for with a single call to ib_poll_cq */
|
||||||
|
#define IB_POLL_BATCH 16
|
||||||
|
|
||||||
|
/* # of WCs to iterate over before yielding */
|
||||||
|
#define IB_POLL_BUDGET_IRQ 256
|
||||||
|
#define IB_POLL_BUDGET_WORKQUEUE 65536
|
||||||
|
|
||||||
|
#define IB_POLL_FLAGS \
|
||||||
|
(IB_CQ_NEXT_COMP | IB_CQ_REPORT_MISSED_EVENTS)
|
||||||
|
|
||||||
|
static int __ib_process_cq(struct ib_cq *cq, int budget)
|
||||||
|
{
|
||||||
|
int i, n, completed = 0;
|
||||||
|
|
||||||
|
while ((n = ib_poll_cq(cq, IB_POLL_BATCH, cq->wc)) > 0) {
|
||||||
|
for (i = 0; i < n; i++) {
|
||||||
|
struct ib_wc *wc = &cq->wc[i];
|
||||||
|
|
||||||
|
if (wc->wr_cqe)
|
||||||
|
wc->wr_cqe->done(cq, wc);
|
||||||
|
else
|
||||||
|
WARN_ON_ONCE(wc->status == IB_WC_SUCCESS);
|
||||||
|
}
|
||||||
|
|
||||||
|
completed += n;
|
||||||
|
|
||||||
|
if (n != IB_POLL_BATCH ||
|
||||||
|
(budget != -1 && completed >= budget))
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return completed;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ib_process_direct_cq - process a CQ in caller context
|
||||||
|
* @cq: CQ to process
|
||||||
|
* @budget: number of CQEs to poll for
|
||||||
|
*
|
||||||
|
* This function is used to process all outstanding CQ entries on a
|
||||||
|
* %IB_POLL_DIRECT CQ. It does not offload CQ processing to a different
|
||||||
|
* context and does not ask for completion interrupts from the HCA.
|
||||||
|
*
|
||||||
|
* Note: for compatibility reasons -1 can be passed in %budget for unlimited
|
||||||
|
* polling. Do not use this feature in new code, it will be removed soon.
|
||||||
|
*/
|
||||||
|
int ib_process_cq_direct(struct ib_cq *cq, int budget)
|
||||||
|
{
|
||||||
|
WARN_ON_ONCE(cq->poll_ctx != IB_POLL_DIRECT);
|
||||||
|
|
||||||
|
return __ib_process_cq(cq, budget);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(ib_process_cq_direct);
|
||||||
|
|
||||||
|
static void ib_cq_completion_direct(struct ib_cq *cq, void *private)
|
||||||
|
{
|
||||||
|
WARN_ONCE(1, "got unsolicited completion for CQ 0x%p\n", cq);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int ib_poll_handler(struct irq_poll *iop, int budget)
|
||||||
|
{
|
||||||
|
struct ib_cq *cq = container_of(iop, struct ib_cq, iop);
|
||||||
|
int completed;
|
||||||
|
|
||||||
|
completed = __ib_process_cq(cq, budget);
|
||||||
|
if (completed < budget) {
|
||||||
|
irq_poll_complete(&cq->iop);
|
||||||
|
if (ib_req_notify_cq(cq, IB_POLL_FLAGS) > 0)
|
||||||
|
irq_poll_sched(&cq->iop);
|
||||||
|
}
|
||||||
|
|
||||||
|
return completed;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void ib_cq_completion_softirq(struct ib_cq *cq, void *private)
|
||||||
|
{
|
||||||
|
irq_poll_sched(&cq->iop);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void ib_cq_poll_work(struct work_struct *work)
|
||||||
|
{
|
||||||
|
struct ib_cq *cq = container_of(work, struct ib_cq, work);
|
||||||
|
int completed;
|
||||||
|
|
||||||
|
completed = __ib_process_cq(cq, IB_POLL_BUDGET_WORKQUEUE);
|
||||||
|
if (completed >= IB_POLL_BUDGET_WORKQUEUE ||
|
||||||
|
ib_req_notify_cq(cq, IB_POLL_FLAGS) > 0)
|
||||||
|
queue_work(ib_comp_wq, &cq->work);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void ib_cq_completion_workqueue(struct ib_cq *cq, void *private)
|
||||||
|
{
|
||||||
|
queue_work(ib_comp_wq, &cq->work);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ib_alloc_cq - allocate a completion queue
|
||||||
|
* @dev: device to allocate the CQ for
|
||||||
|
* @private: driver private data, accessible from cq->cq_context
|
||||||
|
* @nr_cqe: number of CQEs to allocate
|
||||||
|
* @comp_vector: HCA completion vectors for this CQ
|
||||||
|
* @poll_ctx: context to poll the CQ from.
|
||||||
|
*
|
||||||
|
* This is the proper interface to allocate a CQ for in-kernel users. A
|
||||||
|
* CQ allocated with this interface will automatically be polled from the
|
||||||
|
* specified context. The ULP needs must use wr->wr_cqe instead of wr->wr_id
|
||||||
|
* to use this CQ abstraction.
|
||||||
|
*/
|
||||||
|
struct ib_cq *ib_alloc_cq(struct ib_device *dev, void *private,
|
||||||
|
int nr_cqe, int comp_vector, enum ib_poll_context poll_ctx)
|
||||||
|
{
|
||||||
|
struct ib_cq_init_attr cq_attr = {
|
||||||
|
.cqe = nr_cqe,
|
||||||
|
.comp_vector = comp_vector,
|
||||||
|
};
|
||||||
|
struct ib_cq *cq;
|
||||||
|
int ret = -ENOMEM;
|
||||||
|
|
||||||
|
cq = dev->create_cq(dev, &cq_attr, NULL, NULL);
|
||||||
|
if (IS_ERR(cq))
|
||||||
|
return cq;
|
||||||
|
|
||||||
|
cq->device = dev;
|
||||||
|
cq->uobject = NULL;
|
||||||
|
cq->event_handler = NULL;
|
||||||
|
cq->cq_context = private;
|
||||||
|
cq->poll_ctx = poll_ctx;
|
||||||
|
atomic_set(&cq->usecnt, 0);
|
||||||
|
|
||||||
|
cq->wc = kmalloc_array(IB_POLL_BATCH, sizeof(*cq->wc), GFP_KERNEL);
|
||||||
|
if (!cq->wc)
|
||||||
|
goto out_destroy_cq;
|
||||||
|
|
||||||
|
switch (cq->poll_ctx) {
|
||||||
|
case IB_POLL_DIRECT:
|
||||||
|
cq->comp_handler = ib_cq_completion_direct;
|
||||||
|
break;
|
||||||
|
case IB_POLL_SOFTIRQ:
|
||||||
|
cq->comp_handler = ib_cq_completion_softirq;
|
||||||
|
|
||||||
|
irq_poll_init(&cq->iop, IB_POLL_BUDGET_IRQ, ib_poll_handler);
|
||||||
|
ib_req_notify_cq(cq, IB_CQ_NEXT_COMP);
|
||||||
|
break;
|
||||||
|
case IB_POLL_WORKQUEUE:
|
||||||
|
cq->comp_handler = ib_cq_completion_workqueue;
|
||||||
|
INIT_WORK(&cq->work, ib_cq_poll_work);
|
||||||
|
ib_req_notify_cq(cq, IB_CQ_NEXT_COMP);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
ret = -EINVAL;
|
||||||
|
goto out_free_wc;
|
||||||
|
}
|
||||||
|
|
||||||
|
return cq;
|
||||||
|
|
||||||
|
out_free_wc:
|
||||||
|
kfree(cq->wc);
|
||||||
|
out_destroy_cq:
|
||||||
|
cq->device->destroy_cq(cq);
|
||||||
|
return ERR_PTR(ret);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(ib_alloc_cq);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ib_free_cq - free a completion queue
|
||||||
|
* @cq: completion queue to free.
|
||||||
|
*/
|
||||||
|
void ib_free_cq(struct ib_cq *cq)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
if (WARN_ON_ONCE(atomic_read(&cq->usecnt)))
|
||||||
|
return;
|
||||||
|
|
||||||
|
switch (cq->poll_ctx) {
|
||||||
|
case IB_POLL_DIRECT:
|
||||||
|
break;
|
||||||
|
case IB_POLL_SOFTIRQ:
|
||||||
|
irq_poll_disable(&cq->iop);
|
||||||
|
break;
|
||||||
|
case IB_POLL_WORKQUEUE:
|
||||||
|
flush_work(&cq->work);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
WARN_ON_ONCE(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
kfree(cq->wc);
|
||||||
|
ret = cq->device->destroy_cq(cq);
|
||||||
|
WARN_ON_ONCE(ret);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(ib_free_cq);
|
|
@ -58,6 +58,7 @@ struct ib_client_data {
|
||||||
bool going_down;
|
bool going_down;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct workqueue_struct *ib_comp_wq;
|
||||||
struct workqueue_struct *ib_wq;
|
struct workqueue_struct *ib_wq;
|
||||||
EXPORT_SYMBOL_GPL(ib_wq);
|
EXPORT_SYMBOL_GPL(ib_wq);
|
||||||
|
|
||||||
|
@ -325,6 +326,7 @@ int ib_register_device(struct ib_device *device,
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
struct ib_client *client;
|
struct ib_client *client;
|
||||||
|
struct ib_udata uhw = {.outlen = 0, .inlen = 0};
|
||||||
|
|
||||||
mutex_lock(&device_mutex);
|
mutex_lock(&device_mutex);
|
||||||
|
|
||||||
|
@ -352,6 +354,13 @@ int ib_register_device(struct ib_device *device,
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
memset(&device->attrs, 0, sizeof(device->attrs));
|
||||||
|
ret = device->query_device(device, &device->attrs, &uhw);
|
||||||
|
if (ret) {
|
||||||
|
printk(KERN_WARNING "Couldn't query the device attributes\n");
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
ret = ib_device_register_sysfs(device, port_callback);
|
ret = ib_device_register_sysfs(device, port_callback);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
printk(KERN_WARNING "Couldn't register device %s with driver model\n",
|
printk(KERN_WARNING "Couldn't register device %s with driver model\n",
|
||||||
|
@ -627,25 +636,6 @@ void ib_dispatch_event(struct ib_event *event)
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(ib_dispatch_event);
|
EXPORT_SYMBOL(ib_dispatch_event);
|
||||||
|
|
||||||
/**
|
|
||||||
* ib_query_device - Query IB device attributes
|
|
||||||
* @device:Device to query
|
|
||||||
* @device_attr:Device attributes
|
|
||||||
*
|
|
||||||
* ib_query_device() returns the attributes of a device through the
|
|
||||||
* @device_attr pointer.
|
|
||||||
*/
|
|
||||||
int ib_query_device(struct ib_device *device,
|
|
||||||
struct ib_device_attr *device_attr)
|
|
||||||
{
|
|
||||||
struct ib_udata uhw = {.outlen = 0, .inlen = 0};
|
|
||||||
|
|
||||||
memset(device_attr, 0, sizeof(*device_attr));
|
|
||||||
|
|
||||||
return device->query_device(device, device_attr, &uhw);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL(ib_query_device);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* ib_query_port - Query IB port attributes
|
* ib_query_port - Query IB port attributes
|
||||||
* @device:Device to query
|
* @device:Device to query
|
||||||
|
@ -825,26 +815,31 @@ EXPORT_SYMBOL(ib_modify_port);
|
||||||
* a specified GID value occurs.
|
* a specified GID value occurs.
|
||||||
* @device: The device to query.
|
* @device: The device to query.
|
||||||
* @gid: The GID value to search for.
|
* @gid: The GID value to search for.
|
||||||
|
* @gid_type: Type of GID.
|
||||||
* @ndev: The ndev related to the GID to search for.
|
* @ndev: The ndev related to the GID to search for.
|
||||||
* @port_num: The port number of the device where the GID value was found.
|
* @port_num: The port number of the device where the GID value was found.
|
||||||
* @index: The index into the GID table where the GID was found. This
|
* @index: The index into the GID table where the GID was found. This
|
||||||
* parameter may be NULL.
|
* parameter may be NULL.
|
||||||
*/
|
*/
|
||||||
int ib_find_gid(struct ib_device *device, union ib_gid *gid,
|
int ib_find_gid(struct ib_device *device, union ib_gid *gid,
|
||||||
struct net_device *ndev, u8 *port_num, u16 *index)
|
enum ib_gid_type gid_type, struct net_device *ndev,
|
||||||
|
u8 *port_num, u16 *index)
|
||||||
{
|
{
|
||||||
union ib_gid tmp_gid;
|
union ib_gid tmp_gid;
|
||||||
int ret, port, i;
|
int ret, port, i;
|
||||||
|
|
||||||
for (port = rdma_start_port(device); port <= rdma_end_port(device); ++port) {
|
for (port = rdma_start_port(device); port <= rdma_end_port(device); ++port) {
|
||||||
if (rdma_cap_roce_gid_table(device, port)) {
|
if (rdma_cap_roce_gid_table(device, port)) {
|
||||||
if (!ib_find_cached_gid_by_port(device, gid, port,
|
if (!ib_find_cached_gid_by_port(device, gid, gid_type, port,
|
||||||
ndev, index)) {
|
ndev, index)) {
|
||||||
*port_num = port;
|
*port_num = port;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (gid_type != IB_GID_TYPE_IB)
|
||||||
|
continue;
|
||||||
|
|
||||||
for (i = 0; i < device->port_immutable[port].gid_tbl_len; ++i) {
|
for (i = 0; i < device->port_immutable[port].gid_tbl_len; ++i) {
|
||||||
ret = ib_query_gid(device, port, i, &tmp_gid, NULL);
|
ret = ib_query_gid(device, port, i, &tmp_gid, NULL);
|
||||||
if (ret)
|
if (ret)
|
||||||
|
@ -954,10 +949,18 @@ static int __init ib_core_init(void)
|
||||||
if (!ib_wq)
|
if (!ib_wq)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
|
ib_comp_wq = alloc_workqueue("ib-comp-wq",
|
||||||
|
WQ_UNBOUND | WQ_HIGHPRI | WQ_MEM_RECLAIM,
|
||||||
|
WQ_UNBOUND_MAX_ACTIVE);
|
||||||
|
if (!ib_comp_wq) {
|
||||||
|
ret = -ENOMEM;
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
|
|
||||||
ret = class_register(&ib_class);
|
ret = class_register(&ib_class);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
printk(KERN_WARNING "Couldn't create InfiniBand device class\n");
|
printk(KERN_WARNING "Couldn't create InfiniBand device class\n");
|
||||||
goto err;
|
goto err_comp;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = ibnl_init();
|
ret = ibnl_init();
|
||||||
|
@ -972,7 +975,8 @@ static int __init ib_core_init(void)
|
||||||
|
|
||||||
err_sysfs:
|
err_sysfs:
|
||||||
class_unregister(&ib_class);
|
class_unregister(&ib_class);
|
||||||
|
err_comp:
|
||||||
|
destroy_workqueue(ib_comp_wq);
|
||||||
err:
|
err:
|
||||||
destroy_workqueue(ib_wq);
|
destroy_workqueue(ib_wq);
|
||||||
return ret;
|
return ret;
|
||||||
|
@ -983,6 +987,7 @@ static void __exit ib_core_cleanup(void)
|
||||||
ib_cache_cleanup();
|
ib_cache_cleanup();
|
||||||
ibnl_cleanup();
|
ibnl_cleanup();
|
||||||
class_unregister(&ib_class);
|
class_unregister(&ib_class);
|
||||||
|
destroy_workqueue(ib_comp_wq);
|
||||||
/* Make sure that any pending umem accounting work is done. */
|
/* Make sure that any pending umem accounting work is done. */
|
||||||
destroy_workqueue(ib_wq);
|
destroy_workqueue(ib_wq);
|
||||||
}
|
}
|
||||||
|
|
|
@ -212,7 +212,6 @@ struct ib_fmr_pool *ib_create_fmr_pool(struct ib_pd *pd,
|
||||||
{
|
{
|
||||||
struct ib_device *device;
|
struct ib_device *device;
|
||||||
struct ib_fmr_pool *pool;
|
struct ib_fmr_pool *pool;
|
||||||
struct ib_device_attr *attr;
|
|
||||||
int i;
|
int i;
|
||||||
int ret;
|
int ret;
|
||||||
int max_remaps;
|
int max_remaps;
|
||||||
|
@ -228,25 +227,10 @@ struct ib_fmr_pool *ib_create_fmr_pool(struct ib_pd *pd,
|
||||||
return ERR_PTR(-ENOSYS);
|
return ERR_PTR(-ENOSYS);
|
||||||
}
|
}
|
||||||
|
|
||||||
attr = kmalloc(sizeof *attr, GFP_KERNEL);
|
if (!device->attrs.max_map_per_fmr)
|
||||||
if (!attr) {
|
|
||||||
printk(KERN_WARNING PFX "couldn't allocate device attr struct\n");
|
|
||||||
return ERR_PTR(-ENOMEM);
|
|
||||||
}
|
|
||||||
|
|
||||||
ret = ib_query_device(device, attr);
|
|
||||||
if (ret) {
|
|
||||||
printk(KERN_WARNING PFX "couldn't query device: %d\n", ret);
|
|
||||||
kfree(attr);
|
|
||||||
return ERR_PTR(ret);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!attr->max_map_per_fmr)
|
|
||||||
max_remaps = IB_FMR_MAX_REMAPS;
|
max_remaps = IB_FMR_MAX_REMAPS;
|
||||||
else
|
else
|
||||||
max_remaps = attr->max_map_per_fmr;
|
max_remaps = device->attrs.max_map_per_fmr;
|
||||||
|
|
||||||
kfree(attr);
|
|
||||||
|
|
||||||
pool = kmalloc(sizeof *pool, GFP_KERNEL);
|
pool = kmalloc(sizeof *pool, GFP_KERNEL);
|
||||||
if (!pool) {
|
if (!pool) {
|
||||||
|
|
|
@ -84,6 +84,9 @@ static int add_nonoui_reg_req(struct ib_mad_reg_req *mad_reg_req,
|
||||||
u8 mgmt_class);
|
u8 mgmt_class);
|
||||||
static int add_oui_reg_req(struct ib_mad_reg_req *mad_reg_req,
|
static int add_oui_reg_req(struct ib_mad_reg_req *mad_reg_req,
|
||||||
struct ib_mad_agent_private *agent_priv);
|
struct ib_mad_agent_private *agent_priv);
|
||||||
|
static bool ib_mad_send_error(struct ib_mad_port_private *port_priv,
|
||||||
|
struct ib_wc *wc);
|
||||||
|
static void ib_mad_send_done(struct ib_cq *cq, struct ib_wc *wc);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Returns a ib_mad_port_private structure or NULL for a device/port
|
* Returns a ib_mad_port_private structure or NULL for a device/port
|
||||||
|
@ -681,7 +684,7 @@ static void snoop_recv(struct ib_mad_qp_info *qp_info,
|
||||||
|
|
||||||
atomic_inc(&mad_snoop_priv->refcount);
|
atomic_inc(&mad_snoop_priv->refcount);
|
||||||
spin_unlock_irqrestore(&qp_info->snoop_lock, flags);
|
spin_unlock_irqrestore(&qp_info->snoop_lock, flags);
|
||||||
mad_snoop_priv->agent.recv_handler(&mad_snoop_priv->agent,
|
mad_snoop_priv->agent.recv_handler(&mad_snoop_priv->agent, NULL,
|
||||||
mad_recv_wc);
|
mad_recv_wc);
|
||||||
deref_snoop_agent(mad_snoop_priv);
|
deref_snoop_agent(mad_snoop_priv);
|
||||||
spin_lock_irqsave(&qp_info->snoop_lock, flags);
|
spin_lock_irqsave(&qp_info->snoop_lock, flags);
|
||||||
|
@ -689,12 +692,11 @@ static void snoop_recv(struct ib_mad_qp_info *qp_info,
|
||||||
spin_unlock_irqrestore(&qp_info->snoop_lock, flags);
|
spin_unlock_irqrestore(&qp_info->snoop_lock, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void build_smp_wc(struct ib_qp *qp,
|
static void build_smp_wc(struct ib_qp *qp, struct ib_cqe *cqe, u16 slid,
|
||||||
u64 wr_id, u16 slid, u16 pkey_index, u8 port_num,
|
u16 pkey_index, u8 port_num, struct ib_wc *wc)
|
||||||
struct ib_wc *wc)
|
|
||||||
{
|
{
|
||||||
memset(wc, 0, sizeof *wc);
|
memset(wc, 0, sizeof *wc);
|
||||||
wc->wr_id = wr_id;
|
wc->wr_cqe = cqe;
|
||||||
wc->status = IB_WC_SUCCESS;
|
wc->status = IB_WC_SUCCESS;
|
||||||
wc->opcode = IB_WC_RECV;
|
wc->opcode = IB_WC_RECV;
|
||||||
wc->pkey_index = pkey_index;
|
wc->pkey_index = pkey_index;
|
||||||
|
@ -832,7 +834,7 @@ static int handle_outgoing_dr_smp(struct ib_mad_agent_private *mad_agent_priv,
|
||||||
}
|
}
|
||||||
|
|
||||||
build_smp_wc(mad_agent_priv->agent.qp,
|
build_smp_wc(mad_agent_priv->agent.qp,
|
||||||
send_wr->wr.wr_id, drslid,
|
send_wr->wr.wr_cqe, drslid,
|
||||||
send_wr->pkey_index,
|
send_wr->pkey_index,
|
||||||
send_wr->port_num, &mad_wc);
|
send_wr->port_num, &mad_wc);
|
||||||
|
|
||||||
|
@ -1039,7 +1041,9 @@ struct ib_mad_send_buf * ib_create_send_mad(struct ib_mad_agent *mad_agent,
|
||||||
|
|
||||||
mad_send_wr->sg_list[1].lkey = mad_agent->qp->pd->local_dma_lkey;
|
mad_send_wr->sg_list[1].lkey = mad_agent->qp->pd->local_dma_lkey;
|
||||||
|
|
||||||
mad_send_wr->send_wr.wr.wr_id = (unsigned long) mad_send_wr;
|
mad_send_wr->mad_list.cqe.done = ib_mad_send_done;
|
||||||
|
|
||||||
|
mad_send_wr->send_wr.wr.wr_cqe = &mad_send_wr->mad_list.cqe;
|
||||||
mad_send_wr->send_wr.wr.sg_list = mad_send_wr->sg_list;
|
mad_send_wr->send_wr.wr.sg_list = mad_send_wr->sg_list;
|
||||||
mad_send_wr->send_wr.wr.num_sge = 2;
|
mad_send_wr->send_wr.wr.num_sge = 2;
|
||||||
mad_send_wr->send_wr.wr.opcode = IB_WR_SEND;
|
mad_send_wr->send_wr.wr.opcode = IB_WR_SEND;
|
||||||
|
@ -1151,8 +1155,9 @@ int ib_send_mad(struct ib_mad_send_wr_private *mad_send_wr)
|
||||||
|
|
||||||
/* Set WR ID to find mad_send_wr upon completion */
|
/* Set WR ID to find mad_send_wr upon completion */
|
||||||
qp_info = mad_send_wr->mad_agent_priv->qp_info;
|
qp_info = mad_send_wr->mad_agent_priv->qp_info;
|
||||||
mad_send_wr->send_wr.wr.wr_id = (unsigned long)&mad_send_wr->mad_list;
|
|
||||||
mad_send_wr->mad_list.mad_queue = &qp_info->send_queue;
|
mad_send_wr->mad_list.mad_queue = &qp_info->send_queue;
|
||||||
|
mad_send_wr->mad_list.cqe.done = ib_mad_send_done;
|
||||||
|
mad_send_wr->send_wr.wr.wr_cqe = &mad_send_wr->mad_list.cqe;
|
||||||
|
|
||||||
mad_agent = mad_send_wr->send_buf.mad_agent;
|
mad_agent = mad_send_wr->send_buf.mad_agent;
|
||||||
sge = mad_send_wr->sg_list;
|
sge = mad_send_wr->sg_list;
|
||||||
|
@ -1982,9 +1987,9 @@ static void ib_mad_complete_recv(struct ib_mad_agent_private *mad_agent_priv,
|
||||||
/* user rmpp is in effect
|
/* user rmpp is in effect
|
||||||
* and this is an active RMPP MAD
|
* and this is an active RMPP MAD
|
||||||
*/
|
*/
|
||||||
mad_recv_wc->wc->wr_id = 0;
|
mad_agent_priv->agent.recv_handler(
|
||||||
mad_agent_priv->agent.recv_handler(&mad_agent_priv->agent,
|
&mad_agent_priv->agent, NULL,
|
||||||
mad_recv_wc);
|
mad_recv_wc);
|
||||||
atomic_dec(&mad_agent_priv->refcount);
|
atomic_dec(&mad_agent_priv->refcount);
|
||||||
} else {
|
} else {
|
||||||
/* not user rmpp, revert to normal behavior and
|
/* not user rmpp, revert to normal behavior and
|
||||||
|
@ -1998,9 +2003,10 @@ static void ib_mad_complete_recv(struct ib_mad_agent_private *mad_agent_priv,
|
||||||
spin_unlock_irqrestore(&mad_agent_priv->lock, flags);
|
spin_unlock_irqrestore(&mad_agent_priv->lock, flags);
|
||||||
|
|
||||||
/* Defined behavior is to complete response before request */
|
/* Defined behavior is to complete response before request */
|
||||||
mad_recv_wc->wc->wr_id = (unsigned long) &mad_send_wr->send_buf;
|
mad_agent_priv->agent.recv_handler(
|
||||||
mad_agent_priv->agent.recv_handler(&mad_agent_priv->agent,
|
&mad_agent_priv->agent,
|
||||||
mad_recv_wc);
|
&mad_send_wr->send_buf,
|
||||||
|
mad_recv_wc);
|
||||||
atomic_dec(&mad_agent_priv->refcount);
|
atomic_dec(&mad_agent_priv->refcount);
|
||||||
|
|
||||||
mad_send_wc.status = IB_WC_SUCCESS;
|
mad_send_wc.status = IB_WC_SUCCESS;
|
||||||
|
@ -2009,7 +2015,7 @@ static void ib_mad_complete_recv(struct ib_mad_agent_private *mad_agent_priv,
|
||||||
ib_mad_complete_send_wr(mad_send_wr, &mad_send_wc);
|
ib_mad_complete_send_wr(mad_send_wr, &mad_send_wc);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
mad_agent_priv->agent.recv_handler(&mad_agent_priv->agent,
|
mad_agent_priv->agent.recv_handler(&mad_agent_priv->agent, NULL,
|
||||||
mad_recv_wc);
|
mad_recv_wc);
|
||||||
deref_mad_agent(mad_agent_priv);
|
deref_mad_agent(mad_agent_priv);
|
||||||
}
|
}
|
||||||
|
@ -2172,13 +2178,14 @@ handle_smi(struct ib_mad_port_private *port_priv,
|
||||||
return handle_ib_smi(port_priv, qp_info, wc, port_num, recv, response);
|
return handle_ib_smi(port_priv, qp_info, wc, port_num, recv, response);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ib_mad_recv_done_handler(struct ib_mad_port_private *port_priv,
|
static void ib_mad_recv_done(struct ib_cq *cq, struct ib_wc *wc)
|
||||||
struct ib_wc *wc)
|
|
||||||
{
|
{
|
||||||
|
struct ib_mad_port_private *port_priv = cq->cq_context;
|
||||||
|
struct ib_mad_list_head *mad_list =
|
||||||
|
container_of(wc->wr_cqe, struct ib_mad_list_head, cqe);
|
||||||
struct ib_mad_qp_info *qp_info;
|
struct ib_mad_qp_info *qp_info;
|
||||||
struct ib_mad_private_header *mad_priv_hdr;
|
struct ib_mad_private_header *mad_priv_hdr;
|
||||||
struct ib_mad_private *recv, *response = NULL;
|
struct ib_mad_private *recv, *response = NULL;
|
||||||
struct ib_mad_list_head *mad_list;
|
|
||||||
struct ib_mad_agent_private *mad_agent;
|
struct ib_mad_agent_private *mad_agent;
|
||||||
int port_num;
|
int port_num;
|
||||||
int ret = IB_MAD_RESULT_SUCCESS;
|
int ret = IB_MAD_RESULT_SUCCESS;
|
||||||
|
@ -2186,7 +2193,17 @@ static void ib_mad_recv_done_handler(struct ib_mad_port_private *port_priv,
|
||||||
u16 resp_mad_pkey_index = 0;
|
u16 resp_mad_pkey_index = 0;
|
||||||
bool opa;
|
bool opa;
|
||||||
|
|
||||||
mad_list = (struct ib_mad_list_head *)(unsigned long)wc->wr_id;
|
if (list_empty_careful(&port_priv->port_list))
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (wc->status != IB_WC_SUCCESS) {
|
||||||
|
/*
|
||||||
|
* Receive errors indicate that the QP has entered the error
|
||||||
|
* state - error handling/shutdown code will cleanup
|
||||||
|
*/
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
qp_info = mad_list->mad_queue->qp_info;
|
qp_info = mad_list->mad_queue->qp_info;
|
||||||
dequeue_mad(mad_list);
|
dequeue_mad(mad_list);
|
||||||
|
|
||||||
|
@ -2227,7 +2244,7 @@ static void ib_mad_recv_done_handler(struct ib_mad_port_private *port_priv,
|
||||||
response = alloc_mad_private(mad_size, GFP_KERNEL);
|
response = alloc_mad_private(mad_size, GFP_KERNEL);
|
||||||
if (!response) {
|
if (!response) {
|
||||||
dev_err(&port_priv->device->dev,
|
dev_err(&port_priv->device->dev,
|
||||||
"ib_mad_recv_done_handler no memory for response buffer\n");
|
"%s: no memory for response buffer\n", __func__);
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2413,11 +2430,12 @@ void ib_mad_complete_send_wr(struct ib_mad_send_wr_private *mad_send_wr,
|
||||||
spin_unlock_irqrestore(&mad_agent_priv->lock, flags);
|
spin_unlock_irqrestore(&mad_agent_priv->lock, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ib_mad_send_done_handler(struct ib_mad_port_private *port_priv,
|
static void ib_mad_send_done(struct ib_cq *cq, struct ib_wc *wc)
|
||||||
struct ib_wc *wc)
|
|
||||||
{
|
{
|
||||||
|
struct ib_mad_port_private *port_priv = cq->cq_context;
|
||||||
|
struct ib_mad_list_head *mad_list =
|
||||||
|
container_of(wc->wr_cqe, struct ib_mad_list_head, cqe);
|
||||||
struct ib_mad_send_wr_private *mad_send_wr, *queued_send_wr;
|
struct ib_mad_send_wr_private *mad_send_wr, *queued_send_wr;
|
||||||
struct ib_mad_list_head *mad_list;
|
|
||||||
struct ib_mad_qp_info *qp_info;
|
struct ib_mad_qp_info *qp_info;
|
||||||
struct ib_mad_queue *send_queue;
|
struct ib_mad_queue *send_queue;
|
||||||
struct ib_send_wr *bad_send_wr;
|
struct ib_send_wr *bad_send_wr;
|
||||||
|
@ -2425,7 +2443,14 @@ static void ib_mad_send_done_handler(struct ib_mad_port_private *port_priv,
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
mad_list = (struct ib_mad_list_head *)(unsigned long)wc->wr_id;
|
if (list_empty_careful(&port_priv->port_list))
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (wc->status != IB_WC_SUCCESS) {
|
||||||
|
if (!ib_mad_send_error(port_priv, wc))
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
mad_send_wr = container_of(mad_list, struct ib_mad_send_wr_private,
|
mad_send_wr = container_of(mad_list, struct ib_mad_send_wr_private,
|
||||||
mad_list);
|
mad_list);
|
||||||
send_queue = mad_list->mad_queue;
|
send_queue = mad_list->mad_queue;
|
||||||
|
@ -2490,24 +2515,15 @@ static void mark_sends_for_retry(struct ib_mad_qp_info *qp_info)
|
||||||
spin_unlock_irqrestore(&qp_info->send_queue.lock, flags);
|
spin_unlock_irqrestore(&qp_info->send_queue.lock, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void mad_error_handler(struct ib_mad_port_private *port_priv,
|
static bool ib_mad_send_error(struct ib_mad_port_private *port_priv,
|
||||||
struct ib_wc *wc)
|
struct ib_wc *wc)
|
||||||
{
|
{
|
||||||
struct ib_mad_list_head *mad_list;
|
struct ib_mad_list_head *mad_list =
|
||||||
struct ib_mad_qp_info *qp_info;
|
container_of(wc->wr_cqe, struct ib_mad_list_head, cqe);
|
||||||
|
struct ib_mad_qp_info *qp_info = mad_list->mad_queue->qp_info;
|
||||||
struct ib_mad_send_wr_private *mad_send_wr;
|
struct ib_mad_send_wr_private *mad_send_wr;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
/* Determine if failure was a send or receive */
|
|
||||||
mad_list = (struct ib_mad_list_head *)(unsigned long)wc->wr_id;
|
|
||||||
qp_info = mad_list->mad_queue->qp_info;
|
|
||||||
if (mad_list->mad_queue == &qp_info->recv_queue)
|
|
||||||
/*
|
|
||||||
* Receive errors indicate that the QP has entered the error
|
|
||||||
* state - error handling/shutdown code will cleanup
|
|
||||||
*/
|
|
||||||
return;
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Send errors will transition the QP to SQE - move
|
* Send errors will transition the QP to SQE - move
|
||||||
* QP to RTS and repost flushed work requests
|
* QP to RTS and repost flushed work requests
|
||||||
|
@ -2522,10 +2538,9 @@ static void mad_error_handler(struct ib_mad_port_private *port_priv,
|
||||||
mad_send_wr->retry = 0;
|
mad_send_wr->retry = 0;
|
||||||
ret = ib_post_send(qp_info->qp, &mad_send_wr->send_wr.wr,
|
ret = ib_post_send(qp_info->qp, &mad_send_wr->send_wr.wr,
|
||||||
&bad_send_wr);
|
&bad_send_wr);
|
||||||
if (ret)
|
if (!ret)
|
||||||
ib_mad_send_done_handler(port_priv, wc);
|
return false;
|
||||||
} else
|
}
|
||||||
ib_mad_send_done_handler(port_priv, wc);
|
|
||||||
} else {
|
} else {
|
||||||
struct ib_qp_attr *attr;
|
struct ib_qp_attr *attr;
|
||||||
|
|
||||||
|
@ -2539,42 +2554,14 @@ static void mad_error_handler(struct ib_mad_port_private *port_priv,
|
||||||
kfree(attr);
|
kfree(attr);
|
||||||
if (ret)
|
if (ret)
|
||||||
dev_err(&port_priv->device->dev,
|
dev_err(&port_priv->device->dev,
|
||||||
"mad_error_handler - ib_modify_qp to RTS : %d\n",
|
"%s - ib_modify_qp to RTS: %d\n",
|
||||||
ret);
|
__func__, ret);
|
||||||
else
|
else
|
||||||
mark_sends_for_retry(qp_info);
|
mark_sends_for_retry(qp_info);
|
||||||
}
|
}
|
||||||
ib_mad_send_done_handler(port_priv, wc);
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
return true;
|
||||||
* IB MAD completion callback
|
|
||||||
*/
|
|
||||||
static void ib_mad_completion_handler(struct work_struct *work)
|
|
||||||
{
|
|
||||||
struct ib_mad_port_private *port_priv;
|
|
||||||
struct ib_wc wc;
|
|
||||||
|
|
||||||
port_priv = container_of(work, struct ib_mad_port_private, work);
|
|
||||||
ib_req_notify_cq(port_priv->cq, IB_CQ_NEXT_COMP);
|
|
||||||
|
|
||||||
while (ib_poll_cq(port_priv->cq, 1, &wc) == 1) {
|
|
||||||
if (wc.status == IB_WC_SUCCESS) {
|
|
||||||
switch (wc.opcode) {
|
|
||||||
case IB_WC_SEND:
|
|
||||||
ib_mad_send_done_handler(port_priv, &wc);
|
|
||||||
break;
|
|
||||||
case IB_WC_RECV:
|
|
||||||
ib_mad_recv_done_handler(port_priv, &wc);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
BUG_ON(1);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
} else
|
|
||||||
mad_error_handler(port_priv, &wc);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void cancel_mads(struct ib_mad_agent_private *mad_agent_priv)
|
static void cancel_mads(struct ib_mad_agent_private *mad_agent_priv)
|
||||||
|
@ -2716,7 +2703,7 @@ static void local_completions(struct work_struct *work)
|
||||||
* before request
|
* before request
|
||||||
*/
|
*/
|
||||||
build_smp_wc(recv_mad_agent->agent.qp,
|
build_smp_wc(recv_mad_agent->agent.qp,
|
||||||
(unsigned long) local->mad_send_wr,
|
local->mad_send_wr->send_wr.wr.wr_cqe,
|
||||||
be16_to_cpu(IB_LID_PERMISSIVE),
|
be16_to_cpu(IB_LID_PERMISSIVE),
|
||||||
local->mad_send_wr->send_wr.pkey_index,
|
local->mad_send_wr->send_wr.pkey_index,
|
||||||
recv_mad_agent->agent.port_num, &wc);
|
recv_mad_agent->agent.port_num, &wc);
|
||||||
|
@ -2744,6 +2731,7 @@ static void local_completions(struct work_struct *work)
|
||||||
IB_MAD_SNOOP_RECVS);
|
IB_MAD_SNOOP_RECVS);
|
||||||
recv_mad_agent->agent.recv_handler(
|
recv_mad_agent->agent.recv_handler(
|
||||||
&recv_mad_agent->agent,
|
&recv_mad_agent->agent,
|
||||||
|
&local->mad_send_wr->send_buf,
|
||||||
&local->mad_priv->header.recv_wc);
|
&local->mad_priv->header.recv_wc);
|
||||||
spin_lock_irqsave(&recv_mad_agent->lock, flags);
|
spin_lock_irqsave(&recv_mad_agent->lock, flags);
|
||||||
atomic_dec(&recv_mad_agent->refcount);
|
atomic_dec(&recv_mad_agent->refcount);
|
||||||
|
@ -2855,17 +2843,6 @@ static void timeout_sends(struct work_struct *work)
|
||||||
spin_unlock_irqrestore(&mad_agent_priv->lock, flags);
|
spin_unlock_irqrestore(&mad_agent_priv->lock, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ib_mad_thread_completion_handler(struct ib_cq *cq, void *arg)
|
|
||||||
{
|
|
||||||
struct ib_mad_port_private *port_priv = cq->cq_context;
|
|
||||||
unsigned long flags;
|
|
||||||
|
|
||||||
spin_lock_irqsave(&ib_mad_port_list_lock, flags);
|
|
||||||
if (!list_empty(&port_priv->port_list))
|
|
||||||
queue_work(port_priv->wq, &port_priv->work);
|
|
||||||
spin_unlock_irqrestore(&ib_mad_port_list_lock, flags);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Allocate receive MADs and post receive WRs for them
|
* Allocate receive MADs and post receive WRs for them
|
||||||
*/
|
*/
|
||||||
|
@ -2913,8 +2890,9 @@ static int ib_mad_post_receive_mads(struct ib_mad_qp_info *qp_info,
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
mad_priv->header.mapping = sg_list.addr;
|
mad_priv->header.mapping = sg_list.addr;
|
||||||
recv_wr.wr_id = (unsigned long)&mad_priv->header.mad_list;
|
|
||||||
mad_priv->header.mad_list.mad_queue = recv_queue;
|
mad_priv->header.mad_list.mad_queue = recv_queue;
|
||||||
|
mad_priv->header.mad_list.cqe.done = ib_mad_recv_done;
|
||||||
|
recv_wr.wr_cqe = &mad_priv->header.mad_list.cqe;
|
||||||
|
|
||||||
/* Post receive WR */
|
/* Post receive WR */
|
||||||
spin_lock_irqsave(&recv_queue->lock, flags);
|
spin_lock_irqsave(&recv_queue->lock, flags);
|
||||||
|
@ -3151,7 +3129,6 @@ static int ib_mad_port_open(struct ib_device *device,
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
char name[sizeof "ib_mad123"];
|
char name[sizeof "ib_mad123"];
|
||||||
int has_smi;
|
int has_smi;
|
||||||
struct ib_cq_init_attr cq_attr = {};
|
|
||||||
|
|
||||||
if (WARN_ON(rdma_max_mad_size(device, port_num) < IB_MGMT_MAD_SIZE))
|
if (WARN_ON(rdma_max_mad_size(device, port_num) < IB_MGMT_MAD_SIZE))
|
||||||
return -EFAULT;
|
return -EFAULT;
|
||||||
|
@ -3179,10 +3156,8 @@ static int ib_mad_port_open(struct ib_device *device,
|
||||||
if (has_smi)
|
if (has_smi)
|
||||||
cq_size *= 2;
|
cq_size *= 2;
|
||||||
|
|
||||||
cq_attr.cqe = cq_size;
|
port_priv->cq = ib_alloc_cq(port_priv->device, port_priv, cq_size, 0,
|
||||||
port_priv->cq = ib_create_cq(port_priv->device,
|
IB_POLL_WORKQUEUE);
|
||||||
ib_mad_thread_completion_handler,
|
|
||||||
NULL, port_priv, &cq_attr);
|
|
||||||
if (IS_ERR(port_priv->cq)) {
|
if (IS_ERR(port_priv->cq)) {
|
||||||
dev_err(&device->dev, "Couldn't create ib_mad CQ\n");
|
dev_err(&device->dev, "Couldn't create ib_mad CQ\n");
|
||||||
ret = PTR_ERR(port_priv->cq);
|
ret = PTR_ERR(port_priv->cq);
|
||||||
|
@ -3211,7 +3186,6 @@ static int ib_mad_port_open(struct ib_device *device,
|
||||||
ret = -ENOMEM;
|
ret = -ENOMEM;
|
||||||
goto error8;
|
goto error8;
|
||||||
}
|
}
|
||||||
INIT_WORK(&port_priv->work, ib_mad_completion_handler);
|
|
||||||
|
|
||||||
spin_lock_irqsave(&ib_mad_port_list_lock, flags);
|
spin_lock_irqsave(&ib_mad_port_list_lock, flags);
|
||||||
list_add_tail(&port_priv->port_list, &ib_mad_port_list);
|
list_add_tail(&port_priv->port_list, &ib_mad_port_list);
|
||||||
|
@ -3238,7 +3212,7 @@ static int ib_mad_port_open(struct ib_device *device,
|
||||||
error6:
|
error6:
|
||||||
ib_dealloc_pd(port_priv->pd);
|
ib_dealloc_pd(port_priv->pd);
|
||||||
error4:
|
error4:
|
||||||
ib_destroy_cq(port_priv->cq);
|
ib_free_cq(port_priv->cq);
|
||||||
cleanup_recv_queue(&port_priv->qp_info[1]);
|
cleanup_recv_queue(&port_priv->qp_info[1]);
|
||||||
cleanup_recv_queue(&port_priv->qp_info[0]);
|
cleanup_recv_queue(&port_priv->qp_info[0]);
|
||||||
error3:
|
error3:
|
||||||
|
@ -3271,7 +3245,7 @@ static int ib_mad_port_close(struct ib_device *device, int port_num)
|
||||||
destroy_mad_qp(&port_priv->qp_info[1]);
|
destroy_mad_qp(&port_priv->qp_info[1]);
|
||||||
destroy_mad_qp(&port_priv->qp_info[0]);
|
destroy_mad_qp(&port_priv->qp_info[0]);
|
||||||
ib_dealloc_pd(port_priv->pd);
|
ib_dealloc_pd(port_priv->pd);
|
||||||
ib_destroy_cq(port_priv->cq);
|
ib_free_cq(port_priv->cq);
|
||||||
cleanup_recv_queue(&port_priv->qp_info[1]);
|
cleanup_recv_queue(&port_priv->qp_info[1]);
|
||||||
cleanup_recv_queue(&port_priv->qp_info[0]);
|
cleanup_recv_queue(&port_priv->qp_info[0]);
|
||||||
/* XXX: Handle deallocation of MAD registration tables */
|
/* XXX: Handle deallocation of MAD registration tables */
|
||||||
|
|
|
@ -64,6 +64,7 @@
|
||||||
|
|
||||||
struct ib_mad_list_head {
|
struct ib_mad_list_head {
|
||||||
struct list_head list;
|
struct list_head list;
|
||||||
|
struct ib_cqe cqe;
|
||||||
struct ib_mad_queue *mad_queue;
|
struct ib_mad_queue *mad_queue;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -204,7 +205,6 @@ struct ib_mad_port_private {
|
||||||
struct ib_mad_mgmt_version_table version[MAX_MGMT_VERSION];
|
struct ib_mad_mgmt_version_table version[MAX_MGMT_VERSION];
|
||||||
struct list_head agent_list;
|
struct list_head agent_list;
|
||||||
struct workqueue_struct *wq;
|
struct workqueue_struct *wq;
|
||||||
struct work_struct work;
|
|
||||||
struct ib_mad_qp_info qp_info[IB_MAD_QPS_CORE];
|
struct ib_mad_qp_info qp_info[IB_MAD_QPS_CORE];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -723,14 +723,27 @@ EXPORT_SYMBOL(ib_sa_get_mcmember_rec);
|
||||||
|
|
||||||
int ib_init_ah_from_mcmember(struct ib_device *device, u8 port_num,
|
int ib_init_ah_from_mcmember(struct ib_device *device, u8 port_num,
|
||||||
struct ib_sa_mcmember_rec *rec,
|
struct ib_sa_mcmember_rec *rec,
|
||||||
|
struct net_device *ndev,
|
||||||
|
enum ib_gid_type gid_type,
|
||||||
struct ib_ah_attr *ah_attr)
|
struct ib_ah_attr *ah_attr)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
u16 gid_index;
|
u16 gid_index;
|
||||||
u8 p;
|
u8 p;
|
||||||
|
|
||||||
ret = ib_find_cached_gid(device, &rec->port_gid,
|
if (rdma_protocol_roce(device, port_num)) {
|
||||||
NULL, &p, &gid_index);
|
ret = ib_find_cached_gid_by_port(device, &rec->port_gid,
|
||||||
|
gid_type, port_num,
|
||||||
|
ndev,
|
||||||
|
&gid_index);
|
||||||
|
} else if (rdma_protocol_ib(device, port_num)) {
|
||||||
|
ret = ib_find_cached_gid(device, &rec->port_gid,
|
||||||
|
IB_GID_TYPE_IB, NULL, &p,
|
||||||
|
&gid_index);
|
||||||
|
} else {
|
||||||
|
ret = -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
if (ret)
|
if (ret)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
|
|
|
@ -67,17 +67,53 @@ struct netdev_event_work {
|
||||||
struct netdev_event_work_cmd cmds[ROCE_NETDEV_CALLBACK_SZ];
|
struct netdev_event_work_cmd cmds[ROCE_NETDEV_CALLBACK_SZ];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const struct {
|
||||||
|
bool (*is_supported)(const struct ib_device *device, u8 port_num);
|
||||||
|
enum ib_gid_type gid_type;
|
||||||
|
} PORT_CAP_TO_GID_TYPE[] = {
|
||||||
|
{rdma_protocol_roce_eth_encap, IB_GID_TYPE_ROCE},
|
||||||
|
{rdma_protocol_roce_udp_encap, IB_GID_TYPE_ROCE_UDP_ENCAP},
|
||||||
|
};
|
||||||
|
|
||||||
|
#define CAP_TO_GID_TABLE_SIZE ARRAY_SIZE(PORT_CAP_TO_GID_TYPE)
|
||||||
|
|
||||||
|
unsigned long roce_gid_type_mask_support(struct ib_device *ib_dev, u8 port)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
unsigned int ret_flags = 0;
|
||||||
|
|
||||||
|
if (!rdma_protocol_roce(ib_dev, port))
|
||||||
|
return 1UL << IB_GID_TYPE_IB;
|
||||||
|
|
||||||
|
for (i = 0; i < CAP_TO_GID_TABLE_SIZE; i++)
|
||||||
|
if (PORT_CAP_TO_GID_TYPE[i].is_supported(ib_dev, port))
|
||||||
|
ret_flags |= 1UL << PORT_CAP_TO_GID_TYPE[i].gid_type;
|
||||||
|
|
||||||
|
return ret_flags;
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(roce_gid_type_mask_support);
|
||||||
|
|
||||||
static void update_gid(enum gid_op_type gid_op, struct ib_device *ib_dev,
|
static void update_gid(enum gid_op_type gid_op, struct ib_device *ib_dev,
|
||||||
u8 port, union ib_gid *gid,
|
u8 port, union ib_gid *gid,
|
||||||
struct ib_gid_attr *gid_attr)
|
struct ib_gid_attr *gid_attr)
|
||||||
{
|
{
|
||||||
switch (gid_op) {
|
int i;
|
||||||
case GID_ADD:
|
unsigned long gid_type_mask = roce_gid_type_mask_support(ib_dev, port);
|
||||||
ib_cache_gid_add(ib_dev, port, gid, gid_attr);
|
|
||||||
break;
|
for (i = 0; i < IB_GID_TYPE_SIZE; i++) {
|
||||||
case GID_DEL:
|
if ((1UL << i) & gid_type_mask) {
|
||||||
ib_cache_gid_del(ib_dev, port, gid, gid_attr);
|
gid_attr->gid_type = i;
|
||||||
break;
|
switch (gid_op) {
|
||||||
|
case GID_ADD:
|
||||||
|
ib_cache_gid_add(ib_dev, port,
|
||||||
|
gid, gid_attr);
|
||||||
|
break;
|
||||||
|
case GID_DEL:
|
||||||
|
ib_cache_gid_del(ib_dev, port,
|
||||||
|
gid, gid_attr);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -103,18 +139,6 @@ static enum bonding_slave_state is_eth_active_slave_of_bonding_rcu(struct net_de
|
||||||
return BONDING_SLAVE_STATE_NA;
|
return BONDING_SLAVE_STATE_NA;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool is_upper_dev_rcu(struct net_device *dev, struct net_device *upper)
|
|
||||||
{
|
|
||||||
struct net_device *_upper = NULL;
|
|
||||||
struct list_head *iter;
|
|
||||||
|
|
||||||
netdev_for_each_all_upper_dev_rcu(dev, _upper, iter)
|
|
||||||
if (_upper == upper)
|
|
||||||
break;
|
|
||||||
|
|
||||||
return _upper == upper;
|
|
||||||
}
|
|
||||||
|
|
||||||
#define REQUIRED_BOND_STATES (BONDING_SLAVE_STATE_ACTIVE | \
|
#define REQUIRED_BOND_STATES (BONDING_SLAVE_STATE_ACTIVE | \
|
||||||
BONDING_SLAVE_STATE_NA)
|
BONDING_SLAVE_STATE_NA)
|
||||||
static int is_eth_port_of_netdev(struct ib_device *ib_dev, u8 port,
|
static int is_eth_port_of_netdev(struct ib_device *ib_dev, u8 port,
|
||||||
|
@ -132,7 +156,7 @@ static int is_eth_port_of_netdev(struct ib_device *ib_dev, u8 port,
|
||||||
if (!real_dev)
|
if (!real_dev)
|
||||||
real_dev = event_ndev;
|
real_dev = event_ndev;
|
||||||
|
|
||||||
res = ((is_upper_dev_rcu(rdma_ndev, event_ndev) &&
|
res = ((rdma_is_upper_dev_rcu(rdma_ndev, event_ndev) &&
|
||||||
(is_eth_active_slave_of_bonding_rcu(rdma_ndev, real_dev) &
|
(is_eth_active_slave_of_bonding_rcu(rdma_ndev, real_dev) &
|
||||||
REQUIRED_BOND_STATES)) ||
|
REQUIRED_BOND_STATES)) ||
|
||||||
real_dev == rdma_ndev);
|
real_dev == rdma_ndev);
|
||||||
|
@ -178,7 +202,7 @@ static int upper_device_filter(struct ib_device *ib_dev, u8 port,
|
||||||
return 1;
|
return 1;
|
||||||
|
|
||||||
rcu_read_lock();
|
rcu_read_lock();
|
||||||
res = is_upper_dev_rcu(rdma_ndev, event_ndev);
|
res = rdma_is_upper_dev_rcu(rdma_ndev, event_ndev);
|
||||||
rcu_read_unlock();
|
rcu_read_unlock();
|
||||||
|
|
||||||
return res;
|
return res;
|
||||||
|
@ -203,10 +227,12 @@ static void enum_netdev_default_gids(struct ib_device *ib_dev,
|
||||||
u8 port, struct net_device *event_ndev,
|
u8 port, struct net_device *event_ndev,
|
||||||
struct net_device *rdma_ndev)
|
struct net_device *rdma_ndev)
|
||||||
{
|
{
|
||||||
|
unsigned long gid_type_mask;
|
||||||
|
|
||||||
rcu_read_lock();
|
rcu_read_lock();
|
||||||
if (!rdma_ndev ||
|
if (!rdma_ndev ||
|
||||||
((rdma_ndev != event_ndev &&
|
((rdma_ndev != event_ndev &&
|
||||||
!is_upper_dev_rcu(rdma_ndev, event_ndev)) ||
|
!rdma_is_upper_dev_rcu(rdma_ndev, event_ndev)) ||
|
||||||
is_eth_active_slave_of_bonding_rcu(rdma_ndev,
|
is_eth_active_slave_of_bonding_rcu(rdma_ndev,
|
||||||
netdev_master_upper_dev_get_rcu(rdma_ndev)) ==
|
netdev_master_upper_dev_get_rcu(rdma_ndev)) ==
|
||||||
BONDING_SLAVE_STATE_INACTIVE)) {
|
BONDING_SLAVE_STATE_INACTIVE)) {
|
||||||
|
@ -215,7 +241,9 @@ static void enum_netdev_default_gids(struct ib_device *ib_dev,
|
||||||
}
|
}
|
||||||
rcu_read_unlock();
|
rcu_read_unlock();
|
||||||
|
|
||||||
ib_cache_gid_set_default_gid(ib_dev, port, rdma_ndev,
|
gid_type_mask = roce_gid_type_mask_support(ib_dev, port);
|
||||||
|
|
||||||
|
ib_cache_gid_set_default_gid(ib_dev, port, rdma_ndev, gid_type_mask,
|
||||||
IB_CACHE_GID_DEFAULT_MODE_SET);
|
IB_CACHE_GID_DEFAULT_MODE_SET);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -234,12 +262,17 @@ static void bond_delete_netdev_default_gids(struct ib_device *ib_dev,
|
||||||
|
|
||||||
rcu_read_lock();
|
rcu_read_lock();
|
||||||
|
|
||||||
if (is_upper_dev_rcu(rdma_ndev, event_ndev) &&
|
if (rdma_is_upper_dev_rcu(rdma_ndev, event_ndev) &&
|
||||||
is_eth_active_slave_of_bonding_rcu(rdma_ndev, real_dev) ==
|
is_eth_active_slave_of_bonding_rcu(rdma_ndev, real_dev) ==
|
||||||
BONDING_SLAVE_STATE_INACTIVE) {
|
BONDING_SLAVE_STATE_INACTIVE) {
|
||||||
|
unsigned long gid_type_mask;
|
||||||
|
|
||||||
rcu_read_unlock();
|
rcu_read_unlock();
|
||||||
|
|
||||||
|
gid_type_mask = roce_gid_type_mask_support(ib_dev, port);
|
||||||
|
|
||||||
ib_cache_gid_set_default_gid(ib_dev, port, rdma_ndev,
|
ib_cache_gid_set_default_gid(ib_dev, port, rdma_ndev,
|
||||||
|
gid_type_mask,
|
||||||
IB_CACHE_GID_DEFAULT_MODE_DELETE);
|
IB_CACHE_GID_DEFAULT_MODE_DELETE);
|
||||||
} else {
|
} else {
|
||||||
rcu_read_unlock();
|
rcu_read_unlock();
|
||||||
|
|
|
@ -49,7 +49,9 @@
|
||||||
#include <net/netlink.h>
|
#include <net/netlink.h>
|
||||||
#include <uapi/rdma/ib_user_sa.h>
|
#include <uapi/rdma/ib_user_sa.h>
|
||||||
#include <rdma/ib_marshall.h>
|
#include <rdma/ib_marshall.h>
|
||||||
|
#include <rdma/ib_addr.h>
|
||||||
#include "sa.h"
|
#include "sa.h"
|
||||||
|
#include "core_priv.h"
|
||||||
|
|
||||||
MODULE_AUTHOR("Roland Dreier");
|
MODULE_AUTHOR("Roland Dreier");
|
||||||
MODULE_DESCRIPTION("InfiniBand subnet administration query support");
|
MODULE_DESCRIPTION("InfiniBand subnet administration query support");
|
||||||
|
@ -715,7 +717,9 @@ static int ib_nl_handle_set_timeout(struct sk_buff *skb,
|
||||||
struct nlattr *tb[LS_NLA_TYPE_MAX];
|
struct nlattr *tb[LS_NLA_TYPE_MAX];
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
if (!netlink_capable(skb, CAP_NET_ADMIN))
|
if (!(nlh->nlmsg_flags & NLM_F_REQUEST) ||
|
||||||
|
!(NETLINK_CB(skb).sk) ||
|
||||||
|
!netlink_capable(skb, CAP_NET_ADMIN))
|
||||||
return -EPERM;
|
return -EPERM;
|
||||||
|
|
||||||
ret = nla_parse(tb, LS_NLA_TYPE_MAX - 1, nlmsg_data(nlh),
|
ret = nla_parse(tb, LS_NLA_TYPE_MAX - 1, nlmsg_data(nlh),
|
||||||
|
@ -789,7 +793,9 @@ static int ib_nl_handle_resolve_resp(struct sk_buff *skb,
|
||||||
int found = 0;
|
int found = 0;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
if (!netlink_capable(skb, CAP_NET_ADMIN))
|
if ((nlh->nlmsg_flags & NLM_F_REQUEST) ||
|
||||||
|
!(NETLINK_CB(skb).sk) ||
|
||||||
|
!netlink_capable(skb, CAP_NET_ADMIN))
|
||||||
return -EPERM;
|
return -EPERM;
|
||||||
|
|
||||||
spin_lock_irqsave(&ib_nl_request_lock, flags);
|
spin_lock_irqsave(&ib_nl_request_lock, flags);
|
||||||
|
@ -996,7 +1002,8 @@ int ib_init_ah_from_path(struct ib_device *device, u8 port_num,
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
u16 gid_index;
|
u16 gid_index;
|
||||||
int force_grh;
|
int use_roce;
|
||||||
|
struct net_device *ndev = NULL;
|
||||||
|
|
||||||
memset(ah_attr, 0, sizeof *ah_attr);
|
memset(ah_attr, 0, sizeof *ah_attr);
|
||||||
ah_attr->dlid = be16_to_cpu(rec->dlid);
|
ah_attr->dlid = be16_to_cpu(rec->dlid);
|
||||||
|
@ -1006,16 +1013,71 @@ int ib_init_ah_from_path(struct ib_device *device, u8 port_num,
|
||||||
ah_attr->port_num = port_num;
|
ah_attr->port_num = port_num;
|
||||||
ah_attr->static_rate = rec->rate;
|
ah_attr->static_rate = rec->rate;
|
||||||
|
|
||||||
force_grh = rdma_cap_eth_ah(device, port_num);
|
use_roce = rdma_cap_eth_ah(device, port_num);
|
||||||
|
|
||||||
if (rec->hop_limit > 1 || force_grh) {
|
if (use_roce) {
|
||||||
struct net_device *ndev = ib_get_ndev_from_path(rec);
|
struct net_device *idev;
|
||||||
|
struct net_device *resolved_dev;
|
||||||
|
struct rdma_dev_addr dev_addr = {.bound_dev_if = rec->ifindex,
|
||||||
|
.net = rec->net ? rec->net :
|
||||||
|
&init_net};
|
||||||
|
union {
|
||||||
|
struct sockaddr _sockaddr;
|
||||||
|
struct sockaddr_in _sockaddr_in;
|
||||||
|
struct sockaddr_in6 _sockaddr_in6;
|
||||||
|
} sgid_addr, dgid_addr;
|
||||||
|
|
||||||
|
if (!device->get_netdev)
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
|
rdma_gid2ip(&sgid_addr._sockaddr, &rec->sgid);
|
||||||
|
rdma_gid2ip(&dgid_addr._sockaddr, &rec->dgid);
|
||||||
|
|
||||||
|
/* validate the route */
|
||||||
|
ret = rdma_resolve_ip_route(&sgid_addr._sockaddr,
|
||||||
|
&dgid_addr._sockaddr, &dev_addr);
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
if ((dev_addr.network == RDMA_NETWORK_IPV4 ||
|
||||||
|
dev_addr.network == RDMA_NETWORK_IPV6) &&
|
||||||
|
rec->gid_type != IB_GID_TYPE_ROCE_UDP_ENCAP)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
idev = device->get_netdev(device, port_num);
|
||||||
|
if (!idev)
|
||||||
|
return -ENODEV;
|
||||||
|
|
||||||
|
resolved_dev = dev_get_by_index(dev_addr.net,
|
||||||
|
dev_addr.bound_dev_if);
|
||||||
|
if (resolved_dev->flags & IFF_LOOPBACK) {
|
||||||
|
dev_put(resolved_dev);
|
||||||
|
resolved_dev = idev;
|
||||||
|
dev_hold(resolved_dev);
|
||||||
|
}
|
||||||
|
ndev = ib_get_ndev_from_path(rec);
|
||||||
|
rcu_read_lock();
|
||||||
|
if ((ndev && ndev != resolved_dev) ||
|
||||||
|
(resolved_dev != idev &&
|
||||||
|
!rdma_is_upper_dev_rcu(idev, resolved_dev)))
|
||||||
|
ret = -EHOSTUNREACH;
|
||||||
|
rcu_read_unlock();
|
||||||
|
dev_put(idev);
|
||||||
|
dev_put(resolved_dev);
|
||||||
|
if (ret) {
|
||||||
|
if (ndev)
|
||||||
|
dev_put(ndev);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (rec->hop_limit > 1 || use_roce) {
|
||||||
ah_attr->ah_flags = IB_AH_GRH;
|
ah_attr->ah_flags = IB_AH_GRH;
|
||||||
ah_attr->grh.dgid = rec->dgid;
|
ah_attr->grh.dgid = rec->dgid;
|
||||||
|
|
||||||
ret = ib_find_cached_gid(device, &rec->sgid, ndev, &port_num,
|
ret = ib_find_cached_gid_by_port(device, &rec->sgid,
|
||||||
&gid_index);
|
rec->gid_type, port_num, ndev,
|
||||||
|
&gid_index);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
if (ndev)
|
if (ndev)
|
||||||
dev_put(ndev);
|
dev_put(ndev);
|
||||||
|
@ -1029,9 +1091,10 @@ int ib_init_ah_from_path(struct ib_device *device, u8 port_num,
|
||||||
if (ndev)
|
if (ndev)
|
||||||
dev_put(ndev);
|
dev_put(ndev);
|
||||||
}
|
}
|
||||||
if (force_grh) {
|
|
||||||
|
if (use_roce)
|
||||||
memcpy(ah_attr->dmac, rec->dmac, ETH_ALEN);
|
memcpy(ah_attr->dmac, rec->dmac, ETH_ALEN);
|
||||||
}
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(ib_init_ah_from_path);
|
EXPORT_SYMBOL(ib_init_ah_from_path);
|
||||||
|
@ -1157,6 +1220,7 @@ static void ib_sa_path_rec_callback(struct ib_sa_query *sa_query,
|
||||||
mad->data, &rec);
|
mad->data, &rec);
|
||||||
rec.net = NULL;
|
rec.net = NULL;
|
||||||
rec.ifindex = 0;
|
rec.ifindex = 0;
|
||||||
|
rec.gid_type = IB_GID_TYPE_IB;
|
||||||
memset(rec.dmac, 0, ETH_ALEN);
|
memset(rec.dmac, 0, ETH_ALEN);
|
||||||
query->callback(status, &rec, query->context);
|
query->callback(status, &rec, query->context);
|
||||||
} else
|
} else
|
||||||
|
@ -1609,14 +1673,15 @@ static void send_handler(struct ib_mad_agent *agent,
|
||||||
}
|
}
|
||||||
|
|
||||||
static void recv_handler(struct ib_mad_agent *mad_agent,
|
static void recv_handler(struct ib_mad_agent *mad_agent,
|
||||||
|
struct ib_mad_send_buf *send_buf,
|
||||||
struct ib_mad_recv_wc *mad_recv_wc)
|
struct ib_mad_recv_wc *mad_recv_wc)
|
||||||
{
|
{
|
||||||
struct ib_sa_query *query;
|
struct ib_sa_query *query;
|
||||||
struct ib_mad_send_buf *mad_buf;
|
|
||||||
|
|
||||||
mad_buf = (void *) (unsigned long) mad_recv_wc->wc->wr_id;
|
if (!send_buf)
|
||||||
query = mad_buf->context[0];
|
return;
|
||||||
|
|
||||||
|
query = send_buf->context[0];
|
||||||
if (query->callback) {
|
if (query->callback) {
|
||||||
if (mad_recv_wc->wc->status == IB_WC_SUCCESS)
|
if (mad_recv_wc->wc->status == IB_WC_SUCCESS)
|
||||||
query->callback(query,
|
query->callback(query,
|
||||||
|
|
|
@ -37,15 +37,27 @@
|
||||||
#include <linux/slab.h>
|
#include <linux/slab.h>
|
||||||
#include <linux/stat.h>
|
#include <linux/stat.h>
|
||||||
#include <linux/string.h>
|
#include <linux/string.h>
|
||||||
|
#include <linux/netdevice.h>
|
||||||
|
|
||||||
#include <rdma/ib_mad.h>
|
#include <rdma/ib_mad.h>
|
||||||
|
#include <rdma/ib_pma.h>
|
||||||
|
|
||||||
|
struct ib_port;
|
||||||
|
|
||||||
|
struct gid_attr_group {
|
||||||
|
struct ib_port *port;
|
||||||
|
struct kobject kobj;
|
||||||
|
struct attribute_group ndev;
|
||||||
|
struct attribute_group type;
|
||||||
|
};
|
||||||
struct ib_port {
|
struct ib_port {
|
||||||
struct kobject kobj;
|
struct kobject kobj;
|
||||||
struct ib_device *ibdev;
|
struct ib_device *ibdev;
|
||||||
|
struct gid_attr_group *gid_attr_group;
|
||||||
struct attribute_group gid_group;
|
struct attribute_group gid_group;
|
||||||
struct attribute_group pkey_group;
|
struct attribute_group pkey_group;
|
||||||
u8 port_num;
|
u8 port_num;
|
||||||
|
struct attribute_group *pma_table;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct port_attribute {
|
struct port_attribute {
|
||||||
|
@ -65,6 +77,7 @@ struct port_table_attribute {
|
||||||
struct port_attribute attr;
|
struct port_attribute attr;
|
||||||
char name[8];
|
char name[8];
|
||||||
int index;
|
int index;
|
||||||
|
__be16 attr_id;
|
||||||
};
|
};
|
||||||
|
|
||||||
static ssize_t port_attr_show(struct kobject *kobj,
|
static ssize_t port_attr_show(struct kobject *kobj,
|
||||||
|
@ -84,6 +97,24 @@ static const struct sysfs_ops port_sysfs_ops = {
|
||||||
.show = port_attr_show
|
.show = port_attr_show
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static ssize_t gid_attr_show(struct kobject *kobj,
|
||||||
|
struct attribute *attr, char *buf)
|
||||||
|
{
|
||||||
|
struct port_attribute *port_attr =
|
||||||
|
container_of(attr, struct port_attribute, attr);
|
||||||
|
struct ib_port *p = container_of(kobj, struct gid_attr_group,
|
||||||
|
kobj)->port;
|
||||||
|
|
||||||
|
if (!port_attr->show)
|
||||||
|
return -EIO;
|
||||||
|
|
||||||
|
return port_attr->show(p, port_attr, buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
static const struct sysfs_ops gid_attr_sysfs_ops = {
|
||||||
|
.show = gid_attr_show
|
||||||
|
};
|
||||||
|
|
||||||
static ssize_t state_show(struct ib_port *p, struct port_attribute *unused,
|
static ssize_t state_show(struct ib_port *p, struct port_attribute *unused,
|
||||||
char *buf)
|
char *buf)
|
||||||
{
|
{
|
||||||
|
@ -281,6 +312,46 @@ static struct attribute *port_default_attrs[] = {
|
||||||
NULL
|
NULL
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static size_t print_ndev(struct ib_gid_attr *gid_attr, char *buf)
|
||||||
|
{
|
||||||
|
if (!gid_attr->ndev)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
return sprintf(buf, "%s\n", gid_attr->ndev->name);
|
||||||
|
}
|
||||||
|
|
||||||
|
static size_t print_gid_type(struct ib_gid_attr *gid_attr, char *buf)
|
||||||
|
{
|
||||||
|
return sprintf(buf, "%s\n", ib_cache_gid_type_str(gid_attr->gid_type));
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t _show_port_gid_attr(struct ib_port *p,
|
||||||
|
struct port_attribute *attr,
|
||||||
|
char *buf,
|
||||||
|
size_t (*print)(struct ib_gid_attr *gid_attr,
|
||||||
|
char *buf))
|
||||||
|
{
|
||||||
|
struct port_table_attribute *tab_attr =
|
||||||
|
container_of(attr, struct port_table_attribute, attr);
|
||||||
|
union ib_gid gid;
|
||||||
|
struct ib_gid_attr gid_attr = {};
|
||||||
|
ssize_t ret;
|
||||||
|
va_list args;
|
||||||
|
|
||||||
|
ret = ib_query_gid(p->ibdev, p->port_num, tab_attr->index, &gid,
|
||||||
|
&gid_attr);
|
||||||
|
if (ret)
|
||||||
|
goto err;
|
||||||
|
|
||||||
|
ret = print(&gid_attr, buf);
|
||||||
|
|
||||||
|
err:
|
||||||
|
if (gid_attr.ndev)
|
||||||
|
dev_put(gid_attr.ndev);
|
||||||
|
va_end(args);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
static ssize_t show_port_gid(struct ib_port *p, struct port_attribute *attr,
|
static ssize_t show_port_gid(struct ib_port *p, struct port_attribute *attr,
|
||||||
char *buf)
|
char *buf)
|
||||||
{
|
{
|
||||||
|
@ -296,6 +367,19 @@ static ssize_t show_port_gid(struct ib_port *p, struct port_attribute *attr,
|
||||||
return sprintf(buf, "%pI6\n", gid.raw);
|
return sprintf(buf, "%pI6\n", gid.raw);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static ssize_t show_port_gid_attr_ndev(struct ib_port *p,
|
||||||
|
struct port_attribute *attr, char *buf)
|
||||||
|
{
|
||||||
|
return _show_port_gid_attr(p, attr, buf, print_ndev);
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t show_port_gid_attr_gid_type(struct ib_port *p,
|
||||||
|
struct port_attribute *attr,
|
||||||
|
char *buf)
|
||||||
|
{
|
||||||
|
return _show_port_gid_attr(p, attr, buf, print_gid_type);
|
||||||
|
}
|
||||||
|
|
||||||
static ssize_t show_port_pkey(struct ib_port *p, struct port_attribute *attr,
|
static ssize_t show_port_pkey(struct ib_port *p, struct port_attribute *attr,
|
||||||
char *buf)
|
char *buf)
|
||||||
{
|
{
|
||||||
|
@ -314,24 +398,32 @@ static ssize_t show_port_pkey(struct ib_port *p, struct port_attribute *attr,
|
||||||
#define PORT_PMA_ATTR(_name, _counter, _width, _offset) \
|
#define PORT_PMA_ATTR(_name, _counter, _width, _offset) \
|
||||||
struct port_table_attribute port_pma_attr_##_name = { \
|
struct port_table_attribute port_pma_attr_##_name = { \
|
||||||
.attr = __ATTR(_name, S_IRUGO, show_pma_counter, NULL), \
|
.attr = __ATTR(_name, S_IRUGO, show_pma_counter, NULL), \
|
||||||
.index = (_offset) | ((_width) << 16) | ((_counter) << 24) \
|
.index = (_offset) | ((_width) << 16) | ((_counter) << 24), \
|
||||||
|
.attr_id = IB_PMA_PORT_COUNTERS , \
|
||||||
}
|
}
|
||||||
|
|
||||||
static ssize_t show_pma_counter(struct ib_port *p, struct port_attribute *attr,
|
#define PORT_PMA_ATTR_EXT(_name, _width, _offset) \
|
||||||
char *buf)
|
struct port_table_attribute port_pma_attr_ext_##_name = { \
|
||||||
|
.attr = __ATTR(_name, S_IRUGO, show_pma_counter, NULL), \
|
||||||
|
.index = (_offset) | ((_width) << 16), \
|
||||||
|
.attr_id = IB_PMA_PORT_COUNTERS_EXT , \
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Get a Perfmgmt MAD block of data.
|
||||||
|
* Returns error code or the number of bytes retrieved.
|
||||||
|
*/
|
||||||
|
static int get_perf_mad(struct ib_device *dev, int port_num, __be16 attr,
|
||||||
|
void *data, int offset, size_t size)
|
||||||
{
|
{
|
||||||
struct port_table_attribute *tab_attr =
|
struct ib_mad *in_mad;
|
||||||
container_of(attr, struct port_table_attribute, attr);
|
struct ib_mad *out_mad;
|
||||||
int offset = tab_attr->index & 0xffff;
|
|
||||||
int width = (tab_attr->index >> 16) & 0xff;
|
|
||||||
struct ib_mad *in_mad = NULL;
|
|
||||||
struct ib_mad *out_mad = NULL;
|
|
||||||
size_t mad_size = sizeof(*out_mad);
|
size_t mad_size = sizeof(*out_mad);
|
||||||
u16 out_mad_pkey_index = 0;
|
u16 out_mad_pkey_index = 0;
|
||||||
ssize_t ret;
|
ssize_t ret;
|
||||||
|
|
||||||
if (!p->ibdev->process_mad)
|
if (!dev->process_mad)
|
||||||
return sprintf(buf, "N/A (no PMA)\n");
|
return -ENOSYS;
|
||||||
|
|
||||||
in_mad = kzalloc(sizeof *in_mad, GFP_KERNEL);
|
in_mad = kzalloc(sizeof *in_mad, GFP_KERNEL);
|
||||||
out_mad = kmalloc(sizeof *out_mad, GFP_KERNEL);
|
out_mad = kmalloc(sizeof *out_mad, GFP_KERNEL);
|
||||||
|
@ -344,12 +436,13 @@ static ssize_t show_pma_counter(struct ib_port *p, struct port_attribute *attr,
|
||||||
in_mad->mad_hdr.mgmt_class = IB_MGMT_CLASS_PERF_MGMT;
|
in_mad->mad_hdr.mgmt_class = IB_MGMT_CLASS_PERF_MGMT;
|
||||||
in_mad->mad_hdr.class_version = 1;
|
in_mad->mad_hdr.class_version = 1;
|
||||||
in_mad->mad_hdr.method = IB_MGMT_METHOD_GET;
|
in_mad->mad_hdr.method = IB_MGMT_METHOD_GET;
|
||||||
in_mad->mad_hdr.attr_id = cpu_to_be16(0x12); /* PortCounters */
|
in_mad->mad_hdr.attr_id = attr;
|
||||||
|
|
||||||
in_mad->data[41] = p->port_num; /* PortSelect field */
|
if (attr != IB_PMA_CLASS_PORT_INFO)
|
||||||
|
in_mad->data[41] = port_num; /* PortSelect field */
|
||||||
|
|
||||||
if ((p->ibdev->process_mad(p->ibdev, IB_MAD_IGNORE_MKEY,
|
if ((dev->process_mad(dev, IB_MAD_IGNORE_MKEY,
|
||||||
p->port_num, NULL, NULL,
|
port_num, NULL, NULL,
|
||||||
(const struct ib_mad_hdr *)in_mad, mad_size,
|
(const struct ib_mad_hdr *)in_mad, mad_size,
|
||||||
(struct ib_mad_hdr *)out_mad, &mad_size,
|
(struct ib_mad_hdr *)out_mad, &mad_size,
|
||||||
&out_mad_pkey_index) &
|
&out_mad_pkey_index) &
|
||||||
|
@ -358,30 +451,53 @@ static ssize_t show_pma_counter(struct ib_port *p, struct port_attribute *attr,
|
||||||
ret = -EINVAL;
|
ret = -EINVAL;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
memcpy(data, out_mad->data + offset, size);
|
||||||
switch (width) {
|
ret = size;
|
||||||
case 4:
|
|
||||||
ret = sprintf(buf, "%u\n", (out_mad->data[40 + offset / 8] >>
|
|
||||||
(4 - (offset % 8))) & 0xf);
|
|
||||||
break;
|
|
||||||
case 8:
|
|
||||||
ret = sprintf(buf, "%u\n", out_mad->data[40 + offset / 8]);
|
|
||||||
break;
|
|
||||||
case 16:
|
|
||||||
ret = sprintf(buf, "%u\n",
|
|
||||||
be16_to_cpup((__be16 *)(out_mad->data + 40 + offset / 8)));
|
|
||||||
break;
|
|
||||||
case 32:
|
|
||||||
ret = sprintf(buf, "%u\n",
|
|
||||||
be32_to_cpup((__be32 *)(out_mad->data + 40 + offset / 8)));
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
ret = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
out:
|
out:
|
||||||
kfree(in_mad);
|
kfree(in_mad);
|
||||||
kfree(out_mad);
|
kfree(out_mad);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t show_pma_counter(struct ib_port *p, struct port_attribute *attr,
|
||||||
|
char *buf)
|
||||||
|
{
|
||||||
|
struct port_table_attribute *tab_attr =
|
||||||
|
container_of(attr, struct port_table_attribute, attr);
|
||||||
|
int offset = tab_attr->index & 0xffff;
|
||||||
|
int width = (tab_attr->index >> 16) & 0xff;
|
||||||
|
ssize_t ret;
|
||||||
|
u8 data[8];
|
||||||
|
|
||||||
|
ret = get_perf_mad(p->ibdev, p->port_num, tab_attr->attr_id, &data,
|
||||||
|
40 + offset / 8, sizeof(data));
|
||||||
|
if (ret < 0)
|
||||||
|
return sprintf(buf, "N/A (no PMA)\n");
|
||||||
|
|
||||||
|
switch (width) {
|
||||||
|
case 4:
|
||||||
|
ret = sprintf(buf, "%u\n", (*data >>
|
||||||
|
(4 - (offset % 8))) & 0xf);
|
||||||
|
break;
|
||||||
|
case 8:
|
||||||
|
ret = sprintf(buf, "%u\n", *data);
|
||||||
|
break;
|
||||||
|
case 16:
|
||||||
|
ret = sprintf(buf, "%u\n",
|
||||||
|
be16_to_cpup((__be16 *)data));
|
||||||
|
break;
|
||||||
|
case 32:
|
||||||
|
ret = sprintf(buf, "%u\n",
|
||||||
|
be32_to_cpup((__be32 *)data));
|
||||||
|
break;
|
||||||
|
case 64:
|
||||||
|
ret = sprintf(buf, "%llu\n",
|
||||||
|
be64_to_cpup((__be64 *)data));
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
ret = 0;
|
||||||
|
}
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -403,6 +519,18 @@ static PORT_PMA_ATTR(port_rcv_data , 13, 32, 224);
|
||||||
static PORT_PMA_ATTR(port_xmit_packets , 14, 32, 256);
|
static PORT_PMA_ATTR(port_xmit_packets , 14, 32, 256);
|
||||||
static PORT_PMA_ATTR(port_rcv_packets , 15, 32, 288);
|
static PORT_PMA_ATTR(port_rcv_packets , 15, 32, 288);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Counters added by extended set
|
||||||
|
*/
|
||||||
|
static PORT_PMA_ATTR_EXT(port_xmit_data , 64, 64);
|
||||||
|
static PORT_PMA_ATTR_EXT(port_rcv_data , 64, 128);
|
||||||
|
static PORT_PMA_ATTR_EXT(port_xmit_packets , 64, 192);
|
||||||
|
static PORT_PMA_ATTR_EXT(port_rcv_packets , 64, 256);
|
||||||
|
static PORT_PMA_ATTR_EXT(unicast_xmit_packets , 64, 320);
|
||||||
|
static PORT_PMA_ATTR_EXT(unicast_rcv_packets , 64, 384);
|
||||||
|
static PORT_PMA_ATTR_EXT(multicast_xmit_packets , 64, 448);
|
||||||
|
static PORT_PMA_ATTR_EXT(multicast_rcv_packets , 64, 512);
|
||||||
|
|
||||||
static struct attribute *pma_attrs[] = {
|
static struct attribute *pma_attrs[] = {
|
||||||
&port_pma_attr_symbol_error.attr.attr,
|
&port_pma_attr_symbol_error.attr.attr,
|
||||||
&port_pma_attr_link_error_recovery.attr.attr,
|
&port_pma_attr_link_error_recovery.attr.attr,
|
||||||
|
@ -423,11 +551,65 @@ static struct attribute *pma_attrs[] = {
|
||||||
NULL
|
NULL
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static struct attribute *pma_attrs_ext[] = {
|
||||||
|
&port_pma_attr_symbol_error.attr.attr,
|
||||||
|
&port_pma_attr_link_error_recovery.attr.attr,
|
||||||
|
&port_pma_attr_link_downed.attr.attr,
|
||||||
|
&port_pma_attr_port_rcv_errors.attr.attr,
|
||||||
|
&port_pma_attr_port_rcv_remote_physical_errors.attr.attr,
|
||||||
|
&port_pma_attr_port_rcv_switch_relay_errors.attr.attr,
|
||||||
|
&port_pma_attr_port_xmit_discards.attr.attr,
|
||||||
|
&port_pma_attr_port_xmit_constraint_errors.attr.attr,
|
||||||
|
&port_pma_attr_port_rcv_constraint_errors.attr.attr,
|
||||||
|
&port_pma_attr_local_link_integrity_errors.attr.attr,
|
||||||
|
&port_pma_attr_excessive_buffer_overrun_errors.attr.attr,
|
||||||
|
&port_pma_attr_VL15_dropped.attr.attr,
|
||||||
|
&port_pma_attr_ext_port_xmit_data.attr.attr,
|
||||||
|
&port_pma_attr_ext_port_rcv_data.attr.attr,
|
||||||
|
&port_pma_attr_ext_port_xmit_packets.attr.attr,
|
||||||
|
&port_pma_attr_ext_port_rcv_packets.attr.attr,
|
||||||
|
&port_pma_attr_ext_unicast_rcv_packets.attr.attr,
|
||||||
|
&port_pma_attr_ext_unicast_xmit_packets.attr.attr,
|
||||||
|
&port_pma_attr_ext_multicast_rcv_packets.attr.attr,
|
||||||
|
&port_pma_attr_ext_multicast_xmit_packets.attr.attr,
|
||||||
|
NULL
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct attribute *pma_attrs_noietf[] = {
|
||||||
|
&port_pma_attr_symbol_error.attr.attr,
|
||||||
|
&port_pma_attr_link_error_recovery.attr.attr,
|
||||||
|
&port_pma_attr_link_downed.attr.attr,
|
||||||
|
&port_pma_attr_port_rcv_errors.attr.attr,
|
||||||
|
&port_pma_attr_port_rcv_remote_physical_errors.attr.attr,
|
||||||
|
&port_pma_attr_port_rcv_switch_relay_errors.attr.attr,
|
||||||
|
&port_pma_attr_port_xmit_discards.attr.attr,
|
||||||
|
&port_pma_attr_port_xmit_constraint_errors.attr.attr,
|
||||||
|
&port_pma_attr_port_rcv_constraint_errors.attr.attr,
|
||||||
|
&port_pma_attr_local_link_integrity_errors.attr.attr,
|
||||||
|
&port_pma_attr_excessive_buffer_overrun_errors.attr.attr,
|
||||||
|
&port_pma_attr_VL15_dropped.attr.attr,
|
||||||
|
&port_pma_attr_ext_port_xmit_data.attr.attr,
|
||||||
|
&port_pma_attr_ext_port_rcv_data.attr.attr,
|
||||||
|
&port_pma_attr_ext_port_xmit_packets.attr.attr,
|
||||||
|
&port_pma_attr_ext_port_rcv_packets.attr.attr,
|
||||||
|
NULL
|
||||||
|
};
|
||||||
|
|
||||||
static struct attribute_group pma_group = {
|
static struct attribute_group pma_group = {
|
||||||
.name = "counters",
|
.name = "counters",
|
||||||
.attrs = pma_attrs
|
.attrs = pma_attrs
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static struct attribute_group pma_group_ext = {
|
||||||
|
.name = "counters",
|
||||||
|
.attrs = pma_attrs_ext
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct attribute_group pma_group_noietf = {
|
||||||
|
.name = "counters",
|
||||||
|
.attrs = pma_attrs_noietf
|
||||||
|
};
|
||||||
|
|
||||||
static void ib_port_release(struct kobject *kobj)
|
static void ib_port_release(struct kobject *kobj)
|
||||||
{
|
{
|
||||||
struct ib_port *p = container_of(kobj, struct ib_port, kobj);
|
struct ib_port *p = container_of(kobj, struct ib_port, kobj);
|
||||||
|
@ -451,12 +633,41 @@ static void ib_port_release(struct kobject *kobj)
|
||||||
kfree(p);
|
kfree(p);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void ib_port_gid_attr_release(struct kobject *kobj)
|
||||||
|
{
|
||||||
|
struct gid_attr_group *g = container_of(kobj, struct gid_attr_group,
|
||||||
|
kobj);
|
||||||
|
struct attribute *a;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (g->ndev.attrs) {
|
||||||
|
for (i = 0; (a = g->ndev.attrs[i]); ++i)
|
||||||
|
kfree(a);
|
||||||
|
|
||||||
|
kfree(g->ndev.attrs);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (g->type.attrs) {
|
||||||
|
for (i = 0; (a = g->type.attrs[i]); ++i)
|
||||||
|
kfree(a);
|
||||||
|
|
||||||
|
kfree(g->type.attrs);
|
||||||
|
}
|
||||||
|
|
||||||
|
kfree(g);
|
||||||
|
}
|
||||||
|
|
||||||
static struct kobj_type port_type = {
|
static struct kobj_type port_type = {
|
||||||
.release = ib_port_release,
|
.release = ib_port_release,
|
||||||
.sysfs_ops = &port_sysfs_ops,
|
.sysfs_ops = &port_sysfs_ops,
|
||||||
.default_attrs = port_default_attrs
|
.default_attrs = port_default_attrs
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static struct kobj_type gid_attr_type = {
|
||||||
|
.sysfs_ops = &gid_attr_sysfs_ops,
|
||||||
|
.release = ib_port_gid_attr_release
|
||||||
|
};
|
||||||
|
|
||||||
static struct attribute **
|
static struct attribute **
|
||||||
alloc_group_attrs(ssize_t (*show)(struct ib_port *,
|
alloc_group_attrs(ssize_t (*show)(struct ib_port *,
|
||||||
struct port_attribute *, char *buf),
|
struct port_attribute *, char *buf),
|
||||||
|
@ -500,6 +711,31 @@ alloc_group_attrs(ssize_t (*show)(struct ib_port *,
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Figure out which counter table to use depending on
|
||||||
|
* the device capabilities.
|
||||||
|
*/
|
||||||
|
static struct attribute_group *get_counter_table(struct ib_device *dev,
|
||||||
|
int port_num)
|
||||||
|
{
|
||||||
|
struct ib_class_port_info cpi;
|
||||||
|
|
||||||
|
if (get_perf_mad(dev, port_num, IB_PMA_CLASS_PORT_INFO,
|
||||||
|
&cpi, 40, sizeof(cpi)) >= 0) {
|
||||||
|
|
||||||
|
if (cpi.capability_mask && IB_PMA_CLASS_CAP_EXT_WIDTH)
|
||||||
|
/* We have extended counters */
|
||||||
|
return &pma_group_ext;
|
||||||
|
|
||||||
|
if (cpi.capability_mask && IB_PMA_CLASS_CAP_EXT_WIDTH_NOIETF)
|
||||||
|
/* But not the IETF ones */
|
||||||
|
return &pma_group_noietf;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Fall back to normal counters */
|
||||||
|
return &pma_group;
|
||||||
|
}
|
||||||
|
|
||||||
static int add_port(struct ib_device *device, int port_num,
|
static int add_port(struct ib_device *device, int port_num,
|
||||||
int (*port_callback)(struct ib_device *,
|
int (*port_callback)(struct ib_device *,
|
||||||
u8, struct kobject *))
|
u8, struct kobject *))
|
||||||
|
@ -528,9 +764,24 @@ static int add_port(struct ib_device *device, int port_num,
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = sysfs_create_group(&p->kobj, &pma_group);
|
p->gid_attr_group = kzalloc(sizeof(*p->gid_attr_group), GFP_KERNEL);
|
||||||
if (ret)
|
if (!p->gid_attr_group) {
|
||||||
|
ret = -ENOMEM;
|
||||||
goto err_put;
|
goto err_put;
|
||||||
|
}
|
||||||
|
|
||||||
|
p->gid_attr_group->port = p;
|
||||||
|
ret = kobject_init_and_add(&p->gid_attr_group->kobj, &gid_attr_type,
|
||||||
|
&p->kobj, "gid_attrs");
|
||||||
|
if (ret) {
|
||||||
|
kfree(p->gid_attr_group);
|
||||||
|
goto err_put;
|
||||||
|
}
|
||||||
|
|
||||||
|
p->pma_table = get_counter_table(device, port_num);
|
||||||
|
ret = sysfs_create_group(&p->kobj, p->pma_table);
|
||||||
|
if (ret)
|
||||||
|
goto err_put_gid_attrs;
|
||||||
|
|
||||||
p->gid_group.name = "gids";
|
p->gid_group.name = "gids";
|
||||||
p->gid_group.attrs = alloc_group_attrs(show_port_gid, attr.gid_tbl_len);
|
p->gid_group.attrs = alloc_group_attrs(show_port_gid, attr.gid_tbl_len);
|
||||||
|
@ -543,12 +794,38 @@ static int add_port(struct ib_device *device, int port_num,
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err_free_gid;
|
goto err_free_gid;
|
||||||
|
|
||||||
|
p->gid_attr_group->ndev.name = "ndevs";
|
||||||
|
p->gid_attr_group->ndev.attrs = alloc_group_attrs(show_port_gid_attr_ndev,
|
||||||
|
attr.gid_tbl_len);
|
||||||
|
if (!p->gid_attr_group->ndev.attrs) {
|
||||||
|
ret = -ENOMEM;
|
||||||
|
goto err_remove_gid;
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = sysfs_create_group(&p->gid_attr_group->kobj,
|
||||||
|
&p->gid_attr_group->ndev);
|
||||||
|
if (ret)
|
||||||
|
goto err_free_gid_ndev;
|
||||||
|
|
||||||
|
p->gid_attr_group->type.name = "types";
|
||||||
|
p->gid_attr_group->type.attrs = alloc_group_attrs(show_port_gid_attr_gid_type,
|
||||||
|
attr.gid_tbl_len);
|
||||||
|
if (!p->gid_attr_group->type.attrs) {
|
||||||
|
ret = -ENOMEM;
|
||||||
|
goto err_remove_gid_ndev;
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = sysfs_create_group(&p->gid_attr_group->kobj,
|
||||||
|
&p->gid_attr_group->type);
|
||||||
|
if (ret)
|
||||||
|
goto err_free_gid_type;
|
||||||
|
|
||||||
p->pkey_group.name = "pkeys";
|
p->pkey_group.name = "pkeys";
|
||||||
p->pkey_group.attrs = alloc_group_attrs(show_port_pkey,
|
p->pkey_group.attrs = alloc_group_attrs(show_port_pkey,
|
||||||
attr.pkey_tbl_len);
|
attr.pkey_tbl_len);
|
||||||
if (!p->pkey_group.attrs) {
|
if (!p->pkey_group.attrs) {
|
||||||
ret = -ENOMEM;
|
ret = -ENOMEM;
|
||||||
goto err_remove_gid;
|
goto err_remove_gid_type;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = sysfs_create_group(&p->kobj, &p->pkey_group);
|
ret = sysfs_create_group(&p->kobj, &p->pkey_group);
|
||||||
|
@ -576,6 +853,28 @@ static int add_port(struct ib_device *device, int port_num,
|
||||||
kfree(p->pkey_group.attrs);
|
kfree(p->pkey_group.attrs);
|
||||||
p->pkey_group.attrs = NULL;
|
p->pkey_group.attrs = NULL;
|
||||||
|
|
||||||
|
err_remove_gid_type:
|
||||||
|
sysfs_remove_group(&p->gid_attr_group->kobj,
|
||||||
|
&p->gid_attr_group->type);
|
||||||
|
|
||||||
|
err_free_gid_type:
|
||||||
|
for (i = 0; i < attr.gid_tbl_len; ++i)
|
||||||
|
kfree(p->gid_attr_group->type.attrs[i]);
|
||||||
|
|
||||||
|
kfree(p->gid_attr_group->type.attrs);
|
||||||
|
p->gid_attr_group->type.attrs = NULL;
|
||||||
|
|
||||||
|
err_remove_gid_ndev:
|
||||||
|
sysfs_remove_group(&p->gid_attr_group->kobj,
|
||||||
|
&p->gid_attr_group->ndev);
|
||||||
|
|
||||||
|
err_free_gid_ndev:
|
||||||
|
for (i = 0; i < attr.gid_tbl_len; ++i)
|
||||||
|
kfree(p->gid_attr_group->ndev.attrs[i]);
|
||||||
|
|
||||||
|
kfree(p->gid_attr_group->ndev.attrs);
|
||||||
|
p->gid_attr_group->ndev.attrs = NULL;
|
||||||
|
|
||||||
err_remove_gid:
|
err_remove_gid:
|
||||||
sysfs_remove_group(&p->kobj, &p->gid_group);
|
sysfs_remove_group(&p->kobj, &p->gid_group);
|
||||||
|
|
||||||
|
@ -587,7 +886,10 @@ static int add_port(struct ib_device *device, int port_num,
|
||||||
p->gid_group.attrs = NULL;
|
p->gid_group.attrs = NULL;
|
||||||
|
|
||||||
err_remove_pma:
|
err_remove_pma:
|
||||||
sysfs_remove_group(&p->kobj, &pma_group);
|
sysfs_remove_group(&p->kobj, p->pma_table);
|
||||||
|
|
||||||
|
err_put_gid_attrs:
|
||||||
|
kobject_put(&p->gid_attr_group->kobj);
|
||||||
|
|
||||||
err_put:
|
err_put:
|
||||||
kobject_put(&p->kobj);
|
kobject_put(&p->kobj);
|
||||||
|
@ -614,18 +916,12 @@ static ssize_t show_sys_image_guid(struct device *device,
|
||||||
struct device_attribute *dev_attr, char *buf)
|
struct device_attribute *dev_attr, char *buf)
|
||||||
{
|
{
|
||||||
struct ib_device *dev = container_of(device, struct ib_device, dev);
|
struct ib_device *dev = container_of(device, struct ib_device, dev);
|
||||||
struct ib_device_attr attr;
|
|
||||||
ssize_t ret;
|
|
||||||
|
|
||||||
ret = ib_query_device(dev, &attr);
|
|
||||||
if (ret)
|
|
||||||
return ret;
|
|
||||||
|
|
||||||
return sprintf(buf, "%04x:%04x:%04x:%04x\n",
|
return sprintf(buf, "%04x:%04x:%04x:%04x\n",
|
||||||
be16_to_cpu(((__be16 *) &attr.sys_image_guid)[0]),
|
be16_to_cpu(((__be16 *) &dev->attrs.sys_image_guid)[0]),
|
||||||
be16_to_cpu(((__be16 *) &attr.sys_image_guid)[1]),
|
be16_to_cpu(((__be16 *) &dev->attrs.sys_image_guid)[1]),
|
||||||
be16_to_cpu(((__be16 *) &attr.sys_image_guid)[2]),
|
be16_to_cpu(((__be16 *) &dev->attrs.sys_image_guid)[2]),
|
||||||
be16_to_cpu(((__be16 *) &attr.sys_image_guid)[3]));
|
be16_to_cpu(((__be16 *) &dev->attrs.sys_image_guid)[3]));
|
||||||
}
|
}
|
||||||
|
|
||||||
static ssize_t show_node_guid(struct device *device,
|
static ssize_t show_node_guid(struct device *device,
|
||||||
|
@ -800,9 +1096,14 @@ static void free_port_list_attributes(struct ib_device *device)
|
||||||
list_for_each_entry_safe(p, t, &device->port_list, entry) {
|
list_for_each_entry_safe(p, t, &device->port_list, entry) {
|
||||||
struct ib_port *port = container_of(p, struct ib_port, kobj);
|
struct ib_port *port = container_of(p, struct ib_port, kobj);
|
||||||
list_del(&p->entry);
|
list_del(&p->entry);
|
||||||
sysfs_remove_group(p, &pma_group);
|
sysfs_remove_group(p, port->pma_table);
|
||||||
sysfs_remove_group(p, &port->pkey_group);
|
sysfs_remove_group(p, &port->pkey_group);
|
||||||
sysfs_remove_group(p, &port->gid_group);
|
sysfs_remove_group(p, &port->gid_group);
|
||||||
|
sysfs_remove_group(&port->gid_attr_group->kobj,
|
||||||
|
&port->gid_attr_group->ndev);
|
||||||
|
sysfs_remove_group(&port->gid_attr_group->kobj,
|
||||||
|
&port->gid_attr_group->type);
|
||||||
|
kobject_put(&port->gid_attr_group->kobj);
|
||||||
kobject_put(p);
|
kobject_put(p);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -35,6 +35,7 @@
|
||||||
#include <linux/string.h>
|
#include <linux/string.h>
|
||||||
#include <linux/export.h>
|
#include <linux/export.h>
|
||||||
#include <linux/if_ether.h>
|
#include <linux/if_ether.h>
|
||||||
|
#include <linux/ip.h>
|
||||||
|
|
||||||
#include <rdma/ib_pack.h>
|
#include <rdma/ib_pack.h>
|
||||||
|
|
||||||
|
@ -116,6 +117,72 @@ static const struct ib_field vlan_table[] = {
|
||||||
.size_bits = 16 }
|
.size_bits = 16 }
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const struct ib_field ip4_table[] = {
|
||||||
|
{ STRUCT_FIELD(ip4, ver),
|
||||||
|
.offset_words = 0,
|
||||||
|
.offset_bits = 0,
|
||||||
|
.size_bits = 4 },
|
||||||
|
{ STRUCT_FIELD(ip4, hdr_len),
|
||||||
|
.offset_words = 0,
|
||||||
|
.offset_bits = 4,
|
||||||
|
.size_bits = 4 },
|
||||||
|
{ STRUCT_FIELD(ip4, tos),
|
||||||
|
.offset_words = 0,
|
||||||
|
.offset_bits = 8,
|
||||||
|
.size_bits = 8 },
|
||||||
|
{ STRUCT_FIELD(ip4, tot_len),
|
||||||
|
.offset_words = 0,
|
||||||
|
.offset_bits = 16,
|
||||||
|
.size_bits = 16 },
|
||||||
|
{ STRUCT_FIELD(ip4, id),
|
||||||
|
.offset_words = 1,
|
||||||
|
.offset_bits = 0,
|
||||||
|
.size_bits = 16 },
|
||||||
|
{ STRUCT_FIELD(ip4, frag_off),
|
||||||
|
.offset_words = 1,
|
||||||
|
.offset_bits = 16,
|
||||||
|
.size_bits = 16 },
|
||||||
|
{ STRUCT_FIELD(ip4, ttl),
|
||||||
|
.offset_words = 2,
|
||||||
|
.offset_bits = 0,
|
||||||
|
.size_bits = 8 },
|
||||||
|
{ STRUCT_FIELD(ip4, protocol),
|
||||||
|
.offset_words = 2,
|
||||||
|
.offset_bits = 8,
|
||||||
|
.size_bits = 8 },
|
||||||
|
{ STRUCT_FIELD(ip4, check),
|
||||||
|
.offset_words = 2,
|
||||||
|
.offset_bits = 16,
|
||||||
|
.size_bits = 16 },
|
||||||
|
{ STRUCT_FIELD(ip4, saddr),
|
||||||
|
.offset_words = 3,
|
||||||
|
.offset_bits = 0,
|
||||||
|
.size_bits = 32 },
|
||||||
|
{ STRUCT_FIELD(ip4, daddr),
|
||||||
|
.offset_words = 4,
|
||||||
|
.offset_bits = 0,
|
||||||
|
.size_bits = 32 }
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct ib_field udp_table[] = {
|
||||||
|
{ STRUCT_FIELD(udp, sport),
|
||||||
|
.offset_words = 0,
|
||||||
|
.offset_bits = 0,
|
||||||
|
.size_bits = 16 },
|
||||||
|
{ STRUCT_FIELD(udp, dport),
|
||||||
|
.offset_words = 0,
|
||||||
|
.offset_bits = 16,
|
||||||
|
.size_bits = 16 },
|
||||||
|
{ STRUCT_FIELD(udp, length),
|
||||||
|
.offset_words = 1,
|
||||||
|
.offset_bits = 0,
|
||||||
|
.size_bits = 16 },
|
||||||
|
{ STRUCT_FIELD(udp, csum),
|
||||||
|
.offset_words = 1,
|
||||||
|
.offset_bits = 16,
|
||||||
|
.size_bits = 16 }
|
||||||
|
};
|
||||||
|
|
||||||
static const struct ib_field grh_table[] = {
|
static const struct ib_field grh_table[] = {
|
||||||
{ STRUCT_FIELD(grh, ip_version),
|
{ STRUCT_FIELD(grh, ip_version),
|
||||||
.offset_words = 0,
|
.offset_words = 0,
|
||||||
|
@ -213,26 +280,57 @@ static const struct ib_field deth_table[] = {
|
||||||
.size_bits = 24 }
|
.size_bits = 24 }
|
||||||
};
|
};
|
||||||
|
|
||||||
|
__sum16 ib_ud_ip4_csum(struct ib_ud_header *header)
|
||||||
|
{
|
||||||
|
struct iphdr iph;
|
||||||
|
|
||||||
|
iph.ihl = 5;
|
||||||
|
iph.version = 4;
|
||||||
|
iph.tos = header->ip4.tos;
|
||||||
|
iph.tot_len = header->ip4.tot_len;
|
||||||
|
iph.id = header->ip4.id;
|
||||||
|
iph.frag_off = header->ip4.frag_off;
|
||||||
|
iph.ttl = header->ip4.ttl;
|
||||||
|
iph.protocol = header->ip4.protocol;
|
||||||
|
iph.check = 0;
|
||||||
|
iph.saddr = header->ip4.saddr;
|
||||||
|
iph.daddr = header->ip4.daddr;
|
||||||
|
|
||||||
|
return ip_fast_csum((u8 *)&iph, iph.ihl);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(ib_ud_ip4_csum);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* ib_ud_header_init - Initialize UD header structure
|
* ib_ud_header_init - Initialize UD header structure
|
||||||
* @payload_bytes:Length of packet payload
|
* @payload_bytes:Length of packet payload
|
||||||
* @lrh_present: specify if LRH is present
|
* @lrh_present: specify if LRH is present
|
||||||
* @eth_present: specify if Eth header is present
|
* @eth_present: specify if Eth header is present
|
||||||
* @vlan_present: packet is tagged vlan
|
* @vlan_present: packet is tagged vlan
|
||||||
* @grh_present:GRH flag (if non-zero, GRH will be included)
|
* @grh_present: GRH flag (if non-zero, GRH will be included)
|
||||||
|
* @ip_version: if non-zero, IP header, V4 or V6, will be included
|
||||||
|
* @udp_present :if non-zero, UDP header will be included
|
||||||
* @immediate_present: specify if immediate data is present
|
* @immediate_present: specify if immediate data is present
|
||||||
* @header:Structure to initialize
|
* @header:Structure to initialize
|
||||||
*/
|
*/
|
||||||
void ib_ud_header_init(int payload_bytes,
|
int ib_ud_header_init(int payload_bytes,
|
||||||
int lrh_present,
|
int lrh_present,
|
||||||
int eth_present,
|
int eth_present,
|
||||||
int vlan_present,
|
int vlan_present,
|
||||||
int grh_present,
|
int grh_present,
|
||||||
int immediate_present,
|
int ip_version,
|
||||||
struct ib_ud_header *header)
|
int udp_present,
|
||||||
|
int immediate_present,
|
||||||
|
struct ib_ud_header *header)
|
||||||
{
|
{
|
||||||
|
grh_present = grh_present && !ip_version;
|
||||||
memset(header, 0, sizeof *header);
|
memset(header, 0, sizeof *header);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* UDP header without IP header doesn't make sense
|
||||||
|
*/
|
||||||
|
if (udp_present && ip_version != 4 && ip_version != 6)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
if (lrh_present) {
|
if (lrh_present) {
|
||||||
u16 packet_length;
|
u16 packet_length;
|
||||||
|
|
||||||
|
@ -252,7 +350,7 @@ void ib_ud_header_init(int payload_bytes,
|
||||||
if (vlan_present)
|
if (vlan_present)
|
||||||
header->eth.type = cpu_to_be16(ETH_P_8021Q);
|
header->eth.type = cpu_to_be16(ETH_P_8021Q);
|
||||||
|
|
||||||
if (grh_present) {
|
if (ip_version == 6 || grh_present) {
|
||||||
header->grh.ip_version = 6;
|
header->grh.ip_version = 6;
|
||||||
header->grh.payload_length =
|
header->grh.payload_length =
|
||||||
cpu_to_be16((IB_BTH_BYTES +
|
cpu_to_be16((IB_BTH_BYTES +
|
||||||
|
@ -260,9 +358,31 @@ void ib_ud_header_init(int payload_bytes,
|
||||||
payload_bytes +
|
payload_bytes +
|
||||||
4 + /* ICRC */
|
4 + /* ICRC */
|
||||||
3) & ~3); /* round up */
|
3) & ~3); /* round up */
|
||||||
header->grh.next_header = 0x1b;
|
header->grh.next_header = udp_present ? IPPROTO_UDP : 0x1b;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (ip_version == 4) {
|
||||||
|
int udp_bytes = udp_present ? IB_UDP_BYTES : 0;
|
||||||
|
|
||||||
|
header->ip4.ver = 4; /* version 4 */
|
||||||
|
header->ip4.hdr_len = 5; /* 5 words */
|
||||||
|
header->ip4.tot_len =
|
||||||
|
cpu_to_be16(IB_IP4_BYTES +
|
||||||
|
udp_bytes +
|
||||||
|
IB_BTH_BYTES +
|
||||||
|
IB_DETH_BYTES +
|
||||||
|
payload_bytes +
|
||||||
|
4); /* ICRC */
|
||||||
|
header->ip4.protocol = IPPROTO_UDP;
|
||||||
|
}
|
||||||
|
if (udp_present && ip_version)
|
||||||
|
header->udp.length =
|
||||||
|
cpu_to_be16(IB_UDP_BYTES +
|
||||||
|
IB_BTH_BYTES +
|
||||||
|
IB_DETH_BYTES +
|
||||||
|
payload_bytes +
|
||||||
|
4); /* ICRC */
|
||||||
|
|
||||||
if (immediate_present)
|
if (immediate_present)
|
||||||
header->bth.opcode = IB_OPCODE_UD_SEND_ONLY_WITH_IMMEDIATE;
|
header->bth.opcode = IB_OPCODE_UD_SEND_ONLY_WITH_IMMEDIATE;
|
||||||
else
|
else
|
||||||
|
@ -273,8 +393,11 @@ void ib_ud_header_init(int payload_bytes,
|
||||||
header->lrh_present = lrh_present;
|
header->lrh_present = lrh_present;
|
||||||
header->eth_present = eth_present;
|
header->eth_present = eth_present;
|
||||||
header->vlan_present = vlan_present;
|
header->vlan_present = vlan_present;
|
||||||
header->grh_present = grh_present;
|
header->grh_present = grh_present || (ip_version == 6);
|
||||||
|
header->ipv4_present = ip_version == 4;
|
||||||
|
header->udp_present = udp_present;
|
||||||
header->immediate_present = immediate_present;
|
header->immediate_present = immediate_present;
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(ib_ud_header_init);
|
EXPORT_SYMBOL(ib_ud_header_init);
|
||||||
|
|
||||||
|
@ -311,6 +434,16 @@ int ib_ud_header_pack(struct ib_ud_header *header,
|
||||||
&header->grh, buf + len);
|
&header->grh, buf + len);
|
||||||
len += IB_GRH_BYTES;
|
len += IB_GRH_BYTES;
|
||||||
}
|
}
|
||||||
|
if (header->ipv4_present) {
|
||||||
|
ib_pack(ip4_table, ARRAY_SIZE(ip4_table),
|
||||||
|
&header->ip4, buf + len);
|
||||||
|
len += IB_IP4_BYTES;
|
||||||
|
}
|
||||||
|
if (header->udp_present) {
|
||||||
|
ib_pack(udp_table, ARRAY_SIZE(udp_table),
|
||||||
|
&header->udp, buf + len);
|
||||||
|
len += IB_UDP_BYTES;
|
||||||
|
}
|
||||||
|
|
||||||
ib_pack(bth_table, ARRAY_SIZE(bth_table),
|
ib_pack(bth_table, ARRAY_SIZE(bth_table),
|
||||||
&header->bth, buf + len);
|
&header->bth, buf + len);
|
||||||
|
|
|
@ -232,7 +232,7 @@ static void ib_umem_notifier_invalidate_range_end(struct mmu_notifier *mn,
|
||||||
ib_ucontext_notifier_end_account(context);
|
ib_ucontext_notifier_end_account(context);
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct mmu_notifier_ops ib_umem_notifiers = {
|
static const struct mmu_notifier_ops ib_umem_notifiers = {
|
||||||
.release = ib_umem_notifier_release,
|
.release = ib_umem_notifier_release,
|
||||||
.invalidate_page = ib_umem_notifier_invalidate_page,
|
.invalidate_page = ib_umem_notifier_invalidate_page,
|
||||||
.invalidate_range_start = ib_umem_notifier_invalidate_range_start,
|
.invalidate_range_start = ib_umem_notifier_invalidate_range_start,
|
||||||
|
|
|
@ -210,6 +210,7 @@ static void send_handler(struct ib_mad_agent *agent,
|
||||||
}
|
}
|
||||||
|
|
||||||
static void recv_handler(struct ib_mad_agent *agent,
|
static void recv_handler(struct ib_mad_agent *agent,
|
||||||
|
struct ib_mad_send_buf *send_buf,
|
||||||
struct ib_mad_recv_wc *mad_recv_wc)
|
struct ib_mad_recv_wc *mad_recv_wc)
|
||||||
{
|
{
|
||||||
struct ib_umad_file *file = agent->context;
|
struct ib_umad_file *file = agent->context;
|
||||||
|
|
|
@ -204,6 +204,8 @@ void ib_uverbs_event_handler(struct ib_event_handler *handler,
|
||||||
struct ib_event *event);
|
struct ib_event *event);
|
||||||
void ib_uverbs_dealloc_xrcd(struct ib_uverbs_device *dev, struct ib_xrcd *xrcd);
|
void ib_uverbs_dealloc_xrcd(struct ib_uverbs_device *dev, struct ib_xrcd *xrcd);
|
||||||
|
|
||||||
|
int uverbs_dealloc_mw(struct ib_mw *mw);
|
||||||
|
|
||||||
struct ib_uverbs_flow_spec {
|
struct ib_uverbs_flow_spec {
|
||||||
union {
|
union {
|
||||||
union {
|
union {
|
||||||
|
|
|
@ -291,9 +291,6 @@ ssize_t ib_uverbs_get_context(struct ib_uverbs_file *file,
|
||||||
struct ib_uverbs_get_context cmd;
|
struct ib_uverbs_get_context cmd;
|
||||||
struct ib_uverbs_get_context_resp resp;
|
struct ib_uverbs_get_context_resp resp;
|
||||||
struct ib_udata udata;
|
struct ib_udata udata;
|
||||||
#ifdef CONFIG_INFINIBAND_ON_DEMAND_PAGING
|
|
||||||
struct ib_device_attr dev_attr;
|
|
||||||
#endif
|
|
||||||
struct ib_ucontext *ucontext;
|
struct ib_ucontext *ucontext;
|
||||||
struct file *filp;
|
struct file *filp;
|
||||||
int ret;
|
int ret;
|
||||||
|
@ -342,10 +339,7 @@ ssize_t ib_uverbs_get_context(struct ib_uverbs_file *file,
|
||||||
ucontext->odp_mrs_count = 0;
|
ucontext->odp_mrs_count = 0;
|
||||||
INIT_LIST_HEAD(&ucontext->no_private_counters);
|
INIT_LIST_HEAD(&ucontext->no_private_counters);
|
||||||
|
|
||||||
ret = ib_query_device(ib_dev, &dev_attr);
|
if (!(ib_dev->attrs.device_cap_flags & IB_DEVICE_ON_DEMAND_PAGING))
|
||||||
if (ret)
|
|
||||||
goto err_free;
|
|
||||||
if (!(dev_attr.device_cap_flags & IB_DEVICE_ON_DEMAND_PAGING))
|
|
||||||
ucontext->invalidate_range = NULL;
|
ucontext->invalidate_range = NULL;
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@ -447,8 +441,6 @@ ssize_t ib_uverbs_query_device(struct ib_uverbs_file *file,
|
||||||
{
|
{
|
||||||
struct ib_uverbs_query_device cmd;
|
struct ib_uverbs_query_device cmd;
|
||||||
struct ib_uverbs_query_device_resp resp;
|
struct ib_uverbs_query_device_resp resp;
|
||||||
struct ib_device_attr attr;
|
|
||||||
int ret;
|
|
||||||
|
|
||||||
if (out_len < sizeof resp)
|
if (out_len < sizeof resp)
|
||||||
return -ENOSPC;
|
return -ENOSPC;
|
||||||
|
@ -456,12 +448,8 @@ ssize_t ib_uverbs_query_device(struct ib_uverbs_file *file,
|
||||||
if (copy_from_user(&cmd, buf, sizeof cmd))
|
if (copy_from_user(&cmd, buf, sizeof cmd))
|
||||||
return -EFAULT;
|
return -EFAULT;
|
||||||
|
|
||||||
ret = ib_query_device(ib_dev, &attr);
|
|
||||||
if (ret)
|
|
||||||
return ret;
|
|
||||||
|
|
||||||
memset(&resp, 0, sizeof resp);
|
memset(&resp, 0, sizeof resp);
|
||||||
copy_query_dev_fields(file, ib_dev, &resp, &attr);
|
copy_query_dev_fields(file, ib_dev, &resp, &ib_dev->attrs);
|
||||||
|
|
||||||
if (copy_to_user((void __user *) (unsigned long) cmd.response,
|
if (copy_to_user((void __user *) (unsigned long) cmd.response,
|
||||||
&resp, sizeof resp))
|
&resp, sizeof resp))
|
||||||
|
@ -986,11 +974,8 @@ ssize_t ib_uverbs_reg_mr(struct ib_uverbs_file *file,
|
||||||
}
|
}
|
||||||
|
|
||||||
if (cmd.access_flags & IB_ACCESS_ON_DEMAND) {
|
if (cmd.access_flags & IB_ACCESS_ON_DEMAND) {
|
||||||
struct ib_device_attr attr;
|
if (!(pd->device->attrs.device_cap_flags &
|
||||||
|
IB_DEVICE_ON_DEMAND_PAGING)) {
|
||||||
ret = ib_query_device(pd->device, &attr);
|
|
||||||
if (ret || !(attr.device_cap_flags &
|
|
||||||
IB_DEVICE_ON_DEMAND_PAGING)) {
|
|
||||||
pr_debug("ODP support not available\n");
|
pr_debug("ODP support not available\n");
|
||||||
ret = -EINVAL;
|
ret = -EINVAL;
|
||||||
goto err_put;
|
goto err_put;
|
||||||
|
@ -1008,7 +993,6 @@ ssize_t ib_uverbs_reg_mr(struct ib_uverbs_file *file,
|
||||||
mr->pd = pd;
|
mr->pd = pd;
|
||||||
mr->uobject = uobj;
|
mr->uobject = uobj;
|
||||||
atomic_inc(&pd->usecnt);
|
atomic_inc(&pd->usecnt);
|
||||||
atomic_set(&mr->usecnt, 0);
|
|
||||||
|
|
||||||
uobj->object = mr;
|
uobj->object = mr;
|
||||||
ret = idr_add_uobj(&ib_uverbs_mr_idr, uobj);
|
ret = idr_add_uobj(&ib_uverbs_mr_idr, uobj);
|
||||||
|
@ -1106,11 +1090,6 @@ ssize_t ib_uverbs_rereg_mr(struct ib_uverbs_file *file,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (atomic_read(&mr->usecnt)) {
|
|
||||||
ret = -EBUSY;
|
|
||||||
goto put_uobj_pd;
|
|
||||||
}
|
|
||||||
|
|
||||||
old_pd = mr->pd;
|
old_pd = mr->pd;
|
||||||
ret = mr->device->rereg_user_mr(mr, cmd.flags, cmd.start,
|
ret = mr->device->rereg_user_mr(mr, cmd.flags, cmd.start,
|
||||||
cmd.length, cmd.hca_va,
|
cmd.length, cmd.hca_va,
|
||||||
|
@ -1258,7 +1237,7 @@ ssize_t ib_uverbs_alloc_mw(struct ib_uverbs_file *file,
|
||||||
idr_remove_uobj(&ib_uverbs_mw_idr, uobj);
|
idr_remove_uobj(&ib_uverbs_mw_idr, uobj);
|
||||||
|
|
||||||
err_unalloc:
|
err_unalloc:
|
||||||
ib_dealloc_mw(mw);
|
uverbs_dealloc_mw(mw);
|
||||||
|
|
||||||
err_put:
|
err_put:
|
||||||
put_pd_read(pd);
|
put_pd_read(pd);
|
||||||
|
@ -1287,7 +1266,7 @@ ssize_t ib_uverbs_dealloc_mw(struct ib_uverbs_file *file,
|
||||||
|
|
||||||
mw = uobj->object;
|
mw = uobj->object;
|
||||||
|
|
||||||
ret = ib_dealloc_mw(mw);
|
ret = uverbs_dealloc_mw(mw);
|
||||||
if (!ret)
|
if (!ret)
|
||||||
uobj->live = 0;
|
uobj->live = 0;
|
||||||
|
|
||||||
|
@ -1845,7 +1824,10 @@ static int create_qp(struct ib_uverbs_file *file,
|
||||||
sizeof(cmd->create_flags))
|
sizeof(cmd->create_flags))
|
||||||
attr.create_flags = cmd->create_flags;
|
attr.create_flags = cmd->create_flags;
|
||||||
|
|
||||||
if (attr.create_flags & ~IB_QP_CREATE_BLOCK_MULTICAST_LOOPBACK) {
|
if (attr.create_flags & ~(IB_QP_CREATE_BLOCK_MULTICAST_LOOPBACK |
|
||||||
|
IB_QP_CREATE_CROSS_CHANNEL |
|
||||||
|
IB_QP_CREATE_MANAGED_SEND |
|
||||||
|
IB_QP_CREATE_MANAGED_RECV)) {
|
||||||
ret = -EINVAL;
|
ret = -EINVAL;
|
||||||
goto err_put;
|
goto err_put;
|
||||||
}
|
}
|
||||||
|
|
|
@ -133,6 +133,17 @@ static int (*uverbs_ex_cmd_table[])(struct ib_uverbs_file *file,
|
||||||
static void ib_uverbs_add_one(struct ib_device *device);
|
static void ib_uverbs_add_one(struct ib_device *device);
|
||||||
static void ib_uverbs_remove_one(struct ib_device *device, void *client_data);
|
static void ib_uverbs_remove_one(struct ib_device *device, void *client_data);
|
||||||
|
|
||||||
|
int uverbs_dealloc_mw(struct ib_mw *mw)
|
||||||
|
{
|
||||||
|
struct ib_pd *pd = mw->pd;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
ret = mw->device->dealloc_mw(mw);
|
||||||
|
if (!ret)
|
||||||
|
atomic_dec(&pd->usecnt);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
static void ib_uverbs_release_dev(struct kobject *kobj)
|
static void ib_uverbs_release_dev(struct kobject *kobj)
|
||||||
{
|
{
|
||||||
struct ib_uverbs_device *dev =
|
struct ib_uverbs_device *dev =
|
||||||
|
@ -224,7 +235,7 @@ static int ib_uverbs_cleanup_ucontext(struct ib_uverbs_file *file,
|
||||||
struct ib_mw *mw = uobj->object;
|
struct ib_mw *mw = uobj->object;
|
||||||
|
|
||||||
idr_remove_uobj(&ib_uverbs_mw_idr, uobj);
|
idr_remove_uobj(&ib_uverbs_mw_idr, uobj);
|
||||||
ib_dealloc_mw(mw);
|
uverbs_dealloc_mw(mw);
|
||||||
kfree(uobj);
|
kfree(uobj);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -144,5 +144,6 @@ void ib_copy_path_rec_from_user(struct ib_sa_path_rec *dst,
|
||||||
memset(dst->dmac, 0, sizeof(dst->dmac));
|
memset(dst->dmac, 0, sizeof(dst->dmac));
|
||||||
dst->net = NULL;
|
dst->net = NULL;
|
||||||
dst->ifindex = 0;
|
dst->ifindex = 0;
|
||||||
|
dst->gid_type = IB_GID_TYPE_IB;
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(ib_copy_path_rec_from_user);
|
EXPORT_SYMBOL(ib_copy_path_rec_from_user);
|
||||||
|
|
|
@ -229,12 +229,6 @@ EXPORT_SYMBOL(rdma_port_get_link_layer);
|
||||||
struct ib_pd *ib_alloc_pd(struct ib_device *device)
|
struct ib_pd *ib_alloc_pd(struct ib_device *device)
|
||||||
{
|
{
|
||||||
struct ib_pd *pd;
|
struct ib_pd *pd;
|
||||||
struct ib_device_attr devattr;
|
|
||||||
int rc;
|
|
||||||
|
|
||||||
rc = ib_query_device(device, &devattr);
|
|
||||||
if (rc)
|
|
||||||
return ERR_PTR(rc);
|
|
||||||
|
|
||||||
pd = device->alloc_pd(device, NULL, NULL);
|
pd = device->alloc_pd(device, NULL, NULL);
|
||||||
if (IS_ERR(pd))
|
if (IS_ERR(pd))
|
||||||
|
@ -245,7 +239,7 @@ struct ib_pd *ib_alloc_pd(struct ib_device *device)
|
||||||
pd->local_mr = NULL;
|
pd->local_mr = NULL;
|
||||||
atomic_set(&pd->usecnt, 0);
|
atomic_set(&pd->usecnt, 0);
|
||||||
|
|
||||||
if (devattr.device_cap_flags & IB_DEVICE_LOCAL_DMA_LKEY)
|
if (device->attrs.device_cap_flags & IB_DEVICE_LOCAL_DMA_LKEY)
|
||||||
pd->local_dma_lkey = device->local_dma_lkey;
|
pd->local_dma_lkey = device->local_dma_lkey;
|
||||||
else {
|
else {
|
||||||
struct ib_mr *mr;
|
struct ib_mr *mr;
|
||||||
|
@ -311,8 +305,61 @@ struct ib_ah *ib_create_ah(struct ib_pd *pd, struct ib_ah_attr *ah_attr)
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(ib_create_ah);
|
EXPORT_SYMBOL(ib_create_ah);
|
||||||
|
|
||||||
|
static int ib_get_header_version(const union rdma_network_hdr *hdr)
|
||||||
|
{
|
||||||
|
const struct iphdr *ip4h = (struct iphdr *)&hdr->roce4grh;
|
||||||
|
struct iphdr ip4h_checked;
|
||||||
|
const struct ipv6hdr *ip6h = (struct ipv6hdr *)&hdr->ibgrh;
|
||||||
|
|
||||||
|
/* If it's IPv6, the version must be 6, otherwise, the first
|
||||||
|
* 20 bytes (before the IPv4 header) are garbled.
|
||||||
|
*/
|
||||||
|
if (ip6h->version != 6)
|
||||||
|
return (ip4h->version == 4) ? 4 : 0;
|
||||||
|
/* version may be 6 or 4 because the first 20 bytes could be garbled */
|
||||||
|
|
||||||
|
/* RoCE v2 requires no options, thus header length
|
||||||
|
* must be 5 words
|
||||||
|
*/
|
||||||
|
if (ip4h->ihl != 5)
|
||||||
|
return 6;
|
||||||
|
|
||||||
|
/* Verify checksum.
|
||||||
|
* We can't write on scattered buffers so we need to copy to
|
||||||
|
* temp buffer.
|
||||||
|
*/
|
||||||
|
memcpy(&ip4h_checked, ip4h, sizeof(ip4h_checked));
|
||||||
|
ip4h_checked.check = 0;
|
||||||
|
ip4h_checked.check = ip_fast_csum((u8 *)&ip4h_checked, 5);
|
||||||
|
/* if IPv4 header checksum is OK, believe it */
|
||||||
|
if (ip4h->check == ip4h_checked.check)
|
||||||
|
return 4;
|
||||||
|
return 6;
|
||||||
|
}
|
||||||
|
|
||||||
|
static enum rdma_network_type ib_get_net_type_by_grh(struct ib_device *device,
|
||||||
|
u8 port_num,
|
||||||
|
const struct ib_grh *grh)
|
||||||
|
{
|
||||||
|
int grh_version;
|
||||||
|
|
||||||
|
if (rdma_protocol_ib(device, port_num))
|
||||||
|
return RDMA_NETWORK_IB;
|
||||||
|
|
||||||
|
grh_version = ib_get_header_version((union rdma_network_hdr *)grh);
|
||||||
|
|
||||||
|
if (grh_version == 4)
|
||||||
|
return RDMA_NETWORK_IPV4;
|
||||||
|
|
||||||
|
if (grh->next_hdr == IPPROTO_UDP)
|
||||||
|
return RDMA_NETWORK_IPV6;
|
||||||
|
|
||||||
|
return RDMA_NETWORK_ROCE_V1;
|
||||||
|
}
|
||||||
|
|
||||||
struct find_gid_index_context {
|
struct find_gid_index_context {
|
||||||
u16 vlan_id;
|
u16 vlan_id;
|
||||||
|
enum ib_gid_type gid_type;
|
||||||
};
|
};
|
||||||
|
|
||||||
static bool find_gid_index(const union ib_gid *gid,
|
static bool find_gid_index(const union ib_gid *gid,
|
||||||
|
@ -322,6 +369,9 @@ static bool find_gid_index(const union ib_gid *gid,
|
||||||
struct find_gid_index_context *ctx =
|
struct find_gid_index_context *ctx =
|
||||||
(struct find_gid_index_context *)context;
|
(struct find_gid_index_context *)context;
|
||||||
|
|
||||||
|
if (ctx->gid_type != gid_attr->gid_type)
|
||||||
|
return false;
|
||||||
|
|
||||||
if ((!!(ctx->vlan_id != 0xffff) == !is_vlan_dev(gid_attr->ndev)) ||
|
if ((!!(ctx->vlan_id != 0xffff) == !is_vlan_dev(gid_attr->ndev)) ||
|
||||||
(is_vlan_dev(gid_attr->ndev) &&
|
(is_vlan_dev(gid_attr->ndev) &&
|
||||||
vlan_dev_vlan_id(gid_attr->ndev) != ctx->vlan_id))
|
vlan_dev_vlan_id(gid_attr->ndev) != ctx->vlan_id))
|
||||||
|
@ -332,14 +382,49 @@ static bool find_gid_index(const union ib_gid *gid,
|
||||||
|
|
||||||
static int get_sgid_index_from_eth(struct ib_device *device, u8 port_num,
|
static int get_sgid_index_from_eth(struct ib_device *device, u8 port_num,
|
||||||
u16 vlan_id, const union ib_gid *sgid,
|
u16 vlan_id, const union ib_gid *sgid,
|
||||||
|
enum ib_gid_type gid_type,
|
||||||
u16 *gid_index)
|
u16 *gid_index)
|
||||||
{
|
{
|
||||||
struct find_gid_index_context context = {.vlan_id = vlan_id};
|
struct find_gid_index_context context = {.vlan_id = vlan_id,
|
||||||
|
.gid_type = gid_type};
|
||||||
|
|
||||||
return ib_find_gid_by_filter(device, sgid, port_num, find_gid_index,
|
return ib_find_gid_by_filter(device, sgid, port_num, find_gid_index,
|
||||||
&context, gid_index);
|
&context, gid_index);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int get_gids_from_rdma_hdr(union rdma_network_hdr *hdr,
|
||||||
|
enum rdma_network_type net_type,
|
||||||
|
union ib_gid *sgid, union ib_gid *dgid)
|
||||||
|
{
|
||||||
|
struct sockaddr_in src_in;
|
||||||
|
struct sockaddr_in dst_in;
|
||||||
|
__be32 src_saddr, dst_saddr;
|
||||||
|
|
||||||
|
if (!sgid || !dgid)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
if (net_type == RDMA_NETWORK_IPV4) {
|
||||||
|
memcpy(&src_in.sin_addr.s_addr,
|
||||||
|
&hdr->roce4grh.saddr, 4);
|
||||||
|
memcpy(&dst_in.sin_addr.s_addr,
|
||||||
|
&hdr->roce4grh.daddr, 4);
|
||||||
|
src_saddr = src_in.sin_addr.s_addr;
|
||||||
|
dst_saddr = dst_in.sin_addr.s_addr;
|
||||||
|
ipv6_addr_set_v4mapped(src_saddr,
|
||||||
|
(struct in6_addr *)sgid);
|
||||||
|
ipv6_addr_set_v4mapped(dst_saddr,
|
||||||
|
(struct in6_addr *)dgid);
|
||||||
|
return 0;
|
||||||
|
} else if (net_type == RDMA_NETWORK_IPV6 ||
|
||||||
|
net_type == RDMA_NETWORK_IB) {
|
||||||
|
*dgid = hdr->ibgrh.dgid;
|
||||||
|
*sgid = hdr->ibgrh.sgid;
|
||||||
|
return 0;
|
||||||
|
} else {
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
int ib_init_ah_from_wc(struct ib_device *device, u8 port_num,
|
int ib_init_ah_from_wc(struct ib_device *device, u8 port_num,
|
||||||
const struct ib_wc *wc, const struct ib_grh *grh,
|
const struct ib_wc *wc, const struct ib_grh *grh,
|
||||||
struct ib_ah_attr *ah_attr)
|
struct ib_ah_attr *ah_attr)
|
||||||
|
@ -347,33 +432,72 @@ int ib_init_ah_from_wc(struct ib_device *device, u8 port_num,
|
||||||
u32 flow_class;
|
u32 flow_class;
|
||||||
u16 gid_index;
|
u16 gid_index;
|
||||||
int ret;
|
int ret;
|
||||||
|
enum rdma_network_type net_type = RDMA_NETWORK_IB;
|
||||||
|
enum ib_gid_type gid_type = IB_GID_TYPE_IB;
|
||||||
|
int hoplimit = 0xff;
|
||||||
|
union ib_gid dgid;
|
||||||
|
union ib_gid sgid;
|
||||||
|
|
||||||
memset(ah_attr, 0, sizeof *ah_attr);
|
memset(ah_attr, 0, sizeof *ah_attr);
|
||||||
if (rdma_cap_eth_ah(device, port_num)) {
|
if (rdma_cap_eth_ah(device, port_num)) {
|
||||||
|
if (wc->wc_flags & IB_WC_WITH_NETWORK_HDR_TYPE)
|
||||||
|
net_type = wc->network_hdr_type;
|
||||||
|
else
|
||||||
|
net_type = ib_get_net_type_by_grh(device, port_num, grh);
|
||||||
|
gid_type = ib_network_to_gid_type(net_type);
|
||||||
|
}
|
||||||
|
ret = get_gids_from_rdma_hdr((union rdma_network_hdr *)grh, net_type,
|
||||||
|
&sgid, &dgid);
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
if (rdma_protocol_roce(device, port_num)) {
|
||||||
|
int if_index = 0;
|
||||||
u16 vlan_id = wc->wc_flags & IB_WC_WITH_VLAN ?
|
u16 vlan_id = wc->wc_flags & IB_WC_WITH_VLAN ?
|
||||||
wc->vlan_id : 0xffff;
|
wc->vlan_id : 0xffff;
|
||||||
|
struct net_device *idev;
|
||||||
|
struct net_device *resolved_dev;
|
||||||
|
|
||||||
if (!(wc->wc_flags & IB_WC_GRH))
|
if (!(wc->wc_flags & IB_WC_GRH))
|
||||||
return -EPROTOTYPE;
|
return -EPROTOTYPE;
|
||||||
|
|
||||||
if (!(wc->wc_flags & IB_WC_WITH_SMAC) ||
|
if (!device->get_netdev)
|
||||||
!(wc->wc_flags & IB_WC_WITH_VLAN)) {
|
return -EOPNOTSUPP;
|
||||||
ret = rdma_addr_find_dmac_by_grh(&grh->dgid, &grh->sgid,
|
|
||||||
ah_attr->dmac,
|
idev = device->get_netdev(device, port_num);
|
||||||
wc->wc_flags & IB_WC_WITH_VLAN ?
|
if (!idev)
|
||||||
NULL : &vlan_id,
|
return -ENODEV;
|
||||||
0);
|
|
||||||
if (ret)
|
ret = rdma_addr_find_l2_eth_by_grh(&dgid, &sgid,
|
||||||
return ret;
|
ah_attr->dmac,
|
||||||
|
wc->wc_flags & IB_WC_WITH_VLAN ?
|
||||||
|
NULL : &vlan_id,
|
||||||
|
&if_index, &hoplimit);
|
||||||
|
if (ret) {
|
||||||
|
dev_put(idev);
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = get_sgid_index_from_eth(device, port_num, vlan_id,
|
resolved_dev = dev_get_by_index(&init_net, if_index);
|
||||||
&grh->dgid, &gid_index);
|
if (resolved_dev->flags & IFF_LOOPBACK) {
|
||||||
|
dev_put(resolved_dev);
|
||||||
|
resolved_dev = idev;
|
||||||
|
dev_hold(resolved_dev);
|
||||||
|
}
|
||||||
|
rcu_read_lock();
|
||||||
|
if (resolved_dev != idev && !rdma_is_upper_dev_rcu(idev,
|
||||||
|
resolved_dev))
|
||||||
|
ret = -EHOSTUNREACH;
|
||||||
|
rcu_read_unlock();
|
||||||
|
dev_put(idev);
|
||||||
|
dev_put(resolved_dev);
|
||||||
if (ret)
|
if (ret)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
if (wc->wc_flags & IB_WC_WITH_SMAC)
|
ret = get_sgid_index_from_eth(device, port_num, vlan_id,
|
||||||
memcpy(ah_attr->dmac, wc->smac, ETH_ALEN);
|
&dgid, gid_type, &gid_index);
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
ah_attr->dlid = wc->slid;
|
ah_attr->dlid = wc->slid;
|
||||||
|
@ -383,10 +507,11 @@ int ib_init_ah_from_wc(struct ib_device *device, u8 port_num,
|
||||||
|
|
||||||
if (wc->wc_flags & IB_WC_GRH) {
|
if (wc->wc_flags & IB_WC_GRH) {
|
||||||
ah_attr->ah_flags = IB_AH_GRH;
|
ah_attr->ah_flags = IB_AH_GRH;
|
||||||
ah_attr->grh.dgid = grh->sgid;
|
ah_attr->grh.dgid = sgid;
|
||||||
|
|
||||||
if (!rdma_cap_eth_ah(device, port_num)) {
|
if (!rdma_cap_eth_ah(device, port_num)) {
|
||||||
ret = ib_find_cached_gid_by_port(device, &grh->dgid,
|
ret = ib_find_cached_gid_by_port(device, &dgid,
|
||||||
|
IB_GID_TYPE_IB,
|
||||||
port_num, NULL,
|
port_num, NULL,
|
||||||
&gid_index);
|
&gid_index);
|
||||||
if (ret)
|
if (ret)
|
||||||
|
@ -396,7 +521,7 @@ int ib_init_ah_from_wc(struct ib_device *device, u8 port_num,
|
||||||
ah_attr->grh.sgid_index = (u8) gid_index;
|
ah_attr->grh.sgid_index = (u8) gid_index;
|
||||||
flow_class = be32_to_cpu(grh->version_tclass_flow);
|
flow_class = be32_to_cpu(grh->version_tclass_flow);
|
||||||
ah_attr->grh.flow_label = flow_class & 0xFFFFF;
|
ah_attr->grh.flow_label = flow_class & 0xFFFFF;
|
||||||
ah_attr->grh.hop_limit = 0xFF;
|
ah_attr->grh.hop_limit = hoplimit;
|
||||||
ah_attr->grh.traffic_class = (flow_class >> 20) & 0xFF;
|
ah_attr->grh.traffic_class = (flow_class >> 20) & 0xFF;
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -1014,6 +1139,7 @@ int ib_resolve_eth_dmac(struct ib_qp *qp,
|
||||||
union ib_gid sgid;
|
union ib_gid sgid;
|
||||||
struct ib_gid_attr sgid_attr;
|
struct ib_gid_attr sgid_attr;
|
||||||
int ifindex;
|
int ifindex;
|
||||||
|
int hop_limit;
|
||||||
|
|
||||||
ret = ib_query_gid(qp->device,
|
ret = ib_query_gid(qp->device,
|
||||||
qp_attr->ah_attr.port_num,
|
qp_attr->ah_attr.port_num,
|
||||||
|
@ -1028,12 +1154,14 @@ int ib_resolve_eth_dmac(struct ib_qp *qp,
|
||||||
|
|
||||||
ifindex = sgid_attr.ndev->ifindex;
|
ifindex = sgid_attr.ndev->ifindex;
|
||||||
|
|
||||||
ret = rdma_addr_find_dmac_by_grh(&sgid,
|
ret = rdma_addr_find_l2_eth_by_grh(&sgid,
|
||||||
&qp_attr->ah_attr.grh.dgid,
|
&qp_attr->ah_attr.grh.dgid,
|
||||||
qp_attr->ah_attr.dmac,
|
qp_attr->ah_attr.dmac,
|
||||||
NULL, ifindex);
|
NULL, &ifindex, &hop_limit);
|
||||||
|
|
||||||
dev_put(sgid_attr.ndev);
|
dev_put(sgid_attr.ndev);
|
||||||
|
|
||||||
|
qp_attr->ah_attr.grh.hop_limit = hop_limit;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
out:
|
out:
|
||||||
|
@ -1215,29 +1343,17 @@ struct ib_mr *ib_get_dma_mr(struct ib_pd *pd, int mr_access_flags)
|
||||||
mr->pd = pd;
|
mr->pd = pd;
|
||||||
mr->uobject = NULL;
|
mr->uobject = NULL;
|
||||||
atomic_inc(&pd->usecnt);
|
atomic_inc(&pd->usecnt);
|
||||||
atomic_set(&mr->usecnt, 0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return mr;
|
return mr;
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(ib_get_dma_mr);
|
EXPORT_SYMBOL(ib_get_dma_mr);
|
||||||
|
|
||||||
int ib_query_mr(struct ib_mr *mr, struct ib_mr_attr *mr_attr)
|
|
||||||
{
|
|
||||||
return mr->device->query_mr ?
|
|
||||||
mr->device->query_mr(mr, mr_attr) : -ENOSYS;
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL(ib_query_mr);
|
|
||||||
|
|
||||||
int ib_dereg_mr(struct ib_mr *mr)
|
int ib_dereg_mr(struct ib_mr *mr)
|
||||||
{
|
{
|
||||||
struct ib_pd *pd;
|
struct ib_pd *pd = mr->pd;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
if (atomic_read(&mr->usecnt))
|
|
||||||
return -EBUSY;
|
|
||||||
|
|
||||||
pd = mr->pd;
|
|
||||||
ret = mr->device->dereg_mr(mr);
|
ret = mr->device->dereg_mr(mr);
|
||||||
if (!ret)
|
if (!ret)
|
||||||
atomic_dec(&pd->usecnt);
|
atomic_dec(&pd->usecnt);
|
||||||
|
@ -1273,49 +1389,12 @@ struct ib_mr *ib_alloc_mr(struct ib_pd *pd,
|
||||||
mr->pd = pd;
|
mr->pd = pd;
|
||||||
mr->uobject = NULL;
|
mr->uobject = NULL;
|
||||||
atomic_inc(&pd->usecnt);
|
atomic_inc(&pd->usecnt);
|
||||||
atomic_set(&mr->usecnt, 0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return mr;
|
return mr;
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(ib_alloc_mr);
|
EXPORT_SYMBOL(ib_alloc_mr);
|
||||||
|
|
||||||
/* Memory windows */
|
|
||||||
|
|
||||||
struct ib_mw *ib_alloc_mw(struct ib_pd *pd, enum ib_mw_type type)
|
|
||||||
{
|
|
||||||
struct ib_mw *mw;
|
|
||||||
|
|
||||||
if (!pd->device->alloc_mw)
|
|
||||||
return ERR_PTR(-ENOSYS);
|
|
||||||
|
|
||||||
mw = pd->device->alloc_mw(pd, type);
|
|
||||||
if (!IS_ERR(mw)) {
|
|
||||||
mw->device = pd->device;
|
|
||||||
mw->pd = pd;
|
|
||||||
mw->uobject = NULL;
|
|
||||||
mw->type = type;
|
|
||||||
atomic_inc(&pd->usecnt);
|
|
||||||
}
|
|
||||||
|
|
||||||
return mw;
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL(ib_alloc_mw);
|
|
||||||
|
|
||||||
int ib_dealloc_mw(struct ib_mw *mw)
|
|
||||||
{
|
|
||||||
struct ib_pd *pd;
|
|
||||||
int ret;
|
|
||||||
|
|
||||||
pd = mw->pd;
|
|
||||||
ret = mw->device->dealloc_mw(mw);
|
|
||||||
if (!ret)
|
|
||||||
atomic_dec(&pd->usecnt);
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL(ib_dealloc_mw);
|
|
||||||
|
|
||||||
/* "Fast" memory regions */
|
/* "Fast" memory regions */
|
||||||
|
|
||||||
struct ib_fmr *ib_alloc_fmr(struct ib_pd *pd,
|
struct ib_fmr *ib_alloc_fmr(struct ib_pd *pd,
|
||||||
|
@ -1530,7 +1609,7 @@ int ib_sg_to_pages(struct ib_mr *mr,
|
||||||
int (*set_page)(struct ib_mr *, u64))
|
int (*set_page)(struct ib_mr *, u64))
|
||||||
{
|
{
|
||||||
struct scatterlist *sg;
|
struct scatterlist *sg;
|
||||||
u64 last_end_dma_addr = 0, last_page_addr = 0;
|
u64 last_end_dma_addr = 0;
|
||||||
unsigned int last_page_off = 0;
|
unsigned int last_page_off = 0;
|
||||||
u64 page_mask = ~((u64)mr->page_size - 1);
|
u64 page_mask = ~((u64)mr->page_size - 1);
|
||||||
int i, ret;
|
int i, ret;
|
||||||
|
@ -1572,7 +1651,6 @@ int ib_sg_to_pages(struct ib_mr *mr,
|
||||||
|
|
||||||
mr->length += dma_len;
|
mr->length += dma_len;
|
||||||
last_end_dma_addr = end_dma_addr;
|
last_end_dma_addr = end_dma_addr;
|
||||||
last_page_addr = end_dma_addr & page_mask;
|
|
||||||
last_page_off = end_dma_addr & ~page_mask;
|
last_page_off = end_dma_addr & ~page_mask;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -149,7 +149,7 @@ static int iwch_l2t_send(struct t3cdev *tdev, struct sk_buff *skb, struct l2t_en
|
||||||
error = l2t_send(tdev, skb, l2e);
|
error = l2t_send(tdev, skb, l2e);
|
||||||
if (error < 0)
|
if (error < 0)
|
||||||
kfree_skb(skb);
|
kfree_skb(skb);
|
||||||
return error;
|
return error < 0 ? error : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int iwch_cxgb3_ofld_send(struct t3cdev *tdev, struct sk_buff *skb)
|
int iwch_cxgb3_ofld_send(struct t3cdev *tdev, struct sk_buff *skb)
|
||||||
|
@ -165,7 +165,7 @@ int iwch_cxgb3_ofld_send(struct t3cdev *tdev, struct sk_buff *skb)
|
||||||
error = cxgb3_ofld_send(tdev, skb);
|
error = cxgb3_ofld_send(tdev, skb);
|
||||||
if (error < 0)
|
if (error < 0)
|
||||||
kfree_skb(skb);
|
kfree_skb(skb);
|
||||||
return error;
|
return error < 0 ? error : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void release_tid(struct t3cdev *tdev, u32 hwtid, struct sk_buff *skb)
|
static void release_tid(struct t3cdev *tdev, u32 hwtid, struct sk_buff *skb)
|
||||||
|
|
|
@ -115,10 +115,6 @@ static int iwch_poll_cq_one(struct iwch_dev *rhp, struct iwch_cq *chp,
|
||||||
case T3_SEND_WITH_SE_INV:
|
case T3_SEND_WITH_SE_INV:
|
||||||
wc->opcode = IB_WC_SEND;
|
wc->opcode = IB_WC_SEND;
|
||||||
break;
|
break;
|
||||||
case T3_BIND_MW:
|
|
||||||
wc->opcode = IB_WC_BIND_MW;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case T3_LOCAL_INV:
|
case T3_LOCAL_INV:
|
||||||
wc->opcode = IB_WC_LOCAL_INV;
|
wc->opcode = IB_WC_LOCAL_INV;
|
||||||
break;
|
break;
|
||||||
|
|
|
@ -75,37 +75,6 @@ int iwch_register_mem(struct iwch_dev *rhp, struct iwch_pd *php,
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
int iwch_reregister_mem(struct iwch_dev *rhp, struct iwch_pd *php,
|
|
||||||
struct iwch_mr *mhp,
|
|
||||||
int shift,
|
|
||||||
int npages)
|
|
||||||
{
|
|
||||||
u32 stag;
|
|
||||||
int ret;
|
|
||||||
|
|
||||||
/* We could support this... */
|
|
||||||
if (npages > mhp->attr.pbl_size)
|
|
||||||
return -ENOMEM;
|
|
||||||
|
|
||||||
stag = mhp->attr.stag;
|
|
||||||
if (cxio_reregister_phys_mem(&rhp->rdev,
|
|
||||||
&stag, mhp->attr.pdid,
|
|
||||||
mhp->attr.perms,
|
|
||||||
mhp->attr.zbva,
|
|
||||||
mhp->attr.va_fbo,
|
|
||||||
mhp->attr.len,
|
|
||||||
shift - 12,
|
|
||||||
mhp->attr.pbl_size, mhp->attr.pbl_addr))
|
|
||||||
return -ENOMEM;
|
|
||||||
|
|
||||||
ret = iwch_finish_mem_reg(mhp, stag);
|
|
||||||
if (ret)
|
|
||||||
cxio_dereg_mem(&rhp->rdev, mhp->attr.stag, mhp->attr.pbl_size,
|
|
||||||
mhp->attr.pbl_addr);
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
int iwch_alloc_pbl(struct iwch_mr *mhp, int npages)
|
int iwch_alloc_pbl(struct iwch_mr *mhp, int npages)
|
||||||
{
|
{
|
||||||
mhp->attr.pbl_addr = cxio_hal_pblpool_alloc(&mhp->rhp->rdev,
|
mhp->attr.pbl_addr = cxio_hal_pblpool_alloc(&mhp->rhp->rdev,
|
||||||
|
@ -130,74 +99,3 @@ int iwch_write_pbl(struct iwch_mr *mhp, __be64 *pages, int npages, int offset)
|
||||||
return cxio_write_pbl(&mhp->rhp->rdev, pages,
|
return cxio_write_pbl(&mhp->rhp->rdev, pages,
|
||||||
mhp->attr.pbl_addr + (offset << 3), npages);
|
mhp->attr.pbl_addr + (offset << 3), npages);
|
||||||
}
|
}
|
||||||
|
|
||||||
int build_phys_page_list(struct ib_phys_buf *buffer_list,
|
|
||||||
int num_phys_buf,
|
|
||||||
u64 *iova_start,
|
|
||||||
u64 *total_size,
|
|
||||||
int *npages,
|
|
||||||
int *shift,
|
|
||||||
__be64 **page_list)
|
|
||||||
{
|
|
||||||
u64 mask;
|
|
||||||
int i, j, n;
|
|
||||||
|
|
||||||
mask = 0;
|
|
||||||
*total_size = 0;
|
|
||||||
for (i = 0; i < num_phys_buf; ++i) {
|
|
||||||
if (i != 0 && buffer_list[i].addr & ~PAGE_MASK)
|
|
||||||
return -EINVAL;
|
|
||||||
if (i != 0 && i != num_phys_buf - 1 &&
|
|
||||||
(buffer_list[i].size & ~PAGE_MASK))
|
|
||||||
return -EINVAL;
|
|
||||||
*total_size += buffer_list[i].size;
|
|
||||||
if (i > 0)
|
|
||||||
mask |= buffer_list[i].addr;
|
|
||||||
else
|
|
||||||
mask |= buffer_list[i].addr & PAGE_MASK;
|
|
||||||
if (i != num_phys_buf - 1)
|
|
||||||
mask |= buffer_list[i].addr + buffer_list[i].size;
|
|
||||||
else
|
|
||||||
mask |= (buffer_list[i].addr + buffer_list[i].size +
|
|
||||||
PAGE_SIZE - 1) & PAGE_MASK;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (*total_size > 0xFFFFFFFFULL)
|
|
||||||
return -ENOMEM;
|
|
||||||
|
|
||||||
/* Find largest page shift we can use to cover buffers */
|
|
||||||
for (*shift = PAGE_SHIFT; *shift < 27; ++(*shift))
|
|
||||||
if ((1ULL << *shift) & mask)
|
|
||||||
break;
|
|
||||||
|
|
||||||
buffer_list[0].size += buffer_list[0].addr & ((1ULL << *shift) - 1);
|
|
||||||
buffer_list[0].addr &= ~0ull << *shift;
|
|
||||||
|
|
||||||
*npages = 0;
|
|
||||||
for (i = 0; i < num_phys_buf; ++i)
|
|
||||||
*npages += (buffer_list[i].size +
|
|
||||||
(1ULL << *shift) - 1) >> *shift;
|
|
||||||
|
|
||||||
if (!*npages)
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
*page_list = kmalloc(sizeof(u64) * *npages, GFP_KERNEL);
|
|
||||||
if (!*page_list)
|
|
||||||
return -ENOMEM;
|
|
||||||
|
|
||||||
n = 0;
|
|
||||||
for (i = 0; i < num_phys_buf; ++i)
|
|
||||||
for (j = 0;
|
|
||||||
j < (buffer_list[i].size + (1ULL << *shift) - 1) >> *shift;
|
|
||||||
++j)
|
|
||||||
(*page_list)[n++] = cpu_to_be64(buffer_list[i].addr +
|
|
||||||
((u64) j << *shift));
|
|
||||||
|
|
||||||
PDBG("%s va 0x%llx mask 0x%llx shift %d len %lld pbl_size %d\n",
|
|
||||||
__func__, (unsigned long long) *iova_start,
|
|
||||||
(unsigned long long) mask, *shift, (unsigned long long) *total_size,
|
|
||||||
*npages);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
|
@ -458,9 +458,6 @@ static int iwch_dereg_mr(struct ib_mr *ib_mr)
|
||||||
u32 mmid;
|
u32 mmid;
|
||||||
|
|
||||||
PDBG("%s ib_mr %p\n", __func__, ib_mr);
|
PDBG("%s ib_mr %p\n", __func__, ib_mr);
|
||||||
/* There can be no memory windows */
|
|
||||||
if (atomic_read(&ib_mr->usecnt))
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
mhp = to_iwch_mr(ib_mr);
|
mhp = to_iwch_mr(ib_mr);
|
||||||
kfree(mhp->pages);
|
kfree(mhp->pages);
|
||||||
|
@ -479,24 +476,25 @@ static int iwch_dereg_mr(struct ib_mr *ib_mr)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct ib_mr *iwch_register_phys_mem(struct ib_pd *pd,
|
static struct ib_mr *iwch_get_dma_mr(struct ib_pd *pd, int acc)
|
||||||
struct ib_phys_buf *buffer_list,
|
|
||||||
int num_phys_buf,
|
|
||||||
int acc,
|
|
||||||
u64 *iova_start)
|
|
||||||
{
|
{
|
||||||
__be64 *page_list;
|
const u64 total_size = 0xffffffff;
|
||||||
int shift;
|
const u64 mask = (total_size + PAGE_SIZE - 1) & PAGE_MASK;
|
||||||
u64 total_size;
|
struct iwch_pd *php = to_iwch_pd(pd);
|
||||||
int npages;
|
struct iwch_dev *rhp = php->rhp;
|
||||||
struct iwch_dev *rhp;
|
|
||||||
struct iwch_pd *php;
|
|
||||||
struct iwch_mr *mhp;
|
struct iwch_mr *mhp;
|
||||||
int ret;
|
__be64 *page_list;
|
||||||
|
int shift = 26, npages, ret, i;
|
||||||
|
|
||||||
PDBG("%s ib_pd %p\n", __func__, pd);
|
PDBG("%s ib_pd %p\n", __func__, pd);
|
||||||
php = to_iwch_pd(pd);
|
|
||||||
rhp = php->rhp;
|
/*
|
||||||
|
* T3 only supports 32 bits of size.
|
||||||
|
*/
|
||||||
|
if (sizeof(phys_addr_t) > 4) {
|
||||||
|
pr_warn_once(MOD "Cannot support dma_mrs on this platform.\n");
|
||||||
|
return ERR_PTR(-ENOTSUPP);
|
||||||
|
}
|
||||||
|
|
||||||
mhp = kzalloc(sizeof(*mhp), GFP_KERNEL);
|
mhp = kzalloc(sizeof(*mhp), GFP_KERNEL);
|
||||||
if (!mhp)
|
if (!mhp)
|
||||||
|
@ -504,22 +502,23 @@ static struct ib_mr *iwch_register_phys_mem(struct ib_pd *pd,
|
||||||
|
|
||||||
mhp->rhp = rhp;
|
mhp->rhp = rhp;
|
||||||
|
|
||||||
/* First check that we have enough alignment */
|
npages = (total_size + (1ULL << shift) - 1) >> shift;
|
||||||
if ((*iova_start & ~PAGE_MASK) != (buffer_list[0].addr & ~PAGE_MASK)) {
|
if (!npages) {
|
||||||
ret = -EINVAL;
|
ret = -EINVAL;
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (num_phys_buf > 1 &&
|
page_list = kmalloc_array(npages, sizeof(u64), GFP_KERNEL);
|
||||||
((buffer_list[0].addr + buffer_list[0].size) & ~PAGE_MASK)) {
|
if (!page_list) {
|
||||||
ret = -EINVAL;
|
ret = -ENOMEM;
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = build_phys_page_list(buffer_list, num_phys_buf, iova_start,
|
for (i = 0; i < npages; i++)
|
||||||
&total_size, &npages, &shift, &page_list);
|
page_list[i] = cpu_to_be64((u64)i << shift);
|
||||||
if (ret)
|
|
||||||
goto err;
|
PDBG("%s mask 0x%llx shift %d len %lld pbl_size %d\n",
|
||||||
|
__func__, mask, shift, total_size, npages);
|
||||||
|
|
||||||
ret = iwch_alloc_pbl(mhp, npages);
|
ret = iwch_alloc_pbl(mhp, npages);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
|
@ -536,7 +535,7 @@ static struct ib_mr *iwch_register_phys_mem(struct ib_pd *pd,
|
||||||
mhp->attr.zbva = 0;
|
mhp->attr.zbva = 0;
|
||||||
|
|
||||||
mhp->attr.perms = iwch_ib_to_tpt_access(acc);
|
mhp->attr.perms = iwch_ib_to_tpt_access(acc);
|
||||||
mhp->attr.va_fbo = *iova_start;
|
mhp->attr.va_fbo = 0;
|
||||||
mhp->attr.page_size = shift - 12;
|
mhp->attr.page_size = shift - 12;
|
||||||
|
|
||||||
mhp->attr.len = (u32) total_size;
|
mhp->attr.len = (u32) total_size;
|
||||||
|
@ -553,76 +552,8 @@ static struct ib_mr *iwch_register_phys_mem(struct ib_pd *pd,
|
||||||
err:
|
err:
|
||||||
kfree(mhp);
|
kfree(mhp);
|
||||||
return ERR_PTR(ret);
|
return ERR_PTR(ret);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int iwch_reregister_phys_mem(struct ib_mr *mr,
|
|
||||||
int mr_rereg_mask,
|
|
||||||
struct ib_pd *pd,
|
|
||||||
struct ib_phys_buf *buffer_list,
|
|
||||||
int num_phys_buf,
|
|
||||||
int acc, u64 * iova_start)
|
|
||||||
{
|
|
||||||
|
|
||||||
struct iwch_mr mh, *mhp;
|
|
||||||
struct iwch_pd *php;
|
|
||||||
struct iwch_dev *rhp;
|
|
||||||
__be64 *page_list = NULL;
|
|
||||||
int shift = 0;
|
|
||||||
u64 total_size;
|
|
||||||
int npages = 0;
|
|
||||||
int ret;
|
|
||||||
|
|
||||||
PDBG("%s ib_mr %p ib_pd %p\n", __func__, mr, pd);
|
|
||||||
|
|
||||||
/* There can be no memory windows */
|
|
||||||
if (atomic_read(&mr->usecnt))
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
mhp = to_iwch_mr(mr);
|
|
||||||
rhp = mhp->rhp;
|
|
||||||
php = to_iwch_pd(mr->pd);
|
|
||||||
|
|
||||||
/* make sure we are on the same adapter */
|
|
||||||
if (rhp != php->rhp)
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
memcpy(&mh, mhp, sizeof *mhp);
|
|
||||||
|
|
||||||
if (mr_rereg_mask & IB_MR_REREG_PD)
|
|
||||||
php = to_iwch_pd(pd);
|
|
||||||
if (mr_rereg_mask & IB_MR_REREG_ACCESS)
|
|
||||||
mh.attr.perms = iwch_ib_to_tpt_access(acc);
|
|
||||||
if (mr_rereg_mask & IB_MR_REREG_TRANS) {
|
|
||||||
ret = build_phys_page_list(buffer_list, num_phys_buf,
|
|
||||||
iova_start,
|
|
||||||
&total_size, &npages,
|
|
||||||
&shift, &page_list);
|
|
||||||
if (ret)
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
ret = iwch_reregister_mem(rhp, php, &mh, shift, npages);
|
|
||||||
kfree(page_list);
|
|
||||||
if (ret) {
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
if (mr_rereg_mask & IB_MR_REREG_PD)
|
|
||||||
mhp->attr.pdid = php->pdid;
|
|
||||||
if (mr_rereg_mask & IB_MR_REREG_ACCESS)
|
|
||||||
mhp->attr.perms = iwch_ib_to_tpt_access(acc);
|
|
||||||
if (mr_rereg_mask & IB_MR_REREG_TRANS) {
|
|
||||||
mhp->attr.zbva = 0;
|
|
||||||
mhp->attr.va_fbo = *iova_start;
|
|
||||||
mhp->attr.page_size = shift - 12;
|
|
||||||
mhp->attr.len = (u32) total_size;
|
|
||||||
mhp->attr.pbl_size = npages;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static struct ib_mr *iwch_reg_user_mr(struct ib_pd *pd, u64 start, u64 length,
|
static struct ib_mr *iwch_reg_user_mr(struct ib_pd *pd, u64 start, u64 length,
|
||||||
u64 virt, int acc, struct ib_udata *udata)
|
u64 virt, int acc, struct ib_udata *udata)
|
||||||
{
|
{
|
||||||
|
@ -726,28 +657,6 @@ static struct ib_mr *iwch_reg_user_mr(struct ib_pd *pd, u64 start, u64 length,
|
||||||
return ERR_PTR(err);
|
return ERR_PTR(err);
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct ib_mr *iwch_get_dma_mr(struct ib_pd *pd, int acc)
|
|
||||||
{
|
|
||||||
struct ib_phys_buf bl;
|
|
||||||
u64 kva;
|
|
||||||
struct ib_mr *ibmr;
|
|
||||||
|
|
||||||
PDBG("%s ib_pd %p\n", __func__, pd);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* T3 only supports 32 bits of size.
|
|
||||||
*/
|
|
||||||
if (sizeof(phys_addr_t) > 4) {
|
|
||||||
pr_warn_once(MOD "Cannot support dma_mrs on this platform.\n");
|
|
||||||
return ERR_PTR(-ENOTSUPP);
|
|
||||||
}
|
|
||||||
bl.size = 0xffffffff;
|
|
||||||
bl.addr = 0;
|
|
||||||
kva = 0;
|
|
||||||
ibmr = iwch_register_phys_mem(pd, &bl, 1, acc, &kva);
|
|
||||||
return ibmr;
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct ib_mw *iwch_alloc_mw(struct ib_pd *pd, enum ib_mw_type type)
|
static struct ib_mw *iwch_alloc_mw(struct ib_pd *pd, enum ib_mw_type type)
|
||||||
{
|
{
|
||||||
struct iwch_dev *rhp;
|
struct iwch_dev *rhp;
|
||||||
|
@ -1452,12 +1361,9 @@ int iwch_register_device(struct iwch_dev *dev)
|
||||||
dev->ibdev.resize_cq = iwch_resize_cq;
|
dev->ibdev.resize_cq = iwch_resize_cq;
|
||||||
dev->ibdev.poll_cq = iwch_poll_cq;
|
dev->ibdev.poll_cq = iwch_poll_cq;
|
||||||
dev->ibdev.get_dma_mr = iwch_get_dma_mr;
|
dev->ibdev.get_dma_mr = iwch_get_dma_mr;
|
||||||
dev->ibdev.reg_phys_mr = iwch_register_phys_mem;
|
|
||||||
dev->ibdev.rereg_phys_mr = iwch_reregister_phys_mem;
|
|
||||||
dev->ibdev.reg_user_mr = iwch_reg_user_mr;
|
dev->ibdev.reg_user_mr = iwch_reg_user_mr;
|
||||||
dev->ibdev.dereg_mr = iwch_dereg_mr;
|
dev->ibdev.dereg_mr = iwch_dereg_mr;
|
||||||
dev->ibdev.alloc_mw = iwch_alloc_mw;
|
dev->ibdev.alloc_mw = iwch_alloc_mw;
|
||||||
dev->ibdev.bind_mw = iwch_bind_mw;
|
|
||||||
dev->ibdev.dealloc_mw = iwch_dealloc_mw;
|
dev->ibdev.dealloc_mw = iwch_dealloc_mw;
|
||||||
dev->ibdev.alloc_mr = iwch_alloc_mr;
|
dev->ibdev.alloc_mr = iwch_alloc_mr;
|
||||||
dev->ibdev.map_mr_sg = iwch_map_mr_sg;
|
dev->ibdev.map_mr_sg = iwch_map_mr_sg;
|
||||||
|
|
|
@ -330,9 +330,6 @@ int iwch_post_send(struct ib_qp *ibqp, struct ib_send_wr *wr,
|
||||||
struct ib_send_wr **bad_wr);
|
struct ib_send_wr **bad_wr);
|
||||||
int iwch_post_receive(struct ib_qp *ibqp, struct ib_recv_wr *wr,
|
int iwch_post_receive(struct ib_qp *ibqp, struct ib_recv_wr *wr,
|
||||||
struct ib_recv_wr **bad_wr);
|
struct ib_recv_wr **bad_wr);
|
||||||
int iwch_bind_mw(struct ib_qp *qp,
|
|
||||||
struct ib_mw *mw,
|
|
||||||
struct ib_mw_bind *mw_bind);
|
|
||||||
int iwch_poll_cq(struct ib_cq *ibcq, int num_entries, struct ib_wc *wc);
|
int iwch_poll_cq(struct ib_cq *ibcq, int num_entries, struct ib_wc *wc);
|
||||||
int iwch_post_terminate(struct iwch_qp *qhp, struct respQ_msg_t *rsp_msg);
|
int iwch_post_terminate(struct iwch_qp *qhp, struct respQ_msg_t *rsp_msg);
|
||||||
int iwch_post_zb_read(struct iwch_ep *ep);
|
int iwch_post_zb_read(struct iwch_ep *ep);
|
||||||
|
@ -341,21 +338,9 @@ void iwch_unregister_device(struct iwch_dev *dev);
|
||||||
void stop_read_rep_timer(struct iwch_qp *qhp);
|
void stop_read_rep_timer(struct iwch_qp *qhp);
|
||||||
int iwch_register_mem(struct iwch_dev *rhp, struct iwch_pd *php,
|
int iwch_register_mem(struct iwch_dev *rhp, struct iwch_pd *php,
|
||||||
struct iwch_mr *mhp, int shift);
|
struct iwch_mr *mhp, int shift);
|
||||||
int iwch_reregister_mem(struct iwch_dev *rhp, struct iwch_pd *php,
|
|
||||||
struct iwch_mr *mhp,
|
|
||||||
int shift,
|
|
||||||
int npages);
|
|
||||||
int iwch_alloc_pbl(struct iwch_mr *mhp, int npages);
|
int iwch_alloc_pbl(struct iwch_mr *mhp, int npages);
|
||||||
void iwch_free_pbl(struct iwch_mr *mhp);
|
void iwch_free_pbl(struct iwch_mr *mhp);
|
||||||
int iwch_write_pbl(struct iwch_mr *mhp, __be64 *pages, int npages, int offset);
|
int iwch_write_pbl(struct iwch_mr *mhp, __be64 *pages, int npages, int offset);
|
||||||
int build_phys_page_list(struct ib_phys_buf *buffer_list,
|
|
||||||
int num_phys_buf,
|
|
||||||
u64 *iova_start,
|
|
||||||
u64 *total_size,
|
|
||||||
int *npages,
|
|
||||||
int *shift,
|
|
||||||
__be64 **page_list);
|
|
||||||
|
|
||||||
|
|
||||||
#define IWCH_NODE_DESC "cxgb3 Chelsio Communications"
|
#define IWCH_NODE_DESC "cxgb3 Chelsio Communications"
|
||||||
|
|
||||||
|
|
|
@ -526,88 +526,6 @@ int iwch_post_receive(struct ib_qp *ibqp, struct ib_recv_wr *wr,
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
int iwch_bind_mw(struct ib_qp *qp,
|
|
||||||
struct ib_mw *mw,
|
|
||||||
struct ib_mw_bind *mw_bind)
|
|
||||||
{
|
|
||||||
struct iwch_dev *rhp;
|
|
||||||
struct iwch_mw *mhp;
|
|
||||||
struct iwch_qp *qhp;
|
|
||||||
union t3_wr *wqe;
|
|
||||||
u32 pbl_addr;
|
|
||||||
u8 page_size;
|
|
||||||
u32 num_wrs;
|
|
||||||
unsigned long flag;
|
|
||||||
struct ib_sge sgl;
|
|
||||||
int err=0;
|
|
||||||
enum t3_wr_flags t3_wr_flags;
|
|
||||||
u32 idx;
|
|
||||||
struct t3_swsq *sqp;
|
|
||||||
|
|
||||||
qhp = to_iwch_qp(qp);
|
|
||||||
mhp = to_iwch_mw(mw);
|
|
||||||
rhp = qhp->rhp;
|
|
||||||
|
|
||||||
spin_lock_irqsave(&qhp->lock, flag);
|
|
||||||
if (qhp->attr.state > IWCH_QP_STATE_RTS) {
|
|
||||||
spin_unlock_irqrestore(&qhp->lock, flag);
|
|
||||||
return -EINVAL;
|
|
||||||
}
|
|
||||||
num_wrs = Q_FREECNT(qhp->wq.sq_rptr, qhp->wq.sq_wptr,
|
|
||||||
qhp->wq.sq_size_log2);
|
|
||||||
if (num_wrs == 0) {
|
|
||||||
spin_unlock_irqrestore(&qhp->lock, flag);
|
|
||||||
return -ENOMEM;
|
|
||||||
}
|
|
||||||
idx = Q_PTR2IDX(qhp->wq.wptr, qhp->wq.size_log2);
|
|
||||||
PDBG("%s: idx 0x%0x, mw 0x%p, mw_bind 0x%p\n", __func__, idx,
|
|
||||||
mw, mw_bind);
|
|
||||||
wqe = (union t3_wr *) (qhp->wq.queue + idx);
|
|
||||||
|
|
||||||
t3_wr_flags = 0;
|
|
||||||
if (mw_bind->send_flags & IB_SEND_SIGNALED)
|
|
||||||
t3_wr_flags = T3_COMPLETION_FLAG;
|
|
||||||
|
|
||||||
sgl.addr = mw_bind->bind_info.addr;
|
|
||||||
sgl.lkey = mw_bind->bind_info.mr->lkey;
|
|
||||||
sgl.length = mw_bind->bind_info.length;
|
|
||||||
wqe->bind.reserved = 0;
|
|
||||||
wqe->bind.type = TPT_VATO;
|
|
||||||
|
|
||||||
/* TBD: check perms */
|
|
||||||
wqe->bind.perms = iwch_ib_to_tpt_bind_access(
|
|
||||||
mw_bind->bind_info.mw_access_flags);
|
|
||||||
wqe->bind.mr_stag = cpu_to_be32(mw_bind->bind_info.mr->lkey);
|
|
||||||
wqe->bind.mw_stag = cpu_to_be32(mw->rkey);
|
|
||||||
wqe->bind.mw_len = cpu_to_be32(mw_bind->bind_info.length);
|
|
||||||
wqe->bind.mw_va = cpu_to_be64(mw_bind->bind_info.addr);
|
|
||||||
err = iwch_sgl2pbl_map(rhp, &sgl, 1, &pbl_addr, &page_size);
|
|
||||||
if (err) {
|
|
||||||
spin_unlock_irqrestore(&qhp->lock, flag);
|
|
||||||
return err;
|
|
||||||
}
|
|
||||||
wqe->send.wrid.id0.hi = qhp->wq.sq_wptr;
|
|
||||||
sqp = qhp->wq.sq + Q_PTR2IDX(qhp->wq.sq_wptr, qhp->wq.sq_size_log2);
|
|
||||||
sqp->wr_id = mw_bind->wr_id;
|
|
||||||
sqp->opcode = T3_BIND_MW;
|
|
||||||
sqp->sq_wptr = qhp->wq.sq_wptr;
|
|
||||||
sqp->complete = 0;
|
|
||||||
sqp->signaled = (mw_bind->send_flags & IB_SEND_SIGNALED);
|
|
||||||
wqe->bind.mr_pbl_addr = cpu_to_be32(pbl_addr);
|
|
||||||
wqe->bind.mr_pagesz = page_size;
|
|
||||||
build_fw_riwrh((void *)wqe, T3_WR_BIND, t3_wr_flags,
|
|
||||||
Q_GENBIT(qhp->wq.wptr, qhp->wq.size_log2), 0,
|
|
||||||
sizeof(struct t3_bind_mw_wr) >> 3, T3_SOPEOP);
|
|
||||||
++(qhp->wq.wptr);
|
|
||||||
++(qhp->wq.sq_wptr);
|
|
||||||
spin_unlock_irqrestore(&qhp->lock, flag);
|
|
||||||
|
|
||||||
if (cxio_wq_db_enabled(&qhp->wq))
|
|
||||||
ring_doorbell(qhp->wq.doorbell, qhp->wq.qpid);
|
|
||||||
|
|
||||||
return err;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline void build_term_codes(struct respQ_msg_t *rsp_msg,
|
static inline void build_term_codes(struct respQ_msg_t *rsp_msg,
|
||||||
u8 *layer_type, u8 *ecode)
|
u8 *layer_type, u8 *ecode)
|
||||||
{
|
{
|
||||||
|
|
|
@ -3271,6 +3271,12 @@ static int create_server6(struct c4iw_dev *dev, struct c4iw_listen_ep *ep)
|
||||||
struct sockaddr_in6 *sin6 = (struct sockaddr_in6 *)
|
struct sockaddr_in6 *sin6 = (struct sockaddr_in6 *)
|
||||||
&ep->com.mapped_local_addr;
|
&ep->com.mapped_local_addr;
|
||||||
|
|
||||||
|
if (ipv6_addr_type(&sin6->sin6_addr) != IPV6_ADDR_ANY) {
|
||||||
|
err = cxgb4_clip_get(ep->com.dev->rdev.lldi.ports[0],
|
||||||
|
(const u32 *)&sin6->sin6_addr.s6_addr, 1);
|
||||||
|
if (err)
|
||||||
|
return err;
|
||||||
|
}
|
||||||
c4iw_init_wr_wait(&ep->com.wr_wait);
|
c4iw_init_wr_wait(&ep->com.wr_wait);
|
||||||
err = cxgb4_create_server6(ep->com.dev->rdev.lldi.ports[0],
|
err = cxgb4_create_server6(ep->com.dev->rdev.lldi.ports[0],
|
||||||
ep->stid, &sin6->sin6_addr,
|
ep->stid, &sin6->sin6_addr,
|
||||||
|
@ -3282,13 +3288,13 @@ static int create_server6(struct c4iw_dev *dev, struct c4iw_listen_ep *ep)
|
||||||
0, 0, __func__);
|
0, 0, __func__);
|
||||||
else if (err > 0)
|
else if (err > 0)
|
||||||
err = net_xmit_errno(err);
|
err = net_xmit_errno(err);
|
||||||
if (err)
|
if (err) {
|
||||||
|
cxgb4_clip_release(ep->com.dev->rdev.lldi.ports[0],
|
||||||
|
(const u32 *)&sin6->sin6_addr.s6_addr, 1);
|
||||||
pr_err("cxgb4_create_server6/filter failed err %d stid %d laddr %pI6 lport %d\n",
|
pr_err("cxgb4_create_server6/filter failed err %d stid %d laddr %pI6 lport %d\n",
|
||||||
err, ep->stid,
|
err, ep->stid,
|
||||||
sin6->sin6_addr.s6_addr, ntohs(sin6->sin6_port));
|
sin6->sin6_addr.s6_addr, ntohs(sin6->sin6_port));
|
||||||
else
|
}
|
||||||
cxgb4_clip_get(ep->com.dev->rdev.lldi.ports[0],
|
|
||||||
(const u32 *)&sin6->sin6_addr.s6_addr, 1);
|
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -744,9 +744,6 @@ static int c4iw_poll_cq_one(struct c4iw_cq *chp, struct ib_wc *wc)
|
||||||
case FW_RI_SEND_WITH_SE:
|
case FW_RI_SEND_WITH_SE:
|
||||||
wc->opcode = IB_WC_SEND;
|
wc->opcode = IB_WC_SEND;
|
||||||
break;
|
break;
|
||||||
case FW_RI_BIND_MW:
|
|
||||||
wc->opcode = IB_WC_BIND_MW;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case FW_RI_LOCAL_INV:
|
case FW_RI_LOCAL_INV:
|
||||||
wc->opcode = IB_WC_LOCAL_INV;
|
wc->opcode = IB_WC_LOCAL_INV;
|
||||||
|
|
|
@ -315,14 +315,12 @@ static int qp_release(struct inode *inode, struct file *file)
|
||||||
static int qp_open(struct inode *inode, struct file *file)
|
static int qp_open(struct inode *inode, struct file *file)
|
||||||
{
|
{
|
||||||
struct c4iw_debugfs_data *qpd;
|
struct c4iw_debugfs_data *qpd;
|
||||||
int ret = 0;
|
|
||||||
int count = 1;
|
int count = 1;
|
||||||
|
|
||||||
qpd = kmalloc(sizeof *qpd, GFP_KERNEL);
|
qpd = kmalloc(sizeof *qpd, GFP_KERNEL);
|
||||||
if (!qpd) {
|
if (!qpd)
|
||||||
ret = -ENOMEM;
|
return -ENOMEM;
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
qpd->devp = inode->i_private;
|
qpd->devp = inode->i_private;
|
||||||
qpd->pos = 0;
|
qpd->pos = 0;
|
||||||
|
|
||||||
|
@ -333,8 +331,8 @@ static int qp_open(struct inode *inode, struct file *file)
|
||||||
qpd->bufsize = count * 128;
|
qpd->bufsize = count * 128;
|
||||||
qpd->buf = vmalloc(qpd->bufsize);
|
qpd->buf = vmalloc(qpd->bufsize);
|
||||||
if (!qpd->buf) {
|
if (!qpd->buf) {
|
||||||
ret = -ENOMEM;
|
kfree(qpd);
|
||||||
goto err1;
|
return -ENOMEM;
|
||||||
}
|
}
|
||||||
|
|
||||||
spin_lock_irq(&qpd->devp->lock);
|
spin_lock_irq(&qpd->devp->lock);
|
||||||
|
@ -343,11 +341,7 @@ static int qp_open(struct inode *inode, struct file *file)
|
||||||
|
|
||||||
qpd->buf[qpd->pos++] = 0;
|
qpd->buf[qpd->pos++] = 0;
|
||||||
file->private_data = qpd;
|
file->private_data = qpd;
|
||||||
goto out;
|
return 0;
|
||||||
err1:
|
|
||||||
kfree(qpd);
|
|
||||||
out:
|
|
||||||
return ret;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static const struct file_operations qp_debugfs_fops = {
|
static const struct file_operations qp_debugfs_fops = {
|
||||||
|
@ -781,8 +775,7 @@ static int c4iw_rdev_open(struct c4iw_rdev *rdev)
|
||||||
pr_err(MOD "%s: unsupported udb/ucq densities %u/%u\n",
|
pr_err(MOD "%s: unsupported udb/ucq densities %u/%u\n",
|
||||||
pci_name(rdev->lldi.pdev), rdev->lldi.udb_density,
|
pci_name(rdev->lldi.pdev), rdev->lldi.udb_density,
|
||||||
rdev->lldi.ucq_density);
|
rdev->lldi.ucq_density);
|
||||||
err = -EINVAL;
|
return -EINVAL;
|
||||||
goto err1;
|
|
||||||
}
|
}
|
||||||
if (rdev->lldi.vr->qp.start != rdev->lldi.vr->cq.start ||
|
if (rdev->lldi.vr->qp.start != rdev->lldi.vr->cq.start ||
|
||||||
rdev->lldi.vr->qp.size != rdev->lldi.vr->cq.size) {
|
rdev->lldi.vr->qp.size != rdev->lldi.vr->cq.size) {
|
||||||
|
@ -791,8 +784,7 @@ static int c4iw_rdev_open(struct c4iw_rdev *rdev)
|
||||||
pci_name(rdev->lldi.pdev), rdev->lldi.vr->qp.start,
|
pci_name(rdev->lldi.pdev), rdev->lldi.vr->qp.start,
|
||||||
rdev->lldi.vr->qp.size, rdev->lldi.vr->cq.size,
|
rdev->lldi.vr->qp.size, rdev->lldi.vr->cq.size,
|
||||||
rdev->lldi.vr->cq.size);
|
rdev->lldi.vr->cq.size);
|
||||||
err = -EINVAL;
|
return -EINVAL;
|
||||||
goto err1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
rdev->qpmask = rdev->lldi.udb_density - 1;
|
rdev->qpmask = rdev->lldi.udb_density - 1;
|
||||||
|
@ -816,10 +808,8 @@ static int c4iw_rdev_open(struct c4iw_rdev *rdev)
|
||||||
rdev->lldi.db_reg, rdev->lldi.gts_reg,
|
rdev->lldi.db_reg, rdev->lldi.gts_reg,
|
||||||
rdev->qpmask, rdev->cqmask);
|
rdev->qpmask, rdev->cqmask);
|
||||||
|
|
||||||
if (c4iw_num_stags(rdev) == 0) {
|
if (c4iw_num_stags(rdev) == 0)
|
||||||
err = -EINVAL;
|
return -EINVAL;
|
||||||
goto err1;
|
|
||||||
}
|
|
||||||
|
|
||||||
rdev->stats.pd.total = T4_MAX_NUM_PD;
|
rdev->stats.pd.total = T4_MAX_NUM_PD;
|
||||||
rdev->stats.stag.total = rdev->lldi.vr->stag.size;
|
rdev->stats.stag.total = rdev->lldi.vr->stag.size;
|
||||||
|
@ -831,29 +821,31 @@ static int c4iw_rdev_open(struct c4iw_rdev *rdev)
|
||||||
err = c4iw_init_resource(rdev, c4iw_num_stags(rdev), T4_MAX_NUM_PD);
|
err = c4iw_init_resource(rdev, c4iw_num_stags(rdev), T4_MAX_NUM_PD);
|
||||||
if (err) {
|
if (err) {
|
||||||
printk(KERN_ERR MOD "error %d initializing resources\n", err);
|
printk(KERN_ERR MOD "error %d initializing resources\n", err);
|
||||||
goto err1;
|
return err;
|
||||||
}
|
}
|
||||||
err = c4iw_pblpool_create(rdev);
|
err = c4iw_pblpool_create(rdev);
|
||||||
if (err) {
|
if (err) {
|
||||||
printk(KERN_ERR MOD "error %d initializing pbl pool\n", err);
|
printk(KERN_ERR MOD "error %d initializing pbl pool\n", err);
|
||||||
goto err2;
|
goto destroy_resource;
|
||||||
}
|
}
|
||||||
err = c4iw_rqtpool_create(rdev);
|
err = c4iw_rqtpool_create(rdev);
|
||||||
if (err) {
|
if (err) {
|
||||||
printk(KERN_ERR MOD "error %d initializing rqt pool\n", err);
|
printk(KERN_ERR MOD "error %d initializing rqt pool\n", err);
|
||||||
goto err3;
|
goto destroy_pblpool;
|
||||||
}
|
}
|
||||||
err = c4iw_ocqp_pool_create(rdev);
|
err = c4iw_ocqp_pool_create(rdev);
|
||||||
if (err) {
|
if (err) {
|
||||||
printk(KERN_ERR MOD "error %d initializing ocqp pool\n", err);
|
printk(KERN_ERR MOD "error %d initializing ocqp pool\n", err);
|
||||||
goto err4;
|
goto destroy_rqtpool;
|
||||||
}
|
}
|
||||||
rdev->status_page = (struct t4_dev_status_page *)
|
rdev->status_page = (struct t4_dev_status_page *)
|
||||||
__get_free_page(GFP_KERNEL);
|
__get_free_page(GFP_KERNEL);
|
||||||
if (!rdev->status_page) {
|
if (!rdev->status_page)
|
||||||
pr_err(MOD "error allocating status page\n");
|
goto destroy_ocqp_pool;
|
||||||
goto err4;
|
rdev->status_page->qp_start = rdev->lldi.vr->qp.start;
|
||||||
}
|
rdev->status_page->qp_size = rdev->lldi.vr->qp.size;
|
||||||
|
rdev->status_page->cq_start = rdev->lldi.vr->cq.start;
|
||||||
|
rdev->status_page->cq_size = rdev->lldi.vr->cq.size;
|
||||||
|
|
||||||
if (c4iw_wr_log) {
|
if (c4iw_wr_log) {
|
||||||
rdev->wr_log = kzalloc((1 << c4iw_wr_log_size_order) *
|
rdev->wr_log = kzalloc((1 << c4iw_wr_log_size_order) *
|
||||||
|
@ -869,13 +861,14 @@ static int c4iw_rdev_open(struct c4iw_rdev *rdev)
|
||||||
rdev->status_page->db_off = 0;
|
rdev->status_page->db_off = 0;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
err4:
|
destroy_ocqp_pool:
|
||||||
|
c4iw_ocqp_pool_destroy(rdev);
|
||||||
|
destroy_rqtpool:
|
||||||
c4iw_rqtpool_destroy(rdev);
|
c4iw_rqtpool_destroy(rdev);
|
||||||
err3:
|
destroy_pblpool:
|
||||||
c4iw_pblpool_destroy(rdev);
|
c4iw_pblpool_destroy(rdev);
|
||||||
err2:
|
destroy_resource:
|
||||||
c4iw_destroy_resource(&rdev->resource);
|
c4iw_destroy_resource(&rdev->resource);
|
||||||
err1:
|
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -947,8 +947,6 @@ int c4iw_post_send(struct ib_qp *ibqp, struct ib_send_wr *wr,
|
||||||
struct ib_send_wr **bad_wr);
|
struct ib_send_wr **bad_wr);
|
||||||
int c4iw_post_receive(struct ib_qp *ibqp, struct ib_recv_wr *wr,
|
int c4iw_post_receive(struct ib_qp *ibqp, struct ib_recv_wr *wr,
|
||||||
struct ib_recv_wr **bad_wr);
|
struct ib_recv_wr **bad_wr);
|
||||||
int c4iw_bind_mw(struct ib_qp *qp, struct ib_mw *mw,
|
|
||||||
struct ib_mw_bind *mw_bind);
|
|
||||||
int c4iw_connect(struct iw_cm_id *cm_id, struct iw_cm_conn_param *conn_param);
|
int c4iw_connect(struct iw_cm_id *cm_id, struct iw_cm_conn_param *conn_param);
|
||||||
int c4iw_create_listen(struct iw_cm_id *cm_id, int backlog);
|
int c4iw_create_listen(struct iw_cm_id *cm_id, int backlog);
|
||||||
int c4iw_destroy_listen(struct iw_cm_id *cm_id);
|
int c4iw_destroy_listen(struct iw_cm_id *cm_id);
|
||||||
|
@ -968,17 +966,6 @@ struct ib_mr *c4iw_reg_user_mr(struct ib_pd *pd, u64 start,
|
||||||
u64 length, u64 virt, int acc,
|
u64 length, u64 virt, int acc,
|
||||||
struct ib_udata *udata);
|
struct ib_udata *udata);
|
||||||
struct ib_mr *c4iw_get_dma_mr(struct ib_pd *pd, int acc);
|
struct ib_mr *c4iw_get_dma_mr(struct ib_pd *pd, int acc);
|
||||||
struct ib_mr *c4iw_register_phys_mem(struct ib_pd *pd,
|
|
||||||
struct ib_phys_buf *buffer_list,
|
|
||||||
int num_phys_buf,
|
|
||||||
int acc,
|
|
||||||
u64 *iova_start);
|
|
||||||
int c4iw_reregister_phys_mem(struct ib_mr *mr,
|
|
||||||
int mr_rereg_mask,
|
|
||||||
struct ib_pd *pd,
|
|
||||||
struct ib_phys_buf *buffer_list,
|
|
||||||
int num_phys_buf,
|
|
||||||
int acc, u64 *iova_start);
|
|
||||||
int c4iw_dereg_mr(struct ib_mr *ib_mr);
|
int c4iw_dereg_mr(struct ib_mr *ib_mr);
|
||||||
int c4iw_destroy_cq(struct ib_cq *ib_cq);
|
int c4iw_destroy_cq(struct ib_cq *ib_cq);
|
||||||
struct ib_cq *c4iw_create_cq(struct ib_device *ibdev,
|
struct ib_cq *c4iw_create_cq(struct ib_device *ibdev,
|
||||||
|
|
|
@ -392,32 +392,6 @@ static int register_mem(struct c4iw_dev *rhp, struct c4iw_pd *php,
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int reregister_mem(struct c4iw_dev *rhp, struct c4iw_pd *php,
|
|
||||||
struct c4iw_mr *mhp, int shift, int npages)
|
|
||||||
{
|
|
||||||
u32 stag;
|
|
||||||
int ret;
|
|
||||||
|
|
||||||
if (npages > mhp->attr.pbl_size)
|
|
||||||
return -ENOMEM;
|
|
||||||
|
|
||||||
stag = mhp->attr.stag;
|
|
||||||
ret = write_tpt_entry(&rhp->rdev, 0, &stag, 1, mhp->attr.pdid,
|
|
||||||
FW_RI_STAG_NSMR, mhp->attr.perms,
|
|
||||||
mhp->attr.mw_bind_enable, mhp->attr.zbva,
|
|
||||||
mhp->attr.va_fbo, mhp->attr.len, shift - 12,
|
|
||||||
mhp->attr.pbl_size, mhp->attr.pbl_addr);
|
|
||||||
if (ret)
|
|
||||||
return ret;
|
|
||||||
|
|
||||||
ret = finish_mem_reg(mhp, stag);
|
|
||||||
if (ret)
|
|
||||||
dereg_mem(&rhp->rdev, mhp->attr.stag, mhp->attr.pbl_size,
|
|
||||||
mhp->attr.pbl_addr);
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int alloc_pbl(struct c4iw_mr *mhp, int npages)
|
static int alloc_pbl(struct c4iw_mr *mhp, int npages)
|
||||||
{
|
{
|
||||||
mhp->attr.pbl_addr = c4iw_pblpool_alloc(&mhp->rhp->rdev,
|
mhp->attr.pbl_addr = c4iw_pblpool_alloc(&mhp->rhp->rdev,
|
||||||
|
@ -431,228 +405,6 @@ static int alloc_pbl(struct c4iw_mr *mhp, int npages)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int build_phys_page_list(struct ib_phys_buf *buffer_list,
|
|
||||||
int num_phys_buf, u64 *iova_start,
|
|
||||||
u64 *total_size, int *npages,
|
|
||||||
int *shift, __be64 **page_list)
|
|
||||||
{
|
|
||||||
u64 mask;
|
|
||||||
int i, j, n;
|
|
||||||
|
|
||||||
mask = 0;
|
|
||||||
*total_size = 0;
|
|
||||||
for (i = 0; i < num_phys_buf; ++i) {
|
|
||||||
if (i != 0 && buffer_list[i].addr & ~PAGE_MASK)
|
|
||||||
return -EINVAL;
|
|
||||||
if (i != 0 && i != num_phys_buf - 1 &&
|
|
||||||
(buffer_list[i].size & ~PAGE_MASK))
|
|
||||||
return -EINVAL;
|
|
||||||
*total_size += buffer_list[i].size;
|
|
||||||
if (i > 0)
|
|
||||||
mask |= buffer_list[i].addr;
|
|
||||||
else
|
|
||||||
mask |= buffer_list[i].addr & PAGE_MASK;
|
|
||||||
if (i != num_phys_buf - 1)
|
|
||||||
mask |= buffer_list[i].addr + buffer_list[i].size;
|
|
||||||
else
|
|
||||||
mask |= (buffer_list[i].addr + buffer_list[i].size +
|
|
||||||
PAGE_SIZE - 1) & PAGE_MASK;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (*total_size > 0xFFFFFFFFULL)
|
|
||||||
return -ENOMEM;
|
|
||||||
|
|
||||||
/* Find largest page shift we can use to cover buffers */
|
|
||||||
for (*shift = PAGE_SHIFT; *shift < 27; ++(*shift))
|
|
||||||
if ((1ULL << *shift) & mask)
|
|
||||||
break;
|
|
||||||
|
|
||||||
buffer_list[0].size += buffer_list[0].addr & ((1ULL << *shift) - 1);
|
|
||||||
buffer_list[0].addr &= ~0ull << *shift;
|
|
||||||
|
|
||||||
*npages = 0;
|
|
||||||
for (i = 0; i < num_phys_buf; ++i)
|
|
||||||
*npages += (buffer_list[i].size +
|
|
||||||
(1ULL << *shift) - 1) >> *shift;
|
|
||||||
|
|
||||||
if (!*npages)
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
*page_list = kmalloc(sizeof(u64) * *npages, GFP_KERNEL);
|
|
||||||
if (!*page_list)
|
|
||||||
return -ENOMEM;
|
|
||||||
|
|
||||||
n = 0;
|
|
||||||
for (i = 0; i < num_phys_buf; ++i)
|
|
||||||
for (j = 0;
|
|
||||||
j < (buffer_list[i].size + (1ULL << *shift) - 1) >> *shift;
|
|
||||||
++j)
|
|
||||||
(*page_list)[n++] = cpu_to_be64(buffer_list[i].addr +
|
|
||||||
((u64) j << *shift));
|
|
||||||
|
|
||||||
PDBG("%s va 0x%llx mask 0x%llx shift %d len %lld pbl_size %d\n",
|
|
||||||
__func__, (unsigned long long)*iova_start,
|
|
||||||
(unsigned long long)mask, *shift, (unsigned long long)*total_size,
|
|
||||||
*npages);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
int c4iw_reregister_phys_mem(struct ib_mr *mr, int mr_rereg_mask,
|
|
||||||
struct ib_pd *pd, struct ib_phys_buf *buffer_list,
|
|
||||||
int num_phys_buf, int acc, u64 *iova_start)
|
|
||||||
{
|
|
||||||
|
|
||||||
struct c4iw_mr mh, *mhp;
|
|
||||||
struct c4iw_pd *php;
|
|
||||||
struct c4iw_dev *rhp;
|
|
||||||
__be64 *page_list = NULL;
|
|
||||||
int shift = 0;
|
|
||||||
u64 total_size;
|
|
||||||
int npages;
|
|
||||||
int ret;
|
|
||||||
|
|
||||||
PDBG("%s ib_mr %p ib_pd %p\n", __func__, mr, pd);
|
|
||||||
|
|
||||||
/* There can be no memory windows */
|
|
||||||
if (atomic_read(&mr->usecnt))
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
mhp = to_c4iw_mr(mr);
|
|
||||||
rhp = mhp->rhp;
|
|
||||||
php = to_c4iw_pd(mr->pd);
|
|
||||||
|
|
||||||
/* make sure we are on the same adapter */
|
|
||||||
if (rhp != php->rhp)
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
memcpy(&mh, mhp, sizeof *mhp);
|
|
||||||
|
|
||||||
if (mr_rereg_mask & IB_MR_REREG_PD)
|
|
||||||
php = to_c4iw_pd(pd);
|
|
||||||
if (mr_rereg_mask & IB_MR_REREG_ACCESS) {
|
|
||||||
mh.attr.perms = c4iw_ib_to_tpt_access(acc);
|
|
||||||
mh.attr.mw_bind_enable = (acc & IB_ACCESS_MW_BIND) ==
|
|
||||||
IB_ACCESS_MW_BIND;
|
|
||||||
}
|
|
||||||
if (mr_rereg_mask & IB_MR_REREG_TRANS) {
|
|
||||||
ret = build_phys_page_list(buffer_list, num_phys_buf,
|
|
||||||
iova_start,
|
|
||||||
&total_size, &npages,
|
|
||||||
&shift, &page_list);
|
|
||||||
if (ret)
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (mr_exceeds_hw_limits(rhp, total_size)) {
|
|
||||||
kfree(page_list);
|
|
||||||
return -EINVAL;
|
|
||||||
}
|
|
||||||
|
|
||||||
ret = reregister_mem(rhp, php, &mh, shift, npages);
|
|
||||||
kfree(page_list);
|
|
||||||
if (ret)
|
|
||||||
return ret;
|
|
||||||
if (mr_rereg_mask & IB_MR_REREG_PD)
|
|
||||||
mhp->attr.pdid = php->pdid;
|
|
||||||
if (mr_rereg_mask & IB_MR_REREG_ACCESS)
|
|
||||||
mhp->attr.perms = c4iw_ib_to_tpt_access(acc);
|
|
||||||
if (mr_rereg_mask & IB_MR_REREG_TRANS) {
|
|
||||||
mhp->attr.zbva = 0;
|
|
||||||
mhp->attr.va_fbo = *iova_start;
|
|
||||||
mhp->attr.page_size = shift - 12;
|
|
||||||
mhp->attr.len = (u32) total_size;
|
|
||||||
mhp->attr.pbl_size = npages;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
struct ib_mr *c4iw_register_phys_mem(struct ib_pd *pd,
|
|
||||||
struct ib_phys_buf *buffer_list,
|
|
||||||
int num_phys_buf, int acc, u64 *iova_start)
|
|
||||||
{
|
|
||||||
__be64 *page_list;
|
|
||||||
int shift;
|
|
||||||
u64 total_size;
|
|
||||||
int npages;
|
|
||||||
struct c4iw_dev *rhp;
|
|
||||||
struct c4iw_pd *php;
|
|
||||||
struct c4iw_mr *mhp;
|
|
||||||
int ret;
|
|
||||||
|
|
||||||
PDBG("%s ib_pd %p\n", __func__, pd);
|
|
||||||
php = to_c4iw_pd(pd);
|
|
||||||
rhp = php->rhp;
|
|
||||||
|
|
||||||
mhp = kzalloc(sizeof(*mhp), GFP_KERNEL);
|
|
||||||
if (!mhp)
|
|
||||||
return ERR_PTR(-ENOMEM);
|
|
||||||
|
|
||||||
mhp->rhp = rhp;
|
|
||||||
|
|
||||||
/* First check that we have enough alignment */
|
|
||||||
if ((*iova_start & ~PAGE_MASK) != (buffer_list[0].addr & ~PAGE_MASK)) {
|
|
||||||
ret = -EINVAL;
|
|
||||||
goto err;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (num_phys_buf > 1 &&
|
|
||||||
((buffer_list[0].addr + buffer_list[0].size) & ~PAGE_MASK)) {
|
|
||||||
ret = -EINVAL;
|
|
||||||
goto err;
|
|
||||||
}
|
|
||||||
|
|
||||||
ret = build_phys_page_list(buffer_list, num_phys_buf, iova_start,
|
|
||||||
&total_size, &npages, &shift,
|
|
||||||
&page_list);
|
|
||||||
if (ret)
|
|
||||||
goto err;
|
|
||||||
|
|
||||||
if (mr_exceeds_hw_limits(rhp, total_size)) {
|
|
||||||
kfree(page_list);
|
|
||||||
ret = -EINVAL;
|
|
||||||
goto err;
|
|
||||||
}
|
|
||||||
|
|
||||||
ret = alloc_pbl(mhp, npages);
|
|
||||||
if (ret) {
|
|
||||||
kfree(page_list);
|
|
||||||
goto err;
|
|
||||||
}
|
|
||||||
|
|
||||||
ret = write_pbl(&mhp->rhp->rdev, page_list, mhp->attr.pbl_addr,
|
|
||||||
npages);
|
|
||||||
kfree(page_list);
|
|
||||||
if (ret)
|
|
||||||
goto err_pbl;
|
|
||||||
|
|
||||||
mhp->attr.pdid = php->pdid;
|
|
||||||
mhp->attr.zbva = 0;
|
|
||||||
|
|
||||||
mhp->attr.perms = c4iw_ib_to_tpt_access(acc);
|
|
||||||
mhp->attr.va_fbo = *iova_start;
|
|
||||||
mhp->attr.page_size = shift - 12;
|
|
||||||
|
|
||||||
mhp->attr.len = (u32) total_size;
|
|
||||||
mhp->attr.pbl_size = npages;
|
|
||||||
ret = register_mem(rhp, php, mhp, shift);
|
|
||||||
if (ret)
|
|
||||||
goto err_pbl;
|
|
||||||
|
|
||||||
return &mhp->ibmr;
|
|
||||||
|
|
||||||
err_pbl:
|
|
||||||
c4iw_pblpool_free(&mhp->rhp->rdev, mhp->attr.pbl_addr,
|
|
||||||
mhp->attr.pbl_size << 3);
|
|
||||||
|
|
||||||
err:
|
|
||||||
kfree(mhp);
|
|
||||||
return ERR_PTR(ret);
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
struct ib_mr *c4iw_get_dma_mr(struct ib_pd *pd, int acc)
|
struct ib_mr *c4iw_get_dma_mr(struct ib_pd *pd, int acc)
|
||||||
{
|
{
|
||||||
struct c4iw_dev *rhp;
|
struct c4iw_dev *rhp;
|
||||||
|
@ -952,9 +704,6 @@ int c4iw_dereg_mr(struct ib_mr *ib_mr)
|
||||||
u32 mmid;
|
u32 mmid;
|
||||||
|
|
||||||
PDBG("%s ib_mr %p\n", __func__, ib_mr);
|
PDBG("%s ib_mr %p\n", __func__, ib_mr);
|
||||||
/* There can be no memory windows */
|
|
||||||
if (atomic_read(&ib_mr->usecnt))
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
mhp = to_c4iw_mr(ib_mr);
|
mhp = to_c4iw_mr(ib_mr);
|
||||||
rhp = mhp->rhp;
|
rhp = mhp->rhp;
|
||||||
|
|
|
@ -549,12 +549,9 @@ int c4iw_register_device(struct c4iw_dev *dev)
|
||||||
dev->ibdev.resize_cq = c4iw_resize_cq;
|
dev->ibdev.resize_cq = c4iw_resize_cq;
|
||||||
dev->ibdev.poll_cq = c4iw_poll_cq;
|
dev->ibdev.poll_cq = c4iw_poll_cq;
|
||||||
dev->ibdev.get_dma_mr = c4iw_get_dma_mr;
|
dev->ibdev.get_dma_mr = c4iw_get_dma_mr;
|
||||||
dev->ibdev.reg_phys_mr = c4iw_register_phys_mem;
|
|
||||||
dev->ibdev.rereg_phys_mr = c4iw_reregister_phys_mem;
|
|
||||||
dev->ibdev.reg_user_mr = c4iw_reg_user_mr;
|
dev->ibdev.reg_user_mr = c4iw_reg_user_mr;
|
||||||
dev->ibdev.dereg_mr = c4iw_dereg_mr;
|
dev->ibdev.dereg_mr = c4iw_dereg_mr;
|
||||||
dev->ibdev.alloc_mw = c4iw_alloc_mw;
|
dev->ibdev.alloc_mw = c4iw_alloc_mw;
|
||||||
dev->ibdev.bind_mw = c4iw_bind_mw;
|
|
||||||
dev->ibdev.dealloc_mw = c4iw_dealloc_mw;
|
dev->ibdev.dealloc_mw = c4iw_dealloc_mw;
|
||||||
dev->ibdev.alloc_mr = c4iw_alloc_mr;
|
dev->ibdev.alloc_mr = c4iw_alloc_mr;
|
||||||
dev->ibdev.map_mr_sg = c4iw_map_mr_sg;
|
dev->ibdev.map_mr_sg = c4iw_map_mr_sg;
|
||||||
|
|
|
@ -933,11 +933,6 @@ int c4iw_post_receive(struct ib_qp *ibqp, struct ib_recv_wr *wr,
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
int c4iw_bind_mw(struct ib_qp *qp, struct ib_mw *mw, struct ib_mw_bind *mw_bind)
|
|
||||||
{
|
|
||||||
return -ENOSYS;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline void build_term_codes(struct t4_cqe *err_cqe, u8 *layer_type,
|
static inline void build_term_codes(struct t4_cqe *err_cqe, u8 *layer_type,
|
||||||
u8 *ecode)
|
u8 *ecode)
|
||||||
{
|
{
|
||||||
|
|
|
@ -699,4 +699,11 @@ static inline void t4_set_cq_in_error(struct t4_cq *cq)
|
||||||
|
|
||||||
struct t4_dev_status_page {
|
struct t4_dev_status_page {
|
||||||
u8 db_off;
|
u8 db_off;
|
||||||
|
u8 pad1;
|
||||||
|
u16 pad2;
|
||||||
|
u32 pad3;
|
||||||
|
u64 qp_start;
|
||||||
|
u64 qp_size;
|
||||||
|
u64 cq_start;
|
||||||
|
u64 cq_size;
|
||||||
};
|
};
|
||||||
|
|
|
@ -32,7 +32,7 @@
|
||||||
#ifndef __C4IW_USER_H__
|
#ifndef __C4IW_USER_H__
|
||||||
#define __C4IW_USER_H__
|
#define __C4IW_USER_H__
|
||||||
|
|
||||||
#define C4IW_UVERBS_ABI_VERSION 2
|
#define C4IW_UVERBS_ABI_VERSION 3
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Make sure that all structs defined in this file remain laid out so
|
* Make sure that all structs defined in this file remain laid out so
|
||||||
|
|
|
@ -92,7 +92,7 @@ static struct ib_ah *create_iboe_ah(struct ib_pd *pd, struct ib_ah_attr *ah_attr
|
||||||
ah_attr->grh.sgid_index, &sgid, &gid_attr);
|
ah_attr->grh.sgid_index, &sgid, &gid_attr);
|
||||||
if (ret)
|
if (ret)
|
||||||
return ERR_PTR(ret);
|
return ERR_PTR(ret);
|
||||||
memset(ah->av.eth.s_mac, 0, ETH_ALEN);
|
eth_zero_addr(ah->av.eth.s_mac);
|
||||||
if (gid_attr.ndev) {
|
if (gid_attr.ndev) {
|
||||||
if (is_vlan_dev(gid_attr.ndev))
|
if (is_vlan_dev(gid_attr.ndev))
|
||||||
vlan_tag = vlan_dev_vlan_id(gid_attr.ndev);
|
vlan_tag = vlan_dev_vlan_id(gid_attr.ndev);
|
||||||
|
@ -104,6 +104,7 @@ static struct ib_ah *create_iboe_ah(struct ib_pd *pd, struct ib_ah_attr *ah_attr
|
||||||
ah->av.eth.port_pd = cpu_to_be32(to_mpd(pd)->pdn | (ah_attr->port_num << 24));
|
ah->av.eth.port_pd = cpu_to_be32(to_mpd(pd)->pdn | (ah_attr->port_num << 24));
|
||||||
ah->av.eth.gid_index = mlx4_ib_gid_index_to_real_index(ibdev, ah_attr->port_num, ah_attr->grh.sgid_index);
|
ah->av.eth.gid_index = mlx4_ib_gid_index_to_real_index(ibdev, ah_attr->port_num, ah_attr->grh.sgid_index);
|
||||||
ah->av.eth.vlan = cpu_to_be16(vlan_tag);
|
ah->av.eth.vlan = cpu_to_be16(vlan_tag);
|
||||||
|
ah->av.eth.hop_limit = ah_attr->grh.hop_limit;
|
||||||
if (ah_attr->static_rate) {
|
if (ah_attr->static_rate) {
|
||||||
ah->av.eth.stat_rate = ah_attr->static_rate + MLX4_STAT_RATE_OFFSET;
|
ah->av.eth.stat_rate = ah_attr->static_rate + MLX4_STAT_RATE_OFFSET;
|
||||||
while (ah->av.eth.stat_rate > IB_RATE_2_5_GBPS + MLX4_STAT_RATE_OFFSET &&
|
while (ah->av.eth.stat_rate > IB_RATE_2_5_GBPS + MLX4_STAT_RATE_OFFSET &&
|
||||||
|
|
|
@ -811,9 +811,6 @@ static int mlx4_ib_poll_one(struct mlx4_ib_cq *cq,
|
||||||
wc->opcode = IB_WC_MASKED_FETCH_ADD;
|
wc->opcode = IB_WC_MASKED_FETCH_ADD;
|
||||||
wc->byte_len = 8;
|
wc->byte_len = 8;
|
||||||
break;
|
break;
|
||||||
case MLX4_OPCODE_BIND_MW:
|
|
||||||
wc->opcode = IB_WC_BIND_MW;
|
|
||||||
break;
|
|
||||||
case MLX4_OPCODE_LSO:
|
case MLX4_OPCODE_LSO:
|
||||||
wc->opcode = IB_WC_LSO;
|
wc->opcode = IB_WC_LSO;
|
||||||
break;
|
break;
|
||||||
|
|
|
@ -154,9 +154,9 @@ static struct net_device *mlx4_ib_get_netdev(struct ib_device *device, u8 port_n
|
||||||
return dev;
|
return dev;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int mlx4_ib_update_gids(struct gid_entry *gids,
|
static int mlx4_ib_update_gids_v1(struct gid_entry *gids,
|
||||||
struct mlx4_ib_dev *ibdev,
|
struct mlx4_ib_dev *ibdev,
|
||||||
u8 port_num)
|
u8 port_num)
|
||||||
{
|
{
|
||||||
struct mlx4_cmd_mailbox *mailbox;
|
struct mlx4_cmd_mailbox *mailbox;
|
||||||
int err;
|
int err;
|
||||||
|
@ -187,6 +187,63 @@ static int mlx4_ib_update_gids(struct gid_entry *gids,
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int mlx4_ib_update_gids_v1_v2(struct gid_entry *gids,
|
||||||
|
struct mlx4_ib_dev *ibdev,
|
||||||
|
u8 port_num)
|
||||||
|
{
|
||||||
|
struct mlx4_cmd_mailbox *mailbox;
|
||||||
|
int err;
|
||||||
|
struct mlx4_dev *dev = ibdev->dev;
|
||||||
|
int i;
|
||||||
|
struct {
|
||||||
|
union ib_gid gid;
|
||||||
|
__be32 rsrvd1[2];
|
||||||
|
__be16 rsrvd2;
|
||||||
|
u8 type;
|
||||||
|
u8 version;
|
||||||
|
__be32 rsrvd3;
|
||||||
|
} *gid_tbl;
|
||||||
|
|
||||||
|
mailbox = mlx4_alloc_cmd_mailbox(dev);
|
||||||
|
if (IS_ERR(mailbox))
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
gid_tbl = mailbox->buf;
|
||||||
|
for (i = 0; i < MLX4_MAX_PORT_GIDS; ++i) {
|
||||||
|
memcpy(&gid_tbl[i].gid, &gids[i].gid, sizeof(union ib_gid));
|
||||||
|
if (gids[i].gid_type == IB_GID_TYPE_ROCE_UDP_ENCAP) {
|
||||||
|
gid_tbl[i].version = 2;
|
||||||
|
if (!ipv6_addr_v4mapped((struct in6_addr *)&gids[i].gid))
|
||||||
|
gid_tbl[i].type = 1;
|
||||||
|
else
|
||||||
|
memset(&gid_tbl[i].gid, 0, 12);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
err = mlx4_cmd(dev, mailbox->dma,
|
||||||
|
MLX4_SET_PORT_ROCE_ADDR << 8 | port_num,
|
||||||
|
1, MLX4_CMD_SET_PORT, MLX4_CMD_TIME_CLASS_B,
|
||||||
|
MLX4_CMD_WRAPPED);
|
||||||
|
if (mlx4_is_bonded(dev))
|
||||||
|
err += mlx4_cmd(dev, mailbox->dma,
|
||||||
|
MLX4_SET_PORT_ROCE_ADDR << 8 | 2,
|
||||||
|
1, MLX4_CMD_SET_PORT, MLX4_CMD_TIME_CLASS_B,
|
||||||
|
MLX4_CMD_WRAPPED);
|
||||||
|
|
||||||
|
mlx4_free_cmd_mailbox(dev, mailbox);
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int mlx4_ib_update_gids(struct gid_entry *gids,
|
||||||
|
struct mlx4_ib_dev *ibdev,
|
||||||
|
u8 port_num)
|
||||||
|
{
|
||||||
|
if (ibdev->dev->caps.flags2 & MLX4_DEV_CAP_FLAG2_ROCE_V1_V2)
|
||||||
|
return mlx4_ib_update_gids_v1_v2(gids, ibdev, port_num);
|
||||||
|
|
||||||
|
return mlx4_ib_update_gids_v1(gids, ibdev, port_num);
|
||||||
|
}
|
||||||
|
|
||||||
static int mlx4_ib_add_gid(struct ib_device *device,
|
static int mlx4_ib_add_gid(struct ib_device *device,
|
||||||
u8 port_num,
|
u8 port_num,
|
||||||
unsigned int index,
|
unsigned int index,
|
||||||
|
@ -215,7 +272,8 @@ static int mlx4_ib_add_gid(struct ib_device *device,
|
||||||
port_gid_table = &iboe->gids[port_num - 1];
|
port_gid_table = &iboe->gids[port_num - 1];
|
||||||
spin_lock_bh(&iboe->lock);
|
spin_lock_bh(&iboe->lock);
|
||||||
for (i = 0; i < MLX4_MAX_PORT_GIDS; ++i) {
|
for (i = 0; i < MLX4_MAX_PORT_GIDS; ++i) {
|
||||||
if (!memcmp(&port_gid_table->gids[i].gid, gid, sizeof(*gid))) {
|
if (!memcmp(&port_gid_table->gids[i].gid, gid, sizeof(*gid)) &&
|
||||||
|
(port_gid_table->gids[i].gid_type == attr->gid_type)) {
|
||||||
found = i;
|
found = i;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -233,6 +291,7 @@ static int mlx4_ib_add_gid(struct ib_device *device,
|
||||||
} else {
|
} else {
|
||||||
*context = port_gid_table->gids[free].ctx;
|
*context = port_gid_table->gids[free].ctx;
|
||||||
memcpy(&port_gid_table->gids[free].gid, gid, sizeof(*gid));
|
memcpy(&port_gid_table->gids[free].gid, gid, sizeof(*gid));
|
||||||
|
port_gid_table->gids[free].gid_type = attr->gid_type;
|
||||||
port_gid_table->gids[free].ctx->real_index = free;
|
port_gid_table->gids[free].ctx->real_index = free;
|
||||||
port_gid_table->gids[free].ctx->refcount = 1;
|
port_gid_table->gids[free].ctx->refcount = 1;
|
||||||
hw_update = 1;
|
hw_update = 1;
|
||||||
|
@ -248,8 +307,10 @@ static int mlx4_ib_add_gid(struct ib_device *device,
|
||||||
if (!gids) {
|
if (!gids) {
|
||||||
ret = -ENOMEM;
|
ret = -ENOMEM;
|
||||||
} else {
|
} else {
|
||||||
for (i = 0; i < MLX4_MAX_PORT_GIDS; i++)
|
for (i = 0; i < MLX4_MAX_PORT_GIDS; i++) {
|
||||||
memcpy(&gids[i].gid, &port_gid_table->gids[i].gid, sizeof(union ib_gid));
|
memcpy(&gids[i].gid, &port_gid_table->gids[i].gid, sizeof(union ib_gid));
|
||||||
|
gids[i].gid_type = port_gid_table->gids[i].gid_type;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
spin_unlock_bh(&iboe->lock);
|
spin_unlock_bh(&iboe->lock);
|
||||||
|
@ -325,6 +386,7 @@ int mlx4_ib_gid_index_to_real_index(struct mlx4_ib_dev *ibdev,
|
||||||
int i;
|
int i;
|
||||||
int ret;
|
int ret;
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
|
struct ib_gid_attr attr;
|
||||||
|
|
||||||
if (port_num > MLX4_MAX_PORTS)
|
if (port_num > MLX4_MAX_PORTS)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
@ -335,10 +397,13 @@ int mlx4_ib_gid_index_to_real_index(struct mlx4_ib_dev *ibdev,
|
||||||
if (!rdma_cap_roce_gid_table(&ibdev->ib_dev, port_num))
|
if (!rdma_cap_roce_gid_table(&ibdev->ib_dev, port_num))
|
||||||
return index;
|
return index;
|
||||||
|
|
||||||
ret = ib_get_cached_gid(&ibdev->ib_dev, port_num, index, &gid, NULL);
|
ret = ib_get_cached_gid(&ibdev->ib_dev, port_num, index, &gid, &attr);
|
||||||
if (ret)
|
if (ret)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
|
if (attr.ndev)
|
||||||
|
dev_put(attr.ndev);
|
||||||
|
|
||||||
if (!memcmp(&gid, &zgid, sizeof(gid)))
|
if (!memcmp(&gid, &zgid, sizeof(gid)))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
|
@ -346,7 +411,8 @@ int mlx4_ib_gid_index_to_real_index(struct mlx4_ib_dev *ibdev,
|
||||||
port_gid_table = &iboe->gids[port_num - 1];
|
port_gid_table = &iboe->gids[port_num - 1];
|
||||||
|
|
||||||
for (i = 0; i < MLX4_MAX_PORT_GIDS; ++i)
|
for (i = 0; i < MLX4_MAX_PORT_GIDS; ++i)
|
||||||
if (!memcmp(&port_gid_table->gids[i].gid, &gid, sizeof(gid))) {
|
if (!memcmp(&port_gid_table->gids[i].gid, &gid, sizeof(gid)) &&
|
||||||
|
attr.gid_type == port_gid_table->gids[i].gid_type) {
|
||||||
ctx = port_gid_table->gids[i].ctx;
|
ctx = port_gid_table->gids[i].ctx;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -2119,6 +2185,7 @@ static int mlx4_port_immutable(struct ib_device *ibdev, u8 port_num,
|
||||||
struct ib_port_immutable *immutable)
|
struct ib_port_immutable *immutable)
|
||||||
{
|
{
|
||||||
struct ib_port_attr attr;
|
struct ib_port_attr attr;
|
||||||
|
struct mlx4_ib_dev *mdev = to_mdev(ibdev);
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
err = mlx4_ib_query_port(ibdev, port_num, &attr);
|
err = mlx4_ib_query_port(ibdev, port_num, &attr);
|
||||||
|
@ -2128,10 +2195,15 @@ static int mlx4_port_immutable(struct ib_device *ibdev, u8 port_num,
|
||||||
immutable->pkey_tbl_len = attr.pkey_tbl_len;
|
immutable->pkey_tbl_len = attr.pkey_tbl_len;
|
||||||
immutable->gid_tbl_len = attr.gid_tbl_len;
|
immutable->gid_tbl_len = attr.gid_tbl_len;
|
||||||
|
|
||||||
if (mlx4_ib_port_link_layer(ibdev, port_num) == IB_LINK_LAYER_INFINIBAND)
|
if (mlx4_ib_port_link_layer(ibdev, port_num) == IB_LINK_LAYER_INFINIBAND) {
|
||||||
immutable->core_cap_flags = RDMA_CORE_PORT_IBA_IB;
|
immutable->core_cap_flags = RDMA_CORE_PORT_IBA_IB;
|
||||||
else
|
} else {
|
||||||
immutable->core_cap_flags = RDMA_CORE_PORT_IBA_ROCE;
|
if (mdev->dev->caps.flags & MLX4_DEV_CAP_FLAG_IBOE)
|
||||||
|
immutable->core_cap_flags = RDMA_CORE_PORT_IBA_ROCE;
|
||||||
|
if (mdev->dev->caps.flags2 & MLX4_DEV_CAP_FLAG2_ROCE_V1_V2)
|
||||||
|
immutable->core_cap_flags = RDMA_CORE_PORT_IBA_ROCE |
|
||||||
|
RDMA_CORE_PORT_IBA_ROCE_UDP_ENCAP;
|
||||||
|
}
|
||||||
|
|
||||||
immutable->max_mad_size = IB_MGMT_MAD_SIZE;
|
immutable->max_mad_size = IB_MGMT_MAD_SIZE;
|
||||||
|
|
||||||
|
@ -2283,7 +2355,6 @@ static void *mlx4_ib_add(struct mlx4_dev *dev)
|
||||||
if (dev->caps.flags & MLX4_DEV_CAP_FLAG_MEM_WINDOW ||
|
if (dev->caps.flags & MLX4_DEV_CAP_FLAG_MEM_WINDOW ||
|
||||||
dev->caps.bmme_flags & MLX4_BMME_FLAG_TYPE_2_WIN) {
|
dev->caps.bmme_flags & MLX4_BMME_FLAG_TYPE_2_WIN) {
|
||||||
ibdev->ib_dev.alloc_mw = mlx4_ib_alloc_mw;
|
ibdev->ib_dev.alloc_mw = mlx4_ib_alloc_mw;
|
||||||
ibdev->ib_dev.bind_mw = mlx4_ib_bind_mw;
|
|
||||||
ibdev->ib_dev.dealloc_mw = mlx4_ib_dealloc_mw;
|
ibdev->ib_dev.dealloc_mw = mlx4_ib_dealloc_mw;
|
||||||
|
|
||||||
ibdev->ib_dev.uverbs_cmd_mask |=
|
ibdev->ib_dev.uverbs_cmd_mask |=
|
||||||
|
@ -2423,7 +2494,8 @@ static void *mlx4_ib_add(struct mlx4_dev *dev)
|
||||||
if (mlx4_ib_init_sriov(ibdev))
|
if (mlx4_ib_init_sriov(ibdev))
|
||||||
goto err_mad;
|
goto err_mad;
|
||||||
|
|
||||||
if (dev->caps.flags & MLX4_DEV_CAP_FLAG_IBOE) {
|
if (dev->caps.flags & MLX4_DEV_CAP_FLAG_IBOE ||
|
||||||
|
dev->caps.flags2 & MLX4_DEV_CAP_FLAG2_ROCE_V1_V2) {
|
||||||
if (!iboe->nb.notifier_call) {
|
if (!iboe->nb.notifier_call) {
|
||||||
iboe->nb.notifier_call = mlx4_ib_netdev_event;
|
iboe->nb.notifier_call = mlx4_ib_netdev_event;
|
||||||
err = register_netdevice_notifier(&iboe->nb);
|
err = register_netdevice_notifier(&iboe->nb);
|
||||||
|
@ -2432,6 +2504,12 @@ static void *mlx4_ib_add(struct mlx4_dev *dev)
|
||||||
goto err_notif;
|
goto err_notif;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (dev->caps.flags2 & MLX4_DEV_CAP_FLAG2_ROCE_V1_V2) {
|
||||||
|
err = mlx4_config_roce_v2_port(dev, ROCE_V2_UDP_DPORT);
|
||||||
|
if (err) {
|
||||||
|
goto err_notif;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (j = 0; j < ARRAY_SIZE(mlx4_class_attributes); ++j) {
|
for (j = 0; j < ARRAY_SIZE(mlx4_class_attributes); ++j) {
|
||||||
|
|
|
@ -177,11 +177,18 @@ struct mlx4_ib_wq {
|
||||||
unsigned tail;
|
unsigned tail;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
enum {
|
||||||
|
MLX4_IB_QP_CREATE_ROCE_V2_GSI = IB_QP_CREATE_RESERVED_START
|
||||||
|
};
|
||||||
|
|
||||||
enum mlx4_ib_qp_flags {
|
enum mlx4_ib_qp_flags {
|
||||||
MLX4_IB_QP_LSO = IB_QP_CREATE_IPOIB_UD_LSO,
|
MLX4_IB_QP_LSO = IB_QP_CREATE_IPOIB_UD_LSO,
|
||||||
MLX4_IB_QP_BLOCK_MULTICAST_LOOPBACK = IB_QP_CREATE_BLOCK_MULTICAST_LOOPBACK,
|
MLX4_IB_QP_BLOCK_MULTICAST_LOOPBACK = IB_QP_CREATE_BLOCK_MULTICAST_LOOPBACK,
|
||||||
MLX4_IB_QP_NETIF = IB_QP_CREATE_NETIF_QP,
|
MLX4_IB_QP_NETIF = IB_QP_CREATE_NETIF_QP,
|
||||||
MLX4_IB_QP_CREATE_USE_GFP_NOIO = IB_QP_CREATE_USE_GFP_NOIO,
|
MLX4_IB_QP_CREATE_USE_GFP_NOIO = IB_QP_CREATE_USE_GFP_NOIO,
|
||||||
|
|
||||||
|
/* Mellanox specific flags start from IB_QP_CREATE_RESERVED_START */
|
||||||
|
MLX4_IB_ROCE_V2_GSI_QP = MLX4_IB_QP_CREATE_ROCE_V2_GSI,
|
||||||
MLX4_IB_SRIOV_TUNNEL_QP = 1 << 30,
|
MLX4_IB_SRIOV_TUNNEL_QP = 1 << 30,
|
||||||
MLX4_IB_SRIOV_SQP = 1 << 31,
|
MLX4_IB_SRIOV_SQP = 1 << 31,
|
||||||
};
|
};
|
||||||
|
@ -478,6 +485,7 @@ struct gid_cache_context {
|
||||||
|
|
||||||
struct gid_entry {
|
struct gid_entry {
|
||||||
union ib_gid gid;
|
union ib_gid gid;
|
||||||
|
enum ib_gid_type gid_type;
|
||||||
struct gid_cache_context *ctx;
|
struct gid_cache_context *ctx;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -704,8 +712,6 @@ struct ib_mr *mlx4_ib_reg_user_mr(struct ib_pd *pd, u64 start, u64 length,
|
||||||
struct ib_udata *udata);
|
struct ib_udata *udata);
|
||||||
int mlx4_ib_dereg_mr(struct ib_mr *mr);
|
int mlx4_ib_dereg_mr(struct ib_mr *mr);
|
||||||
struct ib_mw *mlx4_ib_alloc_mw(struct ib_pd *pd, enum ib_mw_type type);
|
struct ib_mw *mlx4_ib_alloc_mw(struct ib_pd *pd, enum ib_mw_type type);
|
||||||
int mlx4_ib_bind_mw(struct ib_qp *qp, struct ib_mw *mw,
|
|
||||||
struct ib_mw_bind *mw_bind);
|
|
||||||
int mlx4_ib_dealloc_mw(struct ib_mw *mw);
|
int mlx4_ib_dealloc_mw(struct ib_mw *mw);
|
||||||
struct ib_mr *mlx4_ib_alloc_mr(struct ib_pd *pd,
|
struct ib_mr *mlx4_ib_alloc_mr(struct ib_pd *pd,
|
||||||
enum ib_mr_type mr_type,
|
enum ib_mr_type mr_type,
|
||||||
|
|
|
@ -366,28 +366,6 @@ struct ib_mw *mlx4_ib_alloc_mw(struct ib_pd *pd, enum ib_mw_type type)
|
||||||
return ERR_PTR(err);
|
return ERR_PTR(err);
|
||||||
}
|
}
|
||||||
|
|
||||||
int mlx4_ib_bind_mw(struct ib_qp *qp, struct ib_mw *mw,
|
|
||||||
struct ib_mw_bind *mw_bind)
|
|
||||||
{
|
|
||||||
struct ib_bind_mw_wr wr;
|
|
||||||
struct ib_send_wr *bad_wr;
|
|
||||||
int ret;
|
|
||||||
|
|
||||||
memset(&wr, 0, sizeof(wr));
|
|
||||||
wr.wr.opcode = IB_WR_BIND_MW;
|
|
||||||
wr.wr.wr_id = mw_bind->wr_id;
|
|
||||||
wr.wr.send_flags = mw_bind->send_flags;
|
|
||||||
wr.mw = mw;
|
|
||||||
wr.bind_info = mw_bind->bind_info;
|
|
||||||
wr.rkey = ib_inc_rkey(mw->rkey);
|
|
||||||
|
|
||||||
ret = mlx4_ib_post_send(qp, &wr.wr, &bad_wr);
|
|
||||||
if (!ret)
|
|
||||||
mw->rkey = wr.rkey;
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
int mlx4_ib_dealloc_mw(struct ib_mw *ibmw)
|
int mlx4_ib_dealloc_mw(struct ib_mw *ibmw)
|
||||||
{
|
{
|
||||||
struct mlx4_ib_mw *mw = to_mmw(ibmw);
|
struct mlx4_ib_mw *mw = to_mmw(ibmw);
|
||||||
|
|
|
@ -32,6 +32,8 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <linux/log2.h>
|
#include <linux/log2.h>
|
||||||
|
#include <linux/etherdevice.h>
|
||||||
|
#include <net/ip.h>
|
||||||
#include <linux/slab.h>
|
#include <linux/slab.h>
|
||||||
#include <linux/netdevice.h>
|
#include <linux/netdevice.h>
|
||||||
#include <linux/vmalloc.h>
|
#include <linux/vmalloc.h>
|
||||||
|
@ -85,6 +87,7 @@ struct mlx4_ib_sqp {
|
||||||
u32 send_psn;
|
u32 send_psn;
|
||||||
struct ib_ud_header ud_header;
|
struct ib_ud_header ud_header;
|
||||||
u8 header_buf[MLX4_IB_UD_HEADER_SIZE];
|
u8 header_buf[MLX4_IB_UD_HEADER_SIZE];
|
||||||
|
struct ib_qp *roce_v2_gsi;
|
||||||
};
|
};
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
|
@ -115,7 +118,6 @@ static const __be32 mlx4_ib_opcode[] = {
|
||||||
[IB_WR_REG_MR] = cpu_to_be32(MLX4_OPCODE_FMR),
|
[IB_WR_REG_MR] = cpu_to_be32(MLX4_OPCODE_FMR),
|
||||||
[IB_WR_MASKED_ATOMIC_CMP_AND_SWP] = cpu_to_be32(MLX4_OPCODE_MASKED_ATOMIC_CS),
|
[IB_WR_MASKED_ATOMIC_CMP_AND_SWP] = cpu_to_be32(MLX4_OPCODE_MASKED_ATOMIC_CS),
|
||||||
[IB_WR_MASKED_ATOMIC_FETCH_AND_ADD] = cpu_to_be32(MLX4_OPCODE_MASKED_ATOMIC_FA),
|
[IB_WR_MASKED_ATOMIC_FETCH_AND_ADD] = cpu_to_be32(MLX4_OPCODE_MASKED_ATOMIC_FA),
|
||||||
[IB_WR_BIND_MW] = cpu_to_be32(MLX4_OPCODE_BIND_MW),
|
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct mlx4_ib_sqp *to_msqp(struct mlx4_ib_qp *mqp)
|
static struct mlx4_ib_sqp *to_msqp(struct mlx4_ib_qp *mqp)
|
||||||
|
@ -154,7 +156,10 @@ static int is_sqp(struct mlx4_ib_dev *dev, struct mlx4_ib_qp *qp)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return proxy_sqp;
|
if (proxy_sqp)
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
return !!(qp->flags & MLX4_IB_ROCE_V2_GSI_QP);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* used for INIT/CLOSE port logic */
|
/* used for INIT/CLOSE port logic */
|
||||||
|
@ -796,11 +801,13 @@ static int create_qp_common(struct mlx4_ib_dev *dev, struct ib_pd *pd,
|
||||||
if (err)
|
if (err)
|
||||||
goto err_mtt;
|
goto err_mtt;
|
||||||
|
|
||||||
qp->sq.wrid = kmalloc(qp->sq.wqe_cnt * sizeof(u64), gfp);
|
qp->sq.wrid = kmalloc_array(qp->sq.wqe_cnt, sizeof(u64),
|
||||||
|
gfp | __GFP_NOWARN);
|
||||||
if (!qp->sq.wrid)
|
if (!qp->sq.wrid)
|
||||||
qp->sq.wrid = __vmalloc(qp->sq.wqe_cnt * sizeof(u64),
|
qp->sq.wrid = __vmalloc(qp->sq.wqe_cnt * sizeof(u64),
|
||||||
gfp, PAGE_KERNEL);
|
gfp, PAGE_KERNEL);
|
||||||
qp->rq.wrid = kmalloc(qp->rq.wqe_cnt * sizeof(u64), gfp);
|
qp->rq.wrid = kmalloc_array(qp->rq.wqe_cnt, sizeof(u64),
|
||||||
|
gfp | __GFP_NOWARN);
|
||||||
if (!qp->rq.wrid)
|
if (!qp->rq.wrid)
|
||||||
qp->rq.wrid = __vmalloc(qp->rq.wqe_cnt * sizeof(u64),
|
qp->rq.wrid = __vmalloc(qp->rq.wqe_cnt * sizeof(u64),
|
||||||
gfp, PAGE_KERNEL);
|
gfp, PAGE_KERNEL);
|
||||||
|
@ -1099,9 +1106,9 @@ static u32 get_sqp_num(struct mlx4_ib_dev *dev, struct ib_qp_init_attr *attr)
|
||||||
return dev->dev->caps.qp1_proxy[attr->port_num - 1];
|
return dev->dev->caps.qp1_proxy[attr->port_num - 1];
|
||||||
}
|
}
|
||||||
|
|
||||||
struct ib_qp *mlx4_ib_create_qp(struct ib_pd *pd,
|
static struct ib_qp *_mlx4_ib_create_qp(struct ib_pd *pd,
|
||||||
struct ib_qp_init_attr *init_attr,
|
struct ib_qp_init_attr *init_attr,
|
||||||
struct ib_udata *udata)
|
struct ib_udata *udata)
|
||||||
{
|
{
|
||||||
struct mlx4_ib_qp *qp = NULL;
|
struct mlx4_ib_qp *qp = NULL;
|
||||||
int err;
|
int err;
|
||||||
|
@ -1120,6 +1127,7 @@ struct ib_qp *mlx4_ib_create_qp(struct ib_pd *pd,
|
||||||
MLX4_IB_SRIOV_TUNNEL_QP |
|
MLX4_IB_SRIOV_TUNNEL_QP |
|
||||||
MLX4_IB_SRIOV_SQP |
|
MLX4_IB_SRIOV_SQP |
|
||||||
MLX4_IB_QP_NETIF |
|
MLX4_IB_QP_NETIF |
|
||||||
|
MLX4_IB_QP_CREATE_ROCE_V2_GSI |
|
||||||
MLX4_IB_QP_CREATE_USE_GFP_NOIO))
|
MLX4_IB_QP_CREATE_USE_GFP_NOIO))
|
||||||
return ERR_PTR(-EINVAL);
|
return ERR_PTR(-EINVAL);
|
||||||
|
|
||||||
|
@ -1128,15 +1136,21 @@ struct ib_qp *mlx4_ib_create_qp(struct ib_pd *pd,
|
||||||
return ERR_PTR(-EINVAL);
|
return ERR_PTR(-EINVAL);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (init_attr->create_flags &&
|
if (init_attr->create_flags) {
|
||||||
((udata && init_attr->create_flags & ~(sup_u_create_flags)) ||
|
if (udata && init_attr->create_flags & ~(sup_u_create_flags))
|
||||||
((init_attr->create_flags & ~(MLX4_IB_SRIOV_SQP |
|
return ERR_PTR(-EINVAL);
|
||||||
MLX4_IB_QP_CREATE_USE_GFP_NOIO |
|
|
||||||
MLX4_IB_QP_BLOCK_MULTICAST_LOOPBACK)) &&
|
if ((init_attr->create_flags & ~(MLX4_IB_SRIOV_SQP |
|
||||||
init_attr->qp_type != IB_QPT_UD) ||
|
MLX4_IB_QP_CREATE_USE_GFP_NOIO |
|
||||||
((init_attr->create_flags & MLX4_IB_SRIOV_SQP) &&
|
MLX4_IB_QP_CREATE_ROCE_V2_GSI |
|
||||||
init_attr->qp_type > IB_QPT_GSI)))
|
MLX4_IB_QP_BLOCK_MULTICAST_LOOPBACK) &&
|
||||||
return ERR_PTR(-EINVAL);
|
init_attr->qp_type != IB_QPT_UD) ||
|
||||||
|
(init_attr->create_flags & MLX4_IB_SRIOV_SQP &&
|
||||||
|
init_attr->qp_type > IB_QPT_GSI) ||
|
||||||
|
(init_attr->create_flags & MLX4_IB_QP_CREATE_ROCE_V2_GSI &&
|
||||||
|
init_attr->qp_type != IB_QPT_GSI))
|
||||||
|
return ERR_PTR(-EINVAL);
|
||||||
|
}
|
||||||
|
|
||||||
switch (init_attr->qp_type) {
|
switch (init_attr->qp_type) {
|
||||||
case IB_QPT_XRC_TGT:
|
case IB_QPT_XRC_TGT:
|
||||||
|
@ -1173,19 +1187,29 @@ struct ib_qp *mlx4_ib_create_qp(struct ib_pd *pd,
|
||||||
case IB_QPT_SMI:
|
case IB_QPT_SMI:
|
||||||
case IB_QPT_GSI:
|
case IB_QPT_GSI:
|
||||||
{
|
{
|
||||||
|
int sqpn;
|
||||||
|
|
||||||
/* Userspace is not allowed to create special QPs: */
|
/* Userspace is not allowed to create special QPs: */
|
||||||
if (udata)
|
if (udata)
|
||||||
return ERR_PTR(-EINVAL);
|
return ERR_PTR(-EINVAL);
|
||||||
|
if (init_attr->create_flags & MLX4_IB_QP_CREATE_ROCE_V2_GSI) {
|
||||||
|
int res = mlx4_qp_reserve_range(to_mdev(pd->device)->dev, 1, 1, &sqpn, 0);
|
||||||
|
|
||||||
|
if (res)
|
||||||
|
return ERR_PTR(res);
|
||||||
|
} else {
|
||||||
|
sqpn = get_sqp_num(to_mdev(pd->device), init_attr);
|
||||||
|
}
|
||||||
|
|
||||||
err = create_qp_common(to_mdev(pd->device), pd, init_attr, udata,
|
err = create_qp_common(to_mdev(pd->device), pd, init_attr, udata,
|
||||||
get_sqp_num(to_mdev(pd->device), init_attr),
|
sqpn,
|
||||||
&qp, gfp);
|
&qp, gfp);
|
||||||
if (err)
|
if (err)
|
||||||
return ERR_PTR(err);
|
return ERR_PTR(err);
|
||||||
|
|
||||||
qp->port = init_attr->port_num;
|
qp->port = init_attr->port_num;
|
||||||
qp->ibqp.qp_num = init_attr->qp_type == IB_QPT_SMI ? 0 : 1;
|
qp->ibqp.qp_num = init_attr->qp_type == IB_QPT_SMI ? 0 :
|
||||||
|
init_attr->create_flags & MLX4_IB_QP_CREATE_ROCE_V2_GSI ? sqpn : 1;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
default:
|
default:
|
||||||
|
@ -1196,7 +1220,41 @@ struct ib_qp *mlx4_ib_create_qp(struct ib_pd *pd,
|
||||||
return &qp->ibqp;
|
return &qp->ibqp;
|
||||||
}
|
}
|
||||||
|
|
||||||
int mlx4_ib_destroy_qp(struct ib_qp *qp)
|
struct ib_qp *mlx4_ib_create_qp(struct ib_pd *pd,
|
||||||
|
struct ib_qp_init_attr *init_attr,
|
||||||
|
struct ib_udata *udata) {
|
||||||
|
struct ib_device *device = pd ? pd->device : init_attr->xrcd->device;
|
||||||
|
struct ib_qp *ibqp;
|
||||||
|
struct mlx4_ib_dev *dev = to_mdev(device);
|
||||||
|
|
||||||
|
ibqp = _mlx4_ib_create_qp(pd, init_attr, udata);
|
||||||
|
|
||||||
|
if (!IS_ERR(ibqp) &&
|
||||||
|
(init_attr->qp_type == IB_QPT_GSI) &&
|
||||||
|
!(init_attr->create_flags & MLX4_IB_QP_CREATE_ROCE_V2_GSI)) {
|
||||||
|
struct mlx4_ib_sqp *sqp = to_msqp((to_mqp(ibqp)));
|
||||||
|
int is_eth = rdma_cap_eth_ah(&dev->ib_dev, init_attr->port_num);
|
||||||
|
|
||||||
|
if (is_eth &&
|
||||||
|
dev->dev->caps.flags2 & MLX4_DEV_CAP_FLAG2_ROCE_V1_V2) {
|
||||||
|
init_attr->create_flags |= MLX4_IB_QP_CREATE_ROCE_V2_GSI;
|
||||||
|
sqp->roce_v2_gsi = ib_create_qp(pd, init_attr);
|
||||||
|
|
||||||
|
if (IS_ERR(sqp->roce_v2_gsi)) {
|
||||||
|
pr_err("Failed to create GSI QP for RoCEv2 (%ld)\n", PTR_ERR(sqp->roce_v2_gsi));
|
||||||
|
sqp->roce_v2_gsi = NULL;
|
||||||
|
} else {
|
||||||
|
sqp = to_msqp(to_mqp(sqp->roce_v2_gsi));
|
||||||
|
sqp->qp.flags |= MLX4_IB_ROCE_V2_GSI_QP;
|
||||||
|
}
|
||||||
|
|
||||||
|
init_attr->create_flags &= ~MLX4_IB_QP_CREATE_ROCE_V2_GSI;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ibqp;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int _mlx4_ib_destroy_qp(struct ib_qp *qp)
|
||||||
{
|
{
|
||||||
struct mlx4_ib_dev *dev = to_mdev(qp->device);
|
struct mlx4_ib_dev *dev = to_mdev(qp->device);
|
||||||
struct mlx4_ib_qp *mqp = to_mqp(qp);
|
struct mlx4_ib_qp *mqp = to_mqp(qp);
|
||||||
|
@ -1225,6 +1283,20 @@ int mlx4_ib_destroy_qp(struct ib_qp *qp)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int mlx4_ib_destroy_qp(struct ib_qp *qp)
|
||||||
|
{
|
||||||
|
struct mlx4_ib_qp *mqp = to_mqp(qp);
|
||||||
|
|
||||||
|
if (mqp->mlx4_ib_qp_type == MLX4_IB_QPT_GSI) {
|
||||||
|
struct mlx4_ib_sqp *sqp = to_msqp(mqp);
|
||||||
|
|
||||||
|
if (sqp->roce_v2_gsi)
|
||||||
|
ib_destroy_qp(sqp->roce_v2_gsi);
|
||||||
|
}
|
||||||
|
|
||||||
|
return _mlx4_ib_destroy_qp(qp);
|
||||||
|
}
|
||||||
|
|
||||||
static int to_mlx4_st(struct mlx4_ib_dev *dev, enum mlx4_ib_qp_type type)
|
static int to_mlx4_st(struct mlx4_ib_dev *dev, enum mlx4_ib_qp_type type)
|
||||||
{
|
{
|
||||||
switch (type) {
|
switch (type) {
|
||||||
|
@ -1507,6 +1579,24 @@ static int create_qp_lb_counter(struct mlx4_ib_dev *dev, struct mlx4_ib_qp *qp)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
enum {
|
||||||
|
MLX4_QPC_ROCE_MODE_1 = 0,
|
||||||
|
MLX4_QPC_ROCE_MODE_2 = 2,
|
||||||
|
MLX4_QPC_ROCE_MODE_UNDEFINED = 0xff
|
||||||
|
};
|
||||||
|
|
||||||
|
static u8 gid_type_to_qpc(enum ib_gid_type gid_type)
|
||||||
|
{
|
||||||
|
switch (gid_type) {
|
||||||
|
case IB_GID_TYPE_ROCE:
|
||||||
|
return MLX4_QPC_ROCE_MODE_1;
|
||||||
|
case IB_GID_TYPE_ROCE_UDP_ENCAP:
|
||||||
|
return MLX4_QPC_ROCE_MODE_2;
|
||||||
|
default:
|
||||||
|
return MLX4_QPC_ROCE_MODE_UNDEFINED;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static int __mlx4_ib_modify_qp(struct ib_qp *ibqp,
|
static int __mlx4_ib_modify_qp(struct ib_qp *ibqp,
|
||||||
const struct ib_qp_attr *attr, int attr_mask,
|
const struct ib_qp_attr *attr, int attr_mask,
|
||||||
enum ib_qp_state cur_state, enum ib_qp_state new_state)
|
enum ib_qp_state cur_state, enum ib_qp_state new_state)
|
||||||
|
@ -1633,6 +1723,14 @@ static int __mlx4_ib_modify_qp(struct ib_qp *ibqp,
|
||||||
mlx4_ib_steer_qp_reg(dev, qp, 1);
|
mlx4_ib_steer_qp_reg(dev, qp, 1);
|
||||||
steer_qp = 1;
|
steer_qp = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (ibqp->qp_type == IB_QPT_GSI) {
|
||||||
|
enum ib_gid_type gid_type = qp->flags & MLX4_IB_ROCE_V2_GSI_QP ?
|
||||||
|
IB_GID_TYPE_ROCE_UDP_ENCAP : IB_GID_TYPE_ROCE;
|
||||||
|
u8 qpc_roce_mode = gid_type_to_qpc(gid_type);
|
||||||
|
|
||||||
|
context->rlkey_roce_mode |= (qpc_roce_mode << 6);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (attr_mask & IB_QP_PKEY_INDEX) {
|
if (attr_mask & IB_QP_PKEY_INDEX) {
|
||||||
|
@ -1650,9 +1748,10 @@ static int __mlx4_ib_modify_qp(struct ib_qp *ibqp,
|
||||||
u16 vlan = 0xffff;
|
u16 vlan = 0xffff;
|
||||||
u8 smac[ETH_ALEN];
|
u8 smac[ETH_ALEN];
|
||||||
int status = 0;
|
int status = 0;
|
||||||
|
int is_eth = rdma_cap_eth_ah(&dev->ib_dev, port_num) &&
|
||||||
|
attr->ah_attr.ah_flags & IB_AH_GRH;
|
||||||
|
|
||||||
if (rdma_cap_eth_ah(&dev->ib_dev, port_num) &&
|
if (is_eth) {
|
||||||
attr->ah_attr.ah_flags & IB_AH_GRH) {
|
|
||||||
int index = attr->ah_attr.grh.sgid_index;
|
int index = attr->ah_attr.grh.sgid_index;
|
||||||
|
|
||||||
status = ib_get_cached_gid(ibqp->device, port_num,
|
status = ib_get_cached_gid(ibqp->device, port_num,
|
||||||
|
@ -1674,6 +1773,18 @@ static int __mlx4_ib_modify_qp(struct ib_qp *ibqp,
|
||||||
|
|
||||||
optpar |= (MLX4_QP_OPTPAR_PRIMARY_ADDR_PATH |
|
optpar |= (MLX4_QP_OPTPAR_PRIMARY_ADDR_PATH |
|
||||||
MLX4_QP_OPTPAR_SCHED_QUEUE);
|
MLX4_QP_OPTPAR_SCHED_QUEUE);
|
||||||
|
|
||||||
|
if (is_eth &&
|
||||||
|
(cur_state == IB_QPS_INIT && new_state == IB_QPS_RTR)) {
|
||||||
|
u8 qpc_roce_mode = gid_type_to_qpc(gid_attr.gid_type);
|
||||||
|
|
||||||
|
if (qpc_roce_mode == MLX4_QPC_ROCE_MODE_UNDEFINED) {
|
||||||
|
err = -EINVAL;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
context->rlkey_roce_mode |= (qpc_roce_mode << 6);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (attr_mask & IB_QP_TIMEOUT) {
|
if (attr_mask & IB_QP_TIMEOUT) {
|
||||||
|
@ -1845,7 +1956,7 @@ static int __mlx4_ib_modify_qp(struct ib_qp *ibqp,
|
||||||
sqd_event = 0;
|
sqd_event = 0;
|
||||||
|
|
||||||
if (!ibqp->uobject && cur_state == IB_QPS_RESET && new_state == IB_QPS_INIT)
|
if (!ibqp->uobject && cur_state == IB_QPS_RESET && new_state == IB_QPS_INIT)
|
||||||
context->rlkey |= (1 << 4);
|
context->rlkey_roce_mode |= (1 << 4);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Before passing a kernel QP to the HW, make sure that the
|
* Before passing a kernel QP to the HW, make sure that the
|
||||||
|
@ -2022,8 +2133,8 @@ static int __mlx4_ib_modify_qp(struct ib_qp *ibqp,
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
int mlx4_ib_modify_qp(struct ib_qp *ibqp, struct ib_qp_attr *attr,
|
static int _mlx4_ib_modify_qp(struct ib_qp *ibqp, struct ib_qp_attr *attr,
|
||||||
int attr_mask, struct ib_udata *udata)
|
int attr_mask, struct ib_udata *udata)
|
||||||
{
|
{
|
||||||
struct mlx4_ib_dev *dev = to_mdev(ibqp->device);
|
struct mlx4_ib_dev *dev = to_mdev(ibqp->device);
|
||||||
struct mlx4_ib_qp *qp = to_mqp(ibqp);
|
struct mlx4_ib_qp *qp = to_mqp(ibqp);
|
||||||
|
@ -2126,6 +2237,27 @@ int mlx4_ib_modify_qp(struct ib_qp *ibqp, struct ib_qp_attr *attr,
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int mlx4_ib_modify_qp(struct ib_qp *ibqp, struct ib_qp_attr *attr,
|
||||||
|
int attr_mask, struct ib_udata *udata)
|
||||||
|
{
|
||||||
|
struct mlx4_ib_qp *mqp = to_mqp(ibqp);
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
ret = _mlx4_ib_modify_qp(ibqp, attr, attr_mask, udata);
|
||||||
|
|
||||||
|
if (mqp->mlx4_ib_qp_type == MLX4_IB_QPT_GSI) {
|
||||||
|
struct mlx4_ib_sqp *sqp = to_msqp(mqp);
|
||||||
|
int err = 0;
|
||||||
|
|
||||||
|
if (sqp->roce_v2_gsi)
|
||||||
|
err = ib_modify_qp(sqp->roce_v2_gsi, attr, attr_mask);
|
||||||
|
if (err)
|
||||||
|
pr_err("Failed to modify GSI QP for RoCEv2 (%d)\n",
|
||||||
|
err);
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
static int vf_get_qp0_qkey(struct mlx4_dev *dev, int qpn, u32 *qkey)
|
static int vf_get_qp0_qkey(struct mlx4_dev *dev, int qpn, u32 *qkey)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
|
@ -2168,7 +2300,7 @@ static int build_sriov_qp0_header(struct mlx4_ib_sqp *sqp,
|
||||||
if (sqp->qp.mlx4_ib_qp_type == MLX4_IB_QPT_PROXY_SMI_OWNER)
|
if (sqp->qp.mlx4_ib_qp_type == MLX4_IB_QPT_PROXY_SMI_OWNER)
|
||||||
send_size += sizeof (struct mlx4_ib_tunnel_header);
|
send_size += sizeof (struct mlx4_ib_tunnel_header);
|
||||||
|
|
||||||
ib_ud_header_init(send_size, 1, 0, 0, 0, 0, &sqp->ud_header);
|
ib_ud_header_init(send_size, 1, 0, 0, 0, 0, 0, 0, &sqp->ud_header);
|
||||||
|
|
||||||
if (sqp->qp.mlx4_ib_qp_type == MLX4_IB_QPT_PROXY_SMI_OWNER) {
|
if (sqp->qp.mlx4_ib_qp_type == MLX4_IB_QPT_PROXY_SMI_OWNER) {
|
||||||
sqp->ud_header.lrh.service_level =
|
sqp->ud_header.lrh.service_level =
|
||||||
|
@ -2252,16 +2384,7 @@ static int build_sriov_qp0_header(struct mlx4_ib_sqp *sqp,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void mlx4_u64_to_smac(u8 *dst_mac, u64 src_mac)
|
#define MLX4_ROCEV2_QP1_SPORT 0xC000
|
||||||
{
|
|
||||||
int i;
|
|
||||||
|
|
||||||
for (i = ETH_ALEN; i; i--) {
|
|
||||||
dst_mac[i - 1] = src_mac & 0xff;
|
|
||||||
src_mac >>= 8;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static int build_mlx_header(struct mlx4_ib_sqp *sqp, struct ib_ud_wr *wr,
|
static int build_mlx_header(struct mlx4_ib_sqp *sqp, struct ib_ud_wr *wr,
|
||||||
void *wqe, unsigned *mlx_seg_len)
|
void *wqe, unsigned *mlx_seg_len)
|
||||||
{
|
{
|
||||||
|
@ -2281,6 +2404,8 @@ static int build_mlx_header(struct mlx4_ib_sqp *sqp, struct ib_ud_wr *wr,
|
||||||
bool is_eth;
|
bool is_eth;
|
||||||
bool is_vlan = false;
|
bool is_vlan = false;
|
||||||
bool is_grh;
|
bool is_grh;
|
||||||
|
bool is_udp = false;
|
||||||
|
int ip_version = 0;
|
||||||
|
|
||||||
send_size = 0;
|
send_size = 0;
|
||||||
for (i = 0; i < wr->wr.num_sge; ++i)
|
for (i = 0; i < wr->wr.num_sge; ++i)
|
||||||
|
@ -2289,6 +2414,8 @@ static int build_mlx_header(struct mlx4_ib_sqp *sqp, struct ib_ud_wr *wr,
|
||||||
is_eth = rdma_port_get_link_layer(sqp->qp.ibqp.device, sqp->qp.port) == IB_LINK_LAYER_ETHERNET;
|
is_eth = rdma_port_get_link_layer(sqp->qp.ibqp.device, sqp->qp.port) == IB_LINK_LAYER_ETHERNET;
|
||||||
is_grh = mlx4_ib_ah_grh_present(ah);
|
is_grh = mlx4_ib_ah_grh_present(ah);
|
||||||
if (is_eth) {
|
if (is_eth) {
|
||||||
|
struct ib_gid_attr gid_attr;
|
||||||
|
|
||||||
if (mlx4_is_mfunc(to_mdev(ib_dev)->dev)) {
|
if (mlx4_is_mfunc(to_mdev(ib_dev)->dev)) {
|
||||||
/* When multi-function is enabled, the ib_core gid
|
/* When multi-function is enabled, the ib_core gid
|
||||||
* indexes don't necessarily match the hw ones, so
|
* indexes don't necessarily match the hw ones, so
|
||||||
|
@ -2302,19 +2429,35 @@ static int build_mlx_header(struct mlx4_ib_sqp *sqp, struct ib_ud_wr *wr,
|
||||||
err = ib_get_cached_gid(ib_dev,
|
err = ib_get_cached_gid(ib_dev,
|
||||||
be32_to_cpu(ah->av.ib.port_pd) >> 24,
|
be32_to_cpu(ah->av.ib.port_pd) >> 24,
|
||||||
ah->av.ib.gid_index, &sgid,
|
ah->av.ib.gid_index, &sgid,
|
||||||
NULL);
|
&gid_attr);
|
||||||
if (!err && !memcmp(&sgid, &zgid, sizeof(sgid)))
|
if (!err) {
|
||||||
err = -ENOENT;
|
if (gid_attr.ndev)
|
||||||
if (err)
|
dev_put(gid_attr.ndev);
|
||||||
|
if (!memcmp(&sgid, &zgid, sizeof(sgid)))
|
||||||
|
err = -ENOENT;
|
||||||
|
}
|
||||||
|
if (!err) {
|
||||||
|
is_udp = gid_attr.gid_type == IB_GID_TYPE_ROCE_UDP_ENCAP;
|
||||||
|
if (is_udp) {
|
||||||
|
if (ipv6_addr_v4mapped((struct in6_addr *)&sgid))
|
||||||
|
ip_version = 4;
|
||||||
|
else
|
||||||
|
ip_version = 6;
|
||||||
|
is_grh = false;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
return err;
|
return err;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ah->av.eth.vlan != cpu_to_be16(0xffff)) {
|
if (ah->av.eth.vlan != cpu_to_be16(0xffff)) {
|
||||||
vlan = be16_to_cpu(ah->av.eth.vlan) & 0x0fff;
|
vlan = be16_to_cpu(ah->av.eth.vlan) & 0x0fff;
|
||||||
is_vlan = 1;
|
is_vlan = 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ib_ud_header_init(send_size, !is_eth, is_eth, is_vlan, is_grh, 0, &sqp->ud_header);
|
err = ib_ud_header_init(send_size, !is_eth, is_eth, is_vlan, is_grh,
|
||||||
|
ip_version, is_udp, 0, &sqp->ud_header);
|
||||||
|
if (err)
|
||||||
|
return err;
|
||||||
|
|
||||||
if (!is_eth) {
|
if (!is_eth) {
|
||||||
sqp->ud_header.lrh.service_level =
|
sqp->ud_header.lrh.service_level =
|
||||||
|
@ -2323,7 +2466,7 @@ static int build_mlx_header(struct mlx4_ib_sqp *sqp, struct ib_ud_wr *wr,
|
||||||
sqp->ud_header.lrh.source_lid = cpu_to_be16(ah->av.ib.g_slid & 0x7f);
|
sqp->ud_header.lrh.source_lid = cpu_to_be16(ah->av.ib.g_slid & 0x7f);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (is_grh) {
|
if (is_grh || (ip_version == 6)) {
|
||||||
sqp->ud_header.grh.traffic_class =
|
sqp->ud_header.grh.traffic_class =
|
||||||
(be32_to_cpu(ah->av.ib.sl_tclass_flowlabel) >> 20) & 0xff;
|
(be32_to_cpu(ah->av.ib.sl_tclass_flowlabel) >> 20) & 0xff;
|
||||||
sqp->ud_header.grh.flow_label =
|
sqp->ud_header.grh.flow_label =
|
||||||
|
@ -2352,6 +2495,25 @@ static int build_mlx_header(struct mlx4_ib_sqp *sqp, struct ib_ud_wr *wr,
|
||||||
ah->av.ib.dgid, 16);
|
ah->av.ib.dgid, 16);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (ip_version == 4) {
|
||||||
|
sqp->ud_header.ip4.tos =
|
||||||
|
(be32_to_cpu(ah->av.ib.sl_tclass_flowlabel) >> 20) & 0xff;
|
||||||
|
sqp->ud_header.ip4.id = 0;
|
||||||
|
sqp->ud_header.ip4.frag_off = htons(IP_DF);
|
||||||
|
sqp->ud_header.ip4.ttl = ah->av.eth.hop_limit;
|
||||||
|
|
||||||
|
memcpy(&sqp->ud_header.ip4.saddr,
|
||||||
|
sgid.raw + 12, 4);
|
||||||
|
memcpy(&sqp->ud_header.ip4.daddr, ah->av.ib.dgid + 12, 4);
|
||||||
|
sqp->ud_header.ip4.check = ib_ud_ip4_csum(&sqp->ud_header);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (is_udp) {
|
||||||
|
sqp->ud_header.udp.dport = htons(ROCE_V2_UDP_DPORT);
|
||||||
|
sqp->ud_header.udp.sport = htons(MLX4_ROCEV2_QP1_SPORT);
|
||||||
|
sqp->ud_header.udp.csum = 0;
|
||||||
|
}
|
||||||
|
|
||||||
mlx->flags &= cpu_to_be32(MLX4_WQE_CTRL_CQ_UPDATE);
|
mlx->flags &= cpu_to_be32(MLX4_WQE_CTRL_CQ_UPDATE);
|
||||||
|
|
||||||
if (!is_eth) {
|
if (!is_eth) {
|
||||||
|
@ -2380,34 +2542,27 @@ static int build_mlx_header(struct mlx4_ib_sqp *sqp, struct ib_ud_wr *wr,
|
||||||
|
|
||||||
if (is_eth) {
|
if (is_eth) {
|
||||||
struct in6_addr in6;
|
struct in6_addr in6;
|
||||||
|
u16 ether_type;
|
||||||
u16 pcp = (be32_to_cpu(ah->av.ib.sl_tclass_flowlabel) >> 29) << 13;
|
u16 pcp = (be32_to_cpu(ah->av.ib.sl_tclass_flowlabel) >> 29) << 13;
|
||||||
|
|
||||||
|
ether_type = (!is_udp) ? MLX4_IB_IBOE_ETHERTYPE :
|
||||||
|
(ip_version == 4 ? ETH_P_IP : ETH_P_IPV6);
|
||||||
|
|
||||||
mlx->sched_prio = cpu_to_be16(pcp);
|
mlx->sched_prio = cpu_to_be16(pcp);
|
||||||
|
|
||||||
|
ether_addr_copy(sqp->ud_header.eth.smac_h, ah->av.eth.s_mac);
|
||||||
memcpy(sqp->ud_header.eth.dmac_h, ah->av.eth.mac, 6);
|
memcpy(sqp->ud_header.eth.dmac_h, ah->av.eth.mac, 6);
|
||||||
/* FIXME: cache smac value? */
|
|
||||||
memcpy(&ctrl->srcrb_flags16[0], ah->av.eth.mac, 2);
|
memcpy(&ctrl->srcrb_flags16[0], ah->av.eth.mac, 2);
|
||||||
memcpy(&ctrl->imm, ah->av.eth.mac + 2, 4);
|
memcpy(&ctrl->imm, ah->av.eth.mac + 2, 4);
|
||||||
memcpy(&in6, sgid.raw, sizeof(in6));
|
memcpy(&in6, sgid.raw, sizeof(in6));
|
||||||
|
|
||||||
if (!mlx4_is_mfunc(to_mdev(ib_dev)->dev)) {
|
|
||||||
u64 mac = atomic64_read(&to_mdev(ib_dev)->iboe.mac[sqp->qp.port - 1]);
|
|
||||||
u8 smac[ETH_ALEN];
|
|
||||||
|
|
||||||
mlx4_u64_to_smac(smac, mac);
|
|
||||||
memcpy(sqp->ud_header.eth.smac_h, smac, ETH_ALEN);
|
|
||||||
} else {
|
|
||||||
/* use the src mac of the tunnel */
|
|
||||||
memcpy(sqp->ud_header.eth.smac_h, ah->av.eth.s_mac, ETH_ALEN);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!memcmp(sqp->ud_header.eth.smac_h, sqp->ud_header.eth.dmac_h, 6))
|
if (!memcmp(sqp->ud_header.eth.smac_h, sqp->ud_header.eth.dmac_h, 6))
|
||||||
mlx->flags |= cpu_to_be32(MLX4_WQE_CTRL_FORCE_LOOPBACK);
|
mlx->flags |= cpu_to_be32(MLX4_WQE_CTRL_FORCE_LOOPBACK);
|
||||||
if (!is_vlan) {
|
if (!is_vlan) {
|
||||||
sqp->ud_header.eth.type = cpu_to_be16(MLX4_IB_IBOE_ETHERTYPE);
|
sqp->ud_header.eth.type = cpu_to_be16(ether_type);
|
||||||
} else {
|
} else {
|
||||||
sqp->ud_header.vlan.type = cpu_to_be16(MLX4_IB_IBOE_ETHERTYPE);
|
sqp->ud_header.vlan.type = cpu_to_be16(ether_type);
|
||||||
sqp->ud_header.vlan.tag = cpu_to_be16(vlan | pcp);
|
sqp->ud_header.vlan.tag = cpu_to_be16(vlan | pcp);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
@ -2528,25 +2683,6 @@ static void set_reg_seg(struct mlx4_wqe_fmr_seg *fseg,
|
||||||
fseg->reserved[1] = 0;
|
fseg->reserved[1] = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void set_bind_seg(struct mlx4_wqe_bind_seg *bseg,
|
|
||||||
struct ib_bind_mw_wr *wr)
|
|
||||||
{
|
|
||||||
bseg->flags1 =
|
|
||||||
convert_access(wr->bind_info.mw_access_flags) &
|
|
||||||
cpu_to_be32(MLX4_WQE_FMR_AND_BIND_PERM_REMOTE_READ |
|
|
||||||
MLX4_WQE_FMR_AND_BIND_PERM_REMOTE_WRITE |
|
|
||||||
MLX4_WQE_FMR_AND_BIND_PERM_ATOMIC);
|
|
||||||
bseg->flags2 = 0;
|
|
||||||
if (wr->mw->type == IB_MW_TYPE_2)
|
|
||||||
bseg->flags2 |= cpu_to_be32(MLX4_WQE_BIND_TYPE_2);
|
|
||||||
if (wr->bind_info.mw_access_flags & IB_ZERO_BASED)
|
|
||||||
bseg->flags2 |= cpu_to_be32(MLX4_WQE_BIND_ZERO_BASED);
|
|
||||||
bseg->new_rkey = cpu_to_be32(wr->rkey);
|
|
||||||
bseg->lkey = cpu_to_be32(wr->bind_info.mr->lkey);
|
|
||||||
bseg->addr = cpu_to_be64(wr->bind_info.addr);
|
|
||||||
bseg->length = cpu_to_be64(wr->bind_info.length);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void set_local_inv_seg(struct mlx4_wqe_local_inval_seg *iseg, u32 rkey)
|
static void set_local_inv_seg(struct mlx4_wqe_local_inval_seg *iseg, u32 rkey)
|
||||||
{
|
{
|
||||||
memset(iseg, 0, sizeof(*iseg));
|
memset(iseg, 0, sizeof(*iseg));
|
||||||
|
@ -2766,6 +2902,29 @@ int mlx4_ib_post_send(struct ib_qp *ibqp, struct ib_send_wr *wr,
|
||||||
int i;
|
int i;
|
||||||
struct mlx4_ib_dev *mdev = to_mdev(ibqp->device);
|
struct mlx4_ib_dev *mdev = to_mdev(ibqp->device);
|
||||||
|
|
||||||
|
if (qp->mlx4_ib_qp_type == MLX4_IB_QPT_GSI) {
|
||||||
|
struct mlx4_ib_sqp *sqp = to_msqp(qp);
|
||||||
|
|
||||||
|
if (sqp->roce_v2_gsi) {
|
||||||
|
struct mlx4_ib_ah *ah = to_mah(ud_wr(wr)->ah);
|
||||||
|
struct ib_gid_attr gid_attr;
|
||||||
|
union ib_gid gid;
|
||||||
|
|
||||||
|
if (!ib_get_cached_gid(ibqp->device,
|
||||||
|
be32_to_cpu(ah->av.ib.port_pd) >> 24,
|
||||||
|
ah->av.ib.gid_index, &gid,
|
||||||
|
&gid_attr)) {
|
||||||
|
if (gid_attr.ndev)
|
||||||
|
dev_put(gid_attr.ndev);
|
||||||
|
qp = (gid_attr.gid_type == IB_GID_TYPE_ROCE_UDP_ENCAP) ?
|
||||||
|
to_mqp(sqp->roce_v2_gsi) : qp;
|
||||||
|
} else {
|
||||||
|
pr_err("Failed to get gid at index %d. RoCEv2 will not work properly\n",
|
||||||
|
ah->av.ib.gid_index);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
spin_lock_irqsave(&qp->sq.lock, flags);
|
spin_lock_irqsave(&qp->sq.lock, flags);
|
||||||
if (mdev->dev->persist->state & MLX4_DEVICE_STATE_INTERNAL_ERROR) {
|
if (mdev->dev->persist->state & MLX4_DEVICE_STATE_INTERNAL_ERROR) {
|
||||||
err = -EIO;
|
err = -EIO;
|
||||||
|
@ -2867,13 +3026,6 @@ int mlx4_ib_post_send(struct ib_qp *ibqp, struct ib_send_wr *wr,
|
||||||
size += sizeof(struct mlx4_wqe_fmr_seg) / 16;
|
size += sizeof(struct mlx4_wqe_fmr_seg) / 16;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case IB_WR_BIND_MW:
|
|
||||||
ctrl->srcrb_flags |=
|
|
||||||
cpu_to_be32(MLX4_WQE_CTRL_STRONG_ORDER);
|
|
||||||
set_bind_seg(wqe, bind_mw_wr(wr));
|
|
||||||
wqe += sizeof(struct mlx4_wqe_bind_seg);
|
|
||||||
size += sizeof(struct mlx4_wqe_bind_seg) / 16;
|
|
||||||
break;
|
|
||||||
default:
|
default:
|
||||||
/* No extra segments required for sends */
|
/* No extra segments required for sends */
|
||||||
break;
|
break;
|
||||||
|
|
|
@ -171,7 +171,8 @@ struct ib_srq *mlx4_ib_create_srq(struct ib_pd *pd,
|
||||||
if (err)
|
if (err)
|
||||||
goto err_mtt;
|
goto err_mtt;
|
||||||
|
|
||||||
srq->wrid = kmalloc(srq->msrq.max * sizeof (u64), GFP_KERNEL);
|
srq->wrid = kmalloc_array(srq->msrq.max, sizeof(u64),
|
||||||
|
GFP_KERNEL | __GFP_NOWARN);
|
||||||
if (!srq->wrid) {
|
if (!srq->wrid) {
|
||||||
srq->wrid = __vmalloc(srq->msrq.max * sizeof(u64),
|
srq->wrid = __vmalloc(srq->msrq.max * sizeof(u64),
|
||||||
GFP_KERNEL, PAGE_KERNEL);
|
GFP_KERNEL, PAGE_KERNEL);
|
||||||
|
|
|
@ -32,8 +32,10 @@
|
||||||
|
|
||||||
#include "mlx5_ib.h"
|
#include "mlx5_ib.h"
|
||||||
|
|
||||||
struct ib_ah *create_ib_ah(struct ib_ah_attr *ah_attr,
|
static struct ib_ah *create_ib_ah(struct mlx5_ib_dev *dev,
|
||||||
struct mlx5_ib_ah *ah)
|
struct mlx5_ib_ah *ah,
|
||||||
|
struct ib_ah_attr *ah_attr,
|
||||||
|
enum rdma_link_layer ll)
|
||||||
{
|
{
|
||||||
if (ah_attr->ah_flags & IB_AH_GRH) {
|
if (ah_attr->ah_flags & IB_AH_GRH) {
|
||||||
memcpy(ah->av.rgid, &ah_attr->grh.dgid, 16);
|
memcpy(ah->av.rgid, &ah_attr->grh.dgid, 16);
|
||||||
|
@ -44,9 +46,20 @@ struct ib_ah *create_ib_ah(struct ib_ah_attr *ah_attr,
|
||||||
ah->av.tclass = ah_attr->grh.traffic_class;
|
ah->av.tclass = ah_attr->grh.traffic_class;
|
||||||
}
|
}
|
||||||
|
|
||||||
ah->av.rlid = cpu_to_be16(ah_attr->dlid);
|
ah->av.stat_rate_sl = (ah_attr->static_rate << 4);
|
||||||
ah->av.fl_mlid = ah_attr->src_path_bits & 0x7f;
|
|
||||||
ah->av.stat_rate_sl = (ah_attr->static_rate << 4) | (ah_attr->sl & 0xf);
|
if (ll == IB_LINK_LAYER_ETHERNET) {
|
||||||
|
memcpy(ah->av.rmac, ah_attr->dmac, sizeof(ah_attr->dmac));
|
||||||
|
ah->av.udp_sport =
|
||||||
|
mlx5_get_roce_udp_sport(dev,
|
||||||
|
ah_attr->port_num,
|
||||||
|
ah_attr->grh.sgid_index);
|
||||||
|
ah->av.stat_rate_sl |= (ah_attr->sl & 0x7) << 1;
|
||||||
|
} else {
|
||||||
|
ah->av.rlid = cpu_to_be16(ah_attr->dlid);
|
||||||
|
ah->av.fl_mlid = ah_attr->src_path_bits & 0x7f;
|
||||||
|
ah->av.stat_rate_sl |= (ah_attr->sl & 0xf);
|
||||||
|
}
|
||||||
|
|
||||||
return &ah->ibah;
|
return &ah->ibah;
|
||||||
}
|
}
|
||||||
|
@ -54,12 +67,19 @@ struct ib_ah *create_ib_ah(struct ib_ah_attr *ah_attr,
|
||||||
struct ib_ah *mlx5_ib_create_ah(struct ib_pd *pd, struct ib_ah_attr *ah_attr)
|
struct ib_ah *mlx5_ib_create_ah(struct ib_pd *pd, struct ib_ah_attr *ah_attr)
|
||||||
{
|
{
|
||||||
struct mlx5_ib_ah *ah;
|
struct mlx5_ib_ah *ah;
|
||||||
|
struct mlx5_ib_dev *dev = to_mdev(pd->device);
|
||||||
|
enum rdma_link_layer ll;
|
||||||
|
|
||||||
|
ll = pd->device->get_link_layer(pd->device, ah_attr->port_num);
|
||||||
|
|
||||||
|
if (ll == IB_LINK_LAYER_ETHERNET && !(ah_attr->ah_flags & IB_AH_GRH))
|
||||||
|
return ERR_PTR(-EINVAL);
|
||||||
|
|
||||||
ah = kzalloc(sizeof(*ah), GFP_ATOMIC);
|
ah = kzalloc(sizeof(*ah), GFP_ATOMIC);
|
||||||
if (!ah)
|
if (!ah)
|
||||||
return ERR_PTR(-ENOMEM);
|
return ERR_PTR(-ENOMEM);
|
||||||
|
|
||||||
return create_ib_ah(ah_attr, ah); /* never fails */
|
return create_ib_ah(dev, ah, ah_attr, ll); /* never fails */
|
||||||
}
|
}
|
||||||
|
|
||||||
int mlx5_ib_query_ah(struct ib_ah *ibah, struct ib_ah_attr *ah_attr)
|
int mlx5_ib_query_ah(struct ib_ah *ibah, struct ib_ah_attr *ah_attr)
|
||||||
|
|
|
@ -154,9 +154,6 @@ static void handle_good_req(struct ib_wc *wc, struct mlx5_cqe64 *cqe,
|
||||||
wc->opcode = IB_WC_MASKED_FETCH_ADD;
|
wc->opcode = IB_WC_MASKED_FETCH_ADD;
|
||||||
wc->byte_len = 8;
|
wc->byte_len = 8;
|
||||||
break;
|
break;
|
||||||
case MLX5_OPCODE_BIND_MW:
|
|
||||||
wc->opcode = IB_WC_BIND_MW;
|
|
||||||
break;
|
|
||||||
case MLX5_OPCODE_UMR:
|
case MLX5_OPCODE_UMR:
|
||||||
wc->opcode = get_umr_comp(wq, idx);
|
wc->opcode = get_umr_comp(wq, idx);
|
||||||
break;
|
break;
|
||||||
|
@ -171,6 +168,7 @@ enum {
|
||||||
static void handle_responder(struct ib_wc *wc, struct mlx5_cqe64 *cqe,
|
static void handle_responder(struct ib_wc *wc, struct mlx5_cqe64 *cqe,
|
||||||
struct mlx5_ib_qp *qp)
|
struct mlx5_ib_qp *qp)
|
||||||
{
|
{
|
||||||
|
enum rdma_link_layer ll = rdma_port_get_link_layer(qp->ibqp.device, 1);
|
||||||
struct mlx5_ib_dev *dev = to_mdev(qp->ibqp.device);
|
struct mlx5_ib_dev *dev = to_mdev(qp->ibqp.device);
|
||||||
struct mlx5_ib_srq *srq;
|
struct mlx5_ib_srq *srq;
|
||||||
struct mlx5_ib_wq *wq;
|
struct mlx5_ib_wq *wq;
|
||||||
|
@ -236,6 +234,22 @@ static void handle_responder(struct ib_wc *wc, struct mlx5_cqe64 *cqe,
|
||||||
} else {
|
} else {
|
||||||
wc->pkey_index = 0;
|
wc->pkey_index = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (ll != IB_LINK_LAYER_ETHERNET)
|
||||||
|
return;
|
||||||
|
|
||||||
|
switch (wc->sl & 0x3) {
|
||||||
|
case MLX5_CQE_ROCE_L3_HEADER_TYPE_GRH:
|
||||||
|
wc->network_hdr_type = RDMA_NETWORK_IB;
|
||||||
|
break;
|
||||||
|
case MLX5_CQE_ROCE_L3_HEADER_TYPE_IPV6:
|
||||||
|
wc->network_hdr_type = RDMA_NETWORK_IPV6;
|
||||||
|
break;
|
||||||
|
case MLX5_CQE_ROCE_L3_HEADER_TYPE_IPV4:
|
||||||
|
wc->network_hdr_type = RDMA_NETWORK_IPV4;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
wc->wc_flags |= IB_WC_WITH_NETWORK_HDR_TYPE;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void dump_cqe(struct mlx5_ib_dev *dev, struct mlx5_err_cqe *cqe)
|
static void dump_cqe(struct mlx5_ib_dev *dev, struct mlx5_err_cqe *cqe)
|
||||||
|
@ -760,12 +774,12 @@ struct ib_cq *mlx5_ib_create_cq(struct ib_device *ibdev,
|
||||||
int eqn;
|
int eqn;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
if (attr->flags)
|
|
||||||
return ERR_PTR(-EINVAL);
|
|
||||||
|
|
||||||
if (entries < 0)
|
if (entries < 0)
|
||||||
return ERR_PTR(-EINVAL);
|
return ERR_PTR(-EINVAL);
|
||||||
|
|
||||||
|
if (check_cq_create_flags(attr->flags))
|
||||||
|
return ERR_PTR(-EOPNOTSUPP);
|
||||||
|
|
||||||
entries = roundup_pow_of_two(entries + 1);
|
entries = roundup_pow_of_two(entries + 1);
|
||||||
if (entries > (1 << MLX5_CAP_GEN(dev->mdev, log_max_cq_sz)))
|
if (entries > (1 << MLX5_CAP_GEN(dev->mdev, log_max_cq_sz)))
|
||||||
return ERR_PTR(-EINVAL);
|
return ERR_PTR(-EINVAL);
|
||||||
|
@ -779,6 +793,7 @@ struct ib_cq *mlx5_ib_create_cq(struct ib_device *ibdev,
|
||||||
spin_lock_init(&cq->lock);
|
spin_lock_init(&cq->lock);
|
||||||
cq->resize_buf = NULL;
|
cq->resize_buf = NULL;
|
||||||
cq->resize_umem = NULL;
|
cq->resize_umem = NULL;
|
||||||
|
cq->create_flags = attr->flags;
|
||||||
|
|
||||||
if (context) {
|
if (context) {
|
||||||
err = create_cq_user(dev, udata, context, cq, entries,
|
err = create_cq_user(dev, udata, context, cq, entries,
|
||||||
|
@ -796,6 +811,10 @@ struct ib_cq *mlx5_ib_create_cq(struct ib_device *ibdev,
|
||||||
|
|
||||||
cq->cqe_size = cqe_size;
|
cq->cqe_size = cqe_size;
|
||||||
cqb->ctx.cqe_sz_flags = cqe_sz_to_mlx_sz(cqe_size) << 5;
|
cqb->ctx.cqe_sz_flags = cqe_sz_to_mlx_sz(cqe_size) << 5;
|
||||||
|
|
||||||
|
if (cq->create_flags & IB_CQ_FLAGS_IGNORE_OVERRUN)
|
||||||
|
cqb->ctx.cqe_sz_flags |= (1 << 1);
|
||||||
|
|
||||||
cqb->ctx.log_sz_usr_page = cpu_to_be32((ilog2(entries) << 24) | index);
|
cqb->ctx.log_sz_usr_page = cpu_to_be32((ilog2(entries) << 24) | index);
|
||||||
err = mlx5_vector2eqn(dev->mdev, vector, &eqn, &irqn);
|
err = mlx5_vector2eqn(dev->mdev, vector, &eqn, &irqn);
|
||||||
if (err)
|
if (err)
|
||||||
|
|
|
@ -40,6 +40,8 @@
|
||||||
#include <linux/io-mapping.h>
|
#include <linux/io-mapping.h>
|
||||||
#include <linux/sched.h>
|
#include <linux/sched.h>
|
||||||
#include <rdma/ib_user_verbs.h>
|
#include <rdma/ib_user_verbs.h>
|
||||||
|
#include <rdma/ib_addr.h>
|
||||||
|
#include <rdma/ib_cache.h>
|
||||||
#include <linux/mlx5/vport.h>
|
#include <linux/mlx5/vport.h>
|
||||||
#include <rdma/ib_smi.h>
|
#include <rdma/ib_smi.h>
|
||||||
#include <rdma/ib_umem.h>
|
#include <rdma/ib_umem.h>
|
||||||
|
@ -66,12 +68,14 @@ 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 " (" DRIVER_RELDATE ")\n";
|
||||||
|
|
||||||
static enum rdma_link_layer
|
enum {
|
||||||
mlx5_ib_port_link_layer(struct ib_device *device)
|
MLX5_ATOMIC_SIZE_QP_8BYTES = 1 << 3,
|
||||||
{
|
};
|
||||||
struct mlx5_ib_dev *dev = to_mdev(device);
|
|
||||||
|
|
||||||
switch (MLX5_CAP_GEN(dev->mdev, port_type)) {
|
static enum rdma_link_layer
|
||||||
|
mlx5_port_type_cap_to_rdma_ll(int port_type_cap)
|
||||||
|
{
|
||||||
|
switch (port_type_cap) {
|
||||||
case MLX5_CAP_PORT_TYPE_IB:
|
case MLX5_CAP_PORT_TYPE_IB:
|
||||||
return IB_LINK_LAYER_INFINIBAND;
|
return IB_LINK_LAYER_INFINIBAND;
|
||||||
case MLX5_CAP_PORT_TYPE_ETH:
|
case MLX5_CAP_PORT_TYPE_ETH:
|
||||||
|
@ -81,6 +85,202 @@ mlx5_ib_port_link_layer(struct ib_device *device)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static enum rdma_link_layer
|
||||||
|
mlx5_ib_port_link_layer(struct ib_device *device, u8 port_num)
|
||||||
|
{
|
||||||
|
struct mlx5_ib_dev *dev = to_mdev(device);
|
||||||
|
int port_type_cap = MLX5_CAP_GEN(dev->mdev, port_type);
|
||||||
|
|
||||||
|
return mlx5_port_type_cap_to_rdma_ll(port_type_cap);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int mlx5_netdev_event(struct notifier_block *this,
|
||||||
|
unsigned long event, void *ptr)
|
||||||
|
{
|
||||||
|
struct net_device *ndev = netdev_notifier_info_to_dev(ptr);
|
||||||
|
struct mlx5_ib_dev *ibdev = container_of(this, struct mlx5_ib_dev,
|
||||||
|
roce.nb);
|
||||||
|
|
||||||
|
if ((event != NETDEV_UNREGISTER) && (event != NETDEV_REGISTER))
|
||||||
|
return NOTIFY_DONE;
|
||||||
|
|
||||||
|
write_lock(&ibdev->roce.netdev_lock);
|
||||||
|
if (ndev->dev.parent == &ibdev->mdev->pdev->dev)
|
||||||
|
ibdev->roce.netdev = (event == NETDEV_UNREGISTER) ? NULL : ndev;
|
||||||
|
write_unlock(&ibdev->roce.netdev_lock);
|
||||||
|
|
||||||
|
return NOTIFY_DONE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct net_device *mlx5_ib_get_netdev(struct ib_device *device,
|
||||||
|
u8 port_num)
|
||||||
|
{
|
||||||
|
struct mlx5_ib_dev *ibdev = to_mdev(device);
|
||||||
|
struct net_device *ndev;
|
||||||
|
|
||||||
|
/* Ensure ndev does not disappear before we invoke dev_hold()
|
||||||
|
*/
|
||||||
|
read_lock(&ibdev->roce.netdev_lock);
|
||||||
|
ndev = ibdev->roce.netdev;
|
||||||
|
if (ndev)
|
||||||
|
dev_hold(ndev);
|
||||||
|
read_unlock(&ibdev->roce.netdev_lock);
|
||||||
|
|
||||||
|
return ndev;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int mlx5_query_port_roce(struct ib_device *device, u8 port_num,
|
||||||
|
struct ib_port_attr *props)
|
||||||
|
{
|
||||||
|
struct mlx5_ib_dev *dev = to_mdev(device);
|
||||||
|
struct net_device *ndev;
|
||||||
|
enum ib_mtu ndev_ib_mtu;
|
||||||
|
u16 qkey_viol_cntr;
|
||||||
|
|
||||||
|
memset(props, 0, sizeof(*props));
|
||||||
|
|
||||||
|
props->port_cap_flags |= IB_PORT_CM_SUP;
|
||||||
|
props->port_cap_flags |= IB_PORT_IP_BASED_GIDS;
|
||||||
|
|
||||||
|
props->gid_tbl_len = MLX5_CAP_ROCE(dev->mdev,
|
||||||
|
roce_address_table_size);
|
||||||
|
props->max_mtu = IB_MTU_4096;
|
||||||
|
props->max_msg_sz = 1 << MLX5_CAP_GEN(dev->mdev, log_max_msg);
|
||||||
|
props->pkey_tbl_len = 1;
|
||||||
|
props->state = IB_PORT_DOWN;
|
||||||
|
props->phys_state = 3;
|
||||||
|
|
||||||
|
mlx5_query_nic_vport_qkey_viol_cntr(dev->mdev, &qkey_viol_cntr);
|
||||||
|
props->qkey_viol_cntr = qkey_viol_cntr;
|
||||||
|
|
||||||
|
ndev = mlx5_ib_get_netdev(device, port_num);
|
||||||
|
if (!ndev)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (netif_running(ndev) && netif_carrier_ok(ndev)) {
|
||||||
|
props->state = IB_PORT_ACTIVE;
|
||||||
|
props->phys_state = 5;
|
||||||
|
}
|
||||||
|
|
||||||
|
ndev_ib_mtu = iboe_get_mtu(ndev->mtu);
|
||||||
|
|
||||||
|
dev_put(ndev);
|
||||||
|
|
||||||
|
props->active_mtu = min(props->max_mtu, ndev_ib_mtu);
|
||||||
|
|
||||||
|
props->active_width = IB_WIDTH_4X; /* TODO */
|
||||||
|
props->active_speed = IB_SPEED_QDR; /* TODO */
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void ib_gid_to_mlx5_roce_addr(const union ib_gid *gid,
|
||||||
|
const struct ib_gid_attr *attr,
|
||||||
|
void *mlx5_addr)
|
||||||
|
{
|
||||||
|
#define MLX5_SET_RA(p, f, v) MLX5_SET(roce_addr_layout, p, f, v)
|
||||||
|
char *mlx5_addr_l3_addr = MLX5_ADDR_OF(roce_addr_layout, mlx5_addr,
|
||||||
|
source_l3_address);
|
||||||
|
void *mlx5_addr_mac = MLX5_ADDR_OF(roce_addr_layout, mlx5_addr,
|
||||||
|
source_mac_47_32);
|
||||||
|
|
||||||
|
if (!gid)
|
||||||
|
return;
|
||||||
|
|
||||||
|
ether_addr_copy(mlx5_addr_mac, attr->ndev->dev_addr);
|
||||||
|
|
||||||
|
if (is_vlan_dev(attr->ndev)) {
|
||||||
|
MLX5_SET_RA(mlx5_addr, vlan_valid, 1);
|
||||||
|
MLX5_SET_RA(mlx5_addr, vlan_id, vlan_dev_vlan_id(attr->ndev));
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (attr->gid_type) {
|
||||||
|
case IB_GID_TYPE_IB:
|
||||||
|
MLX5_SET_RA(mlx5_addr, roce_version, MLX5_ROCE_VERSION_1);
|
||||||
|
break;
|
||||||
|
case IB_GID_TYPE_ROCE_UDP_ENCAP:
|
||||||
|
MLX5_SET_RA(mlx5_addr, roce_version, MLX5_ROCE_VERSION_2);
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
WARN_ON(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (attr->gid_type != IB_GID_TYPE_IB) {
|
||||||
|
if (ipv6_addr_v4mapped((void *)gid))
|
||||||
|
MLX5_SET_RA(mlx5_addr, roce_l3_type,
|
||||||
|
MLX5_ROCE_L3_TYPE_IPV4);
|
||||||
|
else
|
||||||
|
MLX5_SET_RA(mlx5_addr, roce_l3_type,
|
||||||
|
MLX5_ROCE_L3_TYPE_IPV6);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((attr->gid_type == IB_GID_TYPE_IB) ||
|
||||||
|
!ipv6_addr_v4mapped((void *)gid))
|
||||||
|
memcpy(mlx5_addr_l3_addr, gid, sizeof(*gid));
|
||||||
|
else
|
||||||
|
memcpy(&mlx5_addr_l3_addr[12], &gid->raw[12], 4);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int set_roce_addr(struct ib_device *device, u8 port_num,
|
||||||
|
unsigned int index,
|
||||||
|
const union ib_gid *gid,
|
||||||
|
const struct ib_gid_attr *attr)
|
||||||
|
{
|
||||||
|
struct mlx5_ib_dev *dev = to_mdev(device);
|
||||||
|
u32 in[MLX5_ST_SZ_DW(set_roce_address_in)];
|
||||||
|
u32 out[MLX5_ST_SZ_DW(set_roce_address_out)];
|
||||||
|
void *in_addr = MLX5_ADDR_OF(set_roce_address_in, in, roce_address);
|
||||||
|
enum rdma_link_layer ll = mlx5_ib_port_link_layer(device, port_num);
|
||||||
|
|
||||||
|
if (ll != IB_LINK_LAYER_ETHERNET)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
memset(in, 0, sizeof(in));
|
||||||
|
|
||||||
|
ib_gid_to_mlx5_roce_addr(gid, attr, in_addr);
|
||||||
|
|
||||||
|
MLX5_SET(set_roce_address_in, in, roce_address_index, index);
|
||||||
|
MLX5_SET(set_roce_address_in, in, opcode, MLX5_CMD_OP_SET_ROCE_ADDRESS);
|
||||||
|
|
||||||
|
memset(out, 0, sizeof(out));
|
||||||
|
return mlx5_cmd_exec(dev->mdev, in, sizeof(in), out, sizeof(out));
|
||||||
|
}
|
||||||
|
|
||||||
|
static int mlx5_ib_add_gid(struct ib_device *device, u8 port_num,
|
||||||
|
unsigned int index, const union ib_gid *gid,
|
||||||
|
const struct ib_gid_attr *attr,
|
||||||
|
__always_unused void **context)
|
||||||
|
{
|
||||||
|
return set_roce_addr(device, port_num, index, gid, attr);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int mlx5_ib_del_gid(struct ib_device *device, u8 port_num,
|
||||||
|
unsigned int index, __always_unused void **context)
|
||||||
|
{
|
||||||
|
return set_roce_addr(device, port_num, index, NULL, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
__be16 mlx5_get_roce_udp_sport(struct mlx5_ib_dev *dev, u8 port_num,
|
||||||
|
int index)
|
||||||
|
{
|
||||||
|
struct ib_gid_attr attr;
|
||||||
|
union ib_gid gid;
|
||||||
|
|
||||||
|
if (ib_get_cached_gid(&dev->ib_dev, port_num, index, &gid, &attr))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (!attr.ndev)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
dev_put(attr.ndev);
|
||||||
|
|
||||||
|
if (attr.gid_type != IB_GID_TYPE_ROCE_UDP_ENCAP)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
return cpu_to_be16(MLX5_CAP_ROCE(dev->mdev, r_roce_min_src_udp_port));
|
||||||
|
}
|
||||||
|
|
||||||
static int mlx5_use_mad_ifc(struct mlx5_ib_dev *dev)
|
static int mlx5_use_mad_ifc(struct mlx5_ib_dev *dev)
|
||||||
{
|
{
|
||||||
return !dev->mdev->issi;
|
return !dev->mdev->issi;
|
||||||
|
@ -97,13 +297,35 @@ static int mlx5_get_vport_access_method(struct ib_device *ibdev)
|
||||||
if (mlx5_use_mad_ifc(to_mdev(ibdev)))
|
if (mlx5_use_mad_ifc(to_mdev(ibdev)))
|
||||||
return MLX5_VPORT_ACCESS_METHOD_MAD;
|
return MLX5_VPORT_ACCESS_METHOD_MAD;
|
||||||
|
|
||||||
if (mlx5_ib_port_link_layer(ibdev) ==
|
if (mlx5_ib_port_link_layer(ibdev, 1) ==
|
||||||
IB_LINK_LAYER_ETHERNET)
|
IB_LINK_LAYER_ETHERNET)
|
||||||
return MLX5_VPORT_ACCESS_METHOD_NIC;
|
return MLX5_VPORT_ACCESS_METHOD_NIC;
|
||||||
|
|
||||||
return MLX5_VPORT_ACCESS_METHOD_HCA;
|
return MLX5_VPORT_ACCESS_METHOD_HCA;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void get_atomic_caps(struct mlx5_ib_dev *dev,
|
||||||
|
struct ib_device_attr *props)
|
||||||
|
{
|
||||||
|
u8 tmp;
|
||||||
|
u8 atomic_operations = MLX5_CAP_ATOMIC(dev->mdev, atomic_operations);
|
||||||
|
u8 atomic_size_qp = MLX5_CAP_ATOMIC(dev->mdev, atomic_size_qp);
|
||||||
|
u8 atomic_req_8B_endianness_mode =
|
||||||
|
MLX5_CAP_ATOMIC(dev->mdev, atomic_req_8B_endianess_mode);
|
||||||
|
|
||||||
|
/* Check if HW supports 8 bytes standard atomic operations and capable
|
||||||
|
* of host endianness respond
|
||||||
|
*/
|
||||||
|
tmp = MLX5_ATOMIC_OPS_CMP_SWAP | MLX5_ATOMIC_OPS_FETCH_ADD;
|
||||||
|
if (((atomic_operations & tmp) == tmp) &&
|
||||||
|
(atomic_size_qp & MLX5_ATOMIC_SIZE_QP_8BYTES) &&
|
||||||
|
(atomic_req_8B_endianness_mode)) {
|
||||||
|
props->atomic_cap = IB_ATOMIC_HCA;
|
||||||
|
} else {
|
||||||
|
props->atomic_cap = IB_ATOMIC_NONE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static int mlx5_query_system_image_guid(struct ib_device *ibdev,
|
static int mlx5_query_system_image_guid(struct ib_device *ibdev,
|
||||||
__be64 *sys_image_guid)
|
__be64 *sys_image_guid)
|
||||||
{
|
{
|
||||||
|
@ -119,13 +341,21 @@ static int mlx5_query_system_image_guid(struct ib_device *ibdev,
|
||||||
|
|
||||||
case MLX5_VPORT_ACCESS_METHOD_HCA:
|
case MLX5_VPORT_ACCESS_METHOD_HCA:
|
||||||
err = mlx5_query_hca_vport_system_image_guid(mdev, &tmp);
|
err = mlx5_query_hca_vport_system_image_guid(mdev, &tmp);
|
||||||
if (!err)
|
break;
|
||||||
*sys_image_guid = cpu_to_be64(tmp);
|
|
||||||
return err;
|
case MLX5_VPORT_ACCESS_METHOD_NIC:
|
||||||
|
err = mlx5_query_nic_vport_system_image_guid(mdev, &tmp);
|
||||||
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!err)
|
||||||
|
*sys_image_guid = cpu_to_be64(tmp);
|
||||||
|
|
||||||
|
return err;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int mlx5_query_max_pkeys(struct ib_device *ibdev,
|
static int mlx5_query_max_pkeys(struct ib_device *ibdev,
|
||||||
|
@ -179,13 +409,20 @@ static int mlx5_query_node_guid(struct mlx5_ib_dev *dev,
|
||||||
|
|
||||||
case MLX5_VPORT_ACCESS_METHOD_HCA:
|
case MLX5_VPORT_ACCESS_METHOD_HCA:
|
||||||
err = mlx5_query_hca_vport_node_guid(dev->mdev, &tmp);
|
err = mlx5_query_hca_vport_node_guid(dev->mdev, &tmp);
|
||||||
if (!err)
|
break;
|
||||||
*node_guid = cpu_to_be64(tmp);
|
|
||||||
return err;
|
case MLX5_VPORT_ACCESS_METHOD_NIC:
|
||||||
|
err = mlx5_query_nic_vport_node_guid(dev->mdev, &tmp);
|
||||||
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!err)
|
||||||
|
*node_guid = cpu_to_be64(tmp);
|
||||||
|
|
||||||
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct mlx5_reg_node_desc {
|
struct mlx5_reg_node_desc {
|
||||||
|
@ -263,6 +500,10 @@ static int mlx5_ib_query_device(struct ib_device *ibdev,
|
||||||
if (MLX5_CAP_GEN(mdev, block_lb_mc))
|
if (MLX5_CAP_GEN(mdev, block_lb_mc))
|
||||||
props->device_cap_flags |= IB_DEVICE_BLOCK_MULTICAST_LOOPBACK;
|
props->device_cap_flags |= IB_DEVICE_BLOCK_MULTICAST_LOOPBACK;
|
||||||
|
|
||||||
|
if (MLX5_CAP_GEN(dev->mdev, eth_net_offloads) &&
|
||||||
|
(MLX5_CAP_ETH(dev->mdev, csum_cap)))
|
||||||
|
props->device_cap_flags |= IB_DEVICE_RAW_IP_CSUM;
|
||||||
|
|
||||||
props->vendor_part_id = mdev->pdev->device;
|
props->vendor_part_id = mdev->pdev->device;
|
||||||
props->hw_ver = mdev->pdev->revision;
|
props->hw_ver = mdev->pdev->revision;
|
||||||
|
|
||||||
|
@ -278,7 +519,7 @@ static int mlx5_ib_query_device(struct ib_device *ibdev,
|
||||||
props->max_sge = min(max_rq_sg, max_sq_sg);
|
props->max_sge = min(max_rq_sg, max_sq_sg);
|
||||||
props->max_sge_rd = props->max_sge;
|
props->max_sge_rd = props->max_sge;
|
||||||
props->max_cq = 1 << MLX5_CAP_GEN(mdev, log_max_cq);
|
props->max_cq = 1 << MLX5_CAP_GEN(mdev, log_max_cq);
|
||||||
props->max_cqe = (1 << MLX5_CAP_GEN(mdev, log_max_eq_sz)) - 1;
|
props->max_cqe = (1 << MLX5_CAP_GEN(mdev, log_max_cq_sz)) - 1;
|
||||||
props->max_mr = 1 << MLX5_CAP_GEN(mdev, log_max_mkey);
|
props->max_mr = 1 << MLX5_CAP_GEN(mdev, log_max_mkey);
|
||||||
props->max_pd = 1 << MLX5_CAP_GEN(mdev, log_max_pd);
|
props->max_pd = 1 << MLX5_CAP_GEN(mdev, log_max_pd);
|
||||||
props->max_qp_rd_atom = 1 << MLX5_CAP_GEN(mdev, log_max_ra_req_qp);
|
props->max_qp_rd_atom = 1 << MLX5_CAP_GEN(mdev, log_max_ra_req_qp);
|
||||||
|
@ -289,13 +530,15 @@ static int mlx5_ib_query_device(struct ib_device *ibdev,
|
||||||
props->max_res_rd_atom = props->max_qp_rd_atom * props->max_qp;
|
props->max_res_rd_atom = props->max_qp_rd_atom * props->max_qp;
|
||||||
props->max_srq_sge = max_rq_sg - 1;
|
props->max_srq_sge = max_rq_sg - 1;
|
||||||
props->max_fast_reg_page_list_len = (unsigned int)-1;
|
props->max_fast_reg_page_list_len = (unsigned int)-1;
|
||||||
props->atomic_cap = IB_ATOMIC_NONE;
|
get_atomic_caps(dev, props);
|
||||||
props->masked_atomic_cap = IB_ATOMIC_NONE;
|
props->masked_atomic_cap = IB_ATOMIC_NONE;
|
||||||
props->max_mcast_grp = 1 << MLX5_CAP_GEN(mdev, log_max_mcg);
|
props->max_mcast_grp = 1 << MLX5_CAP_GEN(mdev, log_max_mcg);
|
||||||
props->max_mcast_qp_attach = MLX5_CAP_GEN(mdev, max_qp_mcg);
|
props->max_mcast_qp_attach = MLX5_CAP_GEN(mdev, max_qp_mcg);
|
||||||
props->max_total_mcast_qp_attach = props->max_mcast_qp_attach *
|
props->max_total_mcast_qp_attach = props->max_mcast_qp_attach *
|
||||||
props->max_mcast_grp;
|
props->max_mcast_grp;
|
||||||
props->max_map_per_fmr = INT_MAX; /* no limit in ConnectIB */
|
props->max_map_per_fmr = INT_MAX; /* no limit in ConnectIB */
|
||||||
|
props->hca_core_clock = MLX5_CAP_GEN(mdev, device_frequency_khz);
|
||||||
|
props->timestamp_mask = 0x7FFFFFFFFFFFFFFFULL;
|
||||||
|
|
||||||
#ifdef CONFIG_INFINIBAND_ON_DEMAND_PAGING
|
#ifdef CONFIG_INFINIBAND_ON_DEMAND_PAGING
|
||||||
if (MLX5_CAP_GEN(mdev, pg))
|
if (MLX5_CAP_GEN(mdev, pg))
|
||||||
|
@ -303,6 +546,9 @@ static int mlx5_ib_query_device(struct ib_device *ibdev,
|
||||||
props->odp_caps = dev->odp_caps;
|
props->odp_caps = dev->odp_caps;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
if (MLX5_CAP_GEN(mdev, cd))
|
||||||
|
props->device_cap_flags |= IB_DEVICE_CROSS_CHANNEL;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -483,6 +729,9 @@ int mlx5_ib_query_port(struct ib_device *ibdev, u8 port,
|
||||||
case MLX5_VPORT_ACCESS_METHOD_HCA:
|
case MLX5_VPORT_ACCESS_METHOD_HCA:
|
||||||
return mlx5_query_hca_port(ibdev, port, props);
|
return mlx5_query_hca_port(ibdev, port, props);
|
||||||
|
|
||||||
|
case MLX5_VPORT_ACCESS_METHOD_NIC:
|
||||||
|
return mlx5_query_port_roce(ibdev, port, props);
|
||||||
|
|
||||||
default:
|
default:
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
@ -583,8 +832,8 @@ static struct ib_ucontext *mlx5_ib_alloc_ucontext(struct ib_device *ibdev,
|
||||||
struct ib_udata *udata)
|
struct ib_udata *udata)
|
||||||
{
|
{
|
||||||
struct mlx5_ib_dev *dev = to_mdev(ibdev);
|
struct mlx5_ib_dev *dev = to_mdev(ibdev);
|
||||||
struct mlx5_ib_alloc_ucontext_req_v2 req;
|
struct mlx5_ib_alloc_ucontext_req_v2 req = {};
|
||||||
struct mlx5_ib_alloc_ucontext_resp resp;
|
struct mlx5_ib_alloc_ucontext_resp resp = {};
|
||||||
struct mlx5_ib_ucontext *context;
|
struct mlx5_ib_ucontext *context;
|
||||||
struct mlx5_uuar_info *uuari;
|
struct mlx5_uuar_info *uuari;
|
||||||
struct mlx5_uar *uars;
|
struct mlx5_uar *uars;
|
||||||
|
@ -599,20 +848,22 @@ static struct ib_ucontext *mlx5_ib_alloc_ucontext(struct ib_device *ibdev,
|
||||||
if (!dev->ib_active)
|
if (!dev->ib_active)
|
||||||
return ERR_PTR(-EAGAIN);
|
return ERR_PTR(-EAGAIN);
|
||||||
|
|
||||||
memset(&req, 0, sizeof(req));
|
if (udata->inlen < sizeof(struct ib_uverbs_cmd_hdr))
|
||||||
|
return ERR_PTR(-EINVAL);
|
||||||
|
|
||||||
reqlen = udata->inlen - sizeof(struct ib_uverbs_cmd_hdr);
|
reqlen = udata->inlen - sizeof(struct ib_uverbs_cmd_hdr);
|
||||||
if (reqlen == sizeof(struct mlx5_ib_alloc_ucontext_req))
|
if (reqlen == sizeof(struct mlx5_ib_alloc_ucontext_req))
|
||||||
ver = 0;
|
ver = 0;
|
||||||
else if (reqlen == sizeof(struct mlx5_ib_alloc_ucontext_req_v2))
|
else if (reqlen >= sizeof(struct mlx5_ib_alloc_ucontext_req_v2))
|
||||||
ver = 2;
|
ver = 2;
|
||||||
else
|
else
|
||||||
return ERR_PTR(-EINVAL);
|
return ERR_PTR(-EINVAL);
|
||||||
|
|
||||||
err = ib_copy_from_udata(&req, udata, reqlen);
|
err = ib_copy_from_udata(&req, udata, min(reqlen, sizeof(req)));
|
||||||
if (err)
|
if (err)
|
||||||
return ERR_PTR(err);
|
return ERR_PTR(err);
|
||||||
|
|
||||||
if (req.flags || req.reserved)
|
if (req.flags)
|
||||||
return ERR_PTR(-EINVAL);
|
return ERR_PTR(-EINVAL);
|
||||||
|
|
||||||
if (req.total_num_uuars > MLX5_MAX_UUARS)
|
if (req.total_num_uuars > MLX5_MAX_UUARS)
|
||||||
|
@ -621,6 +872,14 @@ static struct ib_ucontext *mlx5_ib_alloc_ucontext(struct ib_device *ibdev,
|
||||||
if (req.total_num_uuars == 0)
|
if (req.total_num_uuars == 0)
|
||||||
return ERR_PTR(-EINVAL);
|
return ERR_PTR(-EINVAL);
|
||||||
|
|
||||||
|
if (req.comp_mask || req.reserved0 || req.reserved1 || req.reserved2)
|
||||||
|
return ERR_PTR(-EOPNOTSUPP);
|
||||||
|
|
||||||
|
if (reqlen > sizeof(req) &&
|
||||||
|
!ib_is_udata_cleared(udata, sizeof(req),
|
||||||
|
reqlen - sizeof(req)))
|
||||||
|
return ERR_PTR(-EOPNOTSUPP);
|
||||||
|
|
||||||
req.total_num_uuars = ALIGN(req.total_num_uuars,
|
req.total_num_uuars = ALIGN(req.total_num_uuars,
|
||||||
MLX5_NON_FP_BF_REGS_PER_PAGE);
|
MLX5_NON_FP_BF_REGS_PER_PAGE);
|
||||||
if (req.num_low_latency_uuars > req.total_num_uuars - 1)
|
if (req.num_low_latency_uuars > req.total_num_uuars - 1)
|
||||||
|
@ -636,6 +895,11 @@ static struct ib_ucontext *mlx5_ib_alloc_ucontext(struct ib_device *ibdev,
|
||||||
resp.max_send_wqebb = 1 << MLX5_CAP_GEN(dev->mdev, log_max_qp_sz);
|
resp.max_send_wqebb = 1 << MLX5_CAP_GEN(dev->mdev, log_max_qp_sz);
|
||||||
resp.max_recv_wr = 1 << MLX5_CAP_GEN(dev->mdev, log_max_qp_sz);
|
resp.max_recv_wr = 1 << MLX5_CAP_GEN(dev->mdev, log_max_qp_sz);
|
||||||
resp.max_srq_recv_wr = 1 << MLX5_CAP_GEN(dev->mdev, log_max_srq_sz);
|
resp.max_srq_recv_wr = 1 << MLX5_CAP_GEN(dev->mdev, log_max_srq_sz);
|
||||||
|
resp.cqe_version = min_t(__u8,
|
||||||
|
(__u8)MLX5_CAP_GEN(dev->mdev, cqe_version),
|
||||||
|
req.max_cqe_version);
|
||||||
|
resp.response_length = min(offsetof(typeof(resp), response_length) +
|
||||||
|
sizeof(resp.response_length), udata->outlen);
|
||||||
|
|
||||||
context = kzalloc(sizeof(*context), GFP_KERNEL);
|
context = kzalloc(sizeof(*context), GFP_KERNEL);
|
||||||
if (!context)
|
if (!context)
|
||||||
|
@ -681,22 +945,49 @@ static struct ib_ucontext *mlx5_ib_alloc_ucontext(struct ib_device *ibdev,
|
||||||
context->ibucontext.invalidate_range = &mlx5_ib_invalidate_range;
|
context->ibucontext.invalidate_range = &mlx5_ib_invalidate_range;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
if (MLX5_CAP_GEN(dev->mdev, log_max_transport_domain)) {
|
||||||
|
err = mlx5_core_alloc_transport_domain(dev->mdev,
|
||||||
|
&context->tdn);
|
||||||
|
if (err)
|
||||||
|
goto out_uars;
|
||||||
|
}
|
||||||
|
|
||||||
INIT_LIST_HEAD(&context->db_page_list);
|
INIT_LIST_HEAD(&context->db_page_list);
|
||||||
mutex_init(&context->db_page_mutex);
|
mutex_init(&context->db_page_mutex);
|
||||||
|
|
||||||
resp.tot_uuars = req.total_num_uuars;
|
resp.tot_uuars = req.total_num_uuars;
|
||||||
resp.num_ports = MLX5_CAP_GEN(dev->mdev, num_ports);
|
resp.num_ports = MLX5_CAP_GEN(dev->mdev, num_ports);
|
||||||
err = ib_copy_to_udata(udata, &resp,
|
|
||||||
sizeof(resp) - sizeof(resp.reserved));
|
if (field_avail(typeof(resp), cqe_version, udata->outlen))
|
||||||
|
resp.response_length += sizeof(resp.cqe_version);
|
||||||
|
|
||||||
|
if (field_avail(typeof(resp), hca_core_clock_offset, udata->outlen)) {
|
||||||
|
resp.comp_mask |=
|
||||||
|
MLX5_IB_ALLOC_UCONTEXT_RESP_MASK_CORE_CLOCK_OFFSET;
|
||||||
|
resp.hca_core_clock_offset =
|
||||||
|
offsetof(struct mlx5_init_seg, internal_timer_h) %
|
||||||
|
PAGE_SIZE;
|
||||||
|
resp.response_length += sizeof(resp.hca_core_clock_offset) +
|
||||||
|
sizeof(resp.reserved2) +
|
||||||
|
sizeof(resp.reserved3);
|
||||||
|
}
|
||||||
|
|
||||||
|
err = ib_copy_to_udata(udata, &resp, resp.response_length);
|
||||||
if (err)
|
if (err)
|
||||||
goto out_uars;
|
goto out_td;
|
||||||
|
|
||||||
uuari->ver = ver;
|
uuari->ver = ver;
|
||||||
uuari->num_low_latency_uuars = req.num_low_latency_uuars;
|
uuari->num_low_latency_uuars = req.num_low_latency_uuars;
|
||||||
uuari->uars = uars;
|
uuari->uars = uars;
|
||||||
uuari->num_uars = num_uars;
|
uuari->num_uars = num_uars;
|
||||||
|
context->cqe_version = resp.cqe_version;
|
||||||
|
|
||||||
return &context->ibucontext;
|
return &context->ibucontext;
|
||||||
|
|
||||||
|
out_td:
|
||||||
|
if (MLX5_CAP_GEN(dev->mdev, log_max_transport_domain))
|
||||||
|
mlx5_core_dealloc_transport_domain(dev->mdev, context->tdn);
|
||||||
|
|
||||||
out_uars:
|
out_uars:
|
||||||
for (i--; i >= 0; i--)
|
for (i--; i >= 0; i--)
|
||||||
mlx5_cmd_free_uar(dev->mdev, uars[i].index);
|
mlx5_cmd_free_uar(dev->mdev, uars[i].index);
|
||||||
|
@ -721,6 +1012,9 @@ static int mlx5_ib_dealloc_ucontext(struct ib_ucontext *ibcontext)
|
||||||
struct mlx5_uuar_info *uuari = &context->uuari;
|
struct mlx5_uuar_info *uuari = &context->uuari;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
|
if (MLX5_CAP_GEN(dev->mdev, log_max_transport_domain))
|
||||||
|
mlx5_core_dealloc_transport_domain(dev->mdev, context->tdn);
|
||||||
|
|
||||||
for (i = 0; i < uuari->num_uars; i++) {
|
for (i = 0; i < uuari->num_uars; i++) {
|
||||||
if (mlx5_cmd_free_uar(dev->mdev, uuari->uars[i].index))
|
if (mlx5_cmd_free_uar(dev->mdev, uuari->uars[i].index))
|
||||||
mlx5_ib_warn(dev, "failed to free UAR 0x%x\n", uuari->uars[i].index);
|
mlx5_ib_warn(dev, "failed to free UAR 0x%x\n", uuari->uars[i].index);
|
||||||
|
@ -790,6 +1084,30 @@ static int mlx5_ib_mmap(struct ib_ucontext *ibcontext, struct vm_area_struct *vm
|
||||||
case MLX5_IB_MMAP_GET_CONTIGUOUS_PAGES:
|
case MLX5_IB_MMAP_GET_CONTIGUOUS_PAGES:
|
||||||
return -ENOSYS;
|
return -ENOSYS;
|
||||||
|
|
||||||
|
case MLX5_IB_MMAP_CORE_CLOCK:
|
||||||
|
if (vma->vm_end - vma->vm_start != PAGE_SIZE)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
if (vma->vm_flags & (VM_WRITE | VM_EXEC))
|
||||||
|
return -EPERM;
|
||||||
|
|
||||||
|
/* Don't expose to user-space information it shouldn't have */
|
||||||
|
if (PAGE_SIZE > 4096)
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
|
vma->vm_page_prot = pgprot_noncached(vma->vm_page_prot);
|
||||||
|
pfn = (dev->mdev->iseg_base +
|
||||||
|
offsetof(struct mlx5_init_seg, internal_timer_h)) >>
|
||||||
|
PAGE_SHIFT;
|
||||||
|
if (io_remap_pfn_range(vma, vma->vm_start, pfn,
|
||||||
|
PAGE_SIZE, vma->vm_page_prot))
|
||||||
|
return -EAGAIN;
|
||||||
|
|
||||||
|
mlx5_ib_dbg(dev, "mapped internal timer at 0x%lx, PA 0x%llx\n",
|
||||||
|
vma->vm_start,
|
||||||
|
(unsigned long long)pfn << PAGE_SHIFT);
|
||||||
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
@ -1758,6 +2076,32 @@ static void destroy_dev_resources(struct mlx5_ib_resources *devr)
|
||||||
mlx5_ib_dealloc_pd(devr->p0);
|
mlx5_ib_dealloc_pd(devr->p0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static u32 get_core_cap_flags(struct ib_device *ibdev)
|
||||||
|
{
|
||||||
|
struct mlx5_ib_dev *dev = to_mdev(ibdev);
|
||||||
|
enum rdma_link_layer ll = mlx5_ib_port_link_layer(ibdev, 1);
|
||||||
|
u8 l3_type_cap = MLX5_CAP_ROCE(dev->mdev, l3_type);
|
||||||
|
u8 roce_version_cap = MLX5_CAP_ROCE(dev->mdev, roce_version);
|
||||||
|
u32 ret = 0;
|
||||||
|
|
||||||
|
if (ll == IB_LINK_LAYER_INFINIBAND)
|
||||||
|
return RDMA_CORE_PORT_IBA_IB;
|
||||||
|
|
||||||
|
if (!(l3_type_cap & MLX5_ROCE_L3_TYPE_IPV4_CAP))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (!(l3_type_cap & MLX5_ROCE_L3_TYPE_IPV6_CAP))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (roce_version_cap & MLX5_ROCE_VERSION_1_CAP)
|
||||||
|
ret |= RDMA_CORE_PORT_IBA_ROCE;
|
||||||
|
|
||||||
|
if (roce_version_cap & MLX5_ROCE_VERSION_2_CAP)
|
||||||
|
ret |= RDMA_CORE_PORT_IBA_ROCE_UDP_ENCAP;
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
static int mlx5_port_immutable(struct ib_device *ibdev, u8 port_num,
|
static int mlx5_port_immutable(struct ib_device *ibdev, u8 port_num,
|
||||||
struct ib_port_immutable *immutable)
|
struct ib_port_immutable *immutable)
|
||||||
{
|
{
|
||||||
|
@ -1770,20 +2114,50 @@ static int mlx5_port_immutable(struct ib_device *ibdev, u8 port_num,
|
||||||
|
|
||||||
immutable->pkey_tbl_len = attr.pkey_tbl_len;
|
immutable->pkey_tbl_len = attr.pkey_tbl_len;
|
||||||
immutable->gid_tbl_len = attr.gid_tbl_len;
|
immutable->gid_tbl_len = attr.gid_tbl_len;
|
||||||
immutable->core_cap_flags = RDMA_CORE_PORT_IBA_IB;
|
immutable->core_cap_flags = get_core_cap_flags(ibdev);
|
||||||
immutable->max_mad_size = IB_MGMT_MAD_SIZE;
|
immutable->max_mad_size = IB_MGMT_MAD_SIZE;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int mlx5_enable_roce(struct mlx5_ib_dev *dev)
|
||||||
|
{
|
||||||
|
int err;
|
||||||
|
|
||||||
|
dev->roce.nb.notifier_call = mlx5_netdev_event;
|
||||||
|
err = register_netdevice_notifier(&dev->roce.nb);
|
||||||
|
if (err)
|
||||||
|
return err;
|
||||||
|
|
||||||
|
err = mlx5_nic_vport_enable_roce(dev->mdev);
|
||||||
|
if (err)
|
||||||
|
goto err_unregister_netdevice_notifier;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
err_unregister_netdevice_notifier:
|
||||||
|
unregister_netdevice_notifier(&dev->roce.nb);
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void mlx5_disable_roce(struct mlx5_ib_dev *dev)
|
||||||
|
{
|
||||||
|
mlx5_nic_vport_disable_roce(dev->mdev);
|
||||||
|
unregister_netdevice_notifier(&dev->roce.nb);
|
||||||
|
}
|
||||||
|
|
||||||
static void *mlx5_ib_add(struct mlx5_core_dev *mdev)
|
static void *mlx5_ib_add(struct mlx5_core_dev *mdev)
|
||||||
{
|
{
|
||||||
struct mlx5_ib_dev *dev;
|
struct mlx5_ib_dev *dev;
|
||||||
|
enum rdma_link_layer ll;
|
||||||
|
int port_type_cap;
|
||||||
int err;
|
int err;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
/* don't create IB instance over Eth ports, no RoCE yet! */
|
port_type_cap = MLX5_CAP_GEN(mdev, port_type);
|
||||||
if (MLX5_CAP_GEN(mdev, port_type) == MLX5_CAP_PORT_TYPE_ETH)
|
ll = mlx5_port_type_cap_to_rdma_ll(port_type_cap);
|
||||||
|
|
||||||
|
if ((ll == IB_LINK_LAYER_ETHERNET) && !MLX5_CAP_GEN(mdev, roce))
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
printk_once(KERN_INFO "%s", mlx5_version);
|
printk_once(KERN_INFO "%s", mlx5_version);
|
||||||
|
@ -1794,6 +2168,7 @@ static void *mlx5_ib_add(struct mlx5_core_dev *mdev)
|
||||||
|
|
||||||
dev->mdev = mdev;
|
dev->mdev = mdev;
|
||||||
|
|
||||||
|
rwlock_init(&dev->roce.netdev_lock);
|
||||||
err = get_port_caps(dev);
|
err = get_port_caps(dev);
|
||||||
if (err)
|
if (err)
|
||||||
goto err_dealloc;
|
goto err_dealloc;
|
||||||
|
@ -1843,7 +2218,12 @@ static void *mlx5_ib_add(struct mlx5_core_dev *mdev)
|
||||||
|
|
||||||
dev->ib_dev.query_device = mlx5_ib_query_device;
|
dev->ib_dev.query_device = mlx5_ib_query_device;
|
||||||
dev->ib_dev.query_port = mlx5_ib_query_port;
|
dev->ib_dev.query_port = mlx5_ib_query_port;
|
||||||
|
dev->ib_dev.get_link_layer = mlx5_ib_port_link_layer;
|
||||||
|
if (ll == IB_LINK_LAYER_ETHERNET)
|
||||||
|
dev->ib_dev.get_netdev = mlx5_ib_get_netdev;
|
||||||
dev->ib_dev.query_gid = mlx5_ib_query_gid;
|
dev->ib_dev.query_gid = mlx5_ib_query_gid;
|
||||||
|
dev->ib_dev.add_gid = mlx5_ib_add_gid;
|
||||||
|
dev->ib_dev.del_gid = mlx5_ib_del_gid;
|
||||||
dev->ib_dev.query_pkey = mlx5_ib_query_pkey;
|
dev->ib_dev.query_pkey = mlx5_ib_query_pkey;
|
||||||
dev->ib_dev.modify_device = mlx5_ib_modify_device;
|
dev->ib_dev.modify_device = mlx5_ib_modify_device;
|
||||||
dev->ib_dev.modify_port = mlx5_ib_modify_port;
|
dev->ib_dev.modify_port = mlx5_ib_modify_port;
|
||||||
|
@ -1893,7 +2273,7 @@ static void *mlx5_ib_add(struct mlx5_core_dev *mdev)
|
||||||
(1ull << IB_USER_VERBS_CMD_CLOSE_XRCD);
|
(1ull << IB_USER_VERBS_CMD_CLOSE_XRCD);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mlx5_ib_port_link_layer(&dev->ib_dev) ==
|
if (mlx5_ib_port_link_layer(&dev->ib_dev, 1) ==
|
||||||
IB_LINK_LAYER_ETHERNET) {
|
IB_LINK_LAYER_ETHERNET) {
|
||||||
dev->ib_dev.create_flow = mlx5_ib_create_flow;
|
dev->ib_dev.create_flow = mlx5_ib_create_flow;
|
||||||
dev->ib_dev.destroy_flow = mlx5_ib_destroy_flow;
|
dev->ib_dev.destroy_flow = mlx5_ib_destroy_flow;
|
||||||
|
@ -1908,9 +2288,15 @@ static void *mlx5_ib_add(struct mlx5_core_dev *mdev)
|
||||||
mutex_init(&dev->flow_db.lock);
|
mutex_init(&dev->flow_db.lock);
|
||||||
mutex_init(&dev->cap_mask_mutex);
|
mutex_init(&dev->cap_mask_mutex);
|
||||||
|
|
||||||
|
if (ll == IB_LINK_LAYER_ETHERNET) {
|
||||||
|
err = mlx5_enable_roce(dev);
|
||||||
|
if (err)
|
||||||
|
goto err_dealloc;
|
||||||
|
}
|
||||||
|
|
||||||
err = create_dev_resources(&dev->devr);
|
err = create_dev_resources(&dev->devr);
|
||||||
if (err)
|
if (err)
|
||||||
goto err_dealloc;
|
goto err_disable_roce;
|
||||||
|
|
||||||
err = mlx5_ib_odp_init_one(dev);
|
err = mlx5_ib_odp_init_one(dev);
|
||||||
if (err)
|
if (err)
|
||||||
|
@ -1947,6 +2333,10 @@ static void *mlx5_ib_add(struct mlx5_core_dev *mdev)
|
||||||
err_rsrc:
|
err_rsrc:
|
||||||
destroy_dev_resources(&dev->devr);
|
destroy_dev_resources(&dev->devr);
|
||||||
|
|
||||||
|
err_disable_roce:
|
||||||
|
if (ll == IB_LINK_LAYER_ETHERNET)
|
||||||
|
mlx5_disable_roce(dev);
|
||||||
|
|
||||||
err_dealloc:
|
err_dealloc:
|
||||||
ib_dealloc_device((struct ib_device *)dev);
|
ib_dealloc_device((struct ib_device *)dev);
|
||||||
|
|
||||||
|
@ -1956,11 +2346,14 @@ static void *mlx5_ib_add(struct mlx5_core_dev *mdev)
|
||||||
static void mlx5_ib_remove(struct mlx5_core_dev *mdev, void *context)
|
static void mlx5_ib_remove(struct mlx5_core_dev *mdev, void *context)
|
||||||
{
|
{
|
||||||
struct mlx5_ib_dev *dev = context;
|
struct mlx5_ib_dev *dev = context;
|
||||||
|
enum rdma_link_layer ll = mlx5_ib_port_link_layer(&dev->ib_dev, 1);
|
||||||
|
|
||||||
ib_unregister_device(&dev->ib_dev);
|
ib_unregister_device(&dev->ib_dev);
|
||||||
destroy_umrc_res(dev);
|
destroy_umrc_res(dev);
|
||||||
mlx5_ib_odp_remove_one(dev);
|
mlx5_ib_odp_remove_one(dev);
|
||||||
destroy_dev_resources(&dev->devr);
|
destroy_dev_resources(&dev->devr);
|
||||||
|
if (ll == IB_LINK_LAYER_ETHERNET)
|
||||||
|
mlx5_disable_roce(dev);
|
||||||
ib_dealloc_device(&dev->ib_dev);
|
ib_dealloc_device(&dev->ib_dev);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -42,6 +42,7 @@
|
||||||
#include <linux/mlx5/qp.h>
|
#include <linux/mlx5/qp.h>
|
||||||
#include <linux/mlx5/srq.h>
|
#include <linux/mlx5/srq.h>
|
||||||
#include <linux/types.h>
|
#include <linux/types.h>
|
||||||
|
#include <linux/mlx5/transobj.h>
|
||||||
|
|
||||||
#define mlx5_ib_dbg(dev, format, arg...) \
|
#define mlx5_ib_dbg(dev, format, arg...) \
|
||||||
pr_debug("%s:%s:%d:(pid %d): " format, (dev)->ib_dev.name, __func__, \
|
pr_debug("%s:%s:%d:(pid %d): " format, (dev)->ib_dev.name, __func__, \
|
||||||
|
@ -55,6 +56,11 @@ pr_err("%s:%s:%d:(pid %d): " format, (dev)->ib_dev.name, __func__, \
|
||||||
pr_warn("%s:%s:%d:(pid %d): " format, (dev)->ib_dev.name, __func__, \
|
pr_warn("%s:%s:%d:(pid %d): " format, (dev)->ib_dev.name, __func__, \
|
||||||
__LINE__, current->pid, ##arg)
|
__LINE__, current->pid, ##arg)
|
||||||
|
|
||||||
|
#define field_avail(type, fld, sz) (offsetof(type, fld) + \
|
||||||
|
sizeof(((type *)0)->fld) <= (sz))
|
||||||
|
#define MLX5_IB_DEFAULT_UIDX 0xffffff
|
||||||
|
#define MLX5_USER_ASSIGNED_UIDX_MASK __mlx5_mask(qpc, user_index)
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
MLX5_IB_MMAP_CMD_SHIFT = 8,
|
MLX5_IB_MMAP_CMD_SHIFT = 8,
|
||||||
MLX5_IB_MMAP_CMD_MASK = 0xff,
|
MLX5_IB_MMAP_CMD_MASK = 0xff,
|
||||||
|
@ -62,7 +68,9 @@ enum {
|
||||||
|
|
||||||
enum mlx5_ib_mmap_cmd {
|
enum mlx5_ib_mmap_cmd {
|
||||||
MLX5_IB_MMAP_REGULAR_PAGE = 0,
|
MLX5_IB_MMAP_REGULAR_PAGE = 0,
|
||||||
MLX5_IB_MMAP_GET_CONTIGUOUS_PAGES = 1, /* always last */
|
MLX5_IB_MMAP_GET_CONTIGUOUS_PAGES = 1,
|
||||||
|
/* 5 is chosen in order to be compatible with old versions of libmlx5 */
|
||||||
|
MLX5_IB_MMAP_CORE_CLOCK = 5,
|
||||||
};
|
};
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
|
@ -85,6 +93,15 @@ enum mlx5_ib_mad_ifc_flags {
|
||||||
MLX5_MAD_IFC_NET_VIEW = 4,
|
MLX5_MAD_IFC_NET_VIEW = 4,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
enum {
|
||||||
|
MLX5_CROSS_CHANNEL_UUAR = 0,
|
||||||
|
};
|
||||||
|
|
||||||
|
enum {
|
||||||
|
MLX5_CQE_VERSION_V0,
|
||||||
|
MLX5_CQE_VERSION_V1,
|
||||||
|
};
|
||||||
|
|
||||||
struct mlx5_ib_ucontext {
|
struct mlx5_ib_ucontext {
|
||||||
struct ib_ucontext ibucontext;
|
struct ib_ucontext ibucontext;
|
||||||
struct list_head db_page_list;
|
struct list_head db_page_list;
|
||||||
|
@ -93,6 +110,9 @@ struct mlx5_ib_ucontext {
|
||||||
*/
|
*/
|
||||||
struct mutex db_page_mutex;
|
struct mutex db_page_mutex;
|
||||||
struct mlx5_uuar_info uuari;
|
struct mlx5_uuar_info uuari;
|
||||||
|
u8 cqe_version;
|
||||||
|
/* Transport Domain number */
|
||||||
|
u32 tdn;
|
||||||
};
|
};
|
||||||
|
|
||||||
static inline struct mlx5_ib_ucontext *to_mucontext(struct ib_ucontext *ibucontext)
|
static inline struct mlx5_ib_ucontext *to_mucontext(struct ib_ucontext *ibucontext)
|
||||||
|
@ -201,47 +221,70 @@ struct mlx5_ib_pfault {
|
||||||
struct mlx5_pagefault mpfault;
|
struct mlx5_pagefault mpfault;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct mlx5_ib_ubuffer {
|
||||||
|
struct ib_umem *umem;
|
||||||
|
int buf_size;
|
||||||
|
u64 buf_addr;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct mlx5_ib_qp_base {
|
||||||
|
struct mlx5_ib_qp *container_mibqp;
|
||||||
|
struct mlx5_core_qp mqp;
|
||||||
|
struct mlx5_ib_ubuffer ubuffer;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct mlx5_ib_qp_trans {
|
||||||
|
struct mlx5_ib_qp_base base;
|
||||||
|
u16 xrcdn;
|
||||||
|
u8 alt_port;
|
||||||
|
u8 atomic_rd_en;
|
||||||
|
u8 resp_depth;
|
||||||
|
};
|
||||||
|
|
||||||
struct mlx5_ib_rq {
|
struct mlx5_ib_rq {
|
||||||
|
struct mlx5_ib_qp_base base;
|
||||||
|
struct mlx5_ib_wq *rq;
|
||||||
|
struct mlx5_ib_ubuffer ubuffer;
|
||||||
|
struct mlx5_db *doorbell;
|
||||||
u32 tirn;
|
u32 tirn;
|
||||||
|
u8 state;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct mlx5_ib_sq {
|
||||||
|
struct mlx5_ib_qp_base base;
|
||||||
|
struct mlx5_ib_wq *sq;
|
||||||
|
struct mlx5_ib_ubuffer ubuffer;
|
||||||
|
struct mlx5_db *doorbell;
|
||||||
|
u32 tisn;
|
||||||
|
u8 state;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct mlx5_ib_raw_packet_qp {
|
struct mlx5_ib_raw_packet_qp {
|
||||||
|
struct mlx5_ib_sq sq;
|
||||||
struct mlx5_ib_rq rq;
|
struct mlx5_ib_rq rq;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct mlx5_ib_qp {
|
struct mlx5_ib_qp {
|
||||||
struct ib_qp ibqp;
|
struct ib_qp ibqp;
|
||||||
union {
|
union {
|
||||||
struct mlx5_core_qp mqp;
|
struct mlx5_ib_qp_trans trans_qp;
|
||||||
struct mlx5_ib_raw_packet_qp raw_packet_qp;
|
struct mlx5_ib_raw_packet_qp raw_packet_qp;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct mlx5_buf buf;
|
struct mlx5_buf buf;
|
||||||
|
|
||||||
struct mlx5_db db;
|
struct mlx5_db db;
|
||||||
struct mlx5_ib_wq rq;
|
struct mlx5_ib_wq rq;
|
||||||
|
|
||||||
u32 doorbell_qpn;
|
|
||||||
u8 sq_signal_bits;
|
u8 sq_signal_bits;
|
||||||
u8 fm_cache;
|
u8 fm_cache;
|
||||||
int sq_max_wqes_per_wr;
|
|
||||||
int sq_spare_wqes;
|
|
||||||
struct mlx5_ib_wq sq;
|
struct mlx5_ib_wq sq;
|
||||||
|
|
||||||
struct ib_umem *umem;
|
|
||||||
int buf_size;
|
|
||||||
|
|
||||||
/* serialize qp state modifications
|
/* serialize qp state modifications
|
||||||
*/
|
*/
|
||||||
struct mutex mutex;
|
struct mutex mutex;
|
||||||
u16 xrcdn;
|
|
||||||
u32 flags;
|
u32 flags;
|
||||||
u8 port;
|
u8 port;
|
||||||
u8 alt_port;
|
|
||||||
u8 atomic_rd_en;
|
|
||||||
u8 resp_depth;
|
|
||||||
u8 state;
|
u8 state;
|
||||||
int mlx_type;
|
|
||||||
int wq_sig;
|
int wq_sig;
|
||||||
int scat_cqe;
|
int scat_cqe;
|
||||||
int max_inline_data;
|
int max_inline_data;
|
||||||
|
@ -284,6 +327,9 @@ struct mlx5_ib_cq_buf {
|
||||||
enum mlx5_ib_qp_flags {
|
enum mlx5_ib_qp_flags {
|
||||||
MLX5_IB_QP_BLOCK_MULTICAST_LOOPBACK = 1 << 0,
|
MLX5_IB_QP_BLOCK_MULTICAST_LOOPBACK = 1 << 0,
|
||||||
MLX5_IB_QP_SIGNATURE_HANDLING = 1 << 1,
|
MLX5_IB_QP_SIGNATURE_HANDLING = 1 << 1,
|
||||||
|
MLX5_IB_QP_CROSS_CHANNEL = 1 << 2,
|
||||||
|
MLX5_IB_QP_MANAGED_SEND = 1 << 3,
|
||||||
|
MLX5_IB_QP_MANAGED_RECV = 1 << 4,
|
||||||
};
|
};
|
||||||
|
|
||||||
struct mlx5_umr_wr {
|
struct mlx5_umr_wr {
|
||||||
|
@ -326,6 +372,7 @@ struct mlx5_ib_cq {
|
||||||
struct mlx5_ib_cq_buf *resize_buf;
|
struct mlx5_ib_cq_buf *resize_buf;
|
||||||
struct ib_umem *resize_umem;
|
struct ib_umem *resize_umem;
|
||||||
int cqe_size;
|
int cqe_size;
|
||||||
|
u32 create_flags;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct mlx5_ib_srq {
|
struct mlx5_ib_srq {
|
||||||
|
@ -449,9 +496,19 @@ struct mlx5_ib_resources {
|
||||||
struct ib_srq *s1;
|
struct ib_srq *s1;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct mlx5_roce {
|
||||||
|
/* Protect mlx5_ib_get_netdev from invoking dev_hold() with a NULL
|
||||||
|
* netdev pointer
|
||||||
|
*/
|
||||||
|
rwlock_t netdev_lock;
|
||||||
|
struct net_device *netdev;
|
||||||
|
struct notifier_block nb;
|
||||||
|
};
|
||||||
|
|
||||||
struct mlx5_ib_dev {
|
struct mlx5_ib_dev {
|
||||||
struct ib_device ib_dev;
|
struct ib_device ib_dev;
|
||||||
struct mlx5_core_dev *mdev;
|
struct mlx5_core_dev *mdev;
|
||||||
|
struct mlx5_roce roce;
|
||||||
MLX5_DECLARE_DOORBELL_LOCK(uar_lock);
|
MLX5_DECLARE_DOORBELL_LOCK(uar_lock);
|
||||||
int num_ports;
|
int num_ports;
|
||||||
/* serialize update of capability mask
|
/* serialize update of capability mask
|
||||||
|
@ -498,7 +555,7 @@ static inline struct mlx5_ib_cq *to_mcq(struct ib_cq *ibcq)
|
||||||
|
|
||||||
static inline struct mlx5_ib_qp *to_mibqp(struct mlx5_core_qp *mqp)
|
static inline struct mlx5_ib_qp *to_mibqp(struct mlx5_core_qp *mqp)
|
||||||
{
|
{
|
||||||
return container_of(mqp, struct mlx5_ib_qp, mqp);
|
return container_of(mqp, struct mlx5_ib_qp_base, mqp)->container_mibqp;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline struct mlx5_ib_mr *to_mibmr(struct mlx5_core_mr *mmr)
|
static inline struct mlx5_ib_mr *to_mibmr(struct mlx5_core_mr *mmr)
|
||||||
|
@ -550,8 +607,6 @@ void mlx5_ib_free_srq_wqe(struct mlx5_ib_srq *srq, int wqe_index);
|
||||||
int mlx5_MAD_IFC(struct mlx5_ib_dev *dev, int ignore_mkey, int ignore_bkey,
|
int mlx5_MAD_IFC(struct mlx5_ib_dev *dev, int ignore_mkey, int ignore_bkey,
|
||||||
u8 port, const struct ib_wc *in_wc, const struct ib_grh *in_grh,
|
u8 port, const struct ib_wc *in_wc, const struct ib_grh *in_grh,
|
||||||
const void *in_mad, void *response_mad);
|
const void *in_mad, void *response_mad);
|
||||||
struct ib_ah *create_ib_ah(struct ib_ah_attr *ah_attr,
|
|
||||||
struct mlx5_ib_ah *ah);
|
|
||||||
struct ib_ah *mlx5_ib_create_ah(struct ib_pd *pd, struct ib_ah_attr *ah_attr);
|
struct ib_ah *mlx5_ib_create_ah(struct ib_pd *pd, struct ib_ah_attr *ah_attr);
|
||||||
int mlx5_ib_query_ah(struct ib_ah *ibah, struct ib_ah_attr *ah_attr);
|
int mlx5_ib_query_ah(struct ib_ah *ibah, struct ib_ah_attr *ah_attr);
|
||||||
int mlx5_ib_destroy_ah(struct ib_ah *ah);
|
int mlx5_ib_destroy_ah(struct ib_ah *ah);
|
||||||
|
@ -578,7 +633,8 @@ int mlx5_ib_post_recv(struct ib_qp *ibqp, struct ib_recv_wr *wr,
|
||||||
struct ib_recv_wr **bad_wr);
|
struct ib_recv_wr **bad_wr);
|
||||||
void *mlx5_get_send_wqe(struct mlx5_ib_qp *qp, int n);
|
void *mlx5_get_send_wqe(struct mlx5_ib_qp *qp, int n);
|
||||||
int mlx5_ib_read_user_wqe(struct mlx5_ib_qp *qp, int send, int wqe_index,
|
int mlx5_ib_read_user_wqe(struct mlx5_ib_qp *qp, int send, int wqe_index,
|
||||||
void *buffer, u32 length);
|
void *buffer, u32 length,
|
||||||
|
struct mlx5_ib_qp_base *base);
|
||||||
struct ib_cq *mlx5_ib_create_cq(struct ib_device *ibdev,
|
struct ib_cq *mlx5_ib_create_cq(struct ib_device *ibdev,
|
||||||
const struct ib_cq_init_attr *attr,
|
const struct ib_cq_init_attr *attr,
|
||||||
struct ib_ucontext *context,
|
struct ib_ucontext *context,
|
||||||
|
@ -680,6 +736,9 @@ static inline void mlx5_ib_qp_enable_pagefaults(struct mlx5_ib_qp *qp) {}
|
||||||
|
|
||||||
#endif /* CONFIG_INFINIBAND_ON_DEMAND_PAGING */
|
#endif /* CONFIG_INFINIBAND_ON_DEMAND_PAGING */
|
||||||
|
|
||||||
|
__be16 mlx5_get_roce_udp_sport(struct mlx5_ib_dev *dev, u8 port_num,
|
||||||
|
int index);
|
||||||
|
|
||||||
static inline void init_query_mad(struct ib_smp *mad)
|
static inline void init_query_mad(struct ib_smp *mad)
|
||||||
{
|
{
|
||||||
mad->base_version = 1;
|
mad->base_version = 1;
|
||||||
|
@ -705,4 +764,28 @@ static inline int is_qp1(enum ib_qp_type qp_type)
|
||||||
#define MLX5_MAX_UMR_SHIFT 16
|
#define MLX5_MAX_UMR_SHIFT 16
|
||||||
#define MLX5_MAX_UMR_PAGES (1 << MLX5_MAX_UMR_SHIFT)
|
#define MLX5_MAX_UMR_PAGES (1 << MLX5_MAX_UMR_SHIFT)
|
||||||
|
|
||||||
|
static inline u32 check_cq_create_flags(u32 flags)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* It returns non-zero value for unsupported CQ
|
||||||
|
* create flags, otherwise it returns zero.
|
||||||
|
*/
|
||||||
|
return (flags & ~(IB_CQ_FLAGS_IGNORE_OVERRUN |
|
||||||
|
IB_CQ_FLAGS_TIMESTAMP_COMPLETION));
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline int verify_assign_uidx(u8 cqe_version, u32 cmd_uidx,
|
||||||
|
u32 *user_index)
|
||||||
|
{
|
||||||
|
if (cqe_version) {
|
||||||
|
if ((cmd_uidx == MLX5_IB_DEFAULT_UIDX) ||
|
||||||
|
(cmd_uidx & ~MLX5_USER_ASSIGNED_UIDX_MASK))
|
||||||
|
return -EINVAL;
|
||||||
|
*user_index = cmd_uidx;
|
||||||
|
} else {
|
||||||
|
*user_index = MLX5_IB_DEFAULT_UIDX;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
#endif /* MLX5_IB_H */
|
#endif /* MLX5_IB_H */
|
||||||
|
|
|
@ -153,14 +153,16 @@ static struct mlx5_ib_mr *mlx5_ib_odp_find_mr_lkey(struct mlx5_ib_dev *dev,
|
||||||
|
|
||||||
static void mlx5_ib_page_fault_resume(struct mlx5_ib_qp *qp,
|
static void mlx5_ib_page_fault_resume(struct mlx5_ib_qp *qp,
|
||||||
struct mlx5_ib_pfault *pfault,
|
struct mlx5_ib_pfault *pfault,
|
||||||
int error) {
|
int error)
|
||||||
|
{
|
||||||
struct mlx5_ib_dev *dev = to_mdev(qp->ibqp.pd->device);
|
struct mlx5_ib_dev *dev = to_mdev(qp->ibqp.pd->device);
|
||||||
int ret = mlx5_core_page_fault_resume(dev->mdev, qp->mqp.qpn,
|
u32 qpn = qp->trans_qp.base.mqp.qpn;
|
||||||
|
int ret = mlx5_core_page_fault_resume(dev->mdev,
|
||||||
|
qpn,
|
||||||
pfault->mpfault.flags,
|
pfault->mpfault.flags,
|
||||||
error);
|
error);
|
||||||
if (ret)
|
if (ret)
|
||||||
pr_err("Failed to resolve the page fault on QP 0x%x\n",
|
pr_err("Failed to resolve the page fault on QP 0x%x\n", qpn);
|
||||||
qp->mqp.qpn);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -391,6 +393,7 @@ static int mlx5_ib_mr_initiator_pfault_handler(
|
||||||
#if defined(DEBUG)
|
#if defined(DEBUG)
|
||||||
u32 ctrl_wqe_index, ctrl_qpn;
|
u32 ctrl_wqe_index, ctrl_qpn;
|
||||||
#endif
|
#endif
|
||||||
|
u32 qpn = qp->trans_qp.base.mqp.qpn;
|
||||||
|
|
||||||
ds = be32_to_cpu(ctrl->qpn_ds) & MLX5_WQE_CTRL_DS_MASK;
|
ds = be32_to_cpu(ctrl->qpn_ds) & MLX5_WQE_CTRL_DS_MASK;
|
||||||
if (ds * MLX5_WQE_DS_UNITS > wqe_length) {
|
if (ds * MLX5_WQE_DS_UNITS > wqe_length) {
|
||||||
|
@ -401,7 +404,7 @@ static int mlx5_ib_mr_initiator_pfault_handler(
|
||||||
|
|
||||||
if (ds == 0) {
|
if (ds == 0) {
|
||||||
mlx5_ib_err(dev, "Got WQE with zero DS. wqe_index=%x, qpn=%x\n",
|
mlx5_ib_err(dev, "Got WQE with zero DS. wqe_index=%x, qpn=%x\n",
|
||||||
wqe_index, qp->mqp.qpn);
|
wqe_index, qpn);
|
||||||
return -EFAULT;
|
return -EFAULT;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -411,16 +414,16 @@ static int mlx5_ib_mr_initiator_pfault_handler(
|
||||||
MLX5_WQE_CTRL_WQE_INDEX_SHIFT;
|
MLX5_WQE_CTRL_WQE_INDEX_SHIFT;
|
||||||
if (wqe_index != ctrl_wqe_index) {
|
if (wqe_index != ctrl_wqe_index) {
|
||||||
mlx5_ib_err(dev, "Got WQE with invalid wqe_index. wqe_index=0x%x, qpn=0x%x ctrl->wqe_index=0x%x\n",
|
mlx5_ib_err(dev, "Got WQE with invalid wqe_index. wqe_index=0x%x, qpn=0x%x ctrl->wqe_index=0x%x\n",
|
||||||
wqe_index, qp->mqp.qpn,
|
wqe_index, qpn,
|
||||||
ctrl_wqe_index);
|
ctrl_wqe_index);
|
||||||
return -EFAULT;
|
return -EFAULT;
|
||||||
}
|
}
|
||||||
|
|
||||||
ctrl_qpn = (be32_to_cpu(ctrl->qpn_ds) & MLX5_WQE_CTRL_QPN_MASK) >>
|
ctrl_qpn = (be32_to_cpu(ctrl->qpn_ds) & MLX5_WQE_CTRL_QPN_MASK) >>
|
||||||
MLX5_WQE_CTRL_QPN_SHIFT;
|
MLX5_WQE_CTRL_QPN_SHIFT;
|
||||||
if (qp->mqp.qpn != ctrl_qpn) {
|
if (qpn != ctrl_qpn) {
|
||||||
mlx5_ib_err(dev, "Got WQE with incorrect QP number. wqe_index=0x%x, qpn=0x%x ctrl->qpn=0x%x\n",
|
mlx5_ib_err(dev, "Got WQE with incorrect QP number. wqe_index=0x%x, qpn=0x%x ctrl->qpn=0x%x\n",
|
||||||
wqe_index, qp->mqp.qpn,
|
wqe_index, qpn,
|
||||||
ctrl_qpn);
|
ctrl_qpn);
|
||||||
return -EFAULT;
|
return -EFAULT;
|
||||||
}
|
}
|
||||||
|
@ -537,6 +540,7 @@ static void mlx5_ib_mr_wqe_pfault_handler(struct mlx5_ib_qp *qp,
|
||||||
int resume_with_error = 0;
|
int resume_with_error = 0;
|
||||||
u16 wqe_index = pfault->mpfault.wqe.wqe_index;
|
u16 wqe_index = pfault->mpfault.wqe.wqe_index;
|
||||||
int requestor = pfault->mpfault.flags & MLX5_PFAULT_REQUESTOR;
|
int requestor = pfault->mpfault.flags & MLX5_PFAULT_REQUESTOR;
|
||||||
|
u32 qpn = qp->trans_qp.base.mqp.qpn;
|
||||||
|
|
||||||
buffer = (char *)__get_free_page(GFP_KERNEL);
|
buffer = (char *)__get_free_page(GFP_KERNEL);
|
||||||
if (!buffer) {
|
if (!buffer) {
|
||||||
|
@ -546,10 +550,10 @@ static void mlx5_ib_mr_wqe_pfault_handler(struct mlx5_ib_qp *qp,
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = mlx5_ib_read_user_wqe(qp, requestor, wqe_index, buffer,
|
ret = mlx5_ib_read_user_wqe(qp, requestor, wqe_index, buffer,
|
||||||
PAGE_SIZE);
|
PAGE_SIZE, &qp->trans_qp.base);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
mlx5_ib_err(dev, "Failed reading a WQE following page fault, error=%x, wqe_index=%x, qpn=%x\n",
|
mlx5_ib_err(dev, "Failed reading a WQE following page fault, error=%x, wqe_index=%x, qpn=%x\n",
|
||||||
-ret, wqe_index, qp->mqp.qpn);
|
-ret, wqe_index, qpn);
|
||||||
resume_with_error = 1;
|
resume_with_error = 1;
|
||||||
goto resolve_page_fault;
|
goto resolve_page_fault;
|
||||||
}
|
}
|
||||||
|
@ -586,7 +590,8 @@ static void mlx5_ib_mr_wqe_pfault_handler(struct mlx5_ib_qp *qp,
|
||||||
resolve_page_fault:
|
resolve_page_fault:
|
||||||
mlx5_ib_page_fault_resume(qp, pfault, resume_with_error);
|
mlx5_ib_page_fault_resume(qp, pfault, resume_with_error);
|
||||||
mlx5_ib_dbg(dev, "PAGE FAULT completed. QP 0x%x resume_with_error=%d, flags: 0x%x\n",
|
mlx5_ib_dbg(dev, "PAGE FAULT completed. QP 0x%x resume_with_error=%d, flags: 0x%x\n",
|
||||||
qp->mqp.qpn, resume_with_error, pfault->mpfault.flags);
|
qpn, resume_with_error,
|
||||||
|
pfault->mpfault.flags);
|
||||||
|
|
||||||
free_page((unsigned long)buffer);
|
free_page((unsigned long)buffer);
|
||||||
}
|
}
|
||||||
|
@ -753,7 +758,7 @@ void mlx5_ib_odp_create_qp(struct mlx5_ib_qp *qp)
|
||||||
qp->disable_page_faults = 1;
|
qp->disable_page_faults = 1;
|
||||||
spin_lock_init(&qp->disable_page_faults_lock);
|
spin_lock_init(&qp->disable_page_faults_lock);
|
||||||
|
|
||||||
qp->mqp.pfault_handler = mlx5_ib_pfault_handler;
|
qp->trans_qp.base.mqp.pfault_handler = mlx5_ib_pfault_handler;
|
||||||
|
|
||||||
for (i = 0; i < MLX5_IB_PAGEFAULT_CONTEXTS; ++i)
|
for (i = 0; i < MLX5_IB_PAGEFAULT_CONTEXTS; ++i)
|
||||||
INIT_WORK(&qp->pagefaults[i].work, mlx5_ib_qp_pfault_action);
|
INIT_WORK(&qp->pagefaults[i].work, mlx5_ib_qp_pfault_action);
|
||||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -78,28 +78,41 @@ static int create_srq_user(struct ib_pd *pd, struct mlx5_ib_srq *srq,
|
||||||
struct ib_udata *udata, int buf_size, int *inlen)
|
struct ib_udata *udata, int buf_size, int *inlen)
|
||||||
{
|
{
|
||||||
struct mlx5_ib_dev *dev = to_mdev(pd->device);
|
struct mlx5_ib_dev *dev = to_mdev(pd->device);
|
||||||
struct mlx5_ib_create_srq ucmd;
|
struct mlx5_ib_create_srq ucmd = {};
|
||||||
size_t ucmdlen;
|
size_t ucmdlen;
|
||||||
|
void *xsrqc;
|
||||||
int err;
|
int err;
|
||||||
int npages;
|
int npages;
|
||||||
int page_shift;
|
int page_shift;
|
||||||
int ncont;
|
int ncont;
|
||||||
u32 offset;
|
u32 offset;
|
||||||
|
u32 uidx = MLX5_IB_DEFAULT_UIDX;
|
||||||
|
int drv_data = udata->inlen - sizeof(struct ib_uverbs_cmd_hdr);
|
||||||
|
|
||||||
ucmdlen =
|
if (drv_data < 0)
|
||||||
(udata->inlen - sizeof(struct ib_uverbs_cmd_hdr) <
|
return -EINVAL;
|
||||||
sizeof(ucmd)) ? (sizeof(ucmd) -
|
|
||||||
sizeof(ucmd.reserved)) : sizeof(ucmd);
|
ucmdlen = (drv_data < sizeof(ucmd)) ?
|
||||||
|
drv_data : sizeof(ucmd);
|
||||||
|
|
||||||
if (ib_copy_from_udata(&ucmd, udata, ucmdlen)) {
|
if (ib_copy_from_udata(&ucmd, udata, ucmdlen)) {
|
||||||
mlx5_ib_dbg(dev, "failed copy udata\n");
|
mlx5_ib_dbg(dev, "failed copy udata\n");
|
||||||
return -EFAULT;
|
return -EFAULT;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ucmdlen == sizeof(ucmd) &&
|
if (ucmd.reserved0 || ucmd.reserved1)
|
||||||
ucmd.reserved != 0)
|
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
|
if (drv_data > sizeof(ucmd) &&
|
||||||
|
!ib_is_udata_cleared(udata, sizeof(ucmd),
|
||||||
|
drv_data - sizeof(ucmd)))
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
err = get_srq_user_index(to_mucontext(pd->uobject->context),
|
||||||
|
&ucmd, udata->inlen, &uidx);
|
||||||
|
if (err)
|
||||||
|
return err;
|
||||||
|
|
||||||
srq->wq_sig = !!(ucmd.flags & MLX5_SRQ_FLAG_SIGNATURE);
|
srq->wq_sig = !!(ucmd.flags & MLX5_SRQ_FLAG_SIGNATURE);
|
||||||
|
|
||||||
srq->umem = ib_umem_get(pd->uobject->context, ucmd.buf_addr, buf_size,
|
srq->umem = ib_umem_get(pd->uobject->context, ucmd.buf_addr, buf_size,
|
||||||
|
@ -138,6 +151,12 @@ static int create_srq_user(struct ib_pd *pd, struct mlx5_ib_srq *srq,
|
||||||
(*in)->ctx.log_pg_sz = page_shift - MLX5_ADAPTER_PAGE_SHIFT;
|
(*in)->ctx.log_pg_sz = page_shift - MLX5_ADAPTER_PAGE_SHIFT;
|
||||||
(*in)->ctx.pgoff_cqn = cpu_to_be32(offset << 26);
|
(*in)->ctx.pgoff_cqn = cpu_to_be32(offset << 26);
|
||||||
|
|
||||||
|
if (MLX5_CAP_GEN(dev->mdev, cqe_version) == MLX5_CQE_VERSION_V1) {
|
||||||
|
xsrqc = MLX5_ADDR_OF(create_xrc_srq_in, *in,
|
||||||
|
xrc_srq_context_entry);
|
||||||
|
MLX5_SET(xrc_srqc, xsrqc, user_index, uidx);
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
err_in:
|
err_in:
|
||||||
|
@ -158,6 +177,7 @@ static int create_srq_kernel(struct mlx5_ib_dev *dev, struct mlx5_ib_srq *srq,
|
||||||
struct mlx5_wqe_srq_next_seg *next;
|
struct mlx5_wqe_srq_next_seg *next;
|
||||||
int page_shift;
|
int page_shift;
|
||||||
int npages;
|
int npages;
|
||||||
|
void *xsrqc;
|
||||||
|
|
||||||
err = mlx5_db_alloc(dev->mdev, &srq->db);
|
err = mlx5_db_alloc(dev->mdev, &srq->db);
|
||||||
if (err) {
|
if (err) {
|
||||||
|
@ -204,6 +224,13 @@ static int create_srq_kernel(struct mlx5_ib_dev *dev, struct mlx5_ib_srq *srq,
|
||||||
|
|
||||||
(*in)->ctx.log_pg_sz = page_shift - MLX5_ADAPTER_PAGE_SHIFT;
|
(*in)->ctx.log_pg_sz = page_shift - MLX5_ADAPTER_PAGE_SHIFT;
|
||||||
|
|
||||||
|
if (MLX5_CAP_GEN(dev->mdev, cqe_version) == MLX5_CQE_VERSION_V1) {
|
||||||
|
xsrqc = MLX5_ADDR_OF(create_xrc_srq_in, *in,
|
||||||
|
xrc_srq_context_entry);
|
||||||
|
/* 0xffffff means we ask to work with cqe version 0 */
|
||||||
|
MLX5_SET(xrc_srqc, xsrqc, user_index, MLX5_IB_DEFAULT_UIDX);
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
err_in:
|
err_in:
|
||||||
|
|
|
@ -35,6 +35,8 @@
|
||||||
|
|
||||||
#include <linux/types.h>
|
#include <linux/types.h>
|
||||||
|
|
||||||
|
#include "mlx5_ib.h"
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
MLX5_QP_FLAG_SIGNATURE = 1 << 0,
|
MLX5_QP_FLAG_SIGNATURE = 1 << 0,
|
||||||
MLX5_QP_FLAG_SCATTER_CQE = 1 << 1,
|
MLX5_QP_FLAG_SCATTER_CQE = 1 << 1,
|
||||||
|
@ -66,7 +68,15 @@ struct mlx5_ib_alloc_ucontext_req_v2 {
|
||||||
__u32 total_num_uuars;
|
__u32 total_num_uuars;
|
||||||
__u32 num_low_latency_uuars;
|
__u32 num_low_latency_uuars;
|
||||||
__u32 flags;
|
__u32 flags;
|
||||||
__u32 reserved;
|
__u32 comp_mask;
|
||||||
|
__u8 max_cqe_version;
|
||||||
|
__u8 reserved0;
|
||||||
|
__u16 reserved1;
|
||||||
|
__u32 reserved2;
|
||||||
|
};
|
||||||
|
|
||||||
|
enum mlx5_ib_alloc_ucontext_resp_mask {
|
||||||
|
MLX5_IB_ALLOC_UCONTEXT_RESP_MASK_CORE_CLOCK_OFFSET = 1UL << 0,
|
||||||
};
|
};
|
||||||
|
|
||||||
struct mlx5_ib_alloc_ucontext_resp {
|
struct mlx5_ib_alloc_ucontext_resp {
|
||||||
|
@ -80,7 +90,13 @@ struct mlx5_ib_alloc_ucontext_resp {
|
||||||
__u32 max_recv_wr;
|
__u32 max_recv_wr;
|
||||||
__u32 max_srq_recv_wr;
|
__u32 max_srq_recv_wr;
|
||||||
__u16 num_ports;
|
__u16 num_ports;
|
||||||
__u16 reserved;
|
__u16 reserved1;
|
||||||
|
__u32 comp_mask;
|
||||||
|
__u32 response_length;
|
||||||
|
__u8 cqe_version;
|
||||||
|
__u8 reserved2;
|
||||||
|
__u16 reserved3;
|
||||||
|
__u64 hca_core_clock_offset;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct mlx5_ib_alloc_pd_resp {
|
struct mlx5_ib_alloc_pd_resp {
|
||||||
|
@ -110,7 +126,9 @@ struct mlx5_ib_create_srq {
|
||||||
__u64 buf_addr;
|
__u64 buf_addr;
|
||||||
__u64 db_addr;
|
__u64 db_addr;
|
||||||
__u32 flags;
|
__u32 flags;
|
||||||
__u32 reserved; /* explicit padding (optional on i386) */
|
__u32 reserved0; /* explicit padding (optional on i386) */
|
||||||
|
__u32 uidx;
|
||||||
|
__u32 reserved1;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct mlx5_ib_create_srq_resp {
|
struct mlx5_ib_create_srq_resp {
|
||||||
|
@ -125,9 +143,48 @@ struct mlx5_ib_create_qp {
|
||||||
__u32 rq_wqe_count;
|
__u32 rq_wqe_count;
|
||||||
__u32 rq_wqe_shift;
|
__u32 rq_wqe_shift;
|
||||||
__u32 flags;
|
__u32 flags;
|
||||||
|
__u32 uidx;
|
||||||
|
__u32 reserved0;
|
||||||
|
__u64 sq_buf_addr;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct mlx5_ib_create_qp_resp {
|
struct mlx5_ib_create_qp_resp {
|
||||||
__u32 uuar_index;
|
__u32 uuar_index;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static inline int get_qp_user_index(struct mlx5_ib_ucontext *ucontext,
|
||||||
|
struct mlx5_ib_create_qp *ucmd,
|
||||||
|
int inlen,
|
||||||
|
u32 *user_index)
|
||||||
|
{
|
||||||
|
u8 cqe_version = ucontext->cqe_version;
|
||||||
|
|
||||||
|
if (field_avail(struct mlx5_ib_create_qp, uidx, inlen) &&
|
||||||
|
!cqe_version && (ucmd->uidx == MLX5_IB_DEFAULT_UIDX))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (!!(field_avail(struct mlx5_ib_create_qp, uidx, inlen) !=
|
||||||
|
!!cqe_version))
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
return verify_assign_uidx(cqe_version, ucmd->uidx, user_index);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline int get_srq_user_index(struct mlx5_ib_ucontext *ucontext,
|
||||||
|
struct mlx5_ib_create_srq *ucmd,
|
||||||
|
int inlen,
|
||||||
|
u32 *user_index)
|
||||||
|
{
|
||||||
|
u8 cqe_version = ucontext->cqe_version;
|
||||||
|
|
||||||
|
if (field_avail(struct mlx5_ib_create_srq, uidx, inlen) &&
|
||||||
|
!cqe_version && (ucmd->uidx == MLX5_IB_DEFAULT_UIDX))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (!!(field_avail(struct mlx5_ib_create_srq, uidx, inlen) !=
|
||||||
|
!!cqe_version))
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
return verify_assign_uidx(cqe_version, ucmd->uidx, user_index);
|
||||||
|
}
|
||||||
#endif /* MLX5_IB_USER_H */
|
#endif /* MLX5_IB_USER_H */
|
||||||
|
|
|
@ -608,9 +608,6 @@ static inline int mthca_poll_one(struct mthca_dev *dev,
|
||||||
entry->opcode = IB_WC_FETCH_ADD;
|
entry->opcode = IB_WC_FETCH_ADD;
|
||||||
entry->byte_len = MTHCA_ATOMIC_BYTE_LEN;
|
entry->byte_len = MTHCA_ATOMIC_BYTE_LEN;
|
||||||
break;
|
break;
|
||||||
case MTHCA_OPCODE_BIND_MW:
|
|
||||||
entry->opcode = IB_WC_BIND_MW;
|
|
||||||
break;
|
|
||||||
default:
|
default:
|
||||||
entry->opcode = MTHCA_OPCODE_INVALID;
|
entry->opcode = MTHCA_OPCODE_INVALID;
|
||||||
break;
|
break;
|
||||||
|
|
|
@ -898,89 +898,6 @@ static struct ib_mr *mthca_get_dma_mr(struct ib_pd *pd, int acc)
|
||||||
return &mr->ibmr;
|
return &mr->ibmr;
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct ib_mr *mthca_reg_phys_mr(struct ib_pd *pd,
|
|
||||||
struct ib_phys_buf *buffer_list,
|
|
||||||
int num_phys_buf,
|
|
||||||
int acc,
|
|
||||||
u64 *iova_start)
|
|
||||||
{
|
|
||||||
struct mthca_mr *mr;
|
|
||||||
u64 *page_list;
|
|
||||||
u64 total_size;
|
|
||||||
unsigned long mask;
|
|
||||||
int shift;
|
|
||||||
int npages;
|
|
||||||
int err;
|
|
||||||
int i, j, n;
|
|
||||||
|
|
||||||
mask = buffer_list[0].addr ^ *iova_start;
|
|
||||||
total_size = 0;
|
|
||||||
for (i = 0; i < num_phys_buf; ++i) {
|
|
||||||
if (i != 0)
|
|
||||||
mask |= buffer_list[i].addr;
|
|
||||||
if (i != num_phys_buf - 1)
|
|
||||||
mask |= buffer_list[i].addr + buffer_list[i].size;
|
|
||||||
|
|
||||||
total_size += buffer_list[i].size;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (mask & ~PAGE_MASK)
|
|
||||||
return ERR_PTR(-EINVAL);
|
|
||||||
|
|
||||||
shift = __ffs(mask | 1 << 31);
|
|
||||||
|
|
||||||
buffer_list[0].size += buffer_list[0].addr & ((1ULL << shift) - 1);
|
|
||||||
buffer_list[0].addr &= ~0ull << shift;
|
|
||||||
|
|
||||||
mr = kmalloc(sizeof *mr, GFP_KERNEL);
|
|
||||||
if (!mr)
|
|
||||||
return ERR_PTR(-ENOMEM);
|
|
||||||
|
|
||||||
npages = 0;
|
|
||||||
for (i = 0; i < num_phys_buf; ++i)
|
|
||||||
npages += (buffer_list[i].size + (1ULL << shift) - 1) >> shift;
|
|
||||||
|
|
||||||
if (!npages)
|
|
||||||
return &mr->ibmr;
|
|
||||||
|
|
||||||
page_list = kmalloc(npages * sizeof *page_list, GFP_KERNEL);
|
|
||||||
if (!page_list) {
|
|
||||||
kfree(mr);
|
|
||||||
return ERR_PTR(-ENOMEM);
|
|
||||||
}
|
|
||||||
|
|
||||||
n = 0;
|
|
||||||
for (i = 0; i < num_phys_buf; ++i)
|
|
||||||
for (j = 0;
|
|
||||||
j < (buffer_list[i].size + (1ULL << shift) - 1) >> shift;
|
|
||||||
++j)
|
|
||||||
page_list[n++] = buffer_list[i].addr + ((u64) j << shift);
|
|
||||||
|
|
||||||
mthca_dbg(to_mdev(pd->device), "Registering memory at %llx (iova %llx) "
|
|
||||||
"in PD %x; shift %d, npages %d.\n",
|
|
||||||
(unsigned long long) buffer_list[0].addr,
|
|
||||||
(unsigned long long) *iova_start,
|
|
||||||
to_mpd(pd)->pd_num,
|
|
||||||
shift, npages);
|
|
||||||
|
|
||||||
err = mthca_mr_alloc_phys(to_mdev(pd->device),
|
|
||||||
to_mpd(pd)->pd_num,
|
|
||||||
page_list, shift, npages,
|
|
||||||
*iova_start, total_size,
|
|
||||||
convert_access(acc), mr);
|
|
||||||
|
|
||||||
if (err) {
|
|
||||||
kfree(page_list);
|
|
||||||
kfree(mr);
|
|
||||||
return ERR_PTR(err);
|
|
||||||
}
|
|
||||||
|
|
||||||
kfree(page_list);
|
|
||||||
mr->umem = NULL;
|
|
||||||
|
|
||||||
return &mr->ibmr;
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct ib_mr *mthca_reg_user_mr(struct ib_pd *pd, u64 start, u64 length,
|
static struct ib_mr *mthca_reg_user_mr(struct ib_pd *pd, u64 start, u64 length,
|
||||||
u64 virt, int acc, struct ib_udata *udata)
|
u64 virt, int acc, struct ib_udata *udata)
|
||||||
{
|
{
|
||||||
|
@ -1346,7 +1263,6 @@ int mthca_register_device(struct mthca_dev *dev)
|
||||||
dev->ib_dev.destroy_cq = mthca_destroy_cq;
|
dev->ib_dev.destroy_cq = mthca_destroy_cq;
|
||||||
dev->ib_dev.poll_cq = mthca_poll_cq;
|
dev->ib_dev.poll_cq = mthca_poll_cq;
|
||||||
dev->ib_dev.get_dma_mr = mthca_get_dma_mr;
|
dev->ib_dev.get_dma_mr = mthca_get_dma_mr;
|
||||||
dev->ib_dev.reg_phys_mr = mthca_reg_phys_mr;
|
|
||||||
dev->ib_dev.reg_user_mr = mthca_reg_user_mr;
|
dev->ib_dev.reg_user_mr = mthca_reg_user_mr;
|
||||||
dev->ib_dev.dereg_mr = mthca_dereg_mr;
|
dev->ib_dev.dereg_mr = mthca_dereg_mr;
|
||||||
dev->ib_dev.get_port_immutable = mthca_port_immutable;
|
dev->ib_dev.get_port_immutable = mthca_port_immutable;
|
||||||
|
|
|
@ -1485,7 +1485,7 @@ static int build_mlx_header(struct mthca_dev *dev, struct mthca_sqp *sqp,
|
||||||
u16 pkey;
|
u16 pkey;
|
||||||
|
|
||||||
ib_ud_header_init(256, /* assume a MAD */ 1, 0, 0,
|
ib_ud_header_init(256, /* assume a MAD */ 1, 0, 0,
|
||||||
mthca_ah_grh_present(to_mah(wr->ah)), 0,
|
mthca_ah_grh_present(to_mah(wr->ah)), 0, 0, 0,
|
||||||
&sqp->ud_header);
|
&sqp->ud_header);
|
||||||
|
|
||||||
err = mthca_read_ah(dev, to_mah(wr->ah), &sqp->ud_header);
|
err = mthca_read_ah(dev, to_mah(wr->ah), &sqp->ud_header);
|
||||||
|
|
|
@ -134,7 +134,7 @@ static void record_ird_ord(struct nes_cm_node *, u16, u16);
|
||||||
/* External CM API Interface */
|
/* External CM API Interface */
|
||||||
/* instance of function pointers for client API */
|
/* instance of function pointers for client API */
|
||||||
/* set address of this instance to cm_core->cm_ops at cm_core alloc */
|
/* set address of this instance to cm_core->cm_ops at cm_core alloc */
|
||||||
static struct nes_cm_ops nes_cm_api = {
|
static const struct nes_cm_ops nes_cm_api = {
|
||||||
mini_cm_accelerated,
|
mini_cm_accelerated,
|
||||||
mini_cm_listen,
|
mini_cm_listen,
|
||||||
mini_cm_del_listen,
|
mini_cm_del_listen,
|
||||||
|
@ -3232,7 +3232,6 @@ int nes_accept(struct iw_cm_id *cm_id, struct iw_cm_conn_param *conn_param)
|
||||||
int passive_state;
|
int passive_state;
|
||||||
struct nes_ib_device *nesibdev;
|
struct nes_ib_device *nesibdev;
|
||||||
struct ib_mr *ibmr = NULL;
|
struct ib_mr *ibmr = NULL;
|
||||||
struct ib_phys_buf ibphysbuf;
|
|
||||||
struct nes_pd *nespd;
|
struct nes_pd *nespd;
|
||||||
u64 tagged_offset;
|
u64 tagged_offset;
|
||||||
u8 mpa_frame_offset = 0;
|
u8 mpa_frame_offset = 0;
|
||||||
|
@ -3316,21 +3315,19 @@ int nes_accept(struct iw_cm_id *cm_id, struct iw_cm_conn_param *conn_param)
|
||||||
u64temp = (unsigned long)nesqp;
|
u64temp = (unsigned long)nesqp;
|
||||||
nesibdev = nesvnic->nesibdev;
|
nesibdev = nesvnic->nesibdev;
|
||||||
nespd = nesqp->nespd;
|
nespd = nesqp->nespd;
|
||||||
ibphysbuf.addr = nesqp->ietf_frame_pbase + mpa_frame_offset;
|
|
||||||
ibphysbuf.size = buff_len;
|
|
||||||
tagged_offset = (u64)(unsigned long)*start_buff;
|
tagged_offset = (u64)(unsigned long)*start_buff;
|
||||||
ibmr = nesibdev->ibdev.reg_phys_mr((struct ib_pd *)nespd,
|
ibmr = nes_reg_phys_mr(&nespd->ibpd,
|
||||||
&ibphysbuf, 1,
|
nesqp->ietf_frame_pbase + mpa_frame_offset,
|
||||||
IB_ACCESS_LOCAL_WRITE,
|
buff_len, IB_ACCESS_LOCAL_WRITE,
|
||||||
&tagged_offset);
|
&tagged_offset);
|
||||||
if (!ibmr) {
|
if (IS_ERR(ibmr)) {
|
||||||
nes_debug(NES_DBG_CM, "Unable to register memory region"
|
nes_debug(NES_DBG_CM, "Unable to register memory region"
|
||||||
"for lSMM for cm_node = %p \n",
|
"for lSMM for cm_node = %p \n",
|
||||||
cm_node);
|
cm_node);
|
||||||
pci_free_consistent(nesdev->pcidev,
|
pci_free_consistent(nesdev->pcidev,
|
||||||
nesqp->private_data_len + nesqp->ietf_frame_size,
|
nesqp->private_data_len + nesqp->ietf_frame_size,
|
||||||
nesqp->ietf_frame, nesqp->ietf_frame_pbase);
|
nesqp->ietf_frame, nesqp->ietf_frame_pbase);
|
||||||
return -ENOMEM;
|
return PTR_ERR(ibmr);
|
||||||
}
|
}
|
||||||
|
|
||||||
ibmr->pd = &nespd->ibpd;
|
ibmr->pd = &nespd->ibpd;
|
||||||
|
|
|
@ -423,7 +423,7 @@ struct nes_cm_core {
|
||||||
|
|
||||||
struct timer_list tcp_timer;
|
struct timer_list tcp_timer;
|
||||||
|
|
||||||
struct nes_cm_ops *api;
|
const struct nes_cm_ops *api;
|
||||||
|
|
||||||
int (*post_event)(struct nes_cm_event *event);
|
int (*post_event)(struct nes_cm_event *event);
|
||||||
atomic_t events_posted;
|
atomic_t events_posted;
|
||||||
|
|
|
@ -727,7 +727,7 @@ int nes_arp_table(struct nes_device *nesdev, u32 ip_addr, u8 *mac_addr, u32 acti
|
||||||
if (action == NES_ARP_DELETE) {
|
if (action == NES_ARP_DELETE) {
|
||||||
nes_debug(NES_DBG_NETDEV, "DELETE, arp_index=%d\n", arp_index);
|
nes_debug(NES_DBG_NETDEV, "DELETE, arp_index=%d\n", arp_index);
|
||||||
nesadapter->arp_table[arp_index].ip_addr = 0;
|
nesadapter->arp_table[arp_index].ip_addr = 0;
|
||||||
memset(nesadapter->arp_table[arp_index].mac_addr, 0x00, ETH_ALEN);
|
eth_zero_addr(nesadapter->arp_table[arp_index].mac_addr);
|
||||||
nes_free_resource(nesadapter, nesadapter->allocated_arps, arp_index);
|
nes_free_resource(nesadapter, nesadapter->allocated_arps, arp_index);
|
||||||
return arp_index;
|
return arp_index;
|
||||||
}
|
}
|
||||||
|
|
|
@ -206,80 +206,6 @@ static int nes_dealloc_mw(struct ib_mw *ibmw)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* nes_bind_mw
|
|
||||||
*/
|
|
||||||
static int nes_bind_mw(struct ib_qp *ibqp, struct ib_mw *ibmw,
|
|
||||||
struct ib_mw_bind *ibmw_bind)
|
|
||||||
{
|
|
||||||
u64 u64temp;
|
|
||||||
struct nes_vnic *nesvnic = to_nesvnic(ibqp->device);
|
|
||||||
struct nes_device *nesdev = nesvnic->nesdev;
|
|
||||||
/* struct nes_mr *nesmr = to_nesmw(ibmw); */
|
|
||||||
struct nes_qp *nesqp = to_nesqp(ibqp);
|
|
||||||
struct nes_hw_qp_wqe *wqe;
|
|
||||||
unsigned long flags = 0;
|
|
||||||
u32 head;
|
|
||||||
u32 wqe_misc = 0;
|
|
||||||
u32 qsize;
|
|
||||||
|
|
||||||
if (nesqp->ibqp_state > IB_QPS_RTS)
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
spin_lock_irqsave(&nesqp->lock, flags);
|
|
||||||
|
|
||||||
head = nesqp->hwqp.sq_head;
|
|
||||||
qsize = nesqp->hwqp.sq_tail;
|
|
||||||
|
|
||||||
/* Check for SQ overflow */
|
|
||||||
if (((head + (2 * qsize) - nesqp->hwqp.sq_tail) % qsize) == (qsize - 1)) {
|
|
||||||
spin_unlock_irqrestore(&nesqp->lock, flags);
|
|
||||||
return -ENOMEM;
|
|
||||||
}
|
|
||||||
|
|
||||||
wqe = &nesqp->hwqp.sq_vbase[head];
|
|
||||||
/* nes_debug(NES_DBG_MR, "processing sq wqe at %p, head = %u.\n", wqe, head); */
|
|
||||||
nes_fill_init_qp_wqe(wqe, nesqp, head);
|
|
||||||
u64temp = ibmw_bind->wr_id;
|
|
||||||
set_wqe_64bit_value(wqe->wqe_words, NES_IWARP_SQ_WQE_COMP_SCRATCH_LOW_IDX, u64temp);
|
|
||||||
wqe_misc = NES_IWARP_SQ_OP_BIND;
|
|
||||||
|
|
||||||
wqe_misc |= NES_IWARP_SQ_WQE_LOCAL_FENCE;
|
|
||||||
|
|
||||||
if (ibmw_bind->send_flags & IB_SEND_SIGNALED)
|
|
||||||
wqe_misc |= NES_IWARP_SQ_WQE_SIGNALED_COMPL;
|
|
||||||
|
|
||||||
if (ibmw_bind->bind_info.mw_access_flags & IB_ACCESS_REMOTE_WRITE)
|
|
||||||
wqe_misc |= NES_CQP_STAG_RIGHTS_REMOTE_WRITE;
|
|
||||||
if (ibmw_bind->bind_info.mw_access_flags & IB_ACCESS_REMOTE_READ)
|
|
||||||
wqe_misc |= NES_CQP_STAG_RIGHTS_REMOTE_READ;
|
|
||||||
|
|
||||||
set_wqe_32bit_value(wqe->wqe_words, NES_IWARP_SQ_WQE_MISC_IDX, wqe_misc);
|
|
||||||
set_wqe_32bit_value(wqe->wqe_words, NES_IWARP_SQ_BIND_WQE_MR_IDX,
|
|
||||||
ibmw_bind->bind_info.mr->lkey);
|
|
||||||
set_wqe_32bit_value(wqe->wqe_words, NES_IWARP_SQ_BIND_WQE_MW_IDX, ibmw->rkey);
|
|
||||||
set_wqe_32bit_value(wqe->wqe_words, NES_IWARP_SQ_BIND_WQE_LENGTH_LOW_IDX,
|
|
||||||
ibmw_bind->bind_info.length);
|
|
||||||
wqe->wqe_words[NES_IWARP_SQ_BIND_WQE_LENGTH_HIGH_IDX] = 0;
|
|
||||||
u64temp = (u64)ibmw_bind->bind_info.addr;
|
|
||||||
set_wqe_64bit_value(wqe->wqe_words, NES_IWARP_SQ_BIND_WQE_VA_FBO_LOW_IDX, u64temp);
|
|
||||||
|
|
||||||
head++;
|
|
||||||
if (head >= qsize)
|
|
||||||
head = 0;
|
|
||||||
|
|
||||||
nesqp->hwqp.sq_head = head;
|
|
||||||
barrier();
|
|
||||||
|
|
||||||
nes_write32(nesdev->regs+NES_WQE_ALLOC,
|
|
||||||
(1 << 24) | 0x00800000 | nesqp->hwqp.qp_id);
|
|
||||||
|
|
||||||
spin_unlock_irqrestore(&nesqp->lock, flags);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* nes_alloc_fast_mr
|
* nes_alloc_fast_mr
|
||||||
*/
|
*/
|
||||||
|
@ -2074,9 +2000,8 @@ static int nes_reg_mr(struct nes_device *nesdev, struct nes_pd *nespd,
|
||||||
/**
|
/**
|
||||||
* nes_reg_phys_mr
|
* nes_reg_phys_mr
|
||||||
*/
|
*/
|
||||||
static struct ib_mr *nes_reg_phys_mr(struct ib_pd *ib_pd,
|
struct ib_mr *nes_reg_phys_mr(struct ib_pd *ib_pd, u64 addr, u64 size,
|
||||||
struct ib_phys_buf *buffer_list, int num_phys_buf, int acc,
|
int acc, u64 *iova_start)
|
||||||
u64 * iova_start)
|
|
||||||
{
|
{
|
||||||
u64 region_length;
|
u64 region_length;
|
||||||
struct nes_pd *nespd = to_nespd(ib_pd);
|
struct nes_pd *nespd = to_nespd(ib_pd);
|
||||||
|
@ -2088,13 +2013,10 @@ static struct ib_mr *nes_reg_phys_mr(struct ib_pd *ib_pd,
|
||||||
struct nes_vpbl vpbl;
|
struct nes_vpbl vpbl;
|
||||||
struct nes_root_vpbl root_vpbl;
|
struct nes_root_vpbl root_vpbl;
|
||||||
u32 stag;
|
u32 stag;
|
||||||
u32 i;
|
|
||||||
unsigned long mask;
|
unsigned long mask;
|
||||||
u32 stag_index = 0;
|
u32 stag_index = 0;
|
||||||
u32 next_stag_index = 0;
|
u32 next_stag_index = 0;
|
||||||
u32 driver_key = 0;
|
u32 driver_key = 0;
|
||||||
u32 root_pbl_index = 0;
|
|
||||||
u32 cur_pbl_index = 0;
|
|
||||||
int err = 0;
|
int err = 0;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
u16 pbl_count = 0;
|
u16 pbl_count = 0;
|
||||||
|
@ -2113,11 +2035,8 @@ static struct ib_mr *nes_reg_phys_mr(struct ib_pd *ib_pd,
|
||||||
|
|
||||||
next_stag_index >>= 8;
|
next_stag_index >>= 8;
|
||||||
next_stag_index %= nesadapter->max_mr;
|
next_stag_index %= nesadapter->max_mr;
|
||||||
if (num_phys_buf > (1024*512)) {
|
|
||||||
return ERR_PTR(-E2BIG);
|
|
||||||
}
|
|
||||||
|
|
||||||
if ((buffer_list[0].addr ^ *iova_start) & ~PAGE_MASK)
|
if ((addr ^ *iova_start) & ~PAGE_MASK)
|
||||||
return ERR_PTR(-EINVAL);
|
return ERR_PTR(-EINVAL);
|
||||||
|
|
||||||
err = nes_alloc_resource(nesadapter, nesadapter->allocated_mrs, nesadapter->max_mr,
|
err = nes_alloc_resource(nesadapter, nesadapter->allocated_mrs, nesadapter->max_mr,
|
||||||
|
@ -2132,84 +2051,33 @@ static struct ib_mr *nes_reg_phys_mr(struct ib_pd *ib_pd,
|
||||||
return ERR_PTR(-ENOMEM);
|
return ERR_PTR(-ENOMEM);
|
||||||
}
|
}
|
||||||
|
|
||||||
for (i = 0; i < num_phys_buf; i++) {
|
/* Allocate a 4K buffer for the PBL */
|
||||||
|
vpbl.pbl_vbase = pci_alloc_consistent(nesdev->pcidev, 4096,
|
||||||
if ((i & 0x01FF) == 0) {
|
&vpbl.pbl_pbase);
|
||||||
if (root_pbl_index == 1) {
|
nes_debug(NES_DBG_MR, "Allocating leaf PBL, va = %p, pa = 0x%016lX\n",
|
||||||
/* Allocate the root PBL */
|
vpbl.pbl_vbase, (unsigned long)vpbl.pbl_pbase);
|
||||||
root_vpbl.pbl_vbase = pci_alloc_consistent(nesdev->pcidev, 8192,
|
if (!vpbl.pbl_vbase) {
|
||||||
&root_vpbl.pbl_pbase);
|
nes_free_resource(nesadapter, nesadapter->allocated_mrs, stag_index);
|
||||||
nes_debug(NES_DBG_MR, "Allocating root PBL, va = %p, pa = 0x%08X\n",
|
ibmr = ERR_PTR(-ENOMEM);
|
||||||
root_vpbl.pbl_vbase, (unsigned int)root_vpbl.pbl_pbase);
|
kfree(nesmr);
|
||||||
if (!root_vpbl.pbl_vbase) {
|
goto reg_phys_err;
|
||||||
pci_free_consistent(nesdev->pcidev, 4096, vpbl.pbl_vbase,
|
|
||||||
vpbl.pbl_pbase);
|
|
||||||
nes_free_resource(nesadapter, nesadapter->allocated_mrs, stag_index);
|
|
||||||
kfree(nesmr);
|
|
||||||
return ERR_PTR(-ENOMEM);
|
|
||||||
}
|
|
||||||
root_vpbl.leaf_vpbl = kzalloc(sizeof(*root_vpbl.leaf_vpbl)*1024, GFP_KERNEL);
|
|
||||||
if (!root_vpbl.leaf_vpbl) {
|
|
||||||
pci_free_consistent(nesdev->pcidev, 8192, root_vpbl.pbl_vbase,
|
|
||||||
root_vpbl.pbl_pbase);
|
|
||||||
pci_free_consistent(nesdev->pcidev, 4096, vpbl.pbl_vbase,
|
|
||||||
vpbl.pbl_pbase);
|
|
||||||
nes_free_resource(nesadapter, nesadapter->allocated_mrs, stag_index);
|
|
||||||
kfree(nesmr);
|
|
||||||
return ERR_PTR(-ENOMEM);
|
|
||||||
}
|
|
||||||
root_vpbl.pbl_vbase[0].pa_low = cpu_to_le32((u32)vpbl.pbl_pbase);
|
|
||||||
root_vpbl.pbl_vbase[0].pa_high =
|
|
||||||
cpu_to_le32((u32)((((u64)vpbl.pbl_pbase) >> 32)));
|
|
||||||
root_vpbl.leaf_vpbl[0] = vpbl;
|
|
||||||
}
|
|
||||||
/* Allocate a 4K buffer for the PBL */
|
|
||||||
vpbl.pbl_vbase = pci_alloc_consistent(nesdev->pcidev, 4096,
|
|
||||||
&vpbl.pbl_pbase);
|
|
||||||
nes_debug(NES_DBG_MR, "Allocating leaf PBL, va = %p, pa = 0x%016lX\n",
|
|
||||||
vpbl.pbl_vbase, (unsigned long)vpbl.pbl_pbase);
|
|
||||||
if (!vpbl.pbl_vbase) {
|
|
||||||
nes_free_resource(nesadapter, nesadapter->allocated_mrs, stag_index);
|
|
||||||
ibmr = ERR_PTR(-ENOMEM);
|
|
||||||
kfree(nesmr);
|
|
||||||
goto reg_phys_err;
|
|
||||||
}
|
|
||||||
/* Fill in the root table */
|
|
||||||
if (1 <= root_pbl_index) {
|
|
||||||
root_vpbl.pbl_vbase[root_pbl_index].pa_low =
|
|
||||||
cpu_to_le32((u32)vpbl.pbl_pbase);
|
|
||||||
root_vpbl.pbl_vbase[root_pbl_index].pa_high =
|
|
||||||
cpu_to_le32((u32)((((u64)vpbl.pbl_pbase) >> 32)));
|
|
||||||
root_vpbl.leaf_vpbl[root_pbl_index] = vpbl;
|
|
||||||
}
|
|
||||||
root_pbl_index++;
|
|
||||||
cur_pbl_index = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
mask = !buffer_list[i].size;
|
|
||||||
if (i != 0)
|
|
||||||
mask |= buffer_list[i].addr;
|
|
||||||
if (i != num_phys_buf - 1)
|
|
||||||
mask |= buffer_list[i].addr + buffer_list[i].size;
|
|
||||||
|
|
||||||
if (mask & ~PAGE_MASK) {
|
|
||||||
nes_free_resource(nesadapter, nesadapter->allocated_mrs, stag_index);
|
|
||||||
nes_debug(NES_DBG_MR, "Invalid buffer addr or size\n");
|
|
||||||
ibmr = ERR_PTR(-EINVAL);
|
|
||||||
kfree(nesmr);
|
|
||||||
goto reg_phys_err;
|
|
||||||
}
|
|
||||||
|
|
||||||
region_length += buffer_list[i].size;
|
|
||||||
if ((i != 0) && (single_page)) {
|
|
||||||
if ((buffer_list[i-1].addr+PAGE_SIZE) != buffer_list[i].addr)
|
|
||||||
single_page = 0;
|
|
||||||
}
|
|
||||||
vpbl.pbl_vbase[cur_pbl_index].pa_low = cpu_to_le32((u32)buffer_list[i].addr & PAGE_MASK);
|
|
||||||
vpbl.pbl_vbase[cur_pbl_index++].pa_high =
|
|
||||||
cpu_to_le32((u32)((((u64)buffer_list[i].addr) >> 32)));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
mask = !size;
|
||||||
|
|
||||||
|
if (mask & ~PAGE_MASK) {
|
||||||
|
nes_free_resource(nesadapter, nesadapter->allocated_mrs, stag_index);
|
||||||
|
nes_debug(NES_DBG_MR, "Invalid buffer addr or size\n");
|
||||||
|
ibmr = ERR_PTR(-EINVAL);
|
||||||
|
kfree(nesmr);
|
||||||
|
goto reg_phys_err;
|
||||||
|
}
|
||||||
|
|
||||||
|
region_length += size;
|
||||||
|
vpbl.pbl_vbase[0].pa_low = cpu_to_le32((u32)addr & PAGE_MASK);
|
||||||
|
vpbl.pbl_vbase[0].pa_high = cpu_to_le32((u32)((((u64)addr) >> 32)));
|
||||||
|
|
||||||
stag = stag_index << 8;
|
stag = stag_index << 8;
|
||||||
stag |= driver_key;
|
stag |= driver_key;
|
||||||
stag += (u32)stag_key;
|
stag += (u32)stag_key;
|
||||||
|
@ -2219,17 +2087,15 @@ static struct ib_mr *nes_reg_phys_mr(struct ib_pd *ib_pd,
|
||||||
stag, (unsigned long)*iova_start, (unsigned long)region_length, stag_index);
|
stag, (unsigned long)*iova_start, (unsigned long)region_length, stag_index);
|
||||||
|
|
||||||
/* Make the leaf PBL the root if only one PBL */
|
/* Make the leaf PBL the root if only one PBL */
|
||||||
if (root_pbl_index == 1) {
|
root_vpbl.pbl_pbase = vpbl.pbl_pbase;
|
||||||
root_vpbl.pbl_pbase = vpbl.pbl_pbase;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (single_page) {
|
if (single_page) {
|
||||||
pbl_count = 0;
|
pbl_count = 0;
|
||||||
} else {
|
} else {
|
||||||
pbl_count = root_pbl_index;
|
pbl_count = 1;
|
||||||
}
|
}
|
||||||
ret = nes_reg_mr(nesdev, nespd, stag, region_length, &root_vpbl,
|
ret = nes_reg_mr(nesdev, nespd, stag, region_length, &root_vpbl,
|
||||||
buffer_list[0].addr, pbl_count, (u16)cur_pbl_index, acc, iova_start,
|
addr, pbl_count, 1, acc, iova_start,
|
||||||
&nesmr->pbls_used, &nesmr->pbl_4k);
|
&nesmr->pbls_used, &nesmr->pbl_4k);
|
||||||
|
|
||||||
if (ret == 0) {
|
if (ret == 0) {
|
||||||
|
@ -2242,21 +2108,9 @@ static struct ib_mr *nes_reg_phys_mr(struct ib_pd *ib_pd,
|
||||||
ibmr = ERR_PTR(-ENOMEM);
|
ibmr = ERR_PTR(-ENOMEM);
|
||||||
}
|
}
|
||||||
|
|
||||||
reg_phys_err:
|
reg_phys_err:
|
||||||
/* free the resources */
|
/* single PBL case */
|
||||||
if (root_pbl_index == 1) {
|
pci_free_consistent(nesdev->pcidev, 4096, vpbl.pbl_vbase, vpbl.pbl_pbase);
|
||||||
/* single PBL case */
|
|
||||||
pci_free_consistent(nesdev->pcidev, 4096, vpbl.pbl_vbase, vpbl.pbl_pbase);
|
|
||||||
} else {
|
|
||||||
for (i=0; i<root_pbl_index; i++) {
|
|
||||||
pci_free_consistent(nesdev->pcidev, 4096, root_vpbl.leaf_vpbl[i].pbl_vbase,
|
|
||||||
root_vpbl.leaf_vpbl[i].pbl_pbase);
|
|
||||||
}
|
|
||||||
kfree(root_vpbl.leaf_vpbl);
|
|
||||||
pci_free_consistent(nesdev->pcidev, 8192, root_vpbl.pbl_vbase,
|
|
||||||
root_vpbl.pbl_pbase);
|
|
||||||
}
|
|
||||||
|
|
||||||
return ibmr;
|
return ibmr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2266,17 +2120,13 @@ static struct ib_mr *nes_reg_phys_mr(struct ib_pd *ib_pd,
|
||||||
*/
|
*/
|
||||||
static struct ib_mr *nes_get_dma_mr(struct ib_pd *pd, int acc)
|
static struct ib_mr *nes_get_dma_mr(struct ib_pd *pd, int acc)
|
||||||
{
|
{
|
||||||
struct ib_phys_buf bl;
|
|
||||||
u64 kva = 0;
|
u64 kva = 0;
|
||||||
|
|
||||||
nes_debug(NES_DBG_MR, "\n");
|
nes_debug(NES_DBG_MR, "\n");
|
||||||
|
|
||||||
bl.size = (u64)0xffffffffffULL;
|
return nes_reg_phys_mr(pd, 0, 0xffffffffffULL, acc, &kva);
|
||||||
bl.addr = 0;
|
|
||||||
return nes_reg_phys_mr(pd, &bl, 1, acc, &kva);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* nes_reg_user_mr
|
* nes_reg_user_mr
|
||||||
*/
|
*/
|
||||||
|
@ -3888,12 +3738,10 @@ struct nes_ib_device *nes_init_ofa_device(struct net_device *netdev)
|
||||||
nesibdev->ibdev.destroy_cq = nes_destroy_cq;
|
nesibdev->ibdev.destroy_cq = nes_destroy_cq;
|
||||||
nesibdev->ibdev.poll_cq = nes_poll_cq;
|
nesibdev->ibdev.poll_cq = nes_poll_cq;
|
||||||
nesibdev->ibdev.get_dma_mr = nes_get_dma_mr;
|
nesibdev->ibdev.get_dma_mr = nes_get_dma_mr;
|
||||||
nesibdev->ibdev.reg_phys_mr = nes_reg_phys_mr;
|
|
||||||
nesibdev->ibdev.reg_user_mr = nes_reg_user_mr;
|
nesibdev->ibdev.reg_user_mr = nes_reg_user_mr;
|
||||||
nesibdev->ibdev.dereg_mr = nes_dereg_mr;
|
nesibdev->ibdev.dereg_mr = nes_dereg_mr;
|
||||||
nesibdev->ibdev.alloc_mw = nes_alloc_mw;
|
nesibdev->ibdev.alloc_mw = nes_alloc_mw;
|
||||||
nesibdev->ibdev.dealloc_mw = nes_dealloc_mw;
|
nesibdev->ibdev.dealloc_mw = nes_dealloc_mw;
|
||||||
nesibdev->ibdev.bind_mw = nes_bind_mw;
|
|
||||||
|
|
||||||
nesibdev->ibdev.alloc_mr = nes_alloc_mr;
|
nesibdev->ibdev.alloc_mr = nes_alloc_mr;
|
||||||
nesibdev->ibdev.map_mr_sg = nes_map_mr_sg;
|
nesibdev->ibdev.map_mr_sg = nes_map_mr_sg;
|
||||||
|
|
|
@ -190,4 +190,8 @@ struct nes_qp {
|
||||||
u8 pau_state;
|
u8 pau_state;
|
||||||
__u64 nesuqp_addr;
|
__u64 nesuqp_addr;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct ib_mr *nes_reg_phys_mr(struct ib_pd *ib_pd,
|
||||||
|
u64 addr, u64 size, int acc, u64 *iova_start);
|
||||||
|
|
||||||
#endif /* NES_VERBS_H */
|
#endif /* NES_VERBS_H */
|
||||||
|
|
|
@ -152,9 +152,10 @@ struct ib_ah *ocrdma_create_ah(struct ib_pd *ibpd, struct ib_ah_attr *attr)
|
||||||
if ((pd->uctx) &&
|
if ((pd->uctx) &&
|
||||||
(!rdma_is_multicast_addr((struct in6_addr *)attr->grh.dgid.raw)) &&
|
(!rdma_is_multicast_addr((struct in6_addr *)attr->grh.dgid.raw)) &&
|
||||||
(!rdma_link_local_addr((struct in6_addr *)attr->grh.dgid.raw))) {
|
(!rdma_link_local_addr((struct in6_addr *)attr->grh.dgid.raw))) {
|
||||||
status = rdma_addr_find_dmac_by_grh(&sgid, &attr->grh.dgid,
|
status = rdma_addr_find_l2_eth_by_grh(&sgid, &attr->grh.dgid,
|
||||||
attr->dmac, &vlan_tag,
|
attr->dmac, &vlan_tag,
|
||||||
sgid_attr.ndev->ifindex);
|
&sgid_attr.ndev->ifindex,
|
||||||
|
NULL);
|
||||||
if (status) {
|
if (status) {
|
||||||
pr_err("%s(): Failed to resolve dmac from gid."
|
pr_err("%s(): Failed to resolve dmac from gid."
|
||||||
"status = %d\n", __func__, status);
|
"status = %d\n", __func__, status);
|
||||||
|
|
|
@ -175,7 +175,6 @@ static int ocrdma_register_device(struct ocrdma_dev *dev)
|
||||||
dev->ibdev.req_notify_cq = ocrdma_arm_cq;
|
dev->ibdev.req_notify_cq = ocrdma_arm_cq;
|
||||||
|
|
||||||
dev->ibdev.get_dma_mr = ocrdma_get_dma_mr;
|
dev->ibdev.get_dma_mr = ocrdma_get_dma_mr;
|
||||||
dev->ibdev.reg_phys_mr = ocrdma_reg_kernel_mr;
|
|
||||||
dev->ibdev.dereg_mr = ocrdma_dereg_mr;
|
dev->ibdev.dereg_mr = ocrdma_dereg_mr;
|
||||||
dev->ibdev.reg_user_mr = ocrdma_reg_user_mr;
|
dev->ibdev.reg_user_mr = ocrdma_reg_user_mr;
|
||||||
|
|
||||||
|
|
|
@ -3066,169 +3066,6 @@ struct ib_mr *ocrdma_alloc_mr(struct ib_pd *ibpd,
|
||||||
return ERR_PTR(-ENOMEM);
|
return ERR_PTR(-ENOMEM);
|
||||||
}
|
}
|
||||||
|
|
||||||
#define MAX_KERNEL_PBE_SIZE 65536
|
|
||||||
static inline int count_kernel_pbes(struct ib_phys_buf *buf_list,
|
|
||||||
int buf_cnt, u32 *pbe_size)
|
|
||||||
{
|
|
||||||
u64 total_size = 0;
|
|
||||||
u64 buf_size = 0;
|
|
||||||
int i;
|
|
||||||
*pbe_size = roundup(buf_list[0].size, PAGE_SIZE);
|
|
||||||
*pbe_size = roundup_pow_of_two(*pbe_size);
|
|
||||||
|
|
||||||
/* find the smallest PBE size that we can have */
|
|
||||||
for (i = 0; i < buf_cnt; i++) {
|
|
||||||
/* first addr may not be page aligned, so ignore checking */
|
|
||||||
if ((i != 0) && ((buf_list[i].addr & ~PAGE_MASK) ||
|
|
||||||
(buf_list[i].size & ~PAGE_MASK))) {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* if configured PBE size is greater then the chosen one,
|
|
||||||
* reduce the PBE size.
|
|
||||||
*/
|
|
||||||
buf_size = roundup(buf_list[i].size, PAGE_SIZE);
|
|
||||||
/* pbe_size has to be even multiple of 4K 1,2,4,8...*/
|
|
||||||
buf_size = roundup_pow_of_two(buf_size);
|
|
||||||
if (*pbe_size > buf_size)
|
|
||||||
*pbe_size = buf_size;
|
|
||||||
|
|
||||||
total_size += buf_size;
|
|
||||||
}
|
|
||||||
*pbe_size = *pbe_size > MAX_KERNEL_PBE_SIZE ?
|
|
||||||
(MAX_KERNEL_PBE_SIZE) : (*pbe_size);
|
|
||||||
|
|
||||||
/* num_pbes = total_size / (*pbe_size); this is implemented below. */
|
|
||||||
|
|
||||||
return total_size >> ilog2(*pbe_size);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void build_kernel_pbes(struct ib_phys_buf *buf_list, int ib_buf_cnt,
|
|
||||||
u32 pbe_size, struct ocrdma_pbl *pbl_tbl,
|
|
||||||
struct ocrdma_hw_mr *hwmr)
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
int idx;
|
|
||||||
int pbes_per_buf = 0;
|
|
||||||
u64 buf_addr = 0;
|
|
||||||
int num_pbes;
|
|
||||||
struct ocrdma_pbe *pbe;
|
|
||||||
int total_num_pbes = 0;
|
|
||||||
|
|
||||||
if (!hwmr->num_pbes)
|
|
||||||
return;
|
|
||||||
|
|
||||||
pbe = (struct ocrdma_pbe *)pbl_tbl->va;
|
|
||||||
num_pbes = 0;
|
|
||||||
|
|
||||||
/* go through the OS phy regions & fill hw pbe entries into pbls. */
|
|
||||||
for (i = 0; i < ib_buf_cnt; i++) {
|
|
||||||
buf_addr = buf_list[i].addr;
|
|
||||||
pbes_per_buf =
|
|
||||||
roundup_pow_of_two(roundup(buf_list[i].size, PAGE_SIZE)) /
|
|
||||||
pbe_size;
|
|
||||||
hwmr->len += buf_list[i].size;
|
|
||||||
/* number of pbes can be more for one OS buf, when
|
|
||||||
* buffers are of different sizes.
|
|
||||||
* split the ib_buf to one or more pbes.
|
|
||||||
*/
|
|
||||||
for (idx = 0; idx < pbes_per_buf; idx++) {
|
|
||||||
/* we program always page aligned addresses,
|
|
||||||
* first unaligned address is taken care by fbo.
|
|
||||||
*/
|
|
||||||
if (i == 0) {
|
|
||||||
/* for non zero fbo, assign the
|
|
||||||
* start of the page.
|
|
||||||
*/
|
|
||||||
pbe->pa_lo =
|
|
||||||
cpu_to_le32((u32) (buf_addr & PAGE_MASK));
|
|
||||||
pbe->pa_hi =
|
|
||||||
cpu_to_le32((u32) upper_32_bits(buf_addr));
|
|
||||||
} else {
|
|
||||||
pbe->pa_lo =
|
|
||||||
cpu_to_le32((u32) (buf_addr & 0xffffffff));
|
|
||||||
pbe->pa_hi =
|
|
||||||
cpu_to_le32((u32) upper_32_bits(buf_addr));
|
|
||||||
}
|
|
||||||
buf_addr += pbe_size;
|
|
||||||
num_pbes += 1;
|
|
||||||
total_num_pbes += 1;
|
|
||||||
pbe++;
|
|
||||||
|
|
||||||
if (total_num_pbes == hwmr->num_pbes)
|
|
||||||
goto mr_tbl_done;
|
|
||||||
/* if the pbl is full storing the pbes,
|
|
||||||
* move to next pbl.
|
|
||||||
*/
|
|
||||||
if (num_pbes == (hwmr->pbl_size/sizeof(u64))) {
|
|
||||||
pbl_tbl++;
|
|
||||||
pbe = (struct ocrdma_pbe *)pbl_tbl->va;
|
|
||||||
num_pbes = 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
mr_tbl_done:
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
struct ib_mr *ocrdma_reg_kernel_mr(struct ib_pd *ibpd,
|
|
||||||
struct ib_phys_buf *buf_list,
|
|
||||||
int buf_cnt, int acc, u64 *iova_start)
|
|
||||||
{
|
|
||||||
int status = -ENOMEM;
|
|
||||||
struct ocrdma_mr *mr;
|
|
||||||
struct ocrdma_pd *pd = get_ocrdma_pd(ibpd);
|
|
||||||
struct ocrdma_dev *dev = get_ocrdma_dev(ibpd->device);
|
|
||||||
u32 num_pbes;
|
|
||||||
u32 pbe_size = 0;
|
|
||||||
|
|
||||||
if ((acc & IB_ACCESS_REMOTE_WRITE) && !(acc & IB_ACCESS_LOCAL_WRITE))
|
|
||||||
return ERR_PTR(-EINVAL);
|
|
||||||
|
|
||||||
mr = kzalloc(sizeof(*mr), GFP_KERNEL);
|
|
||||||
if (!mr)
|
|
||||||
return ERR_PTR(status);
|
|
||||||
|
|
||||||
num_pbes = count_kernel_pbes(buf_list, buf_cnt, &pbe_size);
|
|
||||||
if (num_pbes == 0) {
|
|
||||||
status = -EINVAL;
|
|
||||||
goto pbl_err;
|
|
||||||
}
|
|
||||||
status = ocrdma_get_pbl_info(dev, mr, num_pbes);
|
|
||||||
if (status)
|
|
||||||
goto pbl_err;
|
|
||||||
|
|
||||||
mr->hwmr.pbe_size = pbe_size;
|
|
||||||
mr->hwmr.fbo = *iova_start - (buf_list[0].addr & PAGE_MASK);
|
|
||||||
mr->hwmr.va = *iova_start;
|
|
||||||
mr->hwmr.local_rd = 1;
|
|
||||||
mr->hwmr.remote_wr = (acc & IB_ACCESS_REMOTE_WRITE) ? 1 : 0;
|
|
||||||
mr->hwmr.remote_rd = (acc & IB_ACCESS_REMOTE_READ) ? 1 : 0;
|
|
||||||
mr->hwmr.local_wr = (acc & IB_ACCESS_LOCAL_WRITE) ? 1 : 0;
|
|
||||||
mr->hwmr.remote_atomic = (acc & IB_ACCESS_REMOTE_ATOMIC) ? 1 : 0;
|
|
||||||
mr->hwmr.mw_bind = (acc & IB_ACCESS_MW_BIND) ? 1 : 0;
|
|
||||||
|
|
||||||
status = ocrdma_build_pbl_tbl(dev, &mr->hwmr);
|
|
||||||
if (status)
|
|
||||||
goto pbl_err;
|
|
||||||
build_kernel_pbes(buf_list, buf_cnt, pbe_size, mr->hwmr.pbl_table,
|
|
||||||
&mr->hwmr);
|
|
||||||
status = ocrdma_reg_mr(dev, &mr->hwmr, pd->id, acc);
|
|
||||||
if (status)
|
|
||||||
goto mbx_err;
|
|
||||||
|
|
||||||
mr->ibmr.lkey = mr->hwmr.lkey;
|
|
||||||
if (mr->hwmr.remote_wr || mr->hwmr.remote_rd)
|
|
||||||
mr->ibmr.rkey = mr->hwmr.lkey;
|
|
||||||
return &mr->ibmr;
|
|
||||||
|
|
||||||
mbx_err:
|
|
||||||
ocrdma_free_mr_pbl_tbl(dev, &mr->hwmr);
|
|
||||||
pbl_err:
|
|
||||||
kfree(mr);
|
|
||||||
return ERR_PTR(status);
|
|
||||||
}
|
|
||||||
|
|
||||||
static int ocrdma_set_page(struct ib_mr *ibmr, u64 addr)
|
static int ocrdma_set_page(struct ib_mr *ibmr, u64 addr)
|
||||||
{
|
{
|
||||||
struct ocrdma_mr *mr = get_ocrdma_mr(ibmr);
|
struct ocrdma_mr *mr = get_ocrdma_mr(ibmr);
|
||||||
|
|
|
@ -117,9 +117,6 @@ int ocrdma_post_srq_recv(struct ib_srq *, struct ib_recv_wr *,
|
||||||
|
|
||||||
int ocrdma_dereg_mr(struct ib_mr *);
|
int ocrdma_dereg_mr(struct ib_mr *);
|
||||||
struct ib_mr *ocrdma_get_dma_mr(struct ib_pd *, int acc);
|
struct ib_mr *ocrdma_get_dma_mr(struct ib_pd *, int acc);
|
||||||
struct ib_mr *ocrdma_reg_kernel_mr(struct ib_pd *,
|
|
||||||
struct ib_phys_buf *buffer_list,
|
|
||||||
int num_phys_buf, int acc, u64 *iova_start);
|
|
||||||
struct ib_mr *ocrdma_reg_user_mr(struct ib_pd *, u64 start, u64 length,
|
struct ib_mr *ocrdma_reg_user_mr(struct ib_pd *, u64 start, u64 length,
|
||||||
u64 virt, int acc, struct ib_udata *);
|
u64 virt, int acc, struct ib_udata *);
|
||||||
struct ib_mr *ocrdma_alloc_mr(struct ib_pd *pd,
|
struct ib_mr *ocrdma_alloc_mr(struct ib_pd *pd,
|
||||||
|
|
|
@ -150,10 +150,7 @@ static struct qib_mr *alloc_mr(int count, struct ib_pd *pd)
|
||||||
rval = init_qib_mregion(&mr->mr, pd, count);
|
rval = init_qib_mregion(&mr->mr, pd, count);
|
||||||
if (rval)
|
if (rval)
|
||||||
goto bail;
|
goto bail;
|
||||||
/*
|
|
||||||
* ib_reg_phys_mr() will initialize mr->ibmr except for
|
|
||||||
* lkey and rkey.
|
|
||||||
*/
|
|
||||||
rval = qib_alloc_lkey(&mr->mr, 0);
|
rval = qib_alloc_lkey(&mr->mr, 0);
|
||||||
if (rval)
|
if (rval)
|
||||||
goto bail_mregion;
|
goto bail_mregion;
|
||||||
|
@ -170,52 +167,6 @@ static struct qib_mr *alloc_mr(int count, struct ib_pd *pd)
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* qib_reg_phys_mr - register a physical memory region
|
|
||||||
* @pd: protection domain for this memory region
|
|
||||||
* @buffer_list: pointer to the list of physical buffers to register
|
|
||||||
* @num_phys_buf: the number of physical buffers to register
|
|
||||||
* @iova_start: the starting address passed over IB which maps to this MR
|
|
||||||
*
|
|
||||||
* Returns the memory region on success, otherwise returns an errno.
|
|
||||||
*/
|
|
||||||
struct ib_mr *qib_reg_phys_mr(struct ib_pd *pd,
|
|
||||||
struct ib_phys_buf *buffer_list,
|
|
||||||
int num_phys_buf, int acc, u64 *iova_start)
|
|
||||||
{
|
|
||||||
struct qib_mr *mr;
|
|
||||||
int n, m, i;
|
|
||||||
struct ib_mr *ret;
|
|
||||||
|
|
||||||
mr = alloc_mr(num_phys_buf, pd);
|
|
||||||
if (IS_ERR(mr)) {
|
|
||||||
ret = (struct ib_mr *)mr;
|
|
||||||
goto bail;
|
|
||||||
}
|
|
||||||
|
|
||||||
mr->mr.user_base = *iova_start;
|
|
||||||
mr->mr.iova = *iova_start;
|
|
||||||
mr->mr.access_flags = acc;
|
|
||||||
|
|
||||||
m = 0;
|
|
||||||
n = 0;
|
|
||||||
for (i = 0; i < num_phys_buf; i++) {
|
|
||||||
mr->mr.map[m]->segs[n].vaddr = (void *) buffer_list[i].addr;
|
|
||||||
mr->mr.map[m]->segs[n].length = buffer_list[i].size;
|
|
||||||
mr->mr.length += buffer_list[i].size;
|
|
||||||
n++;
|
|
||||||
if (n == QIB_SEGSZ) {
|
|
||||||
m++;
|
|
||||||
n = 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
ret = &mr->ibmr;
|
|
||||||
|
|
||||||
bail:
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* qib_reg_user_mr - register a userspace memory region
|
* qib_reg_user_mr - register a userspace memory region
|
||||||
* @pd: protection domain for this memory region
|
* @pd: protection domain for this memory region
|
||||||
|
|
|
@ -100,9 +100,10 @@ static u32 credit_table[31] = {
|
||||||
32768 /* 1E */
|
32768 /* 1E */
|
||||||
};
|
};
|
||||||
|
|
||||||
static void get_map_page(struct qib_qpn_table *qpt, struct qpn_map *map)
|
static void get_map_page(struct qib_qpn_table *qpt, struct qpn_map *map,
|
||||||
|
gfp_t gfp)
|
||||||
{
|
{
|
||||||
unsigned long page = get_zeroed_page(GFP_KERNEL);
|
unsigned long page = get_zeroed_page(gfp);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Free the page if someone raced with us installing it.
|
* Free the page if someone raced with us installing it.
|
||||||
|
@ -121,7 +122,7 @@ static void get_map_page(struct qib_qpn_table *qpt, struct qpn_map *map)
|
||||||
* zero/one for QP type IB_QPT_SMI/IB_QPT_GSI.
|
* zero/one for QP type IB_QPT_SMI/IB_QPT_GSI.
|
||||||
*/
|
*/
|
||||||
static int alloc_qpn(struct qib_devdata *dd, struct qib_qpn_table *qpt,
|
static int alloc_qpn(struct qib_devdata *dd, struct qib_qpn_table *qpt,
|
||||||
enum ib_qp_type type, u8 port)
|
enum ib_qp_type type, u8 port, gfp_t gfp)
|
||||||
{
|
{
|
||||||
u32 i, offset, max_scan, qpn;
|
u32 i, offset, max_scan, qpn;
|
||||||
struct qpn_map *map;
|
struct qpn_map *map;
|
||||||
|
@ -151,7 +152,7 @@ static int alloc_qpn(struct qib_devdata *dd, struct qib_qpn_table *qpt,
|
||||||
max_scan = qpt->nmaps - !offset;
|
max_scan = qpt->nmaps - !offset;
|
||||||
for (i = 0;;) {
|
for (i = 0;;) {
|
||||||
if (unlikely(!map->page)) {
|
if (unlikely(!map->page)) {
|
||||||
get_map_page(qpt, map);
|
get_map_page(qpt, map, gfp);
|
||||||
if (unlikely(!map->page))
|
if (unlikely(!map->page))
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -983,13 +984,21 @@ struct ib_qp *qib_create_qp(struct ib_pd *ibpd,
|
||||||
size_t sz;
|
size_t sz;
|
||||||
size_t sg_list_sz;
|
size_t sg_list_sz;
|
||||||
struct ib_qp *ret;
|
struct ib_qp *ret;
|
||||||
|
gfp_t gfp;
|
||||||
|
|
||||||
|
|
||||||
if (init_attr->cap.max_send_sge > ib_qib_max_sges ||
|
if (init_attr->cap.max_send_sge > ib_qib_max_sges ||
|
||||||
init_attr->cap.max_send_wr > ib_qib_max_qp_wrs ||
|
init_attr->cap.max_send_wr > ib_qib_max_qp_wrs ||
|
||||||
init_attr->create_flags) {
|
init_attr->create_flags & ~(IB_QP_CREATE_USE_GFP_NOIO))
|
||||||
ret = ERR_PTR(-EINVAL);
|
return ERR_PTR(-EINVAL);
|
||||||
goto bail;
|
|
||||||
}
|
/* GFP_NOIO is applicable in RC QPs only */
|
||||||
|
if (init_attr->create_flags & IB_QP_CREATE_USE_GFP_NOIO &&
|
||||||
|
init_attr->qp_type != IB_QPT_RC)
|
||||||
|
return ERR_PTR(-EINVAL);
|
||||||
|
|
||||||
|
gfp = init_attr->create_flags & IB_QP_CREATE_USE_GFP_NOIO ?
|
||||||
|
GFP_NOIO : GFP_KERNEL;
|
||||||
|
|
||||||
/* Check receive queue parameters if no SRQ is specified. */
|
/* Check receive queue parameters if no SRQ is specified. */
|
||||||
if (!init_attr->srq) {
|
if (!init_attr->srq) {
|
||||||
|
@ -1021,7 +1030,8 @@ struct ib_qp *qib_create_qp(struct ib_pd *ibpd,
|
||||||
sz = sizeof(struct qib_sge) *
|
sz = sizeof(struct qib_sge) *
|
||||||
init_attr->cap.max_send_sge +
|
init_attr->cap.max_send_sge +
|
||||||
sizeof(struct qib_swqe);
|
sizeof(struct qib_swqe);
|
||||||
swq = vmalloc((init_attr->cap.max_send_wr + 1) * sz);
|
swq = __vmalloc((init_attr->cap.max_send_wr + 1) * sz,
|
||||||
|
gfp, PAGE_KERNEL);
|
||||||
if (swq == NULL) {
|
if (swq == NULL) {
|
||||||
ret = ERR_PTR(-ENOMEM);
|
ret = ERR_PTR(-ENOMEM);
|
||||||
goto bail;
|
goto bail;
|
||||||
|
@ -1037,13 +1047,13 @@ struct ib_qp *qib_create_qp(struct ib_pd *ibpd,
|
||||||
} else if (init_attr->cap.max_recv_sge > 1)
|
} else if (init_attr->cap.max_recv_sge > 1)
|
||||||
sg_list_sz = sizeof(*qp->r_sg_list) *
|
sg_list_sz = sizeof(*qp->r_sg_list) *
|
||||||
(init_attr->cap.max_recv_sge - 1);
|
(init_attr->cap.max_recv_sge - 1);
|
||||||
qp = kzalloc(sz + sg_list_sz, GFP_KERNEL);
|
qp = kzalloc(sz + sg_list_sz, gfp);
|
||||||
if (!qp) {
|
if (!qp) {
|
||||||
ret = ERR_PTR(-ENOMEM);
|
ret = ERR_PTR(-ENOMEM);
|
||||||
goto bail_swq;
|
goto bail_swq;
|
||||||
}
|
}
|
||||||
RCU_INIT_POINTER(qp->next, NULL);
|
RCU_INIT_POINTER(qp->next, NULL);
|
||||||
qp->s_hdr = kzalloc(sizeof(*qp->s_hdr), GFP_KERNEL);
|
qp->s_hdr = kzalloc(sizeof(*qp->s_hdr), gfp);
|
||||||
if (!qp->s_hdr) {
|
if (!qp->s_hdr) {
|
||||||
ret = ERR_PTR(-ENOMEM);
|
ret = ERR_PTR(-ENOMEM);
|
||||||
goto bail_qp;
|
goto bail_qp;
|
||||||
|
@ -1058,8 +1068,16 @@ struct ib_qp *qib_create_qp(struct ib_pd *ibpd,
|
||||||
qp->r_rq.max_sge = init_attr->cap.max_recv_sge;
|
qp->r_rq.max_sge = init_attr->cap.max_recv_sge;
|
||||||
sz = (sizeof(struct ib_sge) * qp->r_rq.max_sge) +
|
sz = (sizeof(struct ib_sge) * qp->r_rq.max_sge) +
|
||||||
sizeof(struct qib_rwqe);
|
sizeof(struct qib_rwqe);
|
||||||
qp->r_rq.wq = vmalloc_user(sizeof(struct qib_rwq) +
|
if (gfp != GFP_NOIO)
|
||||||
qp->r_rq.size * sz);
|
qp->r_rq.wq = vmalloc_user(
|
||||||
|
sizeof(struct qib_rwq) +
|
||||||
|
qp->r_rq.size * sz);
|
||||||
|
else
|
||||||
|
qp->r_rq.wq = __vmalloc(
|
||||||
|
sizeof(struct qib_rwq) +
|
||||||
|
qp->r_rq.size * sz,
|
||||||
|
gfp, PAGE_KERNEL);
|
||||||
|
|
||||||
if (!qp->r_rq.wq) {
|
if (!qp->r_rq.wq) {
|
||||||
ret = ERR_PTR(-ENOMEM);
|
ret = ERR_PTR(-ENOMEM);
|
||||||
goto bail_qp;
|
goto bail_qp;
|
||||||
|
@ -1090,7 +1108,7 @@ struct ib_qp *qib_create_qp(struct ib_pd *ibpd,
|
||||||
dev = to_idev(ibpd->device);
|
dev = to_idev(ibpd->device);
|
||||||
dd = dd_from_dev(dev);
|
dd = dd_from_dev(dev);
|
||||||
err = alloc_qpn(dd, &dev->qpn_table, init_attr->qp_type,
|
err = alloc_qpn(dd, &dev->qpn_table, init_attr->qp_type,
|
||||||
init_attr->port_num);
|
init_attr->port_num, gfp);
|
||||||
if (err < 0) {
|
if (err < 0) {
|
||||||
ret = ERR_PTR(err);
|
ret = ERR_PTR(err);
|
||||||
vfree(qp->r_rq.wq);
|
vfree(qp->r_rq.wq);
|
||||||
|
|
|
@ -346,6 +346,7 @@ static int qib_post_one_send(struct qib_qp *qp, struct ib_send_wr *wr,
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
struct qib_lkey_table *rkt;
|
struct qib_lkey_table *rkt;
|
||||||
struct qib_pd *pd;
|
struct qib_pd *pd;
|
||||||
|
int avoid_schedule = 0;
|
||||||
|
|
||||||
spin_lock_irqsave(&qp->s_lock, flags);
|
spin_lock_irqsave(&qp->s_lock, flags);
|
||||||
|
|
||||||
|
@ -438,11 +439,15 @@ static int qib_post_one_send(struct qib_qp *qp, struct ib_send_wr *wr,
|
||||||
qp->ibqp.qp_type == IB_QPT_RC) {
|
qp->ibqp.qp_type == IB_QPT_RC) {
|
||||||
if (wqe->length > 0x80000000U)
|
if (wqe->length > 0x80000000U)
|
||||||
goto bail_inval_free;
|
goto bail_inval_free;
|
||||||
|
if (wqe->length <= qp->pmtu)
|
||||||
|
avoid_schedule = 1;
|
||||||
} else if (wqe->length > (dd_from_ibdev(qp->ibqp.device)->pport +
|
} else if (wqe->length > (dd_from_ibdev(qp->ibqp.device)->pport +
|
||||||
qp->port_num - 1)->ibmtu)
|
qp->port_num - 1)->ibmtu) {
|
||||||
goto bail_inval_free;
|
goto bail_inval_free;
|
||||||
else
|
} else {
|
||||||
atomic_inc(&to_iah(ud_wr(wr)->ah)->refcount);
|
atomic_inc(&to_iah(ud_wr(wr)->ah)->refcount);
|
||||||
|
avoid_schedule = 1;
|
||||||
|
}
|
||||||
wqe->ssn = qp->s_ssn++;
|
wqe->ssn = qp->s_ssn++;
|
||||||
qp->s_head = next;
|
qp->s_head = next;
|
||||||
|
|
||||||
|
@ -458,7 +463,7 @@ static int qib_post_one_send(struct qib_qp *qp, struct ib_send_wr *wr,
|
||||||
bail_inval:
|
bail_inval:
|
||||||
ret = -EINVAL;
|
ret = -EINVAL;
|
||||||
bail:
|
bail:
|
||||||
if (!ret && !wr->next &&
|
if (!ret && !wr->next && !avoid_schedule &&
|
||||||
!qib_sdma_empty(
|
!qib_sdma_empty(
|
||||||
dd_from_ibdev(qp->ibqp.device)->pport + qp->port_num - 1)) {
|
dd_from_ibdev(qp->ibqp.device)->pport + qp->port_num - 1)) {
|
||||||
qib_schedule_send(qp);
|
qib_schedule_send(qp);
|
||||||
|
@ -2256,7 +2261,6 @@ int qib_register_ib_device(struct qib_devdata *dd)
|
||||||
ibdev->poll_cq = qib_poll_cq;
|
ibdev->poll_cq = qib_poll_cq;
|
||||||
ibdev->req_notify_cq = qib_req_notify_cq;
|
ibdev->req_notify_cq = qib_req_notify_cq;
|
||||||
ibdev->get_dma_mr = qib_get_dma_mr;
|
ibdev->get_dma_mr = qib_get_dma_mr;
|
||||||
ibdev->reg_phys_mr = qib_reg_phys_mr;
|
|
||||||
ibdev->reg_user_mr = qib_reg_user_mr;
|
ibdev->reg_user_mr = qib_reg_user_mr;
|
||||||
ibdev->dereg_mr = qib_dereg_mr;
|
ibdev->dereg_mr = qib_dereg_mr;
|
||||||
ibdev->alloc_mr = qib_alloc_mr;
|
ibdev->alloc_mr = qib_alloc_mr;
|
||||||
|
|
|
@ -1032,10 +1032,6 @@ int qib_resize_cq(struct ib_cq *ibcq, int cqe, struct ib_udata *udata);
|
||||||
|
|
||||||
struct ib_mr *qib_get_dma_mr(struct ib_pd *pd, int acc);
|
struct ib_mr *qib_get_dma_mr(struct ib_pd *pd, int acc);
|
||||||
|
|
||||||
struct ib_mr *qib_reg_phys_mr(struct ib_pd *pd,
|
|
||||||
struct ib_phys_buf *buffer_list,
|
|
||||||
int num_phys_buf, int acc, u64 *iova_start);
|
|
||||||
|
|
||||||
struct ib_mr *qib_reg_user_mr(struct ib_pd *pd, u64 start, u64 length,
|
struct ib_mr *qib_reg_user_mr(struct ib_pd *pd, u64 start, u64 length,
|
||||||
u64 virt_addr, int mr_access_flags,
|
u64 virt_addr, int mr_access_flags,
|
||||||
struct ib_udata *udata);
|
struct ib_udata *udata);
|
||||||
|
|
|
@ -286,15 +286,13 @@ int qib_multicast_detach(struct ib_qp *ibqp, union ib_gid *gid, u16 lid)
|
||||||
struct qib_ibdev *dev = to_idev(ibqp->device);
|
struct qib_ibdev *dev = to_idev(ibqp->device);
|
||||||
struct qib_ibport *ibp = to_iport(ibqp->device, qp->port_num);
|
struct qib_ibport *ibp = to_iport(ibqp->device, qp->port_num);
|
||||||
struct qib_mcast *mcast = NULL;
|
struct qib_mcast *mcast = NULL;
|
||||||
struct qib_mcast_qp *p, *tmp;
|
struct qib_mcast_qp *p, *tmp, *delp = NULL;
|
||||||
struct rb_node *n;
|
struct rb_node *n;
|
||||||
int last = 0;
|
int last = 0;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
if (ibqp->qp_num <= 1 || qp->state == IB_QPS_RESET) {
|
if (ibqp->qp_num <= 1 || qp->state == IB_QPS_RESET)
|
||||||
ret = -EINVAL;
|
return -EINVAL;
|
||||||
goto bail;
|
|
||||||
}
|
|
||||||
|
|
||||||
spin_lock_irq(&ibp->lock);
|
spin_lock_irq(&ibp->lock);
|
||||||
|
|
||||||
|
@ -303,8 +301,7 @@ int qib_multicast_detach(struct ib_qp *ibqp, union ib_gid *gid, u16 lid)
|
||||||
while (1) {
|
while (1) {
|
||||||
if (n == NULL) {
|
if (n == NULL) {
|
||||||
spin_unlock_irq(&ibp->lock);
|
spin_unlock_irq(&ibp->lock);
|
||||||
ret = -EINVAL;
|
return -EINVAL;
|
||||||
goto bail;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
mcast = rb_entry(n, struct qib_mcast, rb_node);
|
mcast = rb_entry(n, struct qib_mcast, rb_node);
|
||||||
|
@ -328,6 +325,7 @@ int qib_multicast_detach(struct ib_qp *ibqp, union ib_gid *gid, u16 lid)
|
||||||
*/
|
*/
|
||||||
list_del_rcu(&p->list);
|
list_del_rcu(&p->list);
|
||||||
mcast->n_attached--;
|
mcast->n_attached--;
|
||||||
|
delp = p;
|
||||||
|
|
||||||
/* If this was the last attached QP, remove the GID too. */
|
/* If this was the last attached QP, remove the GID too. */
|
||||||
if (list_empty(&mcast->qp_list)) {
|
if (list_empty(&mcast->qp_list)) {
|
||||||
|
@ -338,15 +336,16 @@ int qib_multicast_detach(struct ib_qp *ibqp, union ib_gid *gid, u16 lid)
|
||||||
}
|
}
|
||||||
|
|
||||||
spin_unlock_irq(&ibp->lock);
|
spin_unlock_irq(&ibp->lock);
|
||||||
|
/* QP not attached */
|
||||||
|
if (!delp)
|
||||||
|
return -EINVAL;
|
||||||
|
/*
|
||||||
|
* Wait for any list walkers to finish before freeing the
|
||||||
|
* list element.
|
||||||
|
*/
|
||||||
|
wait_event(mcast->wait, atomic_read(&mcast->refcount) <= 1);
|
||||||
|
qib_mcast_qp_free(delp);
|
||||||
|
|
||||||
if (p) {
|
|
||||||
/*
|
|
||||||
* Wait for any list walkers to finish before freeing the
|
|
||||||
* list element.
|
|
||||||
*/
|
|
||||||
wait_event(mcast->wait, atomic_read(&mcast->refcount) <= 1);
|
|
||||||
qib_mcast_qp_free(p);
|
|
||||||
}
|
|
||||||
if (last) {
|
if (last) {
|
||||||
atomic_dec(&mcast->refcount);
|
atomic_dec(&mcast->refcount);
|
||||||
wait_event(mcast->wait, !atomic_read(&mcast->refcount));
|
wait_event(mcast->wait, !atomic_read(&mcast->refcount));
|
||||||
|
@ -355,11 +354,7 @@ int qib_multicast_detach(struct ib_qp *ibqp, union ib_gid *gid, u16 lid)
|
||||||
dev->n_mcast_grps_allocated--;
|
dev->n_mcast_grps_allocated--;
|
||||||
spin_unlock_irq(&dev->n_mcast_grps_lock);
|
spin_unlock_irq(&dev->n_mcast_grps_lock);
|
||||||
}
|
}
|
||||||
|
return 0;
|
||||||
ret = 0;
|
|
||||||
|
|
||||||
bail:
|
|
||||||
return ret;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int qib_mcast_tree_empty(struct qib_ibport *ibp)
|
int qib_mcast_tree_empty(struct qib_ibport *ibp)
|
||||||
|
|
|
@ -157,8 +157,9 @@ void usnic_debugfs_flow_add(struct usnic_ib_qp_grp_flow *qp_flow)
|
||||||
qp_flow,
|
qp_flow,
|
||||||
&flowinfo_ops);
|
&flowinfo_ops);
|
||||||
if (IS_ERR_OR_NULL(qp_flow->dbgfs_dentry)) {
|
if (IS_ERR_OR_NULL(qp_flow->dbgfs_dentry)) {
|
||||||
usnic_err("Failed to create dbg fs entry for flow %u\n",
|
usnic_err("Failed to create dbg fs entry for flow %u with error %ld\n",
|
||||||
qp_flow->flow->flow_id);
|
qp_flow->flow->flow_id,
|
||||||
|
PTR_ERR(qp_flow->dbgfs_dentry));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -521,7 +521,7 @@ int usnic_ib_qp_grp_modify(struct usnic_ib_qp_grp *qp_grp,
|
||||||
|
|
||||||
if (!status) {
|
if (!status) {
|
||||||
qp_grp->state = new_state;
|
qp_grp->state = new_state;
|
||||||
usnic_info("Transistioned %u from %s to %s",
|
usnic_info("Transitioned %u from %s to %s",
|
||||||
qp_grp->grp_id,
|
qp_grp->grp_id,
|
||||||
usnic_ib_qp_grp_state_to_string(old_state),
|
usnic_ib_qp_grp_state_to_string(old_state),
|
||||||
usnic_ib_qp_grp_state_to_string(new_state));
|
usnic_ib_qp_grp_state_to_string(new_state));
|
||||||
|
@ -575,7 +575,7 @@ alloc_res_chunk_list(struct usnic_vnic *vnic,
|
||||||
return res_chunk_list;
|
return res_chunk_list;
|
||||||
|
|
||||||
out_free_res:
|
out_free_res:
|
||||||
for (i--; i > 0; i--)
|
for (i--; i >= 0; i--)
|
||||||
usnic_vnic_put_resources(res_chunk_list[i]);
|
usnic_vnic_put_resources(res_chunk_list[i]);
|
||||||
kfree(res_chunk_list);
|
kfree(res_chunk_list);
|
||||||
return ERR_PTR(err);
|
return ERR_PTR(err);
|
||||||
|
|
|
@ -51,7 +51,7 @@
|
||||||
|
|
||||||
static void usnic_ib_fw_string_to_u64(char *fw_ver_str, u64 *fw_ver)
|
static void usnic_ib_fw_string_to_u64(char *fw_ver_str, u64 *fw_ver)
|
||||||
{
|
{
|
||||||
*fw_ver = (u64) *fw_ver_str;
|
*fw_ver = *((u64 *)fw_ver_str);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int usnic_ib_fill_create_qp_resp(struct usnic_ib_qp_grp *qp_grp,
|
static int usnic_ib_fill_create_qp_resp(struct usnic_ib_qp_grp *qp_grp,
|
||||||
|
@ -571,20 +571,20 @@ int usnic_ib_modify_qp(struct ib_qp *ibqp, struct ib_qp_attr *attr,
|
||||||
|
|
||||||
qp_grp = to_uqp_grp(ibqp);
|
qp_grp = to_uqp_grp(ibqp);
|
||||||
|
|
||||||
/* TODO: Future Support All States */
|
|
||||||
mutex_lock(&qp_grp->vf->pf->usdev_lock);
|
mutex_lock(&qp_grp->vf->pf->usdev_lock);
|
||||||
if ((attr_mask & IB_QP_STATE) && attr->qp_state == IB_QPS_INIT) {
|
if ((attr_mask & IB_QP_PORT) && attr->port_num != 1) {
|
||||||
status = usnic_ib_qp_grp_modify(qp_grp, IB_QPS_INIT, NULL);
|
/* usnic devices only have one port */
|
||||||
} else if ((attr_mask & IB_QP_STATE) && attr->qp_state == IB_QPS_RTR) {
|
status = -EINVAL;
|
||||||
status = usnic_ib_qp_grp_modify(qp_grp, IB_QPS_RTR, NULL);
|
goto out_unlock;
|
||||||
} else if ((attr_mask & IB_QP_STATE) && attr->qp_state == IB_QPS_RTS) {
|
}
|
||||||
status = usnic_ib_qp_grp_modify(qp_grp, IB_QPS_RTS, NULL);
|
if (attr_mask & IB_QP_STATE) {
|
||||||
|
status = usnic_ib_qp_grp_modify(qp_grp, attr->qp_state, NULL);
|
||||||
} else {
|
} else {
|
||||||
usnic_err("Unexpected combination mask: %u state: %u\n",
|
usnic_err("Unhandled request, attr_mask=0x%x\n", attr_mask);
|
||||||
attr_mask & IB_QP_STATE, attr->qp_state);
|
|
||||||
status = -EINVAL;
|
status = -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
out_unlock:
|
||||||
mutex_unlock(&qp_grp->vf->pf->usdev_lock);
|
mutex_unlock(&qp_grp->vf->pf->usdev_lock);
|
||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
@ -625,8 +625,8 @@ struct ib_mr *usnic_ib_reg_mr(struct ib_pd *pd, u64 start, u64 length,
|
||||||
virt_addr, length);
|
virt_addr, length);
|
||||||
|
|
||||||
mr = kzalloc(sizeof(*mr), GFP_KERNEL);
|
mr = kzalloc(sizeof(*mr), GFP_KERNEL);
|
||||||
if (IS_ERR_OR_NULL(mr))
|
if (!mr)
|
||||||
return ERR_PTR(mr ? PTR_ERR(mr) : -ENOMEM);
|
return ERR_PTR(-ENOMEM);
|
||||||
|
|
||||||
mr->umem = usnic_uiom_reg_get(to_upd(pd)->umem_pd, start, length,
|
mr->umem = usnic_uiom_reg_get(to_upd(pd)->umem_pd, start, length,
|
||||||
access_flags, 0);
|
access_flags, 0);
|
||||||
|
|
|
@ -43,8 +43,6 @@ int usnic_ib_query_device(struct ib_device *ibdev,
|
||||||
struct ib_udata *uhw);
|
struct ib_udata *uhw);
|
||||||
int usnic_ib_query_port(struct ib_device *ibdev, u8 port,
|
int usnic_ib_query_port(struct ib_device *ibdev, u8 port,
|
||||||
struct ib_port_attr *props);
|
struct ib_port_attr *props);
|
||||||
enum rdma_protocol_type
|
|
||||||
usnic_ib_query_protocol(struct ib_device *device, u8 port_num);
|
|
||||||
int usnic_ib_query_qp(struct ib_qp *qp, struct ib_qp_attr *qp_attr,
|
int usnic_ib_query_qp(struct ib_qp *qp, struct ib_qp_attr *qp_attr,
|
||||||
int qp_attr_mask,
|
int qp_attr_mask,
|
||||||
struct ib_qp_init_attr *qp_init_attr);
|
struct ib_qp_init_attr *qp_init_attr);
|
||||||
|
|
|
@ -237,7 +237,7 @@ usnic_vnic_get_resources(struct usnic_vnic *vnic, enum usnic_vnic_res_type type,
|
||||||
struct usnic_vnic_res *res;
|
struct usnic_vnic_res *res;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
if (usnic_vnic_res_free_cnt(vnic, type) < cnt || cnt < 1 || !owner)
|
if (usnic_vnic_res_free_cnt(vnic, type) < cnt || cnt < 0 || !owner)
|
||||||
return ERR_PTR(-EINVAL);
|
return ERR_PTR(-EINVAL);
|
||||||
|
|
||||||
ret = kzalloc(sizeof(*ret), GFP_ATOMIC);
|
ret = kzalloc(sizeof(*ret), GFP_ATOMIC);
|
||||||
|
@ -247,26 +247,28 @@ usnic_vnic_get_resources(struct usnic_vnic *vnic, enum usnic_vnic_res_type type,
|
||||||
return ERR_PTR(-ENOMEM);
|
return ERR_PTR(-ENOMEM);
|
||||||
}
|
}
|
||||||
|
|
||||||
ret->res = kzalloc(sizeof(*(ret->res))*cnt, GFP_ATOMIC);
|
if (cnt > 0) {
|
||||||
if (!ret->res) {
|
ret->res = kcalloc(cnt, sizeof(*(ret->res)), GFP_ATOMIC);
|
||||||
usnic_err("Failed to allocate resources for %s. Out of memory\n",
|
if (!ret->res) {
|
||||||
usnic_vnic_pci_name(vnic));
|
usnic_err("Failed to allocate resources for %s. Out of memory\n",
|
||||||
kfree(ret);
|
usnic_vnic_pci_name(vnic));
|
||||||
return ERR_PTR(-ENOMEM);
|
kfree(ret);
|
||||||
}
|
return ERR_PTR(-ENOMEM);
|
||||||
|
|
||||||
spin_lock(&vnic->res_lock);
|
|
||||||
src = &vnic->chunks[type];
|
|
||||||
for (i = 0; i < src->cnt && ret->cnt < cnt; i++) {
|
|
||||||
res = src->res[i];
|
|
||||||
if (!res->owner) {
|
|
||||||
src->free_cnt--;
|
|
||||||
res->owner = owner;
|
|
||||||
ret->res[ret->cnt++] = res;
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
spin_unlock(&vnic->res_lock);
|
spin_lock(&vnic->res_lock);
|
||||||
|
src = &vnic->chunks[type];
|
||||||
|
for (i = 0; i < src->cnt && ret->cnt < cnt; i++) {
|
||||||
|
res = src->res[i];
|
||||||
|
if (!res->owner) {
|
||||||
|
src->free_cnt--;
|
||||||
|
res->owner = owner;
|
||||||
|
ret->res[ret->cnt++] = res;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
spin_unlock(&vnic->res_lock);
|
||||||
|
}
|
||||||
ret->type = type;
|
ret->type = type;
|
||||||
ret->vnic = vnic;
|
ret->vnic = vnic;
|
||||||
WARN_ON(ret->cnt != cnt);
|
WARN_ON(ret->cnt != cnt);
|
||||||
|
@ -281,14 +283,16 @@ void usnic_vnic_put_resources(struct usnic_vnic_res_chunk *chunk)
|
||||||
int i;
|
int i;
|
||||||
struct usnic_vnic *vnic = chunk->vnic;
|
struct usnic_vnic *vnic = chunk->vnic;
|
||||||
|
|
||||||
spin_lock(&vnic->res_lock);
|
if (chunk->cnt > 0) {
|
||||||
while ((i = --chunk->cnt) >= 0) {
|
spin_lock(&vnic->res_lock);
|
||||||
res = chunk->res[i];
|
while ((i = --chunk->cnt) >= 0) {
|
||||||
chunk->res[i] = NULL;
|
res = chunk->res[i];
|
||||||
res->owner = NULL;
|
chunk->res[i] = NULL;
|
||||||
vnic->chunks[res->type].free_cnt++;
|
res->owner = NULL;
|
||||||
|
vnic->chunks[res->type].free_cnt++;
|
||||||
|
}
|
||||||
|
spin_unlock(&vnic->res_lock);
|
||||||
}
|
}
|
||||||
spin_unlock(&vnic->res_lock);
|
|
||||||
|
|
||||||
kfree(chunk->res);
|
kfree(chunk->res);
|
||||||
kfree(chunk);
|
kfree(chunk);
|
||||||
|
|
|
@ -495,7 +495,6 @@ void ipoib_dev_cleanup(struct net_device *dev);
|
||||||
void ipoib_mcast_join_task(struct work_struct *work);
|
void ipoib_mcast_join_task(struct work_struct *work);
|
||||||
void ipoib_mcast_carrier_on_task(struct work_struct *work);
|
void ipoib_mcast_carrier_on_task(struct work_struct *work);
|
||||||
void ipoib_mcast_send(struct net_device *dev, u8 *daddr, struct sk_buff *skb);
|
void ipoib_mcast_send(struct net_device *dev, u8 *daddr, struct sk_buff *skb);
|
||||||
void ipoib_mcast_free(struct ipoib_mcast *mc);
|
|
||||||
|
|
||||||
void ipoib_mcast_restart_task(struct work_struct *work);
|
void ipoib_mcast_restart_task(struct work_struct *work);
|
||||||
int ipoib_mcast_start_thread(struct net_device *dev);
|
int ipoib_mcast_start_thread(struct net_device *dev);
|
||||||
|
@ -549,8 +548,9 @@ void ipoib_path_iter_read(struct ipoib_path_iter *iter,
|
||||||
|
|
||||||
int ipoib_mcast_attach(struct net_device *dev, u16 mlid,
|
int ipoib_mcast_attach(struct net_device *dev, u16 mlid,
|
||||||
union ib_gid *mgid, int set_qkey);
|
union ib_gid *mgid, int set_qkey);
|
||||||
int ipoib_mcast_leave(struct net_device *dev, struct ipoib_mcast *mcast);
|
void ipoib_mcast_remove_list(struct list_head *remove_list);
|
||||||
struct ipoib_mcast *__ipoib_mcast_find(struct net_device *dev, void *mgid);
|
void ipoib_check_and_add_mcast_sendonly(struct ipoib_dev_priv *priv, u8 *mgid,
|
||||||
|
struct list_head *remove_list);
|
||||||
|
|
||||||
int ipoib_init_qp(struct net_device *dev);
|
int ipoib_init_qp(struct net_device *dev);
|
||||||
int ipoib_transport_dev_init(struct net_device *dev, struct ib_device *ca);
|
int ipoib_transport_dev_init(struct net_device *dev, struct ib_device *ca);
|
||||||
|
|
|
@ -70,7 +70,6 @@ static struct ib_qp_attr ipoib_cm_err_attr = {
|
||||||
#define IPOIB_CM_RX_DRAIN_WRID 0xffffffff
|
#define IPOIB_CM_RX_DRAIN_WRID 0xffffffff
|
||||||
|
|
||||||
static struct ib_send_wr ipoib_cm_rx_drain_wr = {
|
static struct ib_send_wr ipoib_cm_rx_drain_wr = {
|
||||||
.wr_id = IPOIB_CM_RX_DRAIN_WRID,
|
|
||||||
.opcode = IB_WR_SEND,
|
.opcode = IB_WR_SEND,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -223,6 +222,7 @@ static void ipoib_cm_start_rx_drain(struct ipoib_dev_priv *priv)
|
||||||
* error" WC will be immediately generated for each WR we post.
|
* error" WC will be immediately generated for each WR we post.
|
||||||
*/
|
*/
|
||||||
p = list_entry(priv->cm.rx_flush_list.next, typeof(*p), list);
|
p = list_entry(priv->cm.rx_flush_list.next, typeof(*p), list);
|
||||||
|
ipoib_cm_rx_drain_wr.wr_id = IPOIB_CM_RX_DRAIN_WRID;
|
||||||
if (ib_post_send(p->qp, &ipoib_cm_rx_drain_wr, &bad_wr))
|
if (ib_post_send(p->qp, &ipoib_cm_rx_drain_wr, &bad_wr))
|
||||||
ipoib_warn(priv, "failed to post drain wr\n");
|
ipoib_warn(priv, "failed to post drain wr\n");
|
||||||
|
|
||||||
|
@ -1522,8 +1522,7 @@ static void ipoib_cm_create_srq(struct net_device *dev, int max_sge)
|
||||||
int ipoib_cm_dev_init(struct net_device *dev)
|
int ipoib_cm_dev_init(struct net_device *dev)
|
||||||
{
|
{
|
||||||
struct ipoib_dev_priv *priv = netdev_priv(dev);
|
struct ipoib_dev_priv *priv = netdev_priv(dev);
|
||||||
int i, ret;
|
int max_srq_sge, i;
|
||||||
struct ib_device_attr attr;
|
|
||||||
|
|
||||||
INIT_LIST_HEAD(&priv->cm.passive_ids);
|
INIT_LIST_HEAD(&priv->cm.passive_ids);
|
||||||
INIT_LIST_HEAD(&priv->cm.reap_list);
|
INIT_LIST_HEAD(&priv->cm.reap_list);
|
||||||
|
@ -1540,19 +1539,13 @@ int ipoib_cm_dev_init(struct net_device *dev)
|
||||||
|
|
||||||
skb_queue_head_init(&priv->cm.skb_queue);
|
skb_queue_head_init(&priv->cm.skb_queue);
|
||||||
|
|
||||||
ret = ib_query_device(priv->ca, &attr);
|
ipoib_dbg(priv, "max_srq_sge=%d\n", priv->ca->attrs.max_srq_sge);
|
||||||
if (ret) {
|
|
||||||
printk(KERN_WARNING "ib_query_device() failed with %d\n", ret);
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
ipoib_dbg(priv, "max_srq_sge=%d\n", attr.max_srq_sge);
|
max_srq_sge = min_t(int, IPOIB_CM_RX_SG, priv->ca->attrs.max_srq_sge);
|
||||||
|
ipoib_cm_create_srq(dev, max_srq_sge);
|
||||||
attr.max_srq_sge = min_t(int, IPOIB_CM_RX_SG, attr.max_srq_sge);
|
|
||||||
ipoib_cm_create_srq(dev, attr.max_srq_sge);
|
|
||||||
if (ipoib_cm_has_srq(dev)) {
|
if (ipoib_cm_has_srq(dev)) {
|
||||||
priv->cm.max_cm_mtu = attr.max_srq_sge * PAGE_SIZE - 0x10;
|
priv->cm.max_cm_mtu = max_srq_sge * PAGE_SIZE - 0x10;
|
||||||
priv->cm.num_frags = attr.max_srq_sge;
|
priv->cm.num_frags = max_srq_sge;
|
||||||
ipoib_dbg(priv, "max_cm_mtu = 0x%x, num_frags=%d\n",
|
ipoib_dbg(priv, "max_cm_mtu = 0x%x, num_frags=%d\n",
|
||||||
priv->cm.max_cm_mtu, priv->cm.num_frags);
|
priv->cm.max_cm_mtu, priv->cm.num_frags);
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -40,15 +40,11 @@ static void ipoib_get_drvinfo(struct net_device *netdev,
|
||||||
struct ethtool_drvinfo *drvinfo)
|
struct ethtool_drvinfo *drvinfo)
|
||||||
{
|
{
|
||||||
struct ipoib_dev_priv *priv = netdev_priv(netdev);
|
struct ipoib_dev_priv *priv = netdev_priv(netdev);
|
||||||
struct ib_device_attr *attr;
|
|
||||||
|
|
||||||
attr = kmalloc(sizeof(*attr), GFP_KERNEL);
|
snprintf(drvinfo->fw_version, sizeof(drvinfo->fw_version),
|
||||||
if (attr && !ib_query_device(priv->ca, attr))
|
"%d.%d.%d", (int)(priv->ca->attrs.fw_ver >> 32),
|
||||||
snprintf(drvinfo->fw_version, sizeof(drvinfo->fw_version),
|
(int)(priv->ca->attrs.fw_ver >> 16) & 0xffff,
|
||||||
"%d.%d.%d", (int)(attr->fw_ver >> 32),
|
(int)priv->ca->attrs.fw_ver & 0xffff);
|
||||||
(int)(attr->fw_ver >> 16) & 0xffff,
|
|
||||||
(int)attr->fw_ver & 0xffff);
|
|
||||||
kfree(attr);
|
|
||||||
|
|
||||||
strlcpy(drvinfo->bus_info, dev_name(priv->ca->dma_device),
|
strlcpy(drvinfo->bus_info, dev_name(priv->ca->dma_device),
|
||||||
sizeof(drvinfo->bus_info));
|
sizeof(drvinfo->bus_info));
|
||||||
|
|
|
@ -1150,8 +1150,6 @@ static void __ipoib_reap_neigh(struct ipoib_dev_priv *priv)
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
int i;
|
int i;
|
||||||
LIST_HEAD(remove_list);
|
LIST_HEAD(remove_list);
|
||||||
struct ipoib_mcast *mcast, *tmcast;
|
|
||||||
struct net_device *dev = priv->dev;
|
|
||||||
|
|
||||||
if (test_bit(IPOIB_STOP_NEIGH_GC, &priv->flags))
|
if (test_bit(IPOIB_STOP_NEIGH_GC, &priv->flags))
|
||||||
return;
|
return;
|
||||||
|
@ -1179,18 +1177,8 @@ static void __ipoib_reap_neigh(struct ipoib_dev_priv *priv)
|
||||||
lockdep_is_held(&priv->lock))) != NULL) {
|
lockdep_is_held(&priv->lock))) != NULL) {
|
||||||
/* was the neigh idle for two GC periods */
|
/* was the neigh idle for two GC periods */
|
||||||
if (time_after(neigh_obsolete, neigh->alive)) {
|
if (time_after(neigh_obsolete, neigh->alive)) {
|
||||||
u8 *mgid = neigh->daddr + 4;
|
|
||||||
|
|
||||||
/* Is this multicast ? */
|
ipoib_check_and_add_mcast_sendonly(priv, neigh->daddr + 4, &remove_list);
|
||||||
if (*mgid == 0xff) {
|
|
||||||
mcast = __ipoib_mcast_find(dev, mgid);
|
|
||||||
|
|
||||||
if (mcast && test_bit(IPOIB_MCAST_FLAG_SENDONLY, &mcast->flags)) {
|
|
||||||
list_del(&mcast->list);
|
|
||||||
rb_erase(&mcast->rb_node, &priv->multicast_tree);
|
|
||||||
list_add_tail(&mcast->list, &remove_list);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
rcu_assign_pointer(*np,
|
rcu_assign_pointer(*np,
|
||||||
rcu_dereference_protected(neigh->hnext,
|
rcu_dereference_protected(neigh->hnext,
|
||||||
|
@ -1207,10 +1195,7 @@ static void __ipoib_reap_neigh(struct ipoib_dev_priv *priv)
|
||||||
|
|
||||||
out_unlock:
|
out_unlock:
|
||||||
spin_unlock_irqrestore(&priv->lock, flags);
|
spin_unlock_irqrestore(&priv->lock, flags);
|
||||||
list_for_each_entry_safe(mcast, tmcast, &remove_list, list) {
|
ipoib_mcast_remove_list(&remove_list);
|
||||||
ipoib_mcast_leave(dev, mcast);
|
|
||||||
ipoib_mcast_free(mcast);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ipoib_reap_neigh(struct work_struct *work)
|
static void ipoib_reap_neigh(struct work_struct *work)
|
||||||
|
@ -1777,26 +1762,7 @@ int ipoib_add_pkey_attr(struct net_device *dev)
|
||||||
|
|
||||||
int ipoib_set_dev_features(struct ipoib_dev_priv *priv, struct ib_device *hca)
|
int ipoib_set_dev_features(struct ipoib_dev_priv *priv, struct ib_device *hca)
|
||||||
{
|
{
|
||||||
struct ib_device_attr *device_attr;
|
priv->hca_caps = hca->attrs.device_cap_flags;
|
||||||
int result = -ENOMEM;
|
|
||||||
|
|
||||||
device_attr = kmalloc(sizeof *device_attr, GFP_KERNEL);
|
|
||||||
if (!device_attr) {
|
|
||||||
printk(KERN_WARNING "%s: allocation of %zu bytes failed\n",
|
|
||||||
hca->name, sizeof *device_attr);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
result = ib_query_device(hca, device_attr);
|
|
||||||
if (result) {
|
|
||||||
printk(KERN_WARNING "%s: ib_query_device failed (ret = %d)\n",
|
|
||||||
hca->name, result);
|
|
||||||
kfree(device_attr);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
priv->hca_caps = device_attr->device_cap_flags;
|
|
||||||
|
|
||||||
kfree(device_attr);
|
|
||||||
|
|
||||||
if (priv->hca_caps & IB_DEVICE_UD_IP_CSUM) {
|
if (priv->hca_caps & IB_DEVICE_UD_IP_CSUM) {
|
||||||
priv->dev->hw_features = NETIF_F_SG |
|
priv->dev->hw_features = NETIF_F_SG |
|
||||||
|
|
|
@ -106,7 +106,7 @@ static void __ipoib_mcast_schedule_join_thread(struct ipoib_dev_priv *priv,
|
||||||
queue_delayed_work(priv->wq, &priv->mcast_task, 0);
|
queue_delayed_work(priv->wq, &priv->mcast_task, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ipoib_mcast_free(struct ipoib_mcast *mcast)
|
static void ipoib_mcast_free(struct ipoib_mcast *mcast)
|
||||||
{
|
{
|
||||||
struct net_device *dev = mcast->dev;
|
struct net_device *dev = mcast->dev;
|
||||||
int tx_dropped = 0;
|
int tx_dropped = 0;
|
||||||
|
@ -153,7 +153,7 @@ static struct ipoib_mcast *ipoib_mcast_alloc(struct net_device *dev,
|
||||||
return mcast;
|
return mcast;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct ipoib_mcast *__ipoib_mcast_find(struct net_device *dev, void *mgid)
|
static struct ipoib_mcast *__ipoib_mcast_find(struct net_device *dev, void *mgid)
|
||||||
{
|
{
|
||||||
struct ipoib_dev_priv *priv = netdev_priv(dev);
|
struct ipoib_dev_priv *priv = netdev_priv(dev);
|
||||||
struct rb_node *n = priv->multicast_tree.rb_node;
|
struct rb_node *n = priv->multicast_tree.rb_node;
|
||||||
|
@ -677,7 +677,7 @@ int ipoib_mcast_stop_thread(struct net_device *dev)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int ipoib_mcast_leave(struct net_device *dev, struct ipoib_mcast *mcast)
|
static int ipoib_mcast_leave(struct net_device *dev, struct ipoib_mcast *mcast)
|
||||||
{
|
{
|
||||||
struct ipoib_dev_priv *priv = netdev_priv(dev);
|
struct ipoib_dev_priv *priv = netdev_priv(dev);
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
@ -704,6 +704,35 @@ int ipoib_mcast_leave(struct net_device *dev, struct ipoib_mcast *mcast)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Check if the multicast group is sendonly. If so remove it from the maps
|
||||||
|
* and add to the remove list
|
||||||
|
*/
|
||||||
|
void ipoib_check_and_add_mcast_sendonly(struct ipoib_dev_priv *priv, u8 *mgid,
|
||||||
|
struct list_head *remove_list)
|
||||||
|
{
|
||||||
|
/* Is this multicast ? */
|
||||||
|
if (*mgid == 0xff) {
|
||||||
|
struct ipoib_mcast *mcast = __ipoib_mcast_find(priv->dev, mgid);
|
||||||
|
|
||||||
|
if (mcast && test_bit(IPOIB_MCAST_FLAG_SENDONLY, &mcast->flags)) {
|
||||||
|
list_del(&mcast->list);
|
||||||
|
rb_erase(&mcast->rb_node, &priv->multicast_tree);
|
||||||
|
list_add_tail(&mcast->list, remove_list);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ipoib_mcast_remove_list(struct list_head *remove_list)
|
||||||
|
{
|
||||||
|
struct ipoib_mcast *mcast, *tmcast;
|
||||||
|
|
||||||
|
list_for_each_entry_safe(mcast, tmcast, remove_list, list) {
|
||||||
|
ipoib_mcast_leave(mcast->dev, mcast);
|
||||||
|
ipoib_mcast_free(mcast);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void ipoib_mcast_send(struct net_device *dev, u8 *daddr, struct sk_buff *skb)
|
void ipoib_mcast_send(struct net_device *dev, u8 *daddr, struct sk_buff *skb)
|
||||||
{
|
{
|
||||||
struct ipoib_dev_priv *priv = netdev_priv(dev);
|
struct ipoib_dev_priv *priv = netdev_priv(dev);
|
||||||
|
@ -810,10 +839,7 @@ void ipoib_mcast_dev_flush(struct net_device *dev)
|
||||||
if (test_bit(IPOIB_MCAST_FLAG_BUSY, &mcast->flags))
|
if (test_bit(IPOIB_MCAST_FLAG_BUSY, &mcast->flags))
|
||||||
wait_for_completion(&mcast->done);
|
wait_for_completion(&mcast->done);
|
||||||
|
|
||||||
list_for_each_entry_safe(mcast, tmcast, &remove_list, list) {
|
ipoib_mcast_remove_list(&remove_list);
|
||||||
ipoib_mcast_leave(dev, mcast);
|
|
||||||
ipoib_mcast_free(mcast);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int ipoib_mcast_addr_is_valid(const u8 *addr, const u8 *broadcast)
|
static int ipoib_mcast_addr_is_valid(const u8 *addr, const u8 *broadcast)
|
||||||
|
@ -939,10 +965,7 @@ void ipoib_mcast_restart_task(struct work_struct *work)
|
||||||
if (test_bit(IPOIB_MCAST_FLAG_BUSY, &mcast->flags))
|
if (test_bit(IPOIB_MCAST_FLAG_BUSY, &mcast->flags))
|
||||||
wait_for_completion(&mcast->done);
|
wait_for_completion(&mcast->done);
|
||||||
|
|
||||||
list_for_each_entry_safe(mcast, tmcast, &remove_list, list) {
|
ipoib_mcast_remove_list(&remove_list);
|
||||||
ipoib_mcast_leave(mcast->dev, mcast);
|
|
||||||
ipoib_mcast_free(mcast);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Double check that we are still up
|
* Double check that we are still up
|
||||||
|
|
|
@ -644,7 +644,7 @@ iscsi_iser_session_create(struct iscsi_endpoint *ep,
|
||||||
|
|
||||||
ib_conn = &iser_conn->ib_conn;
|
ib_conn = &iser_conn->ib_conn;
|
||||||
if (ib_conn->pi_support) {
|
if (ib_conn->pi_support) {
|
||||||
u32 sig_caps = ib_conn->device->dev_attr.sig_prot_cap;
|
u32 sig_caps = ib_conn->device->ib_device->attrs.sig_prot_cap;
|
||||||
|
|
||||||
scsi_host_set_prot(shost, iser_dif_prot_caps(sig_caps));
|
scsi_host_set_prot(shost, iser_dif_prot_caps(sig_caps));
|
||||||
scsi_host_set_guard(shost, SHOST_DIX_GUARD_IP |
|
scsi_host_set_guard(shost, SHOST_DIX_GUARD_IP |
|
||||||
|
@ -656,7 +656,7 @@ iscsi_iser_session_create(struct iscsi_endpoint *ep,
|
||||||
* max fastreg page list length.
|
* max fastreg page list length.
|
||||||
*/
|
*/
|
||||||
shost->sg_tablesize = min_t(unsigned short, shost->sg_tablesize,
|
shost->sg_tablesize = min_t(unsigned short, shost->sg_tablesize,
|
||||||
ib_conn->device->dev_attr.max_fast_reg_page_list_len);
|
ib_conn->device->ib_device->attrs.max_fast_reg_page_list_len);
|
||||||
shost->max_sectors = min_t(unsigned int,
|
shost->max_sectors = min_t(unsigned int,
|
||||||
1024, (shost->sg_tablesize * PAGE_SIZE) >> 9);
|
1024, (shost->sg_tablesize * PAGE_SIZE) >> 9);
|
||||||
|
|
||||||
|
@ -1059,7 +1059,8 @@ static int __init iser_init(void)
|
||||||
release_wq = alloc_workqueue("release workqueue", 0, 0);
|
release_wq = alloc_workqueue("release workqueue", 0, 0);
|
||||||
if (!release_wq) {
|
if (!release_wq) {
|
||||||
iser_err("failed to allocate release workqueue\n");
|
iser_err("failed to allocate release workqueue\n");
|
||||||
return -ENOMEM;
|
err = -ENOMEM;
|
||||||
|
goto err_alloc_wq;
|
||||||
}
|
}
|
||||||
|
|
||||||
iscsi_iser_scsi_transport = iscsi_register_transport(
|
iscsi_iser_scsi_transport = iscsi_register_transport(
|
||||||
|
@ -1067,12 +1068,14 @@ static int __init iser_init(void)
|
||||||
if (!iscsi_iser_scsi_transport) {
|
if (!iscsi_iser_scsi_transport) {
|
||||||
iser_err("iscsi_register_transport failed\n");
|
iser_err("iscsi_register_transport failed\n");
|
||||||
err = -EINVAL;
|
err = -EINVAL;
|
||||||
goto register_transport_failure;
|
goto err_reg;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
register_transport_failure:
|
err_reg:
|
||||||
|
destroy_workqueue(release_wq);
|
||||||
|
err_alloc_wq:
|
||||||
kmem_cache_destroy(ig.desc_cache);
|
kmem_cache_destroy(ig.desc_cache);
|
||||||
|
|
||||||
return err;
|
return err;
|
||||||
|
|
|
@ -48,6 +48,7 @@
|
||||||
#include <scsi/scsi_transport_iscsi.h>
|
#include <scsi/scsi_transport_iscsi.h>
|
||||||
#include <scsi/scsi_cmnd.h>
|
#include <scsi/scsi_cmnd.h>
|
||||||
#include <scsi/scsi_device.h>
|
#include <scsi/scsi_device.h>
|
||||||
|
#include <scsi/iser.h>
|
||||||
|
|
||||||
#include <linux/interrupt.h>
|
#include <linux/interrupt.h>
|
||||||
#include <linux/wait.h>
|
#include <linux/wait.h>
|
||||||
|
@ -151,46 +152,10 @@
|
||||||
- ISER_MAX_RX_MISC_PDUS) / \
|
- ISER_MAX_RX_MISC_PDUS) / \
|
||||||
(1 + ISER_INFLIGHT_DATAOUTS))
|
(1 + ISER_INFLIGHT_DATAOUTS))
|
||||||
|
|
||||||
#define ISER_WC_BATCH_COUNT 16
|
|
||||||
#define ISER_SIGNAL_CMD_COUNT 32
|
#define ISER_SIGNAL_CMD_COUNT 32
|
||||||
|
|
||||||
#define ISER_VER 0x10
|
|
||||||
#define ISER_WSV 0x08
|
|
||||||
#define ISER_RSV 0x04
|
|
||||||
|
|
||||||
#define ISER_FASTREG_LI_WRID 0xffffffffffffffffULL
|
|
||||||
#define ISER_BEACON_WRID 0xfffffffffffffffeULL
|
|
||||||
|
|
||||||
/**
|
|
||||||
* struct iser_hdr - iSER header
|
|
||||||
*
|
|
||||||
* @flags: flags support (zbva, remote_inv)
|
|
||||||
* @rsvd: reserved
|
|
||||||
* @write_stag: write rkey
|
|
||||||
* @write_va: write virtual address
|
|
||||||
* @reaf_stag: read rkey
|
|
||||||
* @read_va: read virtual address
|
|
||||||
*/
|
|
||||||
struct iser_hdr {
|
|
||||||
u8 flags;
|
|
||||||
u8 rsvd[3];
|
|
||||||
__be32 write_stag;
|
|
||||||
__be64 write_va;
|
|
||||||
__be32 read_stag;
|
|
||||||
__be64 read_va;
|
|
||||||
} __attribute__((packed));
|
|
||||||
|
|
||||||
|
|
||||||
#define ISER_ZBVA_NOT_SUPPORTED 0x80
|
|
||||||
#define ISER_SEND_W_INV_NOT_SUPPORTED 0x40
|
|
||||||
|
|
||||||
struct iser_cm_hdr {
|
|
||||||
u8 flags;
|
|
||||||
u8 rsvd[3];
|
|
||||||
} __packed;
|
|
||||||
|
|
||||||
/* Constant PDU lengths calculations */
|
/* Constant PDU lengths calculations */
|
||||||
#define ISER_HEADERS_LEN (sizeof(struct iser_hdr) + sizeof(struct iscsi_hdr))
|
#define ISER_HEADERS_LEN (sizeof(struct iser_ctrl) + sizeof(struct iscsi_hdr))
|
||||||
|
|
||||||
#define ISER_RECV_DATA_SEG_LEN 128
|
#define ISER_RECV_DATA_SEG_LEN 128
|
||||||
#define ISER_RX_PAYLOAD_SIZE (ISER_HEADERS_LEN + ISER_RECV_DATA_SEG_LEN)
|
#define ISER_RX_PAYLOAD_SIZE (ISER_HEADERS_LEN + ISER_RECV_DATA_SEG_LEN)
|
||||||
|
@ -269,7 +234,7 @@ enum iser_desc_type {
|
||||||
#define ISER_MAX_WRS 7
|
#define ISER_MAX_WRS 7
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* struct iser_tx_desc - iSER TX descriptor (for send wr_id)
|
* struct iser_tx_desc - iSER TX descriptor
|
||||||
*
|
*
|
||||||
* @iser_header: iser header
|
* @iser_header: iser header
|
||||||
* @iscsi_header: iscsi header
|
* @iscsi_header: iscsi header
|
||||||
|
@ -287,12 +252,13 @@ enum iser_desc_type {
|
||||||
* @sig_attrs: Signature attributes
|
* @sig_attrs: Signature attributes
|
||||||
*/
|
*/
|
||||||
struct iser_tx_desc {
|
struct iser_tx_desc {
|
||||||
struct iser_hdr iser_header;
|
struct iser_ctrl iser_header;
|
||||||
struct iscsi_hdr iscsi_header;
|
struct iscsi_hdr iscsi_header;
|
||||||
enum iser_desc_type type;
|
enum iser_desc_type type;
|
||||||
u64 dma_addr;
|
u64 dma_addr;
|
||||||
struct ib_sge tx_sg[2];
|
struct ib_sge tx_sg[2];
|
||||||
int num_sge;
|
int num_sge;
|
||||||
|
struct ib_cqe cqe;
|
||||||
bool mapped;
|
bool mapped;
|
||||||
u8 wr_idx;
|
u8 wr_idx;
|
||||||
union iser_wr {
|
union iser_wr {
|
||||||
|
@ -306,9 +272,10 @@ struct iser_tx_desc {
|
||||||
};
|
};
|
||||||
|
|
||||||
#define ISER_RX_PAD_SIZE (256 - (ISER_RX_PAYLOAD_SIZE + \
|
#define ISER_RX_PAD_SIZE (256 - (ISER_RX_PAYLOAD_SIZE + \
|
||||||
sizeof(u64) + sizeof(struct ib_sge)))
|
sizeof(u64) + sizeof(struct ib_sge) + \
|
||||||
|
sizeof(struct ib_cqe)))
|
||||||
/**
|
/**
|
||||||
* struct iser_rx_desc - iSER RX descriptor (for recv wr_id)
|
* struct iser_rx_desc - iSER RX descriptor
|
||||||
*
|
*
|
||||||
* @iser_header: iser header
|
* @iser_header: iser header
|
||||||
* @iscsi_header: iscsi header
|
* @iscsi_header: iscsi header
|
||||||
|
@ -318,12 +285,32 @@ struct iser_tx_desc {
|
||||||
* @pad: for sense data TODO: Modify to maximum sense length supported
|
* @pad: for sense data TODO: Modify to maximum sense length supported
|
||||||
*/
|
*/
|
||||||
struct iser_rx_desc {
|
struct iser_rx_desc {
|
||||||
struct iser_hdr iser_header;
|
struct iser_ctrl iser_header;
|
||||||
struct iscsi_hdr iscsi_header;
|
struct iscsi_hdr iscsi_header;
|
||||||
char data[ISER_RECV_DATA_SEG_LEN];
|
char data[ISER_RECV_DATA_SEG_LEN];
|
||||||
u64 dma_addr;
|
u64 dma_addr;
|
||||||
struct ib_sge rx_sg;
|
struct ib_sge rx_sg;
|
||||||
|
struct ib_cqe cqe;
|
||||||
char pad[ISER_RX_PAD_SIZE];
|
char pad[ISER_RX_PAD_SIZE];
|
||||||
|
} __packed;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* struct iser_login_desc - iSER login descriptor
|
||||||
|
*
|
||||||
|
* @req: pointer to login request buffer
|
||||||
|
* @resp: pointer to login response buffer
|
||||||
|
* @req_dma: DMA address of login request buffer
|
||||||
|
* @rsp_dma: DMA address of login response buffer
|
||||||
|
* @sge: IB sge for login post recv
|
||||||
|
* @cqe: completion handler
|
||||||
|
*/
|
||||||
|
struct iser_login_desc {
|
||||||
|
void *req;
|
||||||
|
void *rsp;
|
||||||
|
u64 req_dma;
|
||||||
|
u64 rsp_dma;
|
||||||
|
struct ib_sge sge;
|
||||||
|
struct ib_cqe cqe;
|
||||||
} __attribute__((packed));
|
} __attribute__((packed));
|
||||||
|
|
||||||
struct iser_conn;
|
struct iser_conn;
|
||||||
|
@ -333,18 +320,12 @@ struct iscsi_iser_task;
|
||||||
/**
|
/**
|
||||||
* struct iser_comp - iSER completion context
|
* struct iser_comp - iSER completion context
|
||||||
*
|
*
|
||||||
* @device: pointer to device handle
|
|
||||||
* @cq: completion queue
|
* @cq: completion queue
|
||||||
* @wcs: work completion array
|
|
||||||
* @tasklet: Tasklet handle
|
|
||||||
* @active_qps: Number of active QPs attached
|
* @active_qps: Number of active QPs attached
|
||||||
* to completion context
|
* to completion context
|
||||||
*/
|
*/
|
||||||
struct iser_comp {
|
struct iser_comp {
|
||||||
struct iser_device *device;
|
|
||||||
struct ib_cq *cq;
|
struct ib_cq *cq;
|
||||||
struct ib_wc wcs[ISER_WC_BATCH_COUNT];
|
|
||||||
struct tasklet_struct tasklet;
|
|
||||||
int active_qps;
|
int active_qps;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -380,7 +361,6 @@ struct iser_reg_ops {
|
||||||
*
|
*
|
||||||
* @ib_device: RDMA device
|
* @ib_device: RDMA device
|
||||||
* @pd: Protection Domain for this device
|
* @pd: Protection Domain for this device
|
||||||
* @dev_attr: Device attributes container
|
|
||||||
* @mr: Global DMA memory region
|
* @mr: Global DMA memory region
|
||||||
* @event_handler: IB events handle routine
|
* @event_handler: IB events handle routine
|
||||||
* @ig_list: entry in devices list
|
* @ig_list: entry in devices list
|
||||||
|
@ -389,18 +369,19 @@ struct iser_reg_ops {
|
||||||
* cpus and device max completion vectors
|
* cpus and device max completion vectors
|
||||||
* @comps: Dinamically allocated array of completion handlers
|
* @comps: Dinamically allocated array of completion handlers
|
||||||
* @reg_ops: Registration ops
|
* @reg_ops: Registration ops
|
||||||
|
* @remote_inv_sup: Remote invalidate is supported on this device
|
||||||
*/
|
*/
|
||||||
struct iser_device {
|
struct iser_device {
|
||||||
struct ib_device *ib_device;
|
struct ib_device *ib_device;
|
||||||
struct ib_pd *pd;
|
struct ib_pd *pd;
|
||||||
struct ib_device_attr dev_attr;
|
|
||||||
struct ib_mr *mr;
|
struct ib_mr *mr;
|
||||||
struct ib_event_handler event_handler;
|
struct ib_event_handler event_handler;
|
||||||
struct list_head ig_list;
|
struct list_head ig_list;
|
||||||
int refcount;
|
int refcount;
|
||||||
int comps_used;
|
int comps_used;
|
||||||
struct iser_comp *comps;
|
struct iser_comp *comps;
|
||||||
struct iser_reg_ops *reg_ops;
|
const struct iser_reg_ops *reg_ops;
|
||||||
|
bool remote_inv_sup;
|
||||||
};
|
};
|
||||||
|
|
||||||
#define ISER_CHECK_GUARD 0xc0
|
#define ISER_CHECK_GUARD 0xc0
|
||||||
|
@ -475,10 +456,11 @@ struct iser_fr_pool {
|
||||||
* @rx_wr: receive work request for batch posts
|
* @rx_wr: receive work request for batch posts
|
||||||
* @device: reference to iser device
|
* @device: reference to iser device
|
||||||
* @comp: iser completion context
|
* @comp: iser completion context
|
||||||
* @pi_support: Indicate device T10-PI support
|
|
||||||
* @beacon: beacon send wr to signal all flush errors were drained
|
|
||||||
* @flush_comp: completes when all connection completions consumed
|
|
||||||
* @fr_pool: connection fast registration poool
|
* @fr_pool: connection fast registration poool
|
||||||
|
* @pi_support: Indicate device T10-PI support
|
||||||
|
* @last: last send wr to signal all flush errors were drained
|
||||||
|
* @last_cqe: cqe handler for last wr
|
||||||
|
* @last_comp: completes when all connection completions consumed
|
||||||
*/
|
*/
|
||||||
struct ib_conn {
|
struct ib_conn {
|
||||||
struct rdma_cm_id *cma_id;
|
struct rdma_cm_id *cma_id;
|
||||||
|
@ -488,10 +470,12 @@ struct ib_conn {
|
||||||
struct ib_recv_wr rx_wr[ISER_MIN_POSTED_RX];
|
struct ib_recv_wr rx_wr[ISER_MIN_POSTED_RX];
|
||||||
struct iser_device *device;
|
struct iser_device *device;
|
||||||
struct iser_comp *comp;
|
struct iser_comp *comp;
|
||||||
bool pi_support;
|
|
||||||
struct ib_send_wr beacon;
|
|
||||||
struct completion flush_comp;
|
|
||||||
struct iser_fr_pool fr_pool;
|
struct iser_fr_pool fr_pool;
|
||||||
|
bool pi_support;
|
||||||
|
struct ib_send_wr last;
|
||||||
|
struct ib_cqe last_cqe;
|
||||||
|
struct ib_cqe reg_cqe;
|
||||||
|
struct completion last_comp;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -514,11 +498,7 @@ struct ib_conn {
|
||||||
* @up_completion: connection establishment completed
|
* @up_completion: connection establishment completed
|
||||||
* (state is ISER_CONN_UP)
|
* (state is ISER_CONN_UP)
|
||||||
* @conn_list: entry in ig conn list
|
* @conn_list: entry in ig conn list
|
||||||
* @login_buf: login data buffer (stores login parameters)
|
* @login_desc: login descriptor
|
||||||
* @login_req_buf: login request buffer
|
|
||||||
* @login_req_dma: login request buffer dma address
|
|
||||||
* @login_resp_buf: login response buffer
|
|
||||||
* @login_resp_dma: login response buffer dma address
|
|
||||||
* @rx_desc_head: head of rx_descs cyclic buffer
|
* @rx_desc_head: head of rx_descs cyclic buffer
|
||||||
* @rx_descs: rx buffers array (cyclic buffer)
|
* @rx_descs: rx buffers array (cyclic buffer)
|
||||||
* @num_rx_descs: number of rx descriptors
|
* @num_rx_descs: number of rx descriptors
|
||||||
|
@ -541,15 +521,13 @@ struct iser_conn {
|
||||||
struct completion ib_completion;
|
struct completion ib_completion;
|
||||||
struct completion up_completion;
|
struct completion up_completion;
|
||||||
struct list_head conn_list;
|
struct list_head conn_list;
|
||||||
|
struct iser_login_desc login_desc;
|
||||||
char *login_buf;
|
|
||||||
char *login_req_buf, *login_resp_buf;
|
|
||||||
u64 login_req_dma, login_resp_dma;
|
|
||||||
unsigned int rx_desc_head;
|
unsigned int rx_desc_head;
|
||||||
struct iser_rx_desc *rx_descs;
|
struct iser_rx_desc *rx_descs;
|
||||||
u32 num_rx_descs;
|
u32 num_rx_descs;
|
||||||
unsigned short scsi_sg_tablesize;
|
unsigned short scsi_sg_tablesize;
|
||||||
unsigned int scsi_max_sectors;
|
unsigned int scsi_max_sectors;
|
||||||
|
bool snd_w_inv;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -579,9 +557,8 @@ struct iscsi_iser_task {
|
||||||
|
|
||||||
struct iser_page_vec {
|
struct iser_page_vec {
|
||||||
u64 *pages;
|
u64 *pages;
|
||||||
int length;
|
int npages;
|
||||||
int offset;
|
struct ib_mr fake_mr;
|
||||||
int data_size;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -633,12 +610,14 @@ int iser_conn_terminate(struct iser_conn *iser_conn);
|
||||||
|
|
||||||
void iser_release_work(struct work_struct *work);
|
void iser_release_work(struct work_struct *work);
|
||||||
|
|
||||||
void iser_rcv_completion(struct iser_rx_desc *desc,
|
void iser_err_comp(struct ib_wc *wc, const char *type);
|
||||||
unsigned long dto_xfer_len,
|
void iser_login_rsp(struct ib_cq *cq, struct ib_wc *wc);
|
||||||
struct ib_conn *ib_conn);
|
void iser_task_rsp(struct ib_cq *cq, struct ib_wc *wc);
|
||||||
|
void iser_cmd_comp(struct ib_cq *cq, struct ib_wc *wc);
|
||||||
void iser_snd_completion(struct iser_tx_desc *desc,
|
void iser_ctrl_comp(struct ib_cq *cq, struct ib_wc *wc);
|
||||||
struct ib_conn *ib_conn);
|
void iser_dataout_comp(struct ib_cq *cq, struct ib_wc *wc);
|
||||||
|
void iser_reg_comp(struct ib_cq *cq, struct ib_wc *wc);
|
||||||
|
void iser_last_comp(struct ib_cq *cq, struct ib_wc *wc);
|
||||||
|
|
||||||
void iser_task_rdma_init(struct iscsi_iser_task *task);
|
void iser_task_rdma_init(struct iscsi_iser_task *task);
|
||||||
|
|
||||||
|
@ -651,7 +630,8 @@ void iser_finalize_rdma_unaligned_sg(struct iscsi_iser_task *iser_task,
|
||||||
enum iser_data_dir cmd_dir);
|
enum iser_data_dir cmd_dir);
|
||||||
|
|
||||||
int iser_reg_rdma_mem(struct iscsi_iser_task *task,
|
int iser_reg_rdma_mem(struct iscsi_iser_task *task,
|
||||||
enum iser_data_dir dir);
|
enum iser_data_dir dir,
|
||||||
|
bool all_imm);
|
||||||
void iser_unreg_rdma_mem(struct iscsi_iser_task *task,
|
void iser_unreg_rdma_mem(struct iscsi_iser_task *task,
|
||||||
enum iser_data_dir dir);
|
enum iser_data_dir dir);
|
||||||
|
|
||||||
|
@ -719,4 +699,28 @@ iser_tx_next_wr(struct iser_tx_desc *tx_desc)
|
||||||
return cur_wr;
|
return cur_wr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline struct iser_conn *
|
||||||
|
to_iser_conn(struct ib_conn *ib_conn)
|
||||||
|
{
|
||||||
|
return container_of(ib_conn, struct iser_conn, ib_conn);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline struct iser_rx_desc *
|
||||||
|
iser_rx(struct ib_cqe *cqe)
|
||||||
|
{
|
||||||
|
return container_of(cqe, struct iser_rx_desc, cqe);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline struct iser_tx_desc *
|
||||||
|
iser_tx(struct ib_cqe *cqe)
|
||||||
|
{
|
||||||
|
return container_of(cqe, struct iser_tx_desc, cqe);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline struct iser_login_desc *
|
||||||
|
iser_login(struct ib_cqe *cqe)
|
||||||
|
{
|
||||||
|
return container_of(cqe, struct iser_login_desc, cqe);
|
||||||
|
}
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -51,7 +51,7 @@ static int iser_prepare_read_cmd(struct iscsi_task *task)
|
||||||
struct iscsi_iser_task *iser_task = task->dd_data;
|
struct iscsi_iser_task *iser_task = task->dd_data;
|
||||||
struct iser_mem_reg *mem_reg;
|
struct iser_mem_reg *mem_reg;
|
||||||
int err;
|
int err;
|
||||||
struct iser_hdr *hdr = &iser_task->desc.iser_header;
|
struct iser_ctrl *hdr = &iser_task->desc.iser_header;
|
||||||
struct iser_data_buf *buf_in = &iser_task->data[ISER_DIR_IN];
|
struct iser_data_buf *buf_in = &iser_task->data[ISER_DIR_IN];
|
||||||
|
|
||||||
err = iser_dma_map_task_data(iser_task,
|
err = iser_dma_map_task_data(iser_task,
|
||||||
|
@ -72,7 +72,7 @@ static int iser_prepare_read_cmd(struct iscsi_task *task)
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
err = iser_reg_rdma_mem(iser_task, ISER_DIR_IN);
|
err = iser_reg_rdma_mem(iser_task, ISER_DIR_IN, false);
|
||||||
if (err) {
|
if (err) {
|
||||||
iser_err("Failed to set up Data-IN RDMA\n");
|
iser_err("Failed to set up Data-IN RDMA\n");
|
||||||
return err;
|
return err;
|
||||||
|
@ -104,7 +104,7 @@ iser_prepare_write_cmd(struct iscsi_task *task,
|
||||||
struct iscsi_iser_task *iser_task = task->dd_data;
|
struct iscsi_iser_task *iser_task = task->dd_data;
|
||||||
struct iser_mem_reg *mem_reg;
|
struct iser_mem_reg *mem_reg;
|
||||||
int err;
|
int err;
|
||||||
struct iser_hdr *hdr = &iser_task->desc.iser_header;
|
struct iser_ctrl *hdr = &iser_task->desc.iser_header;
|
||||||
struct iser_data_buf *buf_out = &iser_task->data[ISER_DIR_OUT];
|
struct iser_data_buf *buf_out = &iser_task->data[ISER_DIR_OUT];
|
||||||
struct ib_sge *tx_dsg = &iser_task->desc.tx_sg[1];
|
struct ib_sge *tx_dsg = &iser_task->desc.tx_sg[1];
|
||||||
|
|
||||||
|
@ -126,7 +126,8 @@ iser_prepare_write_cmd(struct iscsi_task *task,
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
err = iser_reg_rdma_mem(iser_task, ISER_DIR_OUT);
|
err = iser_reg_rdma_mem(iser_task, ISER_DIR_OUT,
|
||||||
|
buf_out->data_len == imm_sz);
|
||||||
if (err != 0) {
|
if (err != 0) {
|
||||||
iser_err("Failed to register write cmd RDMA mem\n");
|
iser_err("Failed to register write cmd RDMA mem\n");
|
||||||
return err;
|
return err;
|
||||||
|
@ -166,7 +167,7 @@ static void iser_create_send_desc(struct iser_conn *iser_conn,
|
||||||
ib_dma_sync_single_for_cpu(device->ib_device,
|
ib_dma_sync_single_for_cpu(device->ib_device,
|
||||||
tx_desc->dma_addr, ISER_HEADERS_LEN, DMA_TO_DEVICE);
|
tx_desc->dma_addr, ISER_HEADERS_LEN, DMA_TO_DEVICE);
|
||||||
|
|
||||||
memset(&tx_desc->iser_header, 0, sizeof(struct iser_hdr));
|
memset(&tx_desc->iser_header, 0, sizeof(struct iser_ctrl));
|
||||||
tx_desc->iser_header.flags = ISER_VER;
|
tx_desc->iser_header.flags = ISER_VER;
|
||||||
tx_desc->num_sge = 1;
|
tx_desc->num_sge = 1;
|
||||||
}
|
}
|
||||||
|
@ -174,73 +175,63 @@ static void iser_create_send_desc(struct iser_conn *iser_conn,
|
||||||
static void iser_free_login_buf(struct iser_conn *iser_conn)
|
static void iser_free_login_buf(struct iser_conn *iser_conn)
|
||||||
{
|
{
|
||||||
struct iser_device *device = iser_conn->ib_conn.device;
|
struct iser_device *device = iser_conn->ib_conn.device;
|
||||||
|
struct iser_login_desc *desc = &iser_conn->login_desc;
|
||||||
|
|
||||||
if (!iser_conn->login_buf)
|
if (!desc->req)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if (iser_conn->login_req_dma)
|
ib_dma_unmap_single(device->ib_device, desc->req_dma,
|
||||||
ib_dma_unmap_single(device->ib_device,
|
ISCSI_DEF_MAX_RECV_SEG_LEN, DMA_TO_DEVICE);
|
||||||
iser_conn->login_req_dma,
|
|
||||||
ISCSI_DEF_MAX_RECV_SEG_LEN, DMA_TO_DEVICE);
|
|
||||||
|
|
||||||
if (iser_conn->login_resp_dma)
|
ib_dma_unmap_single(device->ib_device, desc->rsp_dma,
|
||||||
ib_dma_unmap_single(device->ib_device,
|
ISER_RX_LOGIN_SIZE, DMA_FROM_DEVICE);
|
||||||
iser_conn->login_resp_dma,
|
|
||||||
ISER_RX_LOGIN_SIZE, DMA_FROM_DEVICE);
|
|
||||||
|
|
||||||
kfree(iser_conn->login_buf);
|
kfree(desc->req);
|
||||||
|
kfree(desc->rsp);
|
||||||
|
|
||||||
/* make sure we never redo any unmapping */
|
/* make sure we never redo any unmapping */
|
||||||
iser_conn->login_req_dma = 0;
|
desc->req = NULL;
|
||||||
iser_conn->login_resp_dma = 0;
|
desc->rsp = NULL;
|
||||||
iser_conn->login_buf = NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int iser_alloc_login_buf(struct iser_conn *iser_conn)
|
static int iser_alloc_login_buf(struct iser_conn *iser_conn)
|
||||||
{
|
{
|
||||||
struct iser_device *device = iser_conn->ib_conn.device;
|
struct iser_device *device = iser_conn->ib_conn.device;
|
||||||
int req_err, resp_err;
|
struct iser_login_desc *desc = &iser_conn->login_desc;
|
||||||
|
|
||||||
BUG_ON(device == NULL);
|
desc->req = kmalloc(ISCSI_DEF_MAX_RECV_SEG_LEN, GFP_KERNEL);
|
||||||
|
if (!desc->req)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
iser_conn->login_buf = kmalloc(ISCSI_DEF_MAX_RECV_SEG_LEN +
|
desc->req_dma = ib_dma_map_single(device->ib_device, desc->req,
|
||||||
ISER_RX_LOGIN_SIZE, GFP_KERNEL);
|
ISCSI_DEF_MAX_RECV_SEG_LEN,
|
||||||
if (!iser_conn->login_buf)
|
DMA_TO_DEVICE);
|
||||||
goto out_err;
|
if (ib_dma_mapping_error(device->ib_device,
|
||||||
|
desc->req_dma))
|
||||||
|
goto free_req;
|
||||||
|
|
||||||
iser_conn->login_req_buf = iser_conn->login_buf;
|
desc->rsp = kmalloc(ISER_RX_LOGIN_SIZE, GFP_KERNEL);
|
||||||
iser_conn->login_resp_buf = iser_conn->login_buf +
|
if (!desc->rsp)
|
||||||
ISCSI_DEF_MAX_RECV_SEG_LEN;
|
goto unmap_req;
|
||||||
|
|
||||||
iser_conn->login_req_dma = ib_dma_map_single(device->ib_device,
|
desc->rsp_dma = ib_dma_map_single(device->ib_device, desc->rsp,
|
||||||
iser_conn->login_req_buf,
|
ISER_RX_LOGIN_SIZE,
|
||||||
ISCSI_DEF_MAX_RECV_SEG_LEN,
|
DMA_FROM_DEVICE);
|
||||||
DMA_TO_DEVICE);
|
if (ib_dma_mapping_error(device->ib_device,
|
||||||
|
desc->rsp_dma))
|
||||||
|
goto free_rsp;
|
||||||
|
|
||||||
iser_conn->login_resp_dma = ib_dma_map_single(device->ib_device,
|
|
||||||
iser_conn->login_resp_buf,
|
|
||||||
ISER_RX_LOGIN_SIZE,
|
|
||||||
DMA_FROM_DEVICE);
|
|
||||||
|
|
||||||
req_err = ib_dma_mapping_error(device->ib_device,
|
|
||||||
iser_conn->login_req_dma);
|
|
||||||
resp_err = ib_dma_mapping_error(device->ib_device,
|
|
||||||
iser_conn->login_resp_dma);
|
|
||||||
|
|
||||||
if (req_err || resp_err) {
|
|
||||||
if (req_err)
|
|
||||||
iser_conn->login_req_dma = 0;
|
|
||||||
if (resp_err)
|
|
||||||
iser_conn->login_resp_dma = 0;
|
|
||||||
goto free_login_buf;
|
|
||||||
}
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
free_login_buf:
|
free_rsp:
|
||||||
iser_free_login_buf(iser_conn);
|
kfree(desc->rsp);
|
||||||
|
unmap_req:
|
||||||
|
ib_dma_unmap_single(device->ib_device, desc->req_dma,
|
||||||
|
ISCSI_DEF_MAX_RECV_SEG_LEN,
|
||||||
|
DMA_TO_DEVICE);
|
||||||
|
free_req:
|
||||||
|
kfree(desc->req);
|
||||||
|
|
||||||
out_err:
|
|
||||||
iser_err("unable to alloc or map login buf\n");
|
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -280,11 +271,11 @@ int iser_alloc_rx_descriptors(struct iser_conn *iser_conn,
|
||||||
goto rx_desc_dma_map_failed;
|
goto rx_desc_dma_map_failed;
|
||||||
|
|
||||||
rx_desc->dma_addr = dma_addr;
|
rx_desc->dma_addr = dma_addr;
|
||||||
|
rx_desc->cqe.done = iser_task_rsp;
|
||||||
rx_sg = &rx_desc->rx_sg;
|
rx_sg = &rx_desc->rx_sg;
|
||||||
rx_sg->addr = rx_desc->dma_addr;
|
rx_sg->addr = rx_desc->dma_addr;
|
||||||
rx_sg->length = ISER_RX_PAYLOAD_SIZE;
|
rx_sg->length = ISER_RX_PAYLOAD_SIZE;
|
||||||
rx_sg->lkey = device->pd->local_dma_lkey;
|
rx_sg->lkey = device->pd->local_dma_lkey;
|
||||||
}
|
}
|
||||||
|
|
||||||
iser_conn->rx_desc_head = 0;
|
iser_conn->rx_desc_head = 0;
|
||||||
|
@ -383,6 +374,7 @@ int iser_send_command(struct iscsi_conn *conn,
|
||||||
|
|
||||||
/* build the tx desc regd header and add it to the tx desc dto */
|
/* build the tx desc regd header and add it to the tx desc dto */
|
||||||
tx_desc->type = ISCSI_TX_SCSI_COMMAND;
|
tx_desc->type = ISCSI_TX_SCSI_COMMAND;
|
||||||
|
tx_desc->cqe.done = iser_cmd_comp;
|
||||||
iser_create_send_desc(iser_conn, tx_desc);
|
iser_create_send_desc(iser_conn, tx_desc);
|
||||||
|
|
||||||
if (hdr->flags & ISCSI_FLAG_CMD_READ) {
|
if (hdr->flags & ISCSI_FLAG_CMD_READ) {
|
||||||
|
@ -464,6 +456,7 @@ int iser_send_data_out(struct iscsi_conn *conn,
|
||||||
}
|
}
|
||||||
|
|
||||||
tx_desc->type = ISCSI_TX_DATAOUT;
|
tx_desc->type = ISCSI_TX_DATAOUT;
|
||||||
|
tx_desc->cqe.done = iser_dataout_comp;
|
||||||
tx_desc->iser_header.flags = ISER_VER;
|
tx_desc->iser_header.flags = ISER_VER;
|
||||||
memcpy(&tx_desc->iscsi_header, hdr, sizeof(struct iscsi_hdr));
|
memcpy(&tx_desc->iscsi_header, hdr, sizeof(struct iscsi_hdr));
|
||||||
|
|
||||||
|
@ -513,6 +506,7 @@ int iser_send_control(struct iscsi_conn *conn,
|
||||||
|
|
||||||
/* build the tx desc regd header and add it to the tx desc dto */
|
/* build the tx desc regd header and add it to the tx desc dto */
|
||||||
mdesc->type = ISCSI_TX_CONTROL;
|
mdesc->type = ISCSI_TX_CONTROL;
|
||||||
|
mdesc->cqe.done = iser_ctrl_comp;
|
||||||
iser_create_send_desc(iser_conn, mdesc);
|
iser_create_send_desc(iser_conn, mdesc);
|
||||||
|
|
||||||
device = iser_conn->ib_conn.device;
|
device = iser_conn->ib_conn.device;
|
||||||
|
@ -520,25 +514,25 @@ int iser_send_control(struct iscsi_conn *conn,
|
||||||
data_seg_len = ntoh24(task->hdr->dlength);
|
data_seg_len = ntoh24(task->hdr->dlength);
|
||||||
|
|
||||||
if (data_seg_len > 0) {
|
if (data_seg_len > 0) {
|
||||||
|
struct iser_login_desc *desc = &iser_conn->login_desc;
|
||||||
struct ib_sge *tx_dsg = &mdesc->tx_sg[1];
|
struct ib_sge *tx_dsg = &mdesc->tx_sg[1];
|
||||||
|
|
||||||
if (task != conn->login_task) {
|
if (task != conn->login_task) {
|
||||||
iser_err("data present on non login task!!!\n");
|
iser_err("data present on non login task!!!\n");
|
||||||
goto send_control_error;
|
goto send_control_error;
|
||||||
}
|
}
|
||||||
|
|
||||||
ib_dma_sync_single_for_cpu(device->ib_device,
|
ib_dma_sync_single_for_cpu(device->ib_device, desc->req_dma,
|
||||||
iser_conn->login_req_dma, task->data_count,
|
task->data_count, DMA_TO_DEVICE);
|
||||||
DMA_TO_DEVICE);
|
|
||||||
|
|
||||||
memcpy(iser_conn->login_req_buf, task->data, task->data_count);
|
memcpy(desc->req, task->data, task->data_count);
|
||||||
|
|
||||||
ib_dma_sync_single_for_device(device->ib_device,
|
ib_dma_sync_single_for_device(device->ib_device, desc->req_dma,
|
||||||
iser_conn->login_req_dma, task->data_count,
|
task->data_count, DMA_TO_DEVICE);
|
||||||
DMA_TO_DEVICE);
|
|
||||||
|
|
||||||
tx_dsg->addr = iser_conn->login_req_dma;
|
tx_dsg->addr = desc->req_dma;
|
||||||
tx_dsg->length = task->data_count;
|
tx_dsg->length = task->data_count;
|
||||||
tx_dsg->lkey = device->pd->local_dma_lkey;
|
tx_dsg->lkey = device->pd->local_dma_lkey;
|
||||||
mdesc->num_sge = 2;
|
mdesc->num_sge = 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -562,41 +556,126 @@ int iser_send_control(struct iscsi_conn *conn,
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
void iser_login_rsp(struct ib_cq *cq, struct ib_wc *wc)
|
||||||
* iser_rcv_dto_completion - recv DTO completion
|
|
||||||
*/
|
|
||||||
void iser_rcv_completion(struct iser_rx_desc *rx_desc,
|
|
||||||
unsigned long rx_xfer_len,
|
|
||||||
struct ib_conn *ib_conn)
|
|
||||||
{
|
{
|
||||||
struct iser_conn *iser_conn = container_of(ib_conn, struct iser_conn,
|
struct ib_conn *ib_conn = wc->qp->qp_context;
|
||||||
ib_conn);
|
struct iser_conn *iser_conn = to_iser_conn(ib_conn);
|
||||||
|
struct iser_login_desc *desc = iser_login(wc->wr_cqe);
|
||||||
struct iscsi_hdr *hdr;
|
struct iscsi_hdr *hdr;
|
||||||
u64 rx_dma;
|
char *data;
|
||||||
int rx_buflen, outstanding, count, err;
|
int length;
|
||||||
|
|
||||||
/* differentiate between login to all other PDUs */
|
if (unlikely(wc->status != IB_WC_SUCCESS)) {
|
||||||
if ((char *)rx_desc == iser_conn->login_resp_buf) {
|
iser_err_comp(wc, "login_rsp");
|
||||||
rx_dma = iser_conn->login_resp_dma;
|
return;
|
||||||
rx_buflen = ISER_RX_LOGIN_SIZE;
|
|
||||||
} else {
|
|
||||||
rx_dma = rx_desc->dma_addr;
|
|
||||||
rx_buflen = ISER_RX_PAYLOAD_SIZE;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ib_dma_sync_single_for_cpu(ib_conn->device->ib_device, rx_dma,
|
ib_dma_sync_single_for_cpu(ib_conn->device->ib_device,
|
||||||
rx_buflen, DMA_FROM_DEVICE);
|
desc->rsp_dma, ISER_RX_LOGIN_SIZE,
|
||||||
|
DMA_FROM_DEVICE);
|
||||||
|
|
||||||
hdr = &rx_desc->iscsi_header;
|
hdr = desc->rsp + sizeof(struct iser_ctrl);
|
||||||
|
data = desc->rsp + ISER_HEADERS_LEN;
|
||||||
|
length = wc->byte_len - ISER_HEADERS_LEN;
|
||||||
|
|
||||||
iser_dbg("op 0x%x itt 0x%x dlen %d\n", hdr->opcode,
|
iser_dbg("op 0x%x itt 0x%x dlen %d\n", hdr->opcode,
|
||||||
hdr->itt, (int)(rx_xfer_len - ISER_HEADERS_LEN));
|
hdr->itt, length);
|
||||||
|
|
||||||
iscsi_iser_recv(iser_conn->iscsi_conn, hdr, rx_desc->data,
|
iscsi_iser_recv(iser_conn->iscsi_conn, hdr, data, length);
|
||||||
rx_xfer_len - ISER_HEADERS_LEN);
|
|
||||||
|
|
||||||
ib_dma_sync_single_for_device(ib_conn->device->ib_device, rx_dma,
|
ib_dma_sync_single_for_device(ib_conn->device->ib_device,
|
||||||
rx_buflen, DMA_FROM_DEVICE);
|
desc->rsp_dma, ISER_RX_LOGIN_SIZE,
|
||||||
|
DMA_FROM_DEVICE);
|
||||||
|
|
||||||
|
ib_conn->post_recv_buf_count--;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void
|
||||||
|
iser_inv_desc(struct iser_fr_desc *desc, u32 rkey)
|
||||||
|
{
|
||||||
|
if (likely(rkey == desc->rsc.mr->rkey))
|
||||||
|
desc->rsc.mr_valid = 0;
|
||||||
|
else if (likely(rkey == desc->pi_ctx->sig_mr->rkey))
|
||||||
|
desc->pi_ctx->sig_mr_valid = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
iser_check_remote_inv(struct iser_conn *iser_conn,
|
||||||
|
struct ib_wc *wc,
|
||||||
|
struct iscsi_hdr *hdr)
|
||||||
|
{
|
||||||
|
if (wc->wc_flags & IB_WC_WITH_INVALIDATE) {
|
||||||
|
struct iscsi_task *task;
|
||||||
|
u32 rkey = wc->ex.invalidate_rkey;
|
||||||
|
|
||||||
|
iser_dbg("conn %p: remote invalidation for rkey %#x\n",
|
||||||
|
iser_conn, rkey);
|
||||||
|
|
||||||
|
if (unlikely(!iser_conn->snd_w_inv)) {
|
||||||
|
iser_err("conn %p: unexepected remote invalidation, "
|
||||||
|
"terminating connection\n", iser_conn);
|
||||||
|
return -EPROTO;
|
||||||
|
}
|
||||||
|
|
||||||
|
task = iscsi_itt_to_ctask(iser_conn->iscsi_conn, hdr->itt);
|
||||||
|
if (likely(task)) {
|
||||||
|
struct iscsi_iser_task *iser_task = task->dd_data;
|
||||||
|
struct iser_fr_desc *desc;
|
||||||
|
|
||||||
|
if (iser_task->dir[ISER_DIR_IN]) {
|
||||||
|
desc = iser_task->rdma_reg[ISER_DIR_IN].mem_h;
|
||||||
|
iser_inv_desc(desc, rkey);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (iser_task->dir[ISER_DIR_OUT]) {
|
||||||
|
desc = iser_task->rdma_reg[ISER_DIR_OUT].mem_h;
|
||||||
|
iser_inv_desc(desc, rkey);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
iser_err("failed to get task for itt=%d\n", hdr->itt);
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void iser_task_rsp(struct ib_cq *cq, struct ib_wc *wc)
|
||||||
|
{
|
||||||
|
struct ib_conn *ib_conn = wc->qp->qp_context;
|
||||||
|
struct iser_conn *iser_conn = to_iser_conn(ib_conn);
|
||||||
|
struct iser_rx_desc *desc = iser_rx(wc->wr_cqe);
|
||||||
|
struct iscsi_hdr *hdr;
|
||||||
|
int length;
|
||||||
|
int outstanding, count, err;
|
||||||
|
|
||||||
|
if (unlikely(wc->status != IB_WC_SUCCESS)) {
|
||||||
|
iser_err_comp(wc, "task_rsp");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
ib_dma_sync_single_for_cpu(ib_conn->device->ib_device,
|
||||||
|
desc->dma_addr, ISER_RX_PAYLOAD_SIZE,
|
||||||
|
DMA_FROM_DEVICE);
|
||||||
|
|
||||||
|
hdr = &desc->iscsi_header;
|
||||||
|
length = wc->byte_len - ISER_HEADERS_LEN;
|
||||||
|
|
||||||
|
iser_dbg("op 0x%x itt 0x%x dlen %d\n", hdr->opcode,
|
||||||
|
hdr->itt, length);
|
||||||
|
|
||||||
|
if (iser_check_remote_inv(iser_conn, wc, hdr)) {
|
||||||
|
iscsi_conn_failure(iser_conn->iscsi_conn,
|
||||||
|
ISCSI_ERR_CONN_FAILED);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
iscsi_iser_recv(iser_conn->iscsi_conn, hdr, desc->data, length);
|
||||||
|
|
||||||
|
ib_dma_sync_single_for_device(ib_conn->device->ib_device,
|
||||||
|
desc->dma_addr, ISER_RX_PAYLOAD_SIZE,
|
||||||
|
DMA_FROM_DEVICE);
|
||||||
|
|
||||||
/* decrementing conn->post_recv_buf_count only --after-- freeing the *
|
/* decrementing conn->post_recv_buf_count only --after-- freeing the *
|
||||||
* task eliminates the need to worry on tasks which are completed in *
|
* task eliminates the need to worry on tasks which are completed in *
|
||||||
|
@ -604,9 +683,6 @@ void iser_rcv_completion(struct iser_rx_desc *rx_desc,
|
||||||
* for the posted rx bufs refcount to become zero handles everything */
|
* for the posted rx bufs refcount to become zero handles everything */
|
||||||
ib_conn->post_recv_buf_count--;
|
ib_conn->post_recv_buf_count--;
|
||||||
|
|
||||||
if (rx_dma == iser_conn->login_resp_dma)
|
|
||||||
return;
|
|
||||||
|
|
||||||
outstanding = ib_conn->post_recv_buf_count;
|
outstanding = ib_conn->post_recv_buf_count;
|
||||||
if (outstanding + iser_conn->min_posted_rx <= iser_conn->qp_max_recv_dtos) {
|
if (outstanding + iser_conn->min_posted_rx <= iser_conn->qp_max_recv_dtos) {
|
||||||
count = min(iser_conn->qp_max_recv_dtos - outstanding,
|
count = min(iser_conn->qp_max_recv_dtos - outstanding,
|
||||||
|
@ -617,26 +693,47 @@ void iser_rcv_completion(struct iser_rx_desc *rx_desc,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void iser_snd_completion(struct iser_tx_desc *tx_desc,
|
void iser_cmd_comp(struct ib_cq *cq, struct ib_wc *wc)
|
||||||
struct ib_conn *ib_conn)
|
|
||||||
{
|
{
|
||||||
|
if (unlikely(wc->status != IB_WC_SUCCESS))
|
||||||
|
iser_err_comp(wc, "command");
|
||||||
|
}
|
||||||
|
|
||||||
|
void iser_ctrl_comp(struct ib_cq *cq, struct ib_wc *wc)
|
||||||
|
{
|
||||||
|
struct iser_tx_desc *desc = iser_tx(wc->wr_cqe);
|
||||||
struct iscsi_task *task;
|
struct iscsi_task *task;
|
||||||
|
|
||||||
|
if (unlikely(wc->status != IB_WC_SUCCESS)) {
|
||||||
|
iser_err_comp(wc, "control");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* this arithmetic is legal by libiscsi dd_data allocation */
|
||||||
|
task = (void *)desc - sizeof(struct iscsi_task);
|
||||||
|
if (task->hdr->itt == RESERVED_ITT)
|
||||||
|
iscsi_put_task(task);
|
||||||
|
}
|
||||||
|
|
||||||
|
void iser_dataout_comp(struct ib_cq *cq, struct ib_wc *wc)
|
||||||
|
{
|
||||||
|
struct iser_tx_desc *desc = iser_tx(wc->wr_cqe);
|
||||||
|
struct ib_conn *ib_conn = wc->qp->qp_context;
|
||||||
struct iser_device *device = ib_conn->device;
|
struct iser_device *device = ib_conn->device;
|
||||||
|
|
||||||
if (tx_desc->type == ISCSI_TX_DATAOUT) {
|
if (unlikely(wc->status != IB_WC_SUCCESS))
|
||||||
ib_dma_unmap_single(device->ib_device, tx_desc->dma_addr,
|
iser_err_comp(wc, "dataout");
|
||||||
ISER_HEADERS_LEN, DMA_TO_DEVICE);
|
|
||||||
kmem_cache_free(ig.desc_cache, tx_desc);
|
|
||||||
tx_desc = NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (tx_desc && tx_desc->type == ISCSI_TX_CONTROL) {
|
ib_dma_unmap_single(device->ib_device, desc->dma_addr,
|
||||||
/* this arithmetic is legal by libiscsi dd_data allocation */
|
ISER_HEADERS_LEN, DMA_TO_DEVICE);
|
||||||
task = (void *) ((long)(void *)tx_desc -
|
kmem_cache_free(ig.desc_cache, desc);
|
||||||
sizeof(struct iscsi_task));
|
}
|
||||||
if (task->hdr->itt == RESERVED_ITT)
|
|
||||||
iscsi_put_task(task);
|
void iser_last_comp(struct ib_cq *cq, struct ib_wc *wc)
|
||||||
}
|
{
|
||||||
|
struct ib_conn *ib_conn = wc->qp->qp_context;
|
||||||
|
|
||||||
|
complete(&ib_conn->last_comp);
|
||||||
}
|
}
|
||||||
|
|
||||||
void iser_task_rdma_init(struct iscsi_iser_task *iser_task)
|
void iser_task_rdma_init(struct iscsi_iser_task *iser_task)
|
||||||
|
|
|
@ -49,7 +49,7 @@ int iser_fast_reg_mr(struct iscsi_iser_task *iser_task,
|
||||||
struct iser_reg_resources *rsc,
|
struct iser_reg_resources *rsc,
|
||||||
struct iser_mem_reg *mem_reg);
|
struct iser_mem_reg *mem_reg);
|
||||||
|
|
||||||
static struct iser_reg_ops fastreg_ops = {
|
static const struct iser_reg_ops fastreg_ops = {
|
||||||
.alloc_reg_res = iser_alloc_fastreg_pool,
|
.alloc_reg_res = iser_alloc_fastreg_pool,
|
||||||
.free_reg_res = iser_free_fastreg_pool,
|
.free_reg_res = iser_free_fastreg_pool,
|
||||||
.reg_mem = iser_fast_reg_mr,
|
.reg_mem = iser_fast_reg_mr,
|
||||||
|
@ -58,7 +58,7 @@ static struct iser_reg_ops fastreg_ops = {
|
||||||
.reg_desc_put = iser_reg_desc_put_fr,
|
.reg_desc_put = iser_reg_desc_put_fr,
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct iser_reg_ops fmr_ops = {
|
static const struct iser_reg_ops fmr_ops = {
|
||||||
.alloc_reg_res = iser_alloc_fmr_pool,
|
.alloc_reg_res = iser_alloc_fmr_pool,
|
||||||
.free_reg_res = iser_free_fmr_pool,
|
.free_reg_res = iser_free_fmr_pool,
|
||||||
.reg_mem = iser_fast_reg_fmr,
|
.reg_mem = iser_fast_reg_fmr,
|
||||||
|
@ -67,19 +67,24 @@ static struct iser_reg_ops fmr_ops = {
|
||||||
.reg_desc_put = iser_reg_desc_put_fmr,
|
.reg_desc_put = iser_reg_desc_put_fmr,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
void iser_reg_comp(struct ib_cq *cq, struct ib_wc *wc)
|
||||||
|
{
|
||||||
|
iser_err_comp(wc, "memreg");
|
||||||
|
}
|
||||||
|
|
||||||
int iser_assign_reg_ops(struct iser_device *device)
|
int iser_assign_reg_ops(struct iser_device *device)
|
||||||
{
|
{
|
||||||
struct ib_device_attr *dev_attr = &device->dev_attr;
|
struct ib_device *ib_dev = device->ib_device;
|
||||||
|
|
||||||
/* Assign function handles - based on FMR support */
|
/* Assign function handles - based on FMR support */
|
||||||
if (device->ib_device->alloc_fmr && device->ib_device->dealloc_fmr &&
|
if (ib_dev->alloc_fmr && ib_dev->dealloc_fmr &&
|
||||||
device->ib_device->map_phys_fmr && device->ib_device->unmap_fmr) {
|
ib_dev->map_phys_fmr && ib_dev->unmap_fmr) {
|
||||||
iser_info("FMR supported, using FMR for registration\n");
|
iser_info("FMR supported, using FMR for registration\n");
|
||||||
device->reg_ops = &fmr_ops;
|
device->reg_ops = &fmr_ops;
|
||||||
} else
|
} else if (ib_dev->attrs.device_cap_flags & IB_DEVICE_MEM_MGT_EXTENSIONS) {
|
||||||
if (dev_attr->device_cap_flags & IB_DEVICE_MEM_MGT_EXTENSIONS) {
|
|
||||||
iser_info("FastReg supported, using FastReg for registration\n");
|
iser_info("FastReg supported, using FastReg for registration\n");
|
||||||
device->reg_ops = &fastreg_ops;
|
device->reg_ops = &fastreg_ops;
|
||||||
|
device->remote_inv_sup = iser_always_reg;
|
||||||
} else {
|
} else {
|
||||||
iser_err("IB device does not support FMRs nor FastRegs, can't register memory\n");
|
iser_err("IB device does not support FMRs nor FastRegs, can't register memory\n");
|
||||||
return -1;
|
return -1;
|
||||||
|
@ -131,67 +136,6 @@ iser_reg_desc_put_fmr(struct ib_conn *ib_conn,
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
#define IS_4K_ALIGNED(addr) ((((unsigned long)addr) & ~MASK_4K) == 0)
|
|
||||||
|
|
||||||
/**
|
|
||||||
* iser_sg_to_page_vec - Translates scatterlist entries to physical addresses
|
|
||||||
* and returns the length of resulting physical address array (may be less than
|
|
||||||
* the original due to possible compaction).
|
|
||||||
*
|
|
||||||
* we build a "page vec" under the assumption that the SG meets the RDMA
|
|
||||||
* alignment requirements. Other then the first and last SG elements, all
|
|
||||||
* the "internal" elements can be compacted into a list whose elements are
|
|
||||||
* dma addresses of physical pages. The code supports also the weird case
|
|
||||||
* where --few fragments of the same page-- are present in the SG as
|
|
||||||
* consecutive elements. Also, it handles one entry SG.
|
|
||||||
*/
|
|
||||||
|
|
||||||
static int iser_sg_to_page_vec(struct iser_data_buf *data,
|
|
||||||
struct ib_device *ibdev, u64 *pages,
|
|
||||||
int *offset, int *data_size)
|
|
||||||
{
|
|
||||||
struct scatterlist *sg, *sgl = data->sg;
|
|
||||||
u64 start_addr, end_addr, page, chunk_start = 0;
|
|
||||||
unsigned long total_sz = 0;
|
|
||||||
unsigned int dma_len;
|
|
||||||
int i, new_chunk, cur_page, last_ent = data->dma_nents - 1;
|
|
||||||
|
|
||||||
/* compute the offset of first element */
|
|
||||||
*offset = (u64) sgl[0].offset & ~MASK_4K;
|
|
||||||
|
|
||||||
new_chunk = 1;
|
|
||||||
cur_page = 0;
|
|
||||||
for_each_sg(sgl, sg, data->dma_nents, i) {
|
|
||||||
start_addr = ib_sg_dma_address(ibdev, sg);
|
|
||||||
if (new_chunk)
|
|
||||||
chunk_start = start_addr;
|
|
||||||
dma_len = ib_sg_dma_len(ibdev, sg);
|
|
||||||
end_addr = start_addr + dma_len;
|
|
||||||
total_sz += dma_len;
|
|
||||||
|
|
||||||
/* collect page fragments until aligned or end of SG list */
|
|
||||||
if (!IS_4K_ALIGNED(end_addr) && i < last_ent) {
|
|
||||||
new_chunk = 0;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
new_chunk = 1;
|
|
||||||
|
|
||||||
/* address of the first page in the contiguous chunk;
|
|
||||||
masking relevant for the very first SG entry,
|
|
||||||
which might be unaligned */
|
|
||||||
page = chunk_start & MASK_4K;
|
|
||||||
do {
|
|
||||||
pages[cur_page++] = page;
|
|
||||||
page += SIZE_4K;
|
|
||||||
} while (page < end_addr);
|
|
||||||
}
|
|
||||||
|
|
||||||
*data_size = total_sz;
|
|
||||||
iser_dbg("page_vec->data_size:%d cur_page %d\n",
|
|
||||||
*data_size, cur_page);
|
|
||||||
return cur_page;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void iser_data_buf_dump(struct iser_data_buf *data,
|
static void iser_data_buf_dump(struct iser_data_buf *data,
|
||||||
struct ib_device *ibdev)
|
struct ib_device *ibdev)
|
||||||
{
|
{
|
||||||
|
@ -210,10 +154,10 @@ static void iser_dump_page_vec(struct iser_page_vec *page_vec)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
iser_err("page vec length %d data size %d\n",
|
iser_err("page vec npages %d data length %d\n",
|
||||||
page_vec->length, page_vec->data_size);
|
page_vec->npages, page_vec->fake_mr.length);
|
||||||
for (i = 0; i < page_vec->length; i++)
|
for (i = 0; i < page_vec->npages; i++)
|
||||||
iser_err("%d %lx\n",i,(unsigned long)page_vec->pages[i]);
|
iser_err("vec[%d]: %llx\n", i, page_vec->pages[i]);
|
||||||
}
|
}
|
||||||
|
|
||||||
int iser_dma_map_task_data(struct iscsi_iser_task *iser_task,
|
int iser_dma_map_task_data(struct iscsi_iser_task *iser_task,
|
||||||
|
@ -251,7 +195,11 @@ iser_reg_dma(struct iser_device *device, struct iser_data_buf *mem,
|
||||||
struct scatterlist *sg = mem->sg;
|
struct scatterlist *sg = mem->sg;
|
||||||
|
|
||||||
reg->sge.lkey = device->pd->local_dma_lkey;
|
reg->sge.lkey = device->pd->local_dma_lkey;
|
||||||
reg->rkey = device->mr->rkey;
|
/*
|
||||||
|
* FIXME: rework the registration code path to differentiate
|
||||||
|
* rkey/lkey use cases
|
||||||
|
*/
|
||||||
|
reg->rkey = device->mr ? device->mr->rkey : 0;
|
||||||
reg->sge.addr = ib_sg_dma_address(device->ib_device, &sg[0]);
|
reg->sge.addr = ib_sg_dma_address(device->ib_device, &sg[0]);
|
||||||
reg->sge.length = ib_sg_dma_len(device->ib_device, &sg[0]);
|
reg->sge.length = ib_sg_dma_len(device->ib_device, &sg[0]);
|
||||||
|
|
||||||
|
@ -262,11 +210,16 @@ iser_reg_dma(struct iser_device *device, struct iser_data_buf *mem,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
static int iser_set_page(struct ib_mr *mr, u64 addr)
|
||||||
* iser_reg_page_vec - Register physical memory
|
{
|
||||||
*
|
struct iser_page_vec *page_vec =
|
||||||
* returns: 0 on success, errno code on failure
|
container_of(mr, struct iser_page_vec, fake_mr);
|
||||||
*/
|
|
||||||
|
page_vec->pages[page_vec->npages++] = addr;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
static
|
static
|
||||||
int iser_fast_reg_fmr(struct iscsi_iser_task *iser_task,
|
int iser_fast_reg_fmr(struct iscsi_iser_task *iser_task,
|
||||||
struct iser_data_buf *mem,
|
struct iser_data_buf *mem,
|
||||||
|
@ -280,22 +233,19 @@ int iser_fast_reg_fmr(struct iscsi_iser_task *iser_task,
|
||||||
struct ib_pool_fmr *fmr;
|
struct ib_pool_fmr *fmr;
|
||||||
int ret, plen;
|
int ret, plen;
|
||||||
|
|
||||||
plen = iser_sg_to_page_vec(mem, device->ib_device,
|
page_vec->npages = 0;
|
||||||
page_vec->pages,
|
page_vec->fake_mr.page_size = SIZE_4K;
|
||||||
&page_vec->offset,
|
plen = ib_sg_to_pages(&page_vec->fake_mr, mem->sg,
|
||||||
&page_vec->data_size);
|
mem->size, iser_set_page);
|
||||||
page_vec->length = plen;
|
if (unlikely(plen < mem->size)) {
|
||||||
if (plen * SIZE_4K < page_vec->data_size) {
|
|
||||||
iser_err("page vec too short to hold this SG\n");
|
iser_err("page vec too short to hold this SG\n");
|
||||||
iser_data_buf_dump(mem, device->ib_device);
|
iser_data_buf_dump(mem, device->ib_device);
|
||||||
iser_dump_page_vec(page_vec);
|
iser_dump_page_vec(page_vec);
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
fmr = ib_fmr_pool_map_phys(fmr_pool,
|
fmr = ib_fmr_pool_map_phys(fmr_pool, page_vec->pages,
|
||||||
page_vec->pages,
|
page_vec->npages, page_vec->pages[0]);
|
||||||
page_vec->length,
|
|
||||||
page_vec->pages[0]);
|
|
||||||
if (IS_ERR(fmr)) {
|
if (IS_ERR(fmr)) {
|
||||||
ret = PTR_ERR(fmr);
|
ret = PTR_ERR(fmr);
|
||||||
iser_err("ib_fmr_pool_map_phys failed: %d\n", ret);
|
iser_err("ib_fmr_pool_map_phys failed: %d\n", ret);
|
||||||
|
@ -304,8 +254,8 @@ int iser_fast_reg_fmr(struct iscsi_iser_task *iser_task,
|
||||||
|
|
||||||
reg->sge.lkey = fmr->fmr->lkey;
|
reg->sge.lkey = fmr->fmr->lkey;
|
||||||
reg->rkey = fmr->fmr->rkey;
|
reg->rkey = fmr->fmr->rkey;
|
||||||
reg->sge.addr = page_vec->pages[0] + page_vec->offset;
|
reg->sge.addr = page_vec->fake_mr.iova;
|
||||||
reg->sge.length = page_vec->data_size;
|
reg->sge.length = page_vec->fake_mr.length;
|
||||||
reg->mem_h = fmr;
|
reg->mem_h = fmr;
|
||||||
|
|
||||||
iser_dbg("fmr reg: lkey=0x%x, rkey=0x%x, addr=0x%llx,"
|
iser_dbg("fmr reg: lkey=0x%x, rkey=0x%x, addr=0x%llx,"
|
||||||
|
@ -413,19 +363,16 @@ iser_set_prot_checks(struct scsi_cmnd *sc, u8 *mask)
|
||||||
*mask |= ISER_CHECK_GUARD;
|
*mask |= ISER_CHECK_GUARD;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static inline void
|
||||||
iser_inv_rkey(struct ib_send_wr *inv_wr, struct ib_mr *mr)
|
iser_inv_rkey(struct ib_send_wr *inv_wr,
|
||||||
|
struct ib_mr *mr,
|
||||||
|
struct ib_cqe *cqe)
|
||||||
{
|
{
|
||||||
u32 rkey;
|
|
||||||
|
|
||||||
inv_wr->opcode = IB_WR_LOCAL_INV;
|
inv_wr->opcode = IB_WR_LOCAL_INV;
|
||||||
inv_wr->wr_id = ISER_FASTREG_LI_WRID;
|
inv_wr->wr_cqe = cqe;
|
||||||
inv_wr->ex.invalidate_rkey = mr->rkey;
|
inv_wr->ex.invalidate_rkey = mr->rkey;
|
||||||
inv_wr->send_flags = 0;
|
inv_wr->send_flags = 0;
|
||||||
inv_wr->num_sge = 0;
|
inv_wr->num_sge = 0;
|
||||||
|
|
||||||
rkey = ib_inc_rkey(mr->rkey);
|
|
||||||
ib_update_fast_reg_key(mr, rkey);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
|
@ -437,7 +384,9 @@ iser_reg_sig_mr(struct iscsi_iser_task *iser_task,
|
||||||
{
|
{
|
||||||
struct iser_tx_desc *tx_desc = &iser_task->desc;
|
struct iser_tx_desc *tx_desc = &iser_task->desc;
|
||||||
struct ib_sig_attrs *sig_attrs = &tx_desc->sig_attrs;
|
struct ib_sig_attrs *sig_attrs = &tx_desc->sig_attrs;
|
||||||
|
struct ib_cqe *cqe = &iser_task->iser_conn->ib_conn.reg_cqe;
|
||||||
struct ib_sig_handover_wr *wr;
|
struct ib_sig_handover_wr *wr;
|
||||||
|
struct ib_mr *mr = pi_ctx->sig_mr;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
memset(sig_attrs, 0, sizeof(*sig_attrs));
|
memset(sig_attrs, 0, sizeof(*sig_attrs));
|
||||||
|
@ -447,17 +396,19 @@ iser_reg_sig_mr(struct iscsi_iser_task *iser_task,
|
||||||
|
|
||||||
iser_set_prot_checks(iser_task->sc, &sig_attrs->check_mask);
|
iser_set_prot_checks(iser_task->sc, &sig_attrs->check_mask);
|
||||||
|
|
||||||
if (!pi_ctx->sig_mr_valid)
|
if (pi_ctx->sig_mr_valid)
|
||||||
iser_inv_rkey(iser_tx_next_wr(tx_desc), pi_ctx->sig_mr);
|
iser_inv_rkey(iser_tx_next_wr(tx_desc), mr, cqe);
|
||||||
|
|
||||||
|
ib_update_fast_reg_key(mr, ib_inc_rkey(mr->rkey));
|
||||||
|
|
||||||
wr = sig_handover_wr(iser_tx_next_wr(tx_desc));
|
wr = sig_handover_wr(iser_tx_next_wr(tx_desc));
|
||||||
wr->wr.opcode = IB_WR_REG_SIG_MR;
|
wr->wr.opcode = IB_WR_REG_SIG_MR;
|
||||||
wr->wr.wr_id = ISER_FASTREG_LI_WRID;
|
wr->wr.wr_cqe = cqe;
|
||||||
wr->wr.sg_list = &data_reg->sge;
|
wr->wr.sg_list = &data_reg->sge;
|
||||||
wr->wr.num_sge = 1;
|
wr->wr.num_sge = 1;
|
||||||
wr->wr.send_flags = 0;
|
wr->wr.send_flags = 0;
|
||||||
wr->sig_attrs = sig_attrs;
|
wr->sig_attrs = sig_attrs;
|
||||||
wr->sig_mr = pi_ctx->sig_mr;
|
wr->sig_mr = mr;
|
||||||
if (scsi_prot_sg_count(iser_task->sc))
|
if (scsi_prot_sg_count(iser_task->sc))
|
||||||
wr->prot = &prot_reg->sge;
|
wr->prot = &prot_reg->sge;
|
||||||
else
|
else
|
||||||
|
@ -465,10 +416,10 @@ iser_reg_sig_mr(struct iscsi_iser_task *iser_task,
|
||||||
wr->access_flags = IB_ACCESS_LOCAL_WRITE |
|
wr->access_flags = IB_ACCESS_LOCAL_WRITE |
|
||||||
IB_ACCESS_REMOTE_READ |
|
IB_ACCESS_REMOTE_READ |
|
||||||
IB_ACCESS_REMOTE_WRITE;
|
IB_ACCESS_REMOTE_WRITE;
|
||||||
pi_ctx->sig_mr_valid = 0;
|
pi_ctx->sig_mr_valid = 1;
|
||||||
|
|
||||||
sig_reg->sge.lkey = pi_ctx->sig_mr->lkey;
|
sig_reg->sge.lkey = mr->lkey;
|
||||||
sig_reg->rkey = pi_ctx->sig_mr->rkey;
|
sig_reg->rkey = mr->rkey;
|
||||||
sig_reg->sge.addr = 0;
|
sig_reg->sge.addr = 0;
|
||||||
sig_reg->sge.length = scsi_transfer_length(iser_task->sc);
|
sig_reg->sge.length = scsi_transfer_length(iser_task->sc);
|
||||||
|
|
||||||
|
@ -485,12 +436,15 @@ static int iser_fast_reg_mr(struct iscsi_iser_task *iser_task,
|
||||||
struct iser_mem_reg *reg)
|
struct iser_mem_reg *reg)
|
||||||
{
|
{
|
||||||
struct iser_tx_desc *tx_desc = &iser_task->desc;
|
struct iser_tx_desc *tx_desc = &iser_task->desc;
|
||||||
|
struct ib_cqe *cqe = &iser_task->iser_conn->ib_conn.reg_cqe;
|
||||||
struct ib_mr *mr = rsc->mr;
|
struct ib_mr *mr = rsc->mr;
|
||||||
struct ib_reg_wr *wr;
|
struct ib_reg_wr *wr;
|
||||||
int n;
|
int n;
|
||||||
|
|
||||||
if (!rsc->mr_valid)
|
if (rsc->mr_valid)
|
||||||
iser_inv_rkey(iser_tx_next_wr(tx_desc), mr);
|
iser_inv_rkey(iser_tx_next_wr(tx_desc), mr, cqe);
|
||||||
|
|
||||||
|
ib_update_fast_reg_key(mr, ib_inc_rkey(mr->rkey));
|
||||||
|
|
||||||
n = ib_map_mr_sg(mr, mem->sg, mem->size, SIZE_4K);
|
n = ib_map_mr_sg(mr, mem->sg, mem->size, SIZE_4K);
|
||||||
if (unlikely(n != mem->size)) {
|
if (unlikely(n != mem->size)) {
|
||||||
|
@ -501,7 +455,7 @@ static int iser_fast_reg_mr(struct iscsi_iser_task *iser_task,
|
||||||
|
|
||||||
wr = reg_wr(iser_tx_next_wr(tx_desc));
|
wr = reg_wr(iser_tx_next_wr(tx_desc));
|
||||||
wr->wr.opcode = IB_WR_REG_MR;
|
wr->wr.opcode = IB_WR_REG_MR;
|
||||||
wr->wr.wr_id = ISER_FASTREG_LI_WRID;
|
wr->wr.wr_cqe = cqe;
|
||||||
wr->wr.send_flags = 0;
|
wr->wr.send_flags = 0;
|
||||||
wr->wr.num_sge = 0;
|
wr->wr.num_sge = 0;
|
||||||
wr->mr = mr;
|
wr->mr = mr;
|
||||||
|
@ -510,7 +464,7 @@ static int iser_fast_reg_mr(struct iscsi_iser_task *iser_task,
|
||||||
IB_ACCESS_REMOTE_WRITE |
|
IB_ACCESS_REMOTE_WRITE |
|
||||||
IB_ACCESS_REMOTE_READ;
|
IB_ACCESS_REMOTE_READ;
|
||||||
|
|
||||||
rsc->mr_valid = 0;
|
rsc->mr_valid = 1;
|
||||||
|
|
||||||
reg->sge.lkey = mr->lkey;
|
reg->sge.lkey = mr->lkey;
|
||||||
reg->rkey = mr->rkey;
|
reg->rkey = mr->rkey;
|
||||||
|
@ -554,7 +508,8 @@ iser_reg_data_sg(struct iscsi_iser_task *task,
|
||||||
}
|
}
|
||||||
|
|
||||||
int iser_reg_rdma_mem(struct iscsi_iser_task *task,
|
int iser_reg_rdma_mem(struct iscsi_iser_task *task,
|
||||||
enum iser_data_dir dir)
|
enum iser_data_dir dir,
|
||||||
|
bool all_imm)
|
||||||
{
|
{
|
||||||
struct ib_conn *ib_conn = &task->iser_conn->ib_conn;
|
struct ib_conn *ib_conn = &task->iser_conn->ib_conn;
|
||||||
struct iser_device *device = ib_conn->device;
|
struct iser_device *device = ib_conn->device;
|
||||||
|
@ -565,8 +520,8 @@ int iser_reg_rdma_mem(struct iscsi_iser_task *task,
|
||||||
bool use_dma_key;
|
bool use_dma_key;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
use_dma_key = (mem->dma_nents == 1 && !iser_always_reg &&
|
use_dma_key = mem->dma_nents == 1 && (all_imm || !iser_always_reg) &&
|
||||||
scsi_get_prot_op(task->sc) == SCSI_PROT_NORMAL);
|
scsi_get_prot_op(task->sc) == SCSI_PROT_NORMAL;
|
||||||
|
|
||||||
if (!use_dma_key) {
|
if (!use_dma_key) {
|
||||||
desc = device->reg_ops->reg_desc_get(ib_conn);
|
desc = device->reg_ops->reg_desc_get(ib_conn);
|
||||||
|
|
|
@ -44,17 +44,6 @@
|
||||||
#define ISER_MAX_CQ_LEN (ISER_MAX_RX_LEN + ISER_MAX_TX_LEN + \
|
#define ISER_MAX_CQ_LEN (ISER_MAX_RX_LEN + ISER_MAX_TX_LEN + \
|
||||||
ISCSI_ISER_MAX_CONN)
|
ISCSI_ISER_MAX_CONN)
|
||||||
|
|
||||||
static int iser_cq_poll_limit = 512;
|
|
||||||
|
|
||||||
static void iser_cq_tasklet_fn(unsigned long data);
|
|
||||||
static void iser_cq_callback(struct ib_cq *cq, void *cq_context);
|
|
||||||
|
|
||||||
static void iser_cq_event_callback(struct ib_event *cause, void *context)
|
|
||||||
{
|
|
||||||
iser_err("cq event %s (%d)\n",
|
|
||||||
ib_event_msg(cause->event), cause->event);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void iser_qp_event_callback(struct ib_event *cause, void *context)
|
static void iser_qp_event_callback(struct ib_event *cause, void *context)
|
||||||
{
|
{
|
||||||
iser_err("qp event %s (%d)\n",
|
iser_err("qp event %s (%d)\n",
|
||||||
|
@ -78,59 +67,40 @@ static void iser_event_handler(struct ib_event_handler *handler,
|
||||||
*/
|
*/
|
||||||
static int iser_create_device_ib_res(struct iser_device *device)
|
static int iser_create_device_ib_res(struct iser_device *device)
|
||||||
{
|
{
|
||||||
struct ib_device_attr *dev_attr = &device->dev_attr;
|
struct ib_device *ib_dev = device->ib_device;
|
||||||
int ret, i, max_cqe;
|
int ret, i, max_cqe;
|
||||||
|
|
||||||
ret = ib_query_device(device->ib_device, dev_attr);
|
|
||||||
if (ret) {
|
|
||||||
pr_warn("Query device failed for %s\n", device->ib_device->name);
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
ret = iser_assign_reg_ops(device);
|
ret = iser_assign_reg_ops(device);
|
||||||
if (ret)
|
if (ret)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
device->comps_used = min_t(int, num_online_cpus(),
|
device->comps_used = min_t(int, num_online_cpus(),
|
||||||
device->ib_device->num_comp_vectors);
|
ib_dev->num_comp_vectors);
|
||||||
|
|
||||||
device->comps = kcalloc(device->comps_used, sizeof(*device->comps),
|
device->comps = kcalloc(device->comps_used, sizeof(*device->comps),
|
||||||
GFP_KERNEL);
|
GFP_KERNEL);
|
||||||
if (!device->comps)
|
if (!device->comps)
|
||||||
goto comps_err;
|
goto comps_err;
|
||||||
|
|
||||||
max_cqe = min(ISER_MAX_CQ_LEN, dev_attr->max_cqe);
|
max_cqe = min(ISER_MAX_CQ_LEN, ib_dev->attrs.max_cqe);
|
||||||
|
|
||||||
iser_info("using %d CQs, device %s supports %d vectors max_cqe %d\n",
|
iser_info("using %d CQs, device %s supports %d vectors max_cqe %d\n",
|
||||||
device->comps_used, device->ib_device->name,
|
device->comps_used, ib_dev->name,
|
||||||
device->ib_device->num_comp_vectors, max_cqe);
|
ib_dev->num_comp_vectors, max_cqe);
|
||||||
|
|
||||||
device->pd = ib_alloc_pd(device->ib_device);
|
device->pd = ib_alloc_pd(ib_dev);
|
||||||
if (IS_ERR(device->pd))
|
if (IS_ERR(device->pd))
|
||||||
goto pd_err;
|
goto pd_err;
|
||||||
|
|
||||||
for (i = 0; i < device->comps_used; i++) {
|
for (i = 0; i < device->comps_used; i++) {
|
||||||
struct ib_cq_init_attr cq_attr = {};
|
|
||||||
struct iser_comp *comp = &device->comps[i];
|
struct iser_comp *comp = &device->comps[i];
|
||||||
|
|
||||||
comp->device = device;
|
comp->cq = ib_alloc_cq(ib_dev, comp, max_cqe, i,
|
||||||
cq_attr.cqe = max_cqe;
|
IB_POLL_SOFTIRQ);
|
||||||
cq_attr.comp_vector = i;
|
|
||||||
comp->cq = ib_create_cq(device->ib_device,
|
|
||||||
iser_cq_callback,
|
|
||||||
iser_cq_event_callback,
|
|
||||||
(void *)comp,
|
|
||||||
&cq_attr);
|
|
||||||
if (IS_ERR(comp->cq)) {
|
if (IS_ERR(comp->cq)) {
|
||||||
comp->cq = NULL;
|
comp->cq = NULL;
|
||||||
goto cq_err;
|
goto cq_err;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ib_req_notify_cq(comp->cq, IB_CQ_NEXT_COMP))
|
|
||||||
goto cq_err;
|
|
||||||
|
|
||||||
tasklet_init(&comp->tasklet, iser_cq_tasklet_fn,
|
|
||||||
(unsigned long)comp);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!iser_always_reg) {
|
if (!iser_always_reg) {
|
||||||
|
@ -140,11 +110,11 @@ static int iser_create_device_ib_res(struct iser_device *device)
|
||||||
|
|
||||||
device->mr = ib_get_dma_mr(device->pd, access);
|
device->mr = ib_get_dma_mr(device->pd, access);
|
||||||
if (IS_ERR(device->mr))
|
if (IS_ERR(device->mr))
|
||||||
goto dma_mr_err;
|
goto cq_err;
|
||||||
}
|
}
|
||||||
|
|
||||||
INIT_IB_EVENT_HANDLER(&device->event_handler, device->ib_device,
|
INIT_IB_EVENT_HANDLER(&device->event_handler, ib_dev,
|
||||||
iser_event_handler);
|
iser_event_handler);
|
||||||
if (ib_register_event_handler(&device->event_handler))
|
if (ib_register_event_handler(&device->event_handler))
|
||||||
goto handler_err;
|
goto handler_err;
|
||||||
|
|
||||||
|
@ -153,15 +123,12 @@ static int iser_create_device_ib_res(struct iser_device *device)
|
||||||
handler_err:
|
handler_err:
|
||||||
if (device->mr)
|
if (device->mr)
|
||||||
ib_dereg_mr(device->mr);
|
ib_dereg_mr(device->mr);
|
||||||
dma_mr_err:
|
|
||||||
for (i = 0; i < device->comps_used; i++)
|
|
||||||
tasklet_kill(&device->comps[i].tasklet);
|
|
||||||
cq_err:
|
cq_err:
|
||||||
for (i = 0; i < device->comps_used; i++) {
|
for (i = 0; i < device->comps_used; i++) {
|
||||||
struct iser_comp *comp = &device->comps[i];
|
struct iser_comp *comp = &device->comps[i];
|
||||||
|
|
||||||
if (comp->cq)
|
if (comp->cq)
|
||||||
ib_destroy_cq(comp->cq);
|
ib_free_cq(comp->cq);
|
||||||
}
|
}
|
||||||
ib_dealloc_pd(device->pd);
|
ib_dealloc_pd(device->pd);
|
||||||
pd_err:
|
pd_err:
|
||||||
|
@ -182,8 +149,7 @@ static void iser_free_device_ib_res(struct iser_device *device)
|
||||||
for (i = 0; i < device->comps_used; i++) {
|
for (i = 0; i < device->comps_used; i++) {
|
||||||
struct iser_comp *comp = &device->comps[i];
|
struct iser_comp *comp = &device->comps[i];
|
||||||
|
|
||||||
tasklet_kill(&comp->tasklet);
|
ib_free_cq(comp->cq);
|
||||||
ib_destroy_cq(comp->cq);
|
|
||||||
comp->cq = NULL;
|
comp->cq = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -299,7 +265,7 @@ iser_alloc_reg_res(struct ib_device *ib_device,
|
||||||
iser_err("Failed to allocate ib_fast_reg_mr err=%d\n", ret);
|
iser_err("Failed to allocate ib_fast_reg_mr err=%d\n", ret);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
res->mr_valid = 1;
|
res->mr_valid = 0;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -336,7 +302,7 @@ iser_alloc_pi_ctx(struct ib_device *ib_device,
|
||||||
ret = PTR_ERR(pi_ctx->sig_mr);
|
ret = PTR_ERR(pi_ctx->sig_mr);
|
||||||
goto sig_mr_failure;
|
goto sig_mr_failure;
|
||||||
}
|
}
|
||||||
pi_ctx->sig_mr_valid = 1;
|
pi_ctx->sig_mr_valid = 0;
|
||||||
desc->pi_ctx->sig_protected = 0;
|
desc->pi_ctx->sig_protected = 0;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -461,10 +427,9 @@ void iser_free_fastreg_pool(struct ib_conn *ib_conn)
|
||||||
*/
|
*/
|
||||||
static int iser_create_ib_conn_res(struct ib_conn *ib_conn)
|
static int iser_create_ib_conn_res(struct ib_conn *ib_conn)
|
||||||
{
|
{
|
||||||
struct iser_conn *iser_conn = container_of(ib_conn, struct iser_conn,
|
struct iser_conn *iser_conn = to_iser_conn(ib_conn);
|
||||||
ib_conn);
|
|
||||||
struct iser_device *device;
|
struct iser_device *device;
|
||||||
struct ib_device_attr *dev_attr;
|
struct ib_device *ib_dev;
|
||||||
struct ib_qp_init_attr init_attr;
|
struct ib_qp_init_attr init_attr;
|
||||||
int ret = -ENOMEM;
|
int ret = -ENOMEM;
|
||||||
int index, min_index = 0;
|
int index, min_index = 0;
|
||||||
|
@ -472,7 +437,7 @@ static int iser_create_ib_conn_res(struct ib_conn *ib_conn)
|
||||||
BUG_ON(ib_conn->device == NULL);
|
BUG_ON(ib_conn->device == NULL);
|
||||||
|
|
||||||
device = ib_conn->device;
|
device = ib_conn->device;
|
||||||
dev_attr = &device->dev_attr;
|
ib_dev = device->ib_device;
|
||||||
|
|
||||||
memset(&init_attr, 0, sizeof init_attr);
|
memset(&init_attr, 0, sizeof init_attr);
|
||||||
|
|
||||||
|
@ -503,16 +468,16 @@ static int iser_create_ib_conn_res(struct ib_conn *ib_conn)
|
||||||
iser_conn->max_cmds =
|
iser_conn->max_cmds =
|
||||||
ISER_GET_MAX_XMIT_CMDS(ISER_QP_SIG_MAX_REQ_DTOS);
|
ISER_GET_MAX_XMIT_CMDS(ISER_QP_SIG_MAX_REQ_DTOS);
|
||||||
} else {
|
} else {
|
||||||
if (dev_attr->max_qp_wr > ISER_QP_MAX_REQ_DTOS) {
|
if (ib_dev->attrs.max_qp_wr > ISER_QP_MAX_REQ_DTOS) {
|
||||||
init_attr.cap.max_send_wr = ISER_QP_MAX_REQ_DTOS + 1;
|
init_attr.cap.max_send_wr = ISER_QP_MAX_REQ_DTOS + 1;
|
||||||
iser_conn->max_cmds =
|
iser_conn->max_cmds =
|
||||||
ISER_GET_MAX_XMIT_CMDS(ISER_QP_MAX_REQ_DTOS);
|
ISER_GET_MAX_XMIT_CMDS(ISER_QP_MAX_REQ_DTOS);
|
||||||
} else {
|
} else {
|
||||||
init_attr.cap.max_send_wr = dev_attr->max_qp_wr;
|
init_attr.cap.max_send_wr = ib_dev->attrs.max_qp_wr;
|
||||||
iser_conn->max_cmds =
|
iser_conn->max_cmds =
|
||||||
ISER_GET_MAX_XMIT_CMDS(dev_attr->max_qp_wr);
|
ISER_GET_MAX_XMIT_CMDS(ib_dev->attrs.max_qp_wr);
|
||||||
iser_dbg("device %s supports max_send_wr %d\n",
|
iser_dbg("device %s supports max_send_wr %d\n",
|
||||||
device->ib_device->name, dev_attr->max_qp_wr);
|
device->ib_device->name, ib_dev->attrs.max_qp_wr);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -724,13 +689,13 @@ int iser_conn_terminate(struct iser_conn *iser_conn)
|
||||||
iser_conn, err);
|
iser_conn, err);
|
||||||
|
|
||||||
/* post an indication that all flush errors were consumed */
|
/* post an indication that all flush errors were consumed */
|
||||||
err = ib_post_send(ib_conn->qp, &ib_conn->beacon, &bad_wr);
|
err = ib_post_send(ib_conn->qp, &ib_conn->last, &bad_wr);
|
||||||
if (err) {
|
if (err) {
|
||||||
iser_err("conn %p failed to post beacon", ib_conn);
|
iser_err("conn %p failed to post last wr", ib_conn);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
wait_for_completion(&ib_conn->flush_comp);
|
wait_for_completion(&ib_conn->last_comp);
|
||||||
}
|
}
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
|
@ -756,7 +721,7 @@ iser_calc_scsi_params(struct iser_conn *iser_conn,
|
||||||
|
|
||||||
sg_tablesize = DIV_ROUND_UP(max_sectors * 512, SIZE_4K);
|
sg_tablesize = DIV_ROUND_UP(max_sectors * 512, SIZE_4K);
|
||||||
sup_sg_tablesize = min_t(unsigned, ISCSI_ISER_MAX_SG_TABLESIZE,
|
sup_sg_tablesize = min_t(unsigned, ISCSI_ISER_MAX_SG_TABLESIZE,
|
||||||
device->dev_attr.max_fast_reg_page_list_len);
|
device->ib_device->attrs.max_fast_reg_page_list_len);
|
||||||
|
|
||||||
if (sg_tablesize > sup_sg_tablesize) {
|
if (sg_tablesize > sup_sg_tablesize) {
|
||||||
sg_tablesize = sup_sg_tablesize;
|
sg_tablesize = sup_sg_tablesize;
|
||||||
|
@ -799,7 +764,7 @@ static void iser_addr_handler(struct rdma_cm_id *cma_id)
|
||||||
|
|
||||||
/* connection T10-PI support */
|
/* connection T10-PI support */
|
||||||
if (iser_pi_enable) {
|
if (iser_pi_enable) {
|
||||||
if (!(device->dev_attr.device_cap_flags &
|
if (!(device->ib_device->attrs.device_cap_flags &
|
||||||
IB_DEVICE_SIGNATURE_HANDOVER)) {
|
IB_DEVICE_SIGNATURE_HANDOVER)) {
|
||||||
iser_warn("T10-PI requested but not supported on %s, "
|
iser_warn("T10-PI requested but not supported on %s, "
|
||||||
"continue without T10-PI\n",
|
"continue without T10-PI\n",
|
||||||
|
@ -841,16 +806,17 @@ static void iser_route_handler(struct rdma_cm_id *cma_id)
|
||||||
goto failure;
|
goto failure;
|
||||||
|
|
||||||
memset(&conn_param, 0, sizeof conn_param);
|
memset(&conn_param, 0, sizeof conn_param);
|
||||||
conn_param.responder_resources = device->dev_attr.max_qp_rd_atom;
|
conn_param.responder_resources = device->ib_device->attrs.max_qp_rd_atom;
|
||||||
conn_param.initiator_depth = 1;
|
conn_param.initiator_depth = 1;
|
||||||
conn_param.retry_count = 7;
|
conn_param.retry_count = 7;
|
||||||
conn_param.rnr_retry_count = 6;
|
conn_param.rnr_retry_count = 6;
|
||||||
|
|
||||||
memset(&req_hdr, 0, sizeof(req_hdr));
|
memset(&req_hdr, 0, sizeof(req_hdr));
|
||||||
req_hdr.flags = (ISER_ZBVA_NOT_SUPPORTED |
|
req_hdr.flags = ISER_ZBVA_NOT_SUP;
|
||||||
ISER_SEND_W_INV_NOT_SUPPORTED);
|
if (!device->remote_inv_sup)
|
||||||
conn_param.private_data = (void *)&req_hdr;
|
req_hdr.flags |= ISER_SEND_W_INV_NOT_SUP;
|
||||||
conn_param.private_data_len = sizeof(struct iser_cm_hdr);
|
conn_param.private_data = (void *)&req_hdr;
|
||||||
|
conn_param.private_data_len = sizeof(struct iser_cm_hdr);
|
||||||
|
|
||||||
ret = rdma_connect(cma_id, &conn_param);
|
ret = rdma_connect(cma_id, &conn_param);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
|
@ -863,7 +829,8 @@ static void iser_route_handler(struct rdma_cm_id *cma_id)
|
||||||
iser_connect_error(cma_id);
|
iser_connect_error(cma_id);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void iser_connected_handler(struct rdma_cm_id *cma_id)
|
static void iser_connected_handler(struct rdma_cm_id *cma_id,
|
||||||
|
const void *private_data)
|
||||||
{
|
{
|
||||||
struct iser_conn *iser_conn;
|
struct iser_conn *iser_conn;
|
||||||
struct ib_qp_attr attr;
|
struct ib_qp_attr attr;
|
||||||
|
@ -877,6 +844,15 @@ static void iser_connected_handler(struct rdma_cm_id *cma_id)
|
||||||
(void)ib_query_qp(cma_id->qp, &attr, ~0, &init_attr);
|
(void)ib_query_qp(cma_id->qp, &attr, ~0, &init_attr);
|
||||||
iser_info("remote qpn:%x my qpn:%x\n", attr.dest_qp_num, cma_id->qp->qp_num);
|
iser_info("remote qpn:%x my qpn:%x\n", attr.dest_qp_num, cma_id->qp->qp_num);
|
||||||
|
|
||||||
|
if (private_data) {
|
||||||
|
u8 flags = *(u8 *)private_data;
|
||||||
|
|
||||||
|
iser_conn->snd_w_inv = !(flags & ISER_SEND_W_INV_NOT_SUP);
|
||||||
|
}
|
||||||
|
|
||||||
|
iser_info("conn %p: negotiated %s invalidation\n",
|
||||||
|
iser_conn, iser_conn->snd_w_inv ? "remote" : "local");
|
||||||
|
|
||||||
iser_conn->state = ISER_CONN_UP;
|
iser_conn->state = ISER_CONN_UP;
|
||||||
complete(&iser_conn->up_completion);
|
complete(&iser_conn->up_completion);
|
||||||
}
|
}
|
||||||
|
@ -928,7 +904,7 @@ static int iser_cma_handler(struct rdma_cm_id *cma_id, struct rdma_cm_event *eve
|
||||||
iser_route_handler(cma_id);
|
iser_route_handler(cma_id);
|
||||||
break;
|
break;
|
||||||
case RDMA_CM_EVENT_ESTABLISHED:
|
case RDMA_CM_EVENT_ESTABLISHED:
|
||||||
iser_connected_handler(cma_id);
|
iser_connected_handler(cma_id, event->param.conn.private_data);
|
||||||
break;
|
break;
|
||||||
case RDMA_CM_EVENT_ADDR_ERROR:
|
case RDMA_CM_EVENT_ADDR_ERROR:
|
||||||
case RDMA_CM_EVENT_ROUTE_ERROR:
|
case RDMA_CM_EVENT_ROUTE_ERROR:
|
||||||
|
@ -967,14 +943,21 @@ static int iser_cma_handler(struct rdma_cm_id *cma_id, struct rdma_cm_event *eve
|
||||||
|
|
||||||
void iser_conn_init(struct iser_conn *iser_conn)
|
void iser_conn_init(struct iser_conn *iser_conn)
|
||||||
{
|
{
|
||||||
|
struct ib_conn *ib_conn = &iser_conn->ib_conn;
|
||||||
|
|
||||||
iser_conn->state = ISER_CONN_INIT;
|
iser_conn->state = ISER_CONN_INIT;
|
||||||
iser_conn->ib_conn.post_recv_buf_count = 0;
|
|
||||||
init_completion(&iser_conn->ib_conn.flush_comp);
|
|
||||||
init_completion(&iser_conn->stop_completion);
|
init_completion(&iser_conn->stop_completion);
|
||||||
init_completion(&iser_conn->ib_completion);
|
init_completion(&iser_conn->ib_completion);
|
||||||
init_completion(&iser_conn->up_completion);
|
init_completion(&iser_conn->up_completion);
|
||||||
INIT_LIST_HEAD(&iser_conn->conn_list);
|
INIT_LIST_HEAD(&iser_conn->conn_list);
|
||||||
mutex_init(&iser_conn->state_mutex);
|
mutex_init(&iser_conn->state_mutex);
|
||||||
|
|
||||||
|
ib_conn->post_recv_buf_count = 0;
|
||||||
|
ib_conn->reg_cqe.done = iser_reg_comp;
|
||||||
|
ib_conn->last_cqe.done = iser_last_comp;
|
||||||
|
ib_conn->last.wr_cqe = &ib_conn->last_cqe;
|
||||||
|
ib_conn->last.opcode = IB_WR_SEND;
|
||||||
|
init_completion(&ib_conn->last_comp);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1000,9 +983,6 @@ int iser_connect(struct iser_conn *iser_conn,
|
||||||
|
|
||||||
iser_conn->state = ISER_CONN_PENDING;
|
iser_conn->state = ISER_CONN_PENDING;
|
||||||
|
|
||||||
ib_conn->beacon.wr_id = ISER_BEACON_WRID;
|
|
||||||
ib_conn->beacon.opcode = IB_WR_SEND;
|
|
||||||
|
|
||||||
ib_conn->cma_id = rdma_create_id(&init_net, iser_cma_handler,
|
ib_conn->cma_id = rdma_create_id(&init_net, iser_cma_handler,
|
||||||
(void *)iser_conn,
|
(void *)iser_conn,
|
||||||
RDMA_PS_TCP, IB_QPT_RC);
|
RDMA_PS_TCP, IB_QPT_RC);
|
||||||
|
@ -1045,56 +1025,60 @@ int iser_connect(struct iser_conn *iser_conn,
|
||||||
|
|
||||||
int iser_post_recvl(struct iser_conn *iser_conn)
|
int iser_post_recvl(struct iser_conn *iser_conn)
|
||||||
{
|
{
|
||||||
struct ib_recv_wr rx_wr, *rx_wr_failed;
|
|
||||||
struct ib_conn *ib_conn = &iser_conn->ib_conn;
|
struct ib_conn *ib_conn = &iser_conn->ib_conn;
|
||||||
struct ib_sge sge;
|
struct iser_login_desc *desc = &iser_conn->login_desc;
|
||||||
|
struct ib_recv_wr wr, *wr_failed;
|
||||||
int ib_ret;
|
int ib_ret;
|
||||||
|
|
||||||
sge.addr = iser_conn->login_resp_dma;
|
desc->sge.addr = desc->rsp_dma;
|
||||||
sge.length = ISER_RX_LOGIN_SIZE;
|
desc->sge.length = ISER_RX_LOGIN_SIZE;
|
||||||
sge.lkey = ib_conn->device->pd->local_dma_lkey;
|
desc->sge.lkey = ib_conn->device->pd->local_dma_lkey;
|
||||||
|
|
||||||
rx_wr.wr_id = (uintptr_t)iser_conn->login_resp_buf;
|
desc->cqe.done = iser_login_rsp;
|
||||||
rx_wr.sg_list = &sge;
|
wr.wr_cqe = &desc->cqe;
|
||||||
rx_wr.num_sge = 1;
|
wr.sg_list = &desc->sge;
|
||||||
rx_wr.next = NULL;
|
wr.num_sge = 1;
|
||||||
|
wr.next = NULL;
|
||||||
|
|
||||||
ib_conn->post_recv_buf_count++;
|
ib_conn->post_recv_buf_count++;
|
||||||
ib_ret = ib_post_recv(ib_conn->qp, &rx_wr, &rx_wr_failed);
|
ib_ret = ib_post_recv(ib_conn->qp, &wr, &wr_failed);
|
||||||
if (ib_ret) {
|
if (ib_ret) {
|
||||||
iser_err("ib_post_recv failed ret=%d\n", ib_ret);
|
iser_err("ib_post_recv failed ret=%d\n", ib_ret);
|
||||||
ib_conn->post_recv_buf_count--;
|
ib_conn->post_recv_buf_count--;
|
||||||
}
|
}
|
||||||
|
|
||||||
return ib_ret;
|
return ib_ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
int iser_post_recvm(struct iser_conn *iser_conn, int count)
|
int iser_post_recvm(struct iser_conn *iser_conn, int count)
|
||||||
{
|
{
|
||||||
struct ib_recv_wr *rx_wr, *rx_wr_failed;
|
|
||||||
int i, ib_ret;
|
|
||||||
struct ib_conn *ib_conn = &iser_conn->ib_conn;
|
struct ib_conn *ib_conn = &iser_conn->ib_conn;
|
||||||
unsigned int my_rx_head = iser_conn->rx_desc_head;
|
unsigned int my_rx_head = iser_conn->rx_desc_head;
|
||||||
struct iser_rx_desc *rx_desc;
|
struct iser_rx_desc *rx_desc;
|
||||||
|
struct ib_recv_wr *wr, *wr_failed;
|
||||||
|
int i, ib_ret;
|
||||||
|
|
||||||
for (rx_wr = ib_conn->rx_wr, i = 0; i < count; i++, rx_wr++) {
|
for (wr = ib_conn->rx_wr, i = 0; i < count; i++, wr++) {
|
||||||
rx_desc = &iser_conn->rx_descs[my_rx_head];
|
rx_desc = &iser_conn->rx_descs[my_rx_head];
|
||||||
rx_wr->wr_id = (uintptr_t)rx_desc;
|
rx_desc->cqe.done = iser_task_rsp;
|
||||||
rx_wr->sg_list = &rx_desc->rx_sg;
|
wr->wr_cqe = &rx_desc->cqe;
|
||||||
rx_wr->num_sge = 1;
|
wr->sg_list = &rx_desc->rx_sg;
|
||||||
rx_wr->next = rx_wr + 1;
|
wr->num_sge = 1;
|
||||||
|
wr->next = wr + 1;
|
||||||
my_rx_head = (my_rx_head + 1) & iser_conn->qp_max_recv_dtos_mask;
|
my_rx_head = (my_rx_head + 1) & iser_conn->qp_max_recv_dtos_mask;
|
||||||
}
|
}
|
||||||
|
|
||||||
rx_wr--;
|
wr--;
|
||||||
rx_wr->next = NULL; /* mark end of work requests list */
|
wr->next = NULL; /* mark end of work requests list */
|
||||||
|
|
||||||
ib_conn->post_recv_buf_count += count;
|
ib_conn->post_recv_buf_count += count;
|
||||||
ib_ret = ib_post_recv(ib_conn->qp, ib_conn->rx_wr, &rx_wr_failed);
|
ib_ret = ib_post_recv(ib_conn->qp, ib_conn->rx_wr, &wr_failed);
|
||||||
if (ib_ret) {
|
if (ib_ret) {
|
||||||
iser_err("ib_post_recv failed ret=%d\n", ib_ret);
|
iser_err("ib_post_recv failed ret=%d\n", ib_ret);
|
||||||
ib_conn->post_recv_buf_count -= count;
|
ib_conn->post_recv_buf_count -= count;
|
||||||
} else
|
} else
|
||||||
iser_conn->rx_desc_head = my_rx_head;
|
iser_conn->rx_desc_head = my_rx_head;
|
||||||
|
|
||||||
return ib_ret;
|
return ib_ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1115,7 +1099,7 @@ int iser_post_send(struct ib_conn *ib_conn, struct iser_tx_desc *tx_desc,
|
||||||
DMA_TO_DEVICE);
|
DMA_TO_DEVICE);
|
||||||
|
|
||||||
wr->next = NULL;
|
wr->next = NULL;
|
||||||
wr->wr_id = (uintptr_t)tx_desc;
|
wr->wr_cqe = &tx_desc->cqe;
|
||||||
wr->sg_list = tx_desc->tx_sg;
|
wr->sg_list = tx_desc->tx_sg;
|
||||||
wr->num_sge = tx_desc->num_sge;
|
wr->num_sge = tx_desc->num_sge;
|
||||||
wr->opcode = IB_WR_SEND;
|
wr->opcode = IB_WR_SEND;
|
||||||
|
@ -1129,149 +1113,6 @@ int iser_post_send(struct ib_conn *ib_conn, struct iser_tx_desc *tx_desc,
|
||||||
return ib_ret;
|
return ib_ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* is_iser_tx_desc - Indicate if the completion wr_id
|
|
||||||
* is a TX descriptor or not.
|
|
||||||
* @iser_conn: iser connection
|
|
||||||
* @wr_id: completion WR identifier
|
|
||||||
*
|
|
||||||
* Since we cannot rely on wc opcode in FLUSH errors
|
|
||||||
* we must work around it by checking if the wr_id address
|
|
||||||
* falls in the iser connection rx_descs buffer. If so
|
|
||||||
* it is an RX descriptor, otherwize it is a TX.
|
|
||||||
*/
|
|
||||||
static inline bool
|
|
||||||
is_iser_tx_desc(struct iser_conn *iser_conn, void *wr_id)
|
|
||||||
{
|
|
||||||
void *start = iser_conn->rx_descs;
|
|
||||||
int len = iser_conn->num_rx_descs * sizeof(*iser_conn->rx_descs);
|
|
||||||
|
|
||||||
if (wr_id >= start && wr_id < start + len)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* iser_handle_comp_error() - Handle error completion
|
|
||||||
* @ib_conn: connection RDMA resources
|
|
||||||
* @wc: work completion
|
|
||||||
*
|
|
||||||
* Notes: We may handle a FLUSH error completion and in this case
|
|
||||||
* we only cleanup in case TX type was DATAOUT. For non-FLUSH
|
|
||||||
* error completion we should also notify iscsi layer that
|
|
||||||
* connection is failed (in case we passed bind stage).
|
|
||||||
*/
|
|
||||||
static void
|
|
||||||
iser_handle_comp_error(struct ib_conn *ib_conn,
|
|
||||||
struct ib_wc *wc)
|
|
||||||
{
|
|
||||||
void *wr_id = (void *)(uintptr_t)wc->wr_id;
|
|
||||||
struct iser_conn *iser_conn = container_of(ib_conn, struct iser_conn,
|
|
||||||
ib_conn);
|
|
||||||
|
|
||||||
if (wc->status != IB_WC_WR_FLUSH_ERR)
|
|
||||||
if (iser_conn->iscsi_conn)
|
|
||||||
iscsi_conn_failure(iser_conn->iscsi_conn,
|
|
||||||
ISCSI_ERR_CONN_FAILED);
|
|
||||||
|
|
||||||
if (wc->wr_id == ISER_FASTREG_LI_WRID)
|
|
||||||
return;
|
|
||||||
|
|
||||||
if (is_iser_tx_desc(iser_conn, wr_id)) {
|
|
||||||
struct iser_tx_desc *desc = wr_id;
|
|
||||||
|
|
||||||
if (desc->type == ISCSI_TX_DATAOUT)
|
|
||||||
kmem_cache_free(ig.desc_cache, desc);
|
|
||||||
} else {
|
|
||||||
ib_conn->post_recv_buf_count--;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* iser_handle_wc - handle a single work completion
|
|
||||||
* @wc: work completion
|
|
||||||
*
|
|
||||||
* Soft-IRQ context, work completion can be either
|
|
||||||
* SEND or RECV, and can turn out successful or
|
|
||||||
* with error (or flush error).
|
|
||||||
*/
|
|
||||||
static void iser_handle_wc(struct ib_wc *wc)
|
|
||||||
{
|
|
||||||
struct ib_conn *ib_conn;
|
|
||||||
struct iser_tx_desc *tx_desc;
|
|
||||||
struct iser_rx_desc *rx_desc;
|
|
||||||
|
|
||||||
ib_conn = wc->qp->qp_context;
|
|
||||||
if (likely(wc->status == IB_WC_SUCCESS)) {
|
|
||||||
if (wc->opcode == IB_WC_RECV) {
|
|
||||||
rx_desc = (struct iser_rx_desc *)(uintptr_t)wc->wr_id;
|
|
||||||
iser_rcv_completion(rx_desc, wc->byte_len,
|
|
||||||
ib_conn);
|
|
||||||
} else
|
|
||||||
if (wc->opcode == IB_WC_SEND) {
|
|
||||||
tx_desc = (struct iser_tx_desc *)(uintptr_t)wc->wr_id;
|
|
||||||
iser_snd_completion(tx_desc, ib_conn);
|
|
||||||
} else {
|
|
||||||
iser_err("Unknown wc opcode %d\n", wc->opcode);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (wc->status != IB_WC_WR_FLUSH_ERR)
|
|
||||||
iser_err("%s (%d): wr id %llx vend_err %x\n",
|
|
||||||
ib_wc_status_msg(wc->status), wc->status,
|
|
||||||
wc->wr_id, wc->vendor_err);
|
|
||||||
else
|
|
||||||
iser_dbg("%s (%d): wr id %llx\n",
|
|
||||||
ib_wc_status_msg(wc->status), wc->status,
|
|
||||||
wc->wr_id);
|
|
||||||
|
|
||||||
if (wc->wr_id == ISER_BEACON_WRID)
|
|
||||||
/* all flush errors were consumed */
|
|
||||||
complete(&ib_conn->flush_comp);
|
|
||||||
else
|
|
||||||
iser_handle_comp_error(ib_conn, wc);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* iser_cq_tasklet_fn - iSER completion polling loop
|
|
||||||
* @data: iSER completion context
|
|
||||||
*
|
|
||||||
* Soft-IRQ context, polling connection CQ until
|
|
||||||
* either CQ was empty or we exausted polling budget
|
|
||||||
*/
|
|
||||||
static void iser_cq_tasklet_fn(unsigned long data)
|
|
||||||
{
|
|
||||||
struct iser_comp *comp = (struct iser_comp *)data;
|
|
||||||
struct ib_cq *cq = comp->cq;
|
|
||||||
struct ib_wc *const wcs = comp->wcs;
|
|
||||||
int i, n, completed = 0;
|
|
||||||
|
|
||||||
while ((n = ib_poll_cq(cq, ARRAY_SIZE(comp->wcs), wcs)) > 0) {
|
|
||||||
for (i = 0; i < n; i++)
|
|
||||||
iser_handle_wc(&wcs[i]);
|
|
||||||
|
|
||||||
completed += n;
|
|
||||||
if (completed >= iser_cq_poll_limit)
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* It is assumed here that arming CQ only once its empty
|
|
||||||
* would not cause interrupts to be missed.
|
|
||||||
*/
|
|
||||||
ib_req_notify_cq(cq, IB_CQ_NEXT_COMP);
|
|
||||||
|
|
||||||
iser_dbg("got %d completions\n", completed);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void iser_cq_callback(struct ib_cq *cq, void *cq_context)
|
|
||||||
{
|
|
||||||
struct iser_comp *comp = cq_context;
|
|
||||||
|
|
||||||
tasklet_schedule(&comp->tasklet);
|
|
||||||
}
|
|
||||||
|
|
||||||
u8 iser_check_task_pi_status(struct iscsi_iser_task *iser_task,
|
u8 iser_check_task_pi_status(struct iscsi_iser_task *iser_task,
|
||||||
enum iser_data_dir cmd_dir, sector_t *sector)
|
enum iser_data_dir cmd_dir, sector_t *sector)
|
||||||
{
|
{
|
||||||
|
@ -1319,3 +1160,21 @@ u8 iser_check_task_pi_status(struct iscsi_iser_task *iser_task,
|
||||||
/* Not alot we can do here, return ambiguous guard error */
|
/* Not alot we can do here, return ambiguous guard error */
|
||||||
return 0x1;
|
return 0x1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void iser_err_comp(struct ib_wc *wc, const char *type)
|
||||||
|
{
|
||||||
|
if (wc->status != IB_WC_WR_FLUSH_ERR) {
|
||||||
|
struct iser_conn *iser_conn = to_iser_conn(wc->qp->qp_context);
|
||||||
|
|
||||||
|
iser_err("%s failure: %s (%d) vend_err %x\n", type,
|
||||||
|
ib_wc_status_msg(wc->status), wc->status,
|
||||||
|
wc->vendor_err);
|
||||||
|
|
||||||
|
if (iser_conn->iscsi_conn)
|
||||||
|
iscsi_conn_failure(iser_conn->iscsi_conn,
|
||||||
|
ISCSI_ERR_CONN_FAILED);
|
||||||
|
} else {
|
||||||
|
iser_dbg("%s failure: %s (%d)\n", type,
|
||||||
|
ib_wc_status_msg(wc->status), wc->status);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -29,7 +29,6 @@
|
||||||
#include <target/iscsi/iscsi_transport.h>
|
#include <target/iscsi/iscsi_transport.h>
|
||||||
#include <linux/semaphore.h>
|
#include <linux/semaphore.h>
|
||||||
|
|
||||||
#include "isert_proto.h"
|
|
||||||
#include "ib_isert.h"
|
#include "ib_isert.h"
|
||||||
|
|
||||||
#define ISERT_MAX_CONN 8
|
#define ISERT_MAX_CONN 8
|
||||||
|
@ -95,22 +94,6 @@ isert_qp_event_callback(struct ib_event *e, void *context)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
|
||||||
isert_query_device(struct ib_device *ib_dev, struct ib_device_attr *devattr)
|
|
||||||
{
|
|
||||||
int ret;
|
|
||||||
|
|
||||||
ret = ib_query_device(ib_dev, devattr);
|
|
||||||
if (ret) {
|
|
||||||
isert_err("ib_query_device() failed: %d\n", ret);
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
isert_dbg("devattr->max_sge: %d\n", devattr->max_sge);
|
|
||||||
isert_dbg("devattr->max_sge_rd: %d\n", devattr->max_sge_rd);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct isert_comp *
|
static struct isert_comp *
|
||||||
isert_comp_get(struct isert_conn *isert_conn)
|
isert_comp_get(struct isert_conn *isert_conn)
|
||||||
{
|
{
|
||||||
|
@ -157,9 +140,9 @@ isert_create_qp(struct isert_conn *isert_conn,
|
||||||
attr.recv_cq = comp->cq;
|
attr.recv_cq = comp->cq;
|
||||||
attr.cap.max_send_wr = ISERT_QP_MAX_REQ_DTOS;
|
attr.cap.max_send_wr = ISERT_QP_MAX_REQ_DTOS;
|
||||||
attr.cap.max_recv_wr = ISERT_QP_MAX_RECV_DTOS + 1;
|
attr.cap.max_recv_wr = ISERT_QP_MAX_RECV_DTOS + 1;
|
||||||
attr.cap.max_send_sge = device->dev_attr.max_sge;
|
attr.cap.max_send_sge = device->ib_device->attrs.max_sge;
|
||||||
isert_conn->max_sge = min(device->dev_attr.max_sge,
|
isert_conn->max_sge = min(device->ib_device->attrs.max_sge,
|
||||||
device->dev_attr.max_sge_rd);
|
device->ib_device->attrs.max_sge_rd);
|
||||||
attr.cap.max_recv_sge = 1;
|
attr.cap.max_recv_sge = 1;
|
||||||
attr.sq_sig_type = IB_SIGNAL_REQ_WR;
|
attr.sq_sig_type = IB_SIGNAL_REQ_WR;
|
||||||
attr.qp_type = IB_QPT_RC;
|
attr.qp_type = IB_QPT_RC;
|
||||||
|
@ -287,8 +270,7 @@ isert_free_comps(struct isert_device *device)
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
isert_alloc_comps(struct isert_device *device,
|
isert_alloc_comps(struct isert_device *device)
|
||||||
struct ib_device_attr *attr)
|
|
||||||
{
|
{
|
||||||
int i, max_cqe, ret = 0;
|
int i, max_cqe, ret = 0;
|
||||||
|
|
||||||
|
@ -308,7 +290,7 @@ isert_alloc_comps(struct isert_device *device,
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
}
|
||||||
|
|
||||||
max_cqe = min(ISER_MAX_CQ_LEN, attr->max_cqe);
|
max_cqe = min(ISER_MAX_CQ_LEN, device->ib_device->attrs.max_cqe);
|
||||||
|
|
||||||
for (i = 0; i < device->comps_used; i++) {
|
for (i = 0; i < device->comps_used; i++) {
|
||||||
struct ib_cq_init_attr cq_attr = {};
|
struct ib_cq_init_attr cq_attr = {};
|
||||||
|
@ -344,17 +326,15 @@ isert_alloc_comps(struct isert_device *device,
|
||||||
static int
|
static int
|
||||||
isert_create_device_ib_res(struct isert_device *device)
|
isert_create_device_ib_res(struct isert_device *device)
|
||||||
{
|
{
|
||||||
struct ib_device_attr *dev_attr;
|
struct ib_device *ib_dev = device->ib_device;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
dev_attr = &device->dev_attr;
|
isert_dbg("devattr->max_sge: %d\n", ib_dev->attrs.max_sge);
|
||||||
ret = isert_query_device(device->ib_device, dev_attr);
|
isert_dbg("devattr->max_sge_rd: %d\n", ib_dev->attrs.max_sge_rd);
|
||||||
if (ret)
|
|
||||||
goto out;
|
|
||||||
|
|
||||||
/* asign function handlers */
|
/* asign function handlers */
|
||||||
if (dev_attr->device_cap_flags & IB_DEVICE_MEM_MGT_EXTENSIONS &&
|
if (ib_dev->attrs.device_cap_flags & IB_DEVICE_MEM_MGT_EXTENSIONS &&
|
||||||
dev_attr->device_cap_flags & IB_DEVICE_SIGNATURE_HANDOVER) {
|
ib_dev->attrs.device_cap_flags & IB_DEVICE_SIGNATURE_HANDOVER) {
|
||||||
device->use_fastreg = 1;
|
device->use_fastreg = 1;
|
||||||
device->reg_rdma_mem = isert_reg_rdma;
|
device->reg_rdma_mem = isert_reg_rdma;
|
||||||
device->unreg_rdma_mem = isert_unreg_rdma;
|
device->unreg_rdma_mem = isert_unreg_rdma;
|
||||||
|
@ -364,11 +344,11 @@ isert_create_device_ib_res(struct isert_device *device)
|
||||||
device->unreg_rdma_mem = isert_unmap_cmd;
|
device->unreg_rdma_mem = isert_unmap_cmd;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = isert_alloc_comps(device, dev_attr);
|
ret = isert_alloc_comps(device);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
device->pd = ib_alloc_pd(device->ib_device);
|
device->pd = ib_alloc_pd(ib_dev);
|
||||||
if (IS_ERR(device->pd)) {
|
if (IS_ERR(device->pd)) {
|
||||||
ret = PTR_ERR(device->pd);
|
ret = PTR_ERR(device->pd);
|
||||||
isert_err("failed to allocate pd, device %p, ret=%d\n",
|
isert_err("failed to allocate pd, device %p, ret=%d\n",
|
||||||
|
@ -377,7 +357,7 @@ isert_create_device_ib_res(struct isert_device *device)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Check signature cap */
|
/* Check signature cap */
|
||||||
device->pi_capable = dev_attr->device_cap_flags &
|
device->pi_capable = ib_dev->attrs.device_cap_flags &
|
||||||
IB_DEVICE_SIGNATURE_HANDOVER ? true : false;
|
IB_DEVICE_SIGNATURE_HANDOVER ? true : false;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -676,6 +656,32 @@ isert_alloc_login_buf(struct isert_conn *isert_conn,
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
isert_set_nego_params(struct isert_conn *isert_conn,
|
||||||
|
struct rdma_conn_param *param)
|
||||||
|
{
|
||||||
|
struct ib_device_attr *attr = &isert_conn->device->ib_device->attrs;
|
||||||
|
|
||||||
|
/* Set max inflight RDMA READ requests */
|
||||||
|
isert_conn->initiator_depth = min_t(u8, param->initiator_depth,
|
||||||
|
attr->max_qp_init_rd_atom);
|
||||||
|
isert_dbg("Using initiator_depth: %u\n", isert_conn->initiator_depth);
|
||||||
|
|
||||||
|
if (param->private_data) {
|
||||||
|
u8 flags = *(u8 *)param->private_data;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* use remote invalidation if the both initiator
|
||||||
|
* and the HCA support it
|
||||||
|
*/
|
||||||
|
isert_conn->snd_w_inv = !(flags & ISER_SEND_W_INV_NOT_SUP) &&
|
||||||
|
(attr->device_cap_flags &
|
||||||
|
IB_DEVICE_MEM_MGT_EXTENSIONS);
|
||||||
|
if (isert_conn->snd_w_inv)
|
||||||
|
isert_info("Using remote invalidation\n");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
isert_connect_request(struct rdma_cm_id *cma_id, struct rdma_cm_event *event)
|
isert_connect_request(struct rdma_cm_id *cma_id, struct rdma_cm_event *event)
|
||||||
{
|
{
|
||||||
|
@ -714,11 +720,7 @@ isert_connect_request(struct rdma_cm_id *cma_id, struct rdma_cm_event *event)
|
||||||
}
|
}
|
||||||
isert_conn->device = device;
|
isert_conn->device = device;
|
||||||
|
|
||||||
/* Set max inflight RDMA READ requests */
|
isert_set_nego_params(isert_conn, &event->param.conn);
|
||||||
isert_conn->initiator_depth = min_t(u8,
|
|
||||||
event->param.conn.initiator_depth,
|
|
||||||
device->dev_attr.max_qp_init_rd_atom);
|
|
||||||
isert_dbg("Using initiator_depth: %u\n", isert_conn->initiator_depth);
|
|
||||||
|
|
||||||
ret = isert_conn_setup_qp(isert_conn, cma_id);
|
ret = isert_conn_setup_qp(isert_conn, cma_id);
|
||||||
if (ret)
|
if (ret)
|
||||||
|
@ -1050,8 +1052,8 @@ isert_create_send_desc(struct isert_conn *isert_conn,
|
||||||
ib_dma_sync_single_for_cpu(ib_dev, tx_desc->dma_addr,
|
ib_dma_sync_single_for_cpu(ib_dev, tx_desc->dma_addr,
|
||||||
ISER_HEADERS_LEN, DMA_TO_DEVICE);
|
ISER_HEADERS_LEN, DMA_TO_DEVICE);
|
||||||
|
|
||||||
memset(&tx_desc->iser_header, 0, sizeof(struct iser_hdr));
|
memset(&tx_desc->iser_header, 0, sizeof(struct iser_ctrl));
|
||||||
tx_desc->iser_header.flags = ISER_VER;
|
tx_desc->iser_header.flags = ISCSI_CTRL;
|
||||||
|
|
||||||
tx_desc->num_sge = 1;
|
tx_desc->num_sge = 1;
|
||||||
tx_desc->isert_cmd = isert_cmd;
|
tx_desc->isert_cmd = isert_cmd;
|
||||||
|
@ -1097,7 +1099,14 @@ isert_init_send_wr(struct isert_conn *isert_conn, struct isert_cmd *isert_cmd,
|
||||||
|
|
||||||
isert_cmd->rdma_wr.iser_ib_op = ISER_IB_SEND;
|
isert_cmd->rdma_wr.iser_ib_op = ISER_IB_SEND;
|
||||||
send_wr->wr_id = (uintptr_t)&isert_cmd->tx_desc;
|
send_wr->wr_id = (uintptr_t)&isert_cmd->tx_desc;
|
||||||
send_wr->opcode = IB_WR_SEND;
|
|
||||||
|
if (isert_conn->snd_w_inv && isert_cmd->inv_rkey) {
|
||||||
|
send_wr->opcode = IB_WR_SEND_WITH_INV;
|
||||||
|
send_wr->ex.invalidate_rkey = isert_cmd->inv_rkey;
|
||||||
|
} else {
|
||||||
|
send_wr->opcode = IB_WR_SEND;
|
||||||
|
}
|
||||||
|
|
||||||
send_wr->sg_list = &tx_desc->tx_sg[0];
|
send_wr->sg_list = &tx_desc->tx_sg[0];
|
||||||
send_wr->num_sge = isert_cmd->tx_desc.num_sge;
|
send_wr->num_sge = isert_cmd->tx_desc.num_sge;
|
||||||
send_wr->send_flags = IB_SEND_SIGNALED;
|
send_wr->send_flags = IB_SEND_SIGNALED;
|
||||||
|
@ -1486,6 +1495,7 @@ isert_rx_opcode(struct isert_conn *isert_conn, struct iser_rx_desc *rx_desc,
|
||||||
isert_cmd->read_va = read_va;
|
isert_cmd->read_va = read_va;
|
||||||
isert_cmd->write_stag = write_stag;
|
isert_cmd->write_stag = write_stag;
|
||||||
isert_cmd->write_va = write_va;
|
isert_cmd->write_va = write_va;
|
||||||
|
isert_cmd->inv_rkey = read_stag ? read_stag : write_stag;
|
||||||
|
|
||||||
ret = isert_handle_scsi_cmd(isert_conn, isert_cmd, cmd,
|
ret = isert_handle_scsi_cmd(isert_conn, isert_cmd, cmd,
|
||||||
rx_desc, (unsigned char *)hdr);
|
rx_desc, (unsigned char *)hdr);
|
||||||
|
@ -1543,21 +1553,21 @@ isert_rx_opcode(struct isert_conn *isert_conn, struct iser_rx_desc *rx_desc,
|
||||||
static void
|
static void
|
||||||
isert_rx_do_work(struct iser_rx_desc *rx_desc, struct isert_conn *isert_conn)
|
isert_rx_do_work(struct iser_rx_desc *rx_desc, struct isert_conn *isert_conn)
|
||||||
{
|
{
|
||||||
struct iser_hdr *iser_hdr = &rx_desc->iser_header;
|
struct iser_ctrl *iser_ctrl = &rx_desc->iser_header;
|
||||||
uint64_t read_va = 0, write_va = 0;
|
uint64_t read_va = 0, write_va = 0;
|
||||||
uint32_t read_stag = 0, write_stag = 0;
|
uint32_t read_stag = 0, write_stag = 0;
|
||||||
|
|
||||||
switch (iser_hdr->flags & 0xF0) {
|
switch (iser_ctrl->flags & 0xF0) {
|
||||||
case ISCSI_CTRL:
|
case ISCSI_CTRL:
|
||||||
if (iser_hdr->flags & ISER_RSV) {
|
if (iser_ctrl->flags & ISER_RSV) {
|
||||||
read_stag = be32_to_cpu(iser_hdr->read_stag);
|
read_stag = be32_to_cpu(iser_ctrl->read_stag);
|
||||||
read_va = be64_to_cpu(iser_hdr->read_va);
|
read_va = be64_to_cpu(iser_ctrl->read_va);
|
||||||
isert_dbg("ISER_RSV: read_stag: 0x%x read_va: 0x%llx\n",
|
isert_dbg("ISER_RSV: read_stag: 0x%x read_va: 0x%llx\n",
|
||||||
read_stag, (unsigned long long)read_va);
|
read_stag, (unsigned long long)read_va);
|
||||||
}
|
}
|
||||||
if (iser_hdr->flags & ISER_WSV) {
|
if (iser_ctrl->flags & ISER_WSV) {
|
||||||
write_stag = be32_to_cpu(iser_hdr->write_stag);
|
write_stag = be32_to_cpu(iser_ctrl->write_stag);
|
||||||
write_va = be64_to_cpu(iser_hdr->write_va);
|
write_va = be64_to_cpu(iser_ctrl->write_va);
|
||||||
isert_dbg("ISER_WSV: write_stag: 0x%x write_va: 0x%llx\n",
|
isert_dbg("ISER_WSV: write_stag: 0x%x write_va: 0x%llx\n",
|
||||||
write_stag, (unsigned long long)write_va);
|
write_stag, (unsigned long long)write_va);
|
||||||
}
|
}
|
||||||
|
@ -1568,7 +1578,7 @@ isert_rx_do_work(struct iser_rx_desc *rx_desc, struct isert_conn *isert_conn)
|
||||||
isert_err("iSER Hello message\n");
|
isert_err("iSER Hello message\n");
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
isert_warn("Unknown iSER hdr flags: 0x%02x\n", iser_hdr->flags);
|
isert_warn("Unknown iSER hdr flags: 0x%02x\n", iser_ctrl->flags);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3095,12 +3105,20 @@ isert_rdma_accept(struct isert_conn *isert_conn)
|
||||||
struct rdma_cm_id *cm_id = isert_conn->cm_id;
|
struct rdma_cm_id *cm_id = isert_conn->cm_id;
|
||||||
struct rdma_conn_param cp;
|
struct rdma_conn_param cp;
|
||||||
int ret;
|
int ret;
|
||||||
|
struct iser_cm_hdr rsp_hdr;
|
||||||
|
|
||||||
memset(&cp, 0, sizeof(struct rdma_conn_param));
|
memset(&cp, 0, sizeof(struct rdma_conn_param));
|
||||||
cp.initiator_depth = isert_conn->initiator_depth;
|
cp.initiator_depth = isert_conn->initiator_depth;
|
||||||
cp.retry_count = 7;
|
cp.retry_count = 7;
|
||||||
cp.rnr_retry_count = 7;
|
cp.rnr_retry_count = 7;
|
||||||
|
|
||||||
|
memset(&rsp_hdr, 0, sizeof(rsp_hdr));
|
||||||
|
rsp_hdr.flags = ISERT_ZBVA_NOT_USED;
|
||||||
|
if (!isert_conn->snd_w_inv)
|
||||||
|
rsp_hdr.flags = rsp_hdr.flags | ISERT_SEND_W_INV_NOT_USED;
|
||||||
|
cp.private_data = (void *)&rsp_hdr;
|
||||||
|
cp.private_data_len = sizeof(rsp_hdr);
|
||||||
|
|
||||||
ret = rdma_accept(cm_id, &cp);
|
ret = rdma_accept(cm_id, &cp);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
isert_err("rdma_accept() failed with: %d\n", ret);
|
isert_err("rdma_accept() failed with: %d\n", ret);
|
||||||
|
|
|
@ -3,6 +3,8 @@
|
||||||
#include <linux/in6.h>
|
#include <linux/in6.h>
|
||||||
#include <rdma/ib_verbs.h>
|
#include <rdma/ib_verbs.h>
|
||||||
#include <rdma/rdma_cm.h>
|
#include <rdma/rdma_cm.h>
|
||||||
|
#include <scsi/iser.h>
|
||||||
|
|
||||||
|
|
||||||
#define DRV_NAME "isert"
|
#define DRV_NAME "isert"
|
||||||
#define PFX DRV_NAME ": "
|
#define PFX DRV_NAME ": "
|
||||||
|
@ -31,6 +33,38 @@
|
||||||
#define isert_err(fmt, arg...) \
|
#define isert_err(fmt, arg...) \
|
||||||
pr_err(PFX "%s: " fmt, __func__ , ## arg)
|
pr_err(PFX "%s: " fmt, __func__ , ## arg)
|
||||||
|
|
||||||
|
/* Constant PDU lengths calculations */
|
||||||
|
#define ISER_HEADERS_LEN (sizeof(struct iser_ctrl) + \
|
||||||
|
sizeof(struct iscsi_hdr))
|
||||||
|
#define ISER_RECV_DATA_SEG_LEN 8192
|
||||||
|
#define ISER_RX_PAYLOAD_SIZE (ISER_HEADERS_LEN + ISER_RECV_DATA_SEG_LEN)
|
||||||
|
#define ISER_RX_LOGIN_SIZE (ISER_HEADERS_LEN + ISCSI_DEF_MAX_RECV_SEG_LEN)
|
||||||
|
|
||||||
|
/* QP settings */
|
||||||
|
/* Maximal bounds on received asynchronous PDUs */
|
||||||
|
#define ISERT_MAX_TX_MISC_PDUS 4 /* NOOP_IN(2) , ASYNC_EVENT(2) */
|
||||||
|
|
||||||
|
#define ISERT_MAX_RX_MISC_PDUS 6 /*
|
||||||
|
* NOOP_OUT(2), TEXT(1),
|
||||||
|
* SCSI_TMFUNC(2), LOGOUT(1)
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define ISCSI_DEF_XMIT_CMDS_MAX 128 /* from libiscsi.h, must be power of 2 */
|
||||||
|
|
||||||
|
#define ISERT_QP_MAX_RECV_DTOS (ISCSI_DEF_XMIT_CMDS_MAX)
|
||||||
|
|
||||||
|
#define ISERT_MIN_POSTED_RX (ISCSI_DEF_XMIT_CMDS_MAX >> 2)
|
||||||
|
|
||||||
|
#define ISERT_INFLIGHT_DATAOUTS 8
|
||||||
|
|
||||||
|
#define ISERT_QP_MAX_REQ_DTOS (ISCSI_DEF_XMIT_CMDS_MAX * \
|
||||||
|
(1 + ISERT_INFLIGHT_DATAOUTS) + \
|
||||||
|
ISERT_MAX_TX_MISC_PDUS + \
|
||||||
|
ISERT_MAX_RX_MISC_PDUS)
|
||||||
|
|
||||||
|
#define ISER_RX_PAD_SIZE (ISER_RECV_DATA_SEG_LEN + 4096 - \
|
||||||
|
(ISER_RX_PAYLOAD_SIZE + sizeof(u64) + sizeof(struct ib_sge)))
|
||||||
|
|
||||||
#define ISCSI_ISER_SG_TABLESIZE 256
|
#define ISCSI_ISER_SG_TABLESIZE 256
|
||||||
#define ISER_FASTREG_LI_WRID 0xffffffffffffffffULL
|
#define ISER_FASTREG_LI_WRID 0xffffffffffffffffULL
|
||||||
#define ISER_BEACON_WRID 0xfffffffffffffffeULL
|
#define ISER_BEACON_WRID 0xfffffffffffffffeULL
|
||||||
|
@ -56,7 +90,7 @@ enum iser_conn_state {
|
||||||
};
|
};
|
||||||
|
|
||||||
struct iser_rx_desc {
|
struct iser_rx_desc {
|
||||||
struct iser_hdr iser_header;
|
struct iser_ctrl iser_header;
|
||||||
struct iscsi_hdr iscsi_header;
|
struct iscsi_hdr iscsi_header;
|
||||||
char data[ISER_RECV_DATA_SEG_LEN];
|
char data[ISER_RECV_DATA_SEG_LEN];
|
||||||
u64 dma_addr;
|
u64 dma_addr;
|
||||||
|
@ -65,7 +99,7 @@ struct iser_rx_desc {
|
||||||
} __packed;
|
} __packed;
|
||||||
|
|
||||||
struct iser_tx_desc {
|
struct iser_tx_desc {
|
||||||
struct iser_hdr iser_header;
|
struct iser_ctrl iser_header;
|
||||||
struct iscsi_hdr iscsi_header;
|
struct iscsi_hdr iscsi_header;
|
||||||
enum isert_desc_type type;
|
enum isert_desc_type type;
|
||||||
u64 dma_addr;
|
u64 dma_addr;
|
||||||
|
@ -129,6 +163,7 @@ struct isert_cmd {
|
||||||
uint32_t write_stag;
|
uint32_t write_stag;
|
||||||
uint64_t read_va;
|
uint64_t read_va;
|
||||||
uint64_t write_va;
|
uint64_t write_va;
|
||||||
|
uint32_t inv_rkey;
|
||||||
u64 pdu_buf_dma;
|
u64 pdu_buf_dma;
|
||||||
u32 pdu_buf_len;
|
u32 pdu_buf_len;
|
||||||
struct isert_conn *conn;
|
struct isert_conn *conn;
|
||||||
|
@ -176,6 +211,7 @@ struct isert_conn {
|
||||||
struct work_struct release_work;
|
struct work_struct release_work;
|
||||||
struct ib_recv_wr beacon;
|
struct ib_recv_wr beacon;
|
||||||
bool logout_posted;
|
bool logout_posted;
|
||||||
|
bool snd_w_inv;
|
||||||
};
|
};
|
||||||
|
|
||||||
#define ISERT_MAX_CQ 64
|
#define ISERT_MAX_CQ 64
|
||||||
|
@ -207,7 +243,6 @@ struct isert_device {
|
||||||
struct isert_comp *comps;
|
struct isert_comp *comps;
|
||||||
int comps_used;
|
int comps_used;
|
||||||
struct list_head dev_node;
|
struct list_head dev_node;
|
||||||
struct ib_device_attr dev_attr;
|
|
||||||
int (*reg_rdma_mem)(struct iscsi_conn *conn,
|
int (*reg_rdma_mem)(struct iscsi_conn *conn,
|
||||||
struct iscsi_cmd *cmd,
|
struct iscsi_cmd *cmd,
|
||||||
struct isert_rdma_wr *wr);
|
struct isert_rdma_wr *wr);
|
||||||
|
|
|
@ -1,47 +0,0 @@
|
||||||
/* From iscsi_iser.h */
|
|
||||||
|
|
||||||
struct iser_hdr {
|
|
||||||
u8 flags;
|
|
||||||
u8 rsvd[3];
|
|
||||||
__be32 write_stag; /* write rkey */
|
|
||||||
__be64 write_va;
|
|
||||||
__be32 read_stag; /* read rkey */
|
|
||||||
__be64 read_va;
|
|
||||||
} __packed;
|
|
||||||
|
|
||||||
/*Constant PDU lengths calculations */
|
|
||||||
#define ISER_HEADERS_LEN (sizeof(struct iser_hdr) + sizeof(struct iscsi_hdr))
|
|
||||||
|
|
||||||
#define ISER_RECV_DATA_SEG_LEN 8192
|
|
||||||
#define ISER_RX_PAYLOAD_SIZE (ISER_HEADERS_LEN + ISER_RECV_DATA_SEG_LEN)
|
|
||||||
#define ISER_RX_LOGIN_SIZE (ISER_HEADERS_LEN + ISCSI_DEF_MAX_RECV_SEG_LEN)
|
|
||||||
|
|
||||||
/* QP settings */
|
|
||||||
/* Maximal bounds on received asynchronous PDUs */
|
|
||||||
#define ISERT_MAX_TX_MISC_PDUS 4 /* NOOP_IN(2) , ASYNC_EVENT(2) */
|
|
||||||
|
|
||||||
#define ISERT_MAX_RX_MISC_PDUS 6 /* NOOP_OUT(2), TEXT(1), *
|
|
||||||
* SCSI_TMFUNC(2), LOGOUT(1) */
|
|
||||||
|
|
||||||
#define ISCSI_DEF_XMIT_CMDS_MAX 128 /* from libiscsi.h, must be power of 2 */
|
|
||||||
|
|
||||||
#define ISERT_QP_MAX_RECV_DTOS (ISCSI_DEF_XMIT_CMDS_MAX)
|
|
||||||
|
|
||||||
#define ISERT_MIN_POSTED_RX (ISCSI_DEF_XMIT_CMDS_MAX >> 2)
|
|
||||||
|
|
||||||
#define ISERT_INFLIGHT_DATAOUTS 8
|
|
||||||
|
|
||||||
#define ISERT_QP_MAX_REQ_DTOS (ISCSI_DEF_XMIT_CMDS_MAX * \
|
|
||||||
(1 + ISERT_INFLIGHT_DATAOUTS) + \
|
|
||||||
ISERT_MAX_TX_MISC_PDUS + \
|
|
||||||
ISERT_MAX_RX_MISC_PDUS)
|
|
||||||
|
|
||||||
#define ISER_RX_PAD_SIZE (ISER_RECV_DATA_SEG_LEN + 4096 - \
|
|
||||||
(ISER_RX_PAYLOAD_SIZE + sizeof(u64) + sizeof(struct ib_sge)))
|
|
||||||
|
|
||||||
#define ISER_VER 0x10
|
|
||||||
#define ISER_WSV 0x08
|
|
||||||
#define ISER_RSV 0x04
|
|
||||||
#define ISCSI_CTRL 0x10
|
|
||||||
#define ISER_HELLO 0x20
|
|
||||||
#define ISER_HELLORPLY 0x30
|
|
|
@ -132,8 +132,9 @@ MODULE_PARM_DESC(ch_count,
|
||||||
|
|
||||||
static void srp_add_one(struct ib_device *device);
|
static void srp_add_one(struct ib_device *device);
|
||||||
static void srp_remove_one(struct ib_device *device, void *client_data);
|
static void srp_remove_one(struct ib_device *device, void *client_data);
|
||||||
static void srp_recv_completion(struct ib_cq *cq, void *ch_ptr);
|
static void srp_recv_done(struct ib_cq *cq, struct ib_wc *wc);
|
||||||
static void srp_send_completion(struct ib_cq *cq, void *ch_ptr);
|
static void srp_handle_qp_err(struct ib_cq *cq, struct ib_wc *wc,
|
||||||
|
const char *opname);
|
||||||
static int srp_cm_handler(struct ib_cm_id *cm_id, struct ib_cm_event *event);
|
static int srp_cm_handler(struct ib_cm_id *cm_id, struct ib_cm_event *event);
|
||||||
|
|
||||||
static struct scsi_transport_template *ib_srp_transport_template;
|
static struct scsi_transport_template *ib_srp_transport_template;
|
||||||
|
@ -445,6 +446,17 @@ static struct srp_fr_pool *srp_alloc_fr_pool(struct srp_target_port *target)
|
||||||
dev->max_pages_per_mr);
|
dev->max_pages_per_mr);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void srp_drain_done(struct ib_cq *cq, struct ib_wc *wc)
|
||||||
|
{
|
||||||
|
struct srp_rdma_ch *ch = cq->cq_context;
|
||||||
|
|
||||||
|
complete(&ch->done);
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct ib_cqe srp_drain_cqe = {
|
||||||
|
.done = srp_drain_done,
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* srp_destroy_qp() - destroy an RDMA queue pair
|
* srp_destroy_qp() - destroy an RDMA queue pair
|
||||||
* @ch: SRP RDMA channel.
|
* @ch: SRP RDMA channel.
|
||||||
|
@ -457,10 +469,11 @@ static struct srp_fr_pool *srp_alloc_fr_pool(struct srp_target_port *target)
|
||||||
static void srp_destroy_qp(struct srp_rdma_ch *ch)
|
static void srp_destroy_qp(struct srp_rdma_ch *ch)
|
||||||
{
|
{
|
||||||
static struct ib_qp_attr attr = { .qp_state = IB_QPS_ERR };
|
static struct ib_qp_attr attr = { .qp_state = IB_QPS_ERR };
|
||||||
static struct ib_recv_wr wr = { .wr_id = SRP_LAST_WR_ID };
|
static struct ib_recv_wr wr = { 0 };
|
||||||
struct ib_recv_wr *bad_wr;
|
struct ib_recv_wr *bad_wr;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
wr.wr_cqe = &srp_drain_cqe;
|
||||||
/* Destroying a QP and reusing ch->done is only safe if not connected */
|
/* Destroying a QP and reusing ch->done is only safe if not connected */
|
||||||
WARN_ON_ONCE(ch->connected);
|
WARN_ON_ONCE(ch->connected);
|
||||||
|
|
||||||
|
@ -489,34 +502,27 @@ static int srp_create_ch_ib(struct srp_rdma_ch *ch)
|
||||||
struct ib_fmr_pool *fmr_pool = NULL;
|
struct ib_fmr_pool *fmr_pool = NULL;
|
||||||
struct srp_fr_pool *fr_pool = NULL;
|
struct srp_fr_pool *fr_pool = NULL;
|
||||||
const int m = dev->use_fast_reg ? 3 : 1;
|
const int m = dev->use_fast_reg ? 3 : 1;
|
||||||
struct ib_cq_init_attr cq_attr = {};
|
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
init_attr = kzalloc(sizeof *init_attr, GFP_KERNEL);
|
init_attr = kzalloc(sizeof *init_attr, GFP_KERNEL);
|
||||||
if (!init_attr)
|
if (!init_attr)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
/* + 1 for SRP_LAST_WR_ID */
|
/* queue_size + 1 for ib_drain_qp */
|
||||||
cq_attr.cqe = target->queue_size + 1;
|
recv_cq = ib_alloc_cq(dev->dev, ch, target->queue_size + 1,
|
||||||
cq_attr.comp_vector = ch->comp_vector;
|
ch->comp_vector, IB_POLL_SOFTIRQ);
|
||||||
recv_cq = ib_create_cq(dev->dev, srp_recv_completion, NULL, ch,
|
|
||||||
&cq_attr);
|
|
||||||
if (IS_ERR(recv_cq)) {
|
if (IS_ERR(recv_cq)) {
|
||||||
ret = PTR_ERR(recv_cq);
|
ret = PTR_ERR(recv_cq);
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
cq_attr.cqe = m * target->queue_size;
|
send_cq = ib_alloc_cq(dev->dev, ch, m * target->queue_size,
|
||||||
cq_attr.comp_vector = ch->comp_vector;
|
ch->comp_vector, IB_POLL_DIRECT);
|
||||||
send_cq = ib_create_cq(dev->dev, srp_send_completion, NULL, ch,
|
|
||||||
&cq_attr);
|
|
||||||
if (IS_ERR(send_cq)) {
|
if (IS_ERR(send_cq)) {
|
||||||
ret = PTR_ERR(send_cq);
|
ret = PTR_ERR(send_cq);
|
||||||
goto err_recv_cq;
|
goto err_recv_cq;
|
||||||
}
|
}
|
||||||
|
|
||||||
ib_req_notify_cq(recv_cq, IB_CQ_NEXT_COMP);
|
|
||||||
|
|
||||||
init_attr->event_handler = srp_qp_event;
|
init_attr->event_handler = srp_qp_event;
|
||||||
init_attr->cap.max_send_wr = m * target->queue_size;
|
init_attr->cap.max_send_wr = m * target->queue_size;
|
||||||
init_attr->cap.max_recv_wr = target->queue_size + 1;
|
init_attr->cap.max_recv_wr = target->queue_size + 1;
|
||||||
|
@ -558,9 +564,9 @@ static int srp_create_ch_ib(struct srp_rdma_ch *ch)
|
||||||
if (ch->qp)
|
if (ch->qp)
|
||||||
srp_destroy_qp(ch);
|
srp_destroy_qp(ch);
|
||||||
if (ch->recv_cq)
|
if (ch->recv_cq)
|
||||||
ib_destroy_cq(ch->recv_cq);
|
ib_free_cq(ch->recv_cq);
|
||||||
if (ch->send_cq)
|
if (ch->send_cq)
|
||||||
ib_destroy_cq(ch->send_cq);
|
ib_free_cq(ch->send_cq);
|
||||||
|
|
||||||
ch->qp = qp;
|
ch->qp = qp;
|
||||||
ch->recv_cq = recv_cq;
|
ch->recv_cq = recv_cq;
|
||||||
|
@ -580,13 +586,13 @@ static int srp_create_ch_ib(struct srp_rdma_ch *ch)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
err_qp:
|
err_qp:
|
||||||
ib_destroy_qp(qp);
|
srp_destroy_qp(ch);
|
||||||
|
|
||||||
err_send_cq:
|
err_send_cq:
|
||||||
ib_destroy_cq(send_cq);
|
ib_free_cq(send_cq);
|
||||||
|
|
||||||
err_recv_cq:
|
err_recv_cq:
|
||||||
ib_destroy_cq(recv_cq);
|
ib_free_cq(recv_cq);
|
||||||
|
|
||||||
err:
|
err:
|
||||||
kfree(init_attr);
|
kfree(init_attr);
|
||||||
|
@ -622,9 +628,10 @@ static void srp_free_ch_ib(struct srp_target_port *target,
|
||||||
if (ch->fmr_pool)
|
if (ch->fmr_pool)
|
||||||
ib_destroy_fmr_pool(ch->fmr_pool);
|
ib_destroy_fmr_pool(ch->fmr_pool);
|
||||||
}
|
}
|
||||||
|
|
||||||
srp_destroy_qp(ch);
|
srp_destroy_qp(ch);
|
||||||
ib_destroy_cq(ch->send_cq);
|
ib_free_cq(ch->send_cq);
|
||||||
ib_destroy_cq(ch->recv_cq);
|
ib_free_cq(ch->recv_cq);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Avoid that the SCSI error handler tries to use this channel after
|
* Avoid that the SCSI error handler tries to use this channel after
|
||||||
|
@ -1041,18 +1048,25 @@ static int srp_connect_ch(struct srp_rdma_ch *ch, bool multich)
|
||||||
return ret <= 0 ? ret : -ENODEV;
|
return ret <= 0 ? ret : -ENODEV;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int srp_inv_rkey(struct srp_rdma_ch *ch, u32 rkey)
|
static void srp_inv_rkey_err_done(struct ib_cq *cq, struct ib_wc *wc)
|
||||||
|
{
|
||||||
|
srp_handle_qp_err(cq, wc, "INV RKEY");
|
||||||
|
}
|
||||||
|
|
||||||
|
static int srp_inv_rkey(struct srp_request *req, struct srp_rdma_ch *ch,
|
||||||
|
u32 rkey)
|
||||||
{
|
{
|
||||||
struct ib_send_wr *bad_wr;
|
struct ib_send_wr *bad_wr;
|
||||||
struct ib_send_wr wr = {
|
struct ib_send_wr wr = {
|
||||||
.opcode = IB_WR_LOCAL_INV,
|
.opcode = IB_WR_LOCAL_INV,
|
||||||
.wr_id = LOCAL_INV_WR_ID_MASK,
|
|
||||||
.next = NULL,
|
.next = NULL,
|
||||||
.num_sge = 0,
|
.num_sge = 0,
|
||||||
.send_flags = 0,
|
.send_flags = 0,
|
||||||
.ex.invalidate_rkey = rkey,
|
.ex.invalidate_rkey = rkey,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
wr.wr_cqe = &req->reg_cqe;
|
||||||
|
req->reg_cqe.done = srp_inv_rkey_err_done;
|
||||||
return ib_post_send(ch->qp, &wr, &bad_wr);
|
return ib_post_send(ch->qp, &wr, &bad_wr);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1074,7 +1088,7 @@ static void srp_unmap_data(struct scsi_cmnd *scmnd,
|
||||||
struct srp_fr_desc **pfr;
|
struct srp_fr_desc **pfr;
|
||||||
|
|
||||||
for (i = req->nmdesc, pfr = req->fr_list; i > 0; i--, pfr++) {
|
for (i = req->nmdesc, pfr = req->fr_list; i > 0; i--, pfr++) {
|
||||||
res = srp_inv_rkey(ch, (*pfr)->mr->rkey);
|
res = srp_inv_rkey(req, ch, (*pfr)->mr->rkey);
|
||||||
if (res < 0) {
|
if (res < 0) {
|
||||||
shost_printk(KERN_ERR, target->scsi_host, PFX
|
shost_printk(KERN_ERR, target->scsi_host, PFX
|
||||||
"Queueing INV WR for rkey %#x failed (%d)\n",
|
"Queueing INV WR for rkey %#x failed (%d)\n",
|
||||||
|
@ -1312,7 +1326,13 @@ static int srp_map_finish_fmr(struct srp_map_state *state,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void srp_reg_mr_err_done(struct ib_cq *cq, struct ib_wc *wc)
|
||||||
|
{
|
||||||
|
srp_handle_qp_err(cq, wc, "FAST REG");
|
||||||
|
}
|
||||||
|
|
||||||
static int srp_map_finish_fr(struct srp_map_state *state,
|
static int srp_map_finish_fr(struct srp_map_state *state,
|
||||||
|
struct srp_request *req,
|
||||||
struct srp_rdma_ch *ch, int sg_nents)
|
struct srp_rdma_ch *ch, int sg_nents)
|
||||||
{
|
{
|
||||||
struct srp_target_port *target = ch->target;
|
struct srp_target_port *target = ch->target;
|
||||||
|
@ -1349,9 +1369,11 @@ static int srp_map_finish_fr(struct srp_map_state *state,
|
||||||
if (unlikely(n < 0))
|
if (unlikely(n < 0))
|
||||||
return n;
|
return n;
|
||||||
|
|
||||||
|
req->reg_cqe.done = srp_reg_mr_err_done;
|
||||||
|
|
||||||
wr.wr.next = NULL;
|
wr.wr.next = NULL;
|
||||||
wr.wr.opcode = IB_WR_REG_MR;
|
wr.wr.opcode = IB_WR_REG_MR;
|
||||||
wr.wr.wr_id = FAST_REG_WR_ID_MASK;
|
wr.wr.wr_cqe = &req->reg_cqe;
|
||||||
wr.wr.num_sge = 0;
|
wr.wr.num_sge = 0;
|
||||||
wr.wr.send_flags = 0;
|
wr.wr.send_flags = 0;
|
||||||
wr.mr = desc->mr;
|
wr.mr = desc->mr;
|
||||||
|
@ -1455,7 +1477,7 @@ static int srp_map_sg_fr(struct srp_map_state *state, struct srp_rdma_ch *ch,
|
||||||
while (count) {
|
while (count) {
|
||||||
int i, n;
|
int i, n;
|
||||||
|
|
||||||
n = srp_map_finish_fr(state, ch, count);
|
n = srp_map_finish_fr(state, req, ch, count);
|
||||||
if (unlikely(n < 0))
|
if (unlikely(n < 0))
|
||||||
return n;
|
return n;
|
||||||
|
|
||||||
|
@ -1524,7 +1546,7 @@ static int srp_map_idb(struct srp_rdma_ch *ch, struct srp_request *req,
|
||||||
#ifdef CONFIG_NEED_SG_DMA_LENGTH
|
#ifdef CONFIG_NEED_SG_DMA_LENGTH
|
||||||
idb_sg->dma_length = idb_sg->length; /* hack^2 */
|
idb_sg->dma_length = idb_sg->length; /* hack^2 */
|
||||||
#endif
|
#endif
|
||||||
ret = srp_map_finish_fr(&state, ch, 1);
|
ret = srp_map_finish_fr(&state, req, ch, 1);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return ret;
|
return ret;
|
||||||
} else if (dev->use_fmr) {
|
} else if (dev->use_fmr) {
|
||||||
|
@ -1719,7 +1741,7 @@ static struct srp_iu *__srp_get_tx_iu(struct srp_rdma_ch *ch,
|
||||||
s32 rsv = (iu_type == SRP_IU_TSK_MGMT) ? 0 : SRP_TSK_MGMT_SQ_SIZE;
|
s32 rsv = (iu_type == SRP_IU_TSK_MGMT) ? 0 : SRP_TSK_MGMT_SQ_SIZE;
|
||||||
struct srp_iu *iu;
|
struct srp_iu *iu;
|
||||||
|
|
||||||
srp_send_completion(ch->send_cq, ch);
|
ib_process_cq_direct(ch->send_cq, -1);
|
||||||
|
|
||||||
if (list_empty(&ch->free_tx))
|
if (list_empty(&ch->free_tx))
|
||||||
return NULL;
|
return NULL;
|
||||||
|
@ -1739,6 +1761,19 @@ static struct srp_iu *__srp_get_tx_iu(struct srp_rdma_ch *ch,
|
||||||
return iu;
|
return iu;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void srp_send_done(struct ib_cq *cq, struct ib_wc *wc)
|
||||||
|
{
|
||||||
|
struct srp_iu *iu = container_of(wc->wr_cqe, struct srp_iu, cqe);
|
||||||
|
struct srp_rdma_ch *ch = cq->cq_context;
|
||||||
|
|
||||||
|
if (unlikely(wc->status != IB_WC_SUCCESS)) {
|
||||||
|
srp_handle_qp_err(cq, wc, "SEND");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
list_add(&iu->list, &ch->free_tx);
|
||||||
|
}
|
||||||
|
|
||||||
static int srp_post_send(struct srp_rdma_ch *ch, struct srp_iu *iu, int len)
|
static int srp_post_send(struct srp_rdma_ch *ch, struct srp_iu *iu, int len)
|
||||||
{
|
{
|
||||||
struct srp_target_port *target = ch->target;
|
struct srp_target_port *target = ch->target;
|
||||||
|
@ -1749,8 +1784,10 @@ static int srp_post_send(struct srp_rdma_ch *ch, struct srp_iu *iu, int len)
|
||||||
list.length = len;
|
list.length = len;
|
||||||
list.lkey = target->lkey;
|
list.lkey = target->lkey;
|
||||||
|
|
||||||
|
iu->cqe.done = srp_send_done;
|
||||||
|
|
||||||
wr.next = NULL;
|
wr.next = NULL;
|
||||||
wr.wr_id = (uintptr_t) iu;
|
wr.wr_cqe = &iu->cqe;
|
||||||
wr.sg_list = &list;
|
wr.sg_list = &list;
|
||||||
wr.num_sge = 1;
|
wr.num_sge = 1;
|
||||||
wr.opcode = IB_WR_SEND;
|
wr.opcode = IB_WR_SEND;
|
||||||
|
@ -1769,8 +1806,10 @@ static int srp_post_recv(struct srp_rdma_ch *ch, struct srp_iu *iu)
|
||||||
list.length = iu->size;
|
list.length = iu->size;
|
||||||
list.lkey = target->lkey;
|
list.lkey = target->lkey;
|
||||||
|
|
||||||
|
iu->cqe.done = srp_recv_done;
|
||||||
|
|
||||||
wr.next = NULL;
|
wr.next = NULL;
|
||||||
wr.wr_id = (uintptr_t) iu;
|
wr.wr_cqe = &iu->cqe;
|
||||||
wr.sg_list = &list;
|
wr.sg_list = &list;
|
||||||
wr.num_sge = 1;
|
wr.num_sge = 1;
|
||||||
|
|
||||||
|
@ -1902,14 +1941,20 @@ static void srp_process_aer_req(struct srp_rdma_ch *ch,
|
||||||
"problems processing SRP_AER_REQ\n");
|
"problems processing SRP_AER_REQ\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
static void srp_handle_recv(struct srp_rdma_ch *ch, struct ib_wc *wc)
|
static void srp_recv_done(struct ib_cq *cq, struct ib_wc *wc)
|
||||||
{
|
{
|
||||||
|
struct srp_iu *iu = container_of(wc->wr_cqe, struct srp_iu, cqe);
|
||||||
|
struct srp_rdma_ch *ch = cq->cq_context;
|
||||||
struct srp_target_port *target = ch->target;
|
struct srp_target_port *target = ch->target;
|
||||||
struct ib_device *dev = target->srp_host->srp_dev->dev;
|
struct ib_device *dev = target->srp_host->srp_dev->dev;
|
||||||
struct srp_iu *iu = (struct srp_iu *) (uintptr_t) wc->wr_id;
|
|
||||||
int res;
|
int res;
|
||||||
u8 opcode;
|
u8 opcode;
|
||||||
|
|
||||||
|
if (unlikely(wc->status != IB_WC_SUCCESS)) {
|
||||||
|
srp_handle_qp_err(cq, wc, "RECV");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
ib_dma_sync_single_for_cpu(dev, iu->dma, ch->max_ti_iu_len,
|
ib_dma_sync_single_for_cpu(dev, iu->dma, ch->max_ti_iu_len,
|
||||||
DMA_FROM_DEVICE);
|
DMA_FROM_DEVICE);
|
||||||
|
|
||||||
|
@ -1972,68 +2017,22 @@ static void srp_tl_err_work(struct work_struct *work)
|
||||||
srp_start_tl_fail_timers(target->rport);
|
srp_start_tl_fail_timers(target->rport);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void srp_handle_qp_err(u64 wr_id, enum ib_wc_status wc_status,
|
static void srp_handle_qp_err(struct ib_cq *cq, struct ib_wc *wc,
|
||||||
bool send_err, struct srp_rdma_ch *ch)
|
const char *opname)
|
||||||
{
|
{
|
||||||
|
struct srp_rdma_ch *ch = cq->cq_context;
|
||||||
struct srp_target_port *target = ch->target;
|
struct srp_target_port *target = ch->target;
|
||||||
|
|
||||||
if (wr_id == SRP_LAST_WR_ID) {
|
|
||||||
complete(&ch->done);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (ch->connected && !target->qp_in_error) {
|
if (ch->connected && !target->qp_in_error) {
|
||||||
if (wr_id & LOCAL_INV_WR_ID_MASK) {
|
shost_printk(KERN_ERR, target->scsi_host,
|
||||||
shost_printk(KERN_ERR, target->scsi_host, PFX
|
PFX "failed %s status %s (%d) for CQE %p\n",
|
||||||
"LOCAL_INV failed with status %s (%d)\n",
|
opname, ib_wc_status_msg(wc->status), wc->status,
|
||||||
ib_wc_status_msg(wc_status), wc_status);
|
wc->wr_cqe);
|
||||||
} else if (wr_id & FAST_REG_WR_ID_MASK) {
|
|
||||||
shost_printk(KERN_ERR, target->scsi_host, PFX
|
|
||||||
"FAST_REG_MR failed status %s (%d)\n",
|
|
||||||
ib_wc_status_msg(wc_status), wc_status);
|
|
||||||
} else {
|
|
||||||
shost_printk(KERN_ERR, target->scsi_host,
|
|
||||||
PFX "failed %s status %s (%d) for iu %p\n",
|
|
||||||
send_err ? "send" : "receive",
|
|
||||||
ib_wc_status_msg(wc_status), wc_status,
|
|
||||||
(void *)(uintptr_t)wr_id);
|
|
||||||
}
|
|
||||||
queue_work(system_long_wq, &target->tl_err_work);
|
queue_work(system_long_wq, &target->tl_err_work);
|
||||||
}
|
}
|
||||||
target->qp_in_error = true;
|
target->qp_in_error = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void srp_recv_completion(struct ib_cq *cq, void *ch_ptr)
|
|
||||||
{
|
|
||||||
struct srp_rdma_ch *ch = ch_ptr;
|
|
||||||
struct ib_wc wc;
|
|
||||||
|
|
||||||
ib_req_notify_cq(cq, IB_CQ_NEXT_COMP);
|
|
||||||
while (ib_poll_cq(cq, 1, &wc) > 0) {
|
|
||||||
if (likely(wc.status == IB_WC_SUCCESS)) {
|
|
||||||
srp_handle_recv(ch, &wc);
|
|
||||||
} else {
|
|
||||||
srp_handle_qp_err(wc.wr_id, wc.status, false, ch);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void srp_send_completion(struct ib_cq *cq, void *ch_ptr)
|
|
||||||
{
|
|
||||||
struct srp_rdma_ch *ch = ch_ptr;
|
|
||||||
struct ib_wc wc;
|
|
||||||
struct srp_iu *iu;
|
|
||||||
|
|
||||||
while (ib_poll_cq(cq, 1, &wc) > 0) {
|
|
||||||
if (likely(wc.status == IB_WC_SUCCESS)) {
|
|
||||||
iu = (struct srp_iu *) (uintptr_t) wc.wr_id;
|
|
||||||
list_add(&iu->list, &ch->free_tx);
|
|
||||||
} else {
|
|
||||||
srp_handle_qp_err(wc.wr_id, wc.status, true, ch);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static int srp_queuecommand(struct Scsi_Host *shost, struct scsi_cmnd *scmnd)
|
static int srp_queuecommand(struct Scsi_Host *shost, struct scsi_cmnd *scmnd)
|
||||||
{
|
{
|
||||||
struct srp_target_port *target = host_to_target(shost);
|
struct srp_target_port *target = host_to_target(shost);
|
||||||
|
@ -3439,27 +3438,17 @@ static struct srp_host *srp_add_port(struct srp_device *device, u8 port)
|
||||||
static void srp_add_one(struct ib_device *device)
|
static void srp_add_one(struct ib_device *device)
|
||||||
{
|
{
|
||||||
struct srp_device *srp_dev;
|
struct srp_device *srp_dev;
|
||||||
struct ib_device_attr *dev_attr;
|
|
||||||
struct srp_host *host;
|
struct srp_host *host;
|
||||||
int mr_page_shift, p;
|
int mr_page_shift, p;
|
||||||
u64 max_pages_per_mr;
|
u64 max_pages_per_mr;
|
||||||
|
|
||||||
dev_attr = kmalloc(sizeof *dev_attr, GFP_KERNEL);
|
|
||||||
if (!dev_attr)
|
|
||||||
return;
|
|
||||||
|
|
||||||
if (ib_query_device(device, dev_attr)) {
|
|
||||||
pr_warn("Query device failed for %s\n", device->name);
|
|
||||||
goto free_attr;
|
|
||||||
}
|
|
||||||
|
|
||||||
srp_dev = kmalloc(sizeof *srp_dev, GFP_KERNEL);
|
srp_dev = kmalloc(sizeof *srp_dev, GFP_KERNEL);
|
||||||
if (!srp_dev)
|
if (!srp_dev)
|
||||||
goto free_attr;
|
return;
|
||||||
|
|
||||||
srp_dev->has_fmr = (device->alloc_fmr && device->dealloc_fmr &&
|
srp_dev->has_fmr = (device->alloc_fmr && device->dealloc_fmr &&
|
||||||
device->map_phys_fmr && device->unmap_fmr);
|
device->map_phys_fmr && device->unmap_fmr);
|
||||||
srp_dev->has_fr = (dev_attr->device_cap_flags &
|
srp_dev->has_fr = (device->attrs.device_cap_flags &
|
||||||
IB_DEVICE_MEM_MGT_EXTENSIONS);
|
IB_DEVICE_MEM_MGT_EXTENSIONS);
|
||||||
if (!srp_dev->has_fmr && !srp_dev->has_fr)
|
if (!srp_dev->has_fmr && !srp_dev->has_fr)
|
||||||
dev_warn(&device->dev, "neither FMR nor FR is supported\n");
|
dev_warn(&device->dev, "neither FMR nor FR is supported\n");
|
||||||
|
@ -3473,23 +3462,23 @@ static void srp_add_one(struct ib_device *device)
|
||||||
* minimum of 4096 bytes. We're unlikely to build large sglists
|
* minimum of 4096 bytes. We're unlikely to build large sglists
|
||||||
* out of smaller entries.
|
* out of smaller entries.
|
||||||
*/
|
*/
|
||||||
mr_page_shift = max(12, ffs(dev_attr->page_size_cap) - 1);
|
mr_page_shift = max(12, ffs(device->attrs.page_size_cap) - 1);
|
||||||
srp_dev->mr_page_size = 1 << mr_page_shift;
|
srp_dev->mr_page_size = 1 << mr_page_shift;
|
||||||
srp_dev->mr_page_mask = ~((u64) srp_dev->mr_page_size - 1);
|
srp_dev->mr_page_mask = ~((u64) srp_dev->mr_page_size - 1);
|
||||||
max_pages_per_mr = dev_attr->max_mr_size;
|
max_pages_per_mr = device->attrs.max_mr_size;
|
||||||
do_div(max_pages_per_mr, srp_dev->mr_page_size);
|
do_div(max_pages_per_mr, srp_dev->mr_page_size);
|
||||||
srp_dev->max_pages_per_mr = min_t(u64, SRP_MAX_PAGES_PER_MR,
|
srp_dev->max_pages_per_mr = min_t(u64, SRP_MAX_PAGES_PER_MR,
|
||||||
max_pages_per_mr);
|
max_pages_per_mr);
|
||||||
if (srp_dev->use_fast_reg) {
|
if (srp_dev->use_fast_reg) {
|
||||||
srp_dev->max_pages_per_mr =
|
srp_dev->max_pages_per_mr =
|
||||||
min_t(u32, srp_dev->max_pages_per_mr,
|
min_t(u32, srp_dev->max_pages_per_mr,
|
||||||
dev_attr->max_fast_reg_page_list_len);
|
device->attrs.max_fast_reg_page_list_len);
|
||||||
}
|
}
|
||||||
srp_dev->mr_max_size = srp_dev->mr_page_size *
|
srp_dev->mr_max_size = srp_dev->mr_page_size *
|
||||||
srp_dev->max_pages_per_mr;
|
srp_dev->max_pages_per_mr;
|
||||||
pr_debug("%s: mr_page_shift = %d, dev_attr->max_mr_size = %#llx, dev_attr->max_fast_reg_page_list_len = %u, max_pages_per_mr = %d, mr_max_size = %#x\n",
|
pr_debug("%s: mr_page_shift = %d, device->max_mr_size = %#llx, device->max_fast_reg_page_list_len = %u, max_pages_per_mr = %d, mr_max_size = %#x\n",
|
||||||
device->name, mr_page_shift, dev_attr->max_mr_size,
|
device->name, mr_page_shift, device->attrs.max_mr_size,
|
||||||
dev_attr->max_fast_reg_page_list_len,
|
device->attrs.max_fast_reg_page_list_len,
|
||||||
srp_dev->max_pages_per_mr, srp_dev->mr_max_size);
|
srp_dev->max_pages_per_mr, srp_dev->mr_max_size);
|
||||||
|
|
||||||
INIT_LIST_HEAD(&srp_dev->dev_list);
|
INIT_LIST_HEAD(&srp_dev->dev_list);
|
||||||
|
@ -3517,17 +3506,13 @@ static void srp_add_one(struct ib_device *device)
|
||||||
}
|
}
|
||||||
|
|
||||||
ib_set_client_data(device, &srp_client, srp_dev);
|
ib_set_client_data(device, &srp_client, srp_dev);
|
||||||
|
return;
|
||||||
goto free_attr;
|
|
||||||
|
|
||||||
err_pd:
|
err_pd:
|
||||||
ib_dealloc_pd(srp_dev->pd);
|
ib_dealloc_pd(srp_dev->pd);
|
||||||
|
|
||||||
free_dev:
|
free_dev:
|
||||||
kfree(srp_dev);
|
kfree(srp_dev);
|
||||||
|
|
||||||
free_attr:
|
|
||||||
kfree(dev_attr);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void srp_remove_one(struct ib_device *device, void *client_data)
|
static void srp_remove_one(struct ib_device *device, void *client_data)
|
||||||
|
@ -3587,8 +3572,6 @@ static int __init srp_init_module(void)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
BUILD_BUG_ON(FIELD_SIZEOF(struct ib_wc, wr_id) < sizeof(void *));
|
|
||||||
|
|
||||||
if (srp_sg_tablesize) {
|
if (srp_sg_tablesize) {
|
||||||
pr_warn("srp_sg_tablesize is deprecated, please use cmd_sg_entries\n");
|
pr_warn("srp_sg_tablesize is deprecated, please use cmd_sg_entries\n");
|
||||||
if (!cmd_sg_entries)
|
if (!cmd_sg_entries)
|
||||||
|
|
|
@ -66,11 +66,6 @@ enum {
|
||||||
SRP_TAG_TSK_MGMT = 1U << 31,
|
SRP_TAG_TSK_MGMT = 1U << 31,
|
||||||
|
|
||||||
SRP_MAX_PAGES_PER_MR = 512,
|
SRP_MAX_PAGES_PER_MR = 512,
|
||||||
|
|
||||||
LOCAL_INV_WR_ID_MASK = 1,
|
|
||||||
FAST_REG_WR_ID_MASK = 2,
|
|
||||||
|
|
||||||
SRP_LAST_WR_ID = 0xfffffffcU,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
enum srp_target_state {
|
enum srp_target_state {
|
||||||
|
@ -128,6 +123,7 @@ struct srp_request {
|
||||||
struct srp_direct_buf *indirect_desc;
|
struct srp_direct_buf *indirect_desc;
|
||||||
dma_addr_t indirect_dma_addr;
|
dma_addr_t indirect_dma_addr;
|
||||||
short nmdesc;
|
short nmdesc;
|
||||||
|
struct ib_cqe reg_cqe;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -231,6 +227,7 @@ struct srp_iu {
|
||||||
void *buf;
|
void *buf;
|
||||||
size_t size;
|
size_t size;
|
||||||
enum dma_data_direction direction;
|
enum dma_data_direction direction;
|
||||||
|
struct ib_cqe cqe;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -93,6 +93,8 @@ MODULE_PARM_DESC(srpt_service_guid,
|
||||||
static struct ib_client srpt_client;
|
static struct ib_client srpt_client;
|
||||||
static void srpt_release_channel(struct srpt_rdma_ch *ch);
|
static void srpt_release_channel(struct srpt_rdma_ch *ch);
|
||||||
static int srpt_queue_status(struct se_cmd *cmd);
|
static int srpt_queue_status(struct se_cmd *cmd);
|
||||||
|
static void srpt_recv_done(struct ib_cq *cq, struct ib_wc *wc);
|
||||||
|
static void srpt_send_done(struct ib_cq *cq, struct ib_wc *wc);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* opposite_dma_dir() - Swap DMA_TO_DEVICE and DMA_FROM_DEVICE.
|
* opposite_dma_dir() - Swap DMA_TO_DEVICE and DMA_FROM_DEVICE.
|
||||||
|
@ -341,10 +343,10 @@ static void srpt_get_ioc(struct srpt_port *sport, u32 slot,
|
||||||
memset(iocp, 0, sizeof *iocp);
|
memset(iocp, 0, sizeof *iocp);
|
||||||
strcpy(iocp->id_string, SRPT_ID_STRING);
|
strcpy(iocp->id_string, SRPT_ID_STRING);
|
||||||
iocp->guid = cpu_to_be64(srpt_service_guid);
|
iocp->guid = cpu_to_be64(srpt_service_guid);
|
||||||
iocp->vendor_id = cpu_to_be32(sdev->dev_attr.vendor_id);
|
iocp->vendor_id = cpu_to_be32(sdev->device->attrs.vendor_id);
|
||||||
iocp->device_id = cpu_to_be32(sdev->dev_attr.vendor_part_id);
|
iocp->device_id = cpu_to_be32(sdev->device->attrs.vendor_part_id);
|
||||||
iocp->device_version = cpu_to_be16(sdev->dev_attr.hw_ver);
|
iocp->device_version = cpu_to_be16(sdev->device->attrs.hw_ver);
|
||||||
iocp->subsys_vendor_id = cpu_to_be32(sdev->dev_attr.vendor_id);
|
iocp->subsys_vendor_id = cpu_to_be32(sdev->device->attrs.vendor_id);
|
||||||
iocp->subsys_device_id = 0x0;
|
iocp->subsys_device_id = 0x0;
|
||||||
iocp->io_class = cpu_to_be16(SRP_REV16A_IB_IO_CLASS);
|
iocp->io_class = cpu_to_be16(SRP_REV16A_IB_IO_CLASS);
|
||||||
iocp->io_subclass = cpu_to_be16(SRP_IO_SUBCLASS);
|
iocp->io_subclass = cpu_to_be16(SRP_IO_SUBCLASS);
|
||||||
|
@ -453,6 +455,7 @@ static void srpt_mad_send_handler(struct ib_mad_agent *mad_agent,
|
||||||
* srpt_mad_recv_handler() - MAD reception callback function.
|
* srpt_mad_recv_handler() - MAD reception callback function.
|
||||||
*/
|
*/
|
||||||
static void srpt_mad_recv_handler(struct ib_mad_agent *mad_agent,
|
static void srpt_mad_recv_handler(struct ib_mad_agent *mad_agent,
|
||||||
|
struct ib_mad_send_buf *send_buf,
|
||||||
struct ib_mad_recv_wc *mad_wc)
|
struct ib_mad_recv_wc *mad_wc)
|
||||||
{
|
{
|
||||||
struct srpt_port *sport = (struct srpt_port *)mad_agent->context;
|
struct srpt_port *sport = (struct srpt_port *)mad_agent->context;
|
||||||
|
@ -778,12 +781,12 @@ static int srpt_post_recv(struct srpt_device *sdev,
|
||||||
struct ib_recv_wr wr, *bad_wr;
|
struct ib_recv_wr wr, *bad_wr;
|
||||||
|
|
||||||
BUG_ON(!sdev);
|
BUG_ON(!sdev);
|
||||||
wr.wr_id = encode_wr_id(SRPT_RECV, ioctx->ioctx.index);
|
|
||||||
|
|
||||||
list.addr = ioctx->ioctx.dma;
|
list.addr = ioctx->ioctx.dma;
|
||||||
list.length = srp_max_req_size;
|
list.length = srp_max_req_size;
|
||||||
list.lkey = sdev->pd->local_dma_lkey;
|
list.lkey = sdev->pd->local_dma_lkey;
|
||||||
|
|
||||||
|
ioctx->ioctx.cqe.done = srpt_recv_done;
|
||||||
|
wr.wr_cqe = &ioctx->ioctx.cqe;
|
||||||
wr.next = NULL;
|
wr.next = NULL;
|
||||||
wr.sg_list = &list;
|
wr.sg_list = &list;
|
||||||
wr.num_sge = 1;
|
wr.num_sge = 1;
|
||||||
|
@ -819,8 +822,9 @@ static int srpt_post_send(struct srpt_rdma_ch *ch,
|
||||||
list.length = len;
|
list.length = len;
|
||||||
list.lkey = sdev->pd->local_dma_lkey;
|
list.lkey = sdev->pd->local_dma_lkey;
|
||||||
|
|
||||||
|
ioctx->ioctx.cqe.done = srpt_send_done;
|
||||||
wr.next = NULL;
|
wr.next = NULL;
|
||||||
wr.wr_id = encode_wr_id(SRPT_SEND, ioctx->ioctx.index);
|
wr.wr_cqe = &ioctx->ioctx.cqe;
|
||||||
wr.sg_list = &list;
|
wr.sg_list = &list;
|
||||||
wr.num_sge = 1;
|
wr.num_sge = 1;
|
||||||
wr.opcode = IB_WR_SEND;
|
wr.opcode = IB_WR_SEND;
|
||||||
|
@ -1052,13 +1056,13 @@ static void srpt_unmap_sg_to_ib_sge(struct srpt_rdma_ch *ch,
|
||||||
|
|
||||||
BUG_ON(!ch);
|
BUG_ON(!ch);
|
||||||
BUG_ON(!ioctx);
|
BUG_ON(!ioctx);
|
||||||
BUG_ON(ioctx->n_rdma && !ioctx->rdma_ius);
|
BUG_ON(ioctx->n_rdma && !ioctx->rdma_wrs);
|
||||||
|
|
||||||
while (ioctx->n_rdma)
|
while (ioctx->n_rdma)
|
||||||
kfree(ioctx->rdma_ius[--ioctx->n_rdma].sge);
|
kfree(ioctx->rdma_wrs[--ioctx->n_rdma].wr.sg_list);
|
||||||
|
|
||||||
kfree(ioctx->rdma_ius);
|
kfree(ioctx->rdma_wrs);
|
||||||
ioctx->rdma_ius = NULL;
|
ioctx->rdma_wrs = NULL;
|
||||||
|
|
||||||
if (ioctx->mapped_sg_count) {
|
if (ioctx->mapped_sg_count) {
|
||||||
sg = ioctx->sg;
|
sg = ioctx->sg;
|
||||||
|
@ -1082,7 +1086,7 @@ static int srpt_map_sg_to_ib_sge(struct srpt_rdma_ch *ch,
|
||||||
struct scatterlist *sg, *sg_orig;
|
struct scatterlist *sg, *sg_orig;
|
||||||
int sg_cnt;
|
int sg_cnt;
|
||||||
enum dma_data_direction dir;
|
enum dma_data_direction dir;
|
||||||
struct rdma_iu *riu;
|
struct ib_rdma_wr *riu;
|
||||||
struct srp_direct_buf *db;
|
struct srp_direct_buf *db;
|
||||||
dma_addr_t dma_addr;
|
dma_addr_t dma_addr;
|
||||||
struct ib_sge *sge;
|
struct ib_sge *sge;
|
||||||
|
@ -1109,23 +1113,24 @@ static int srpt_map_sg_to_ib_sge(struct srpt_rdma_ch *ch,
|
||||||
|
|
||||||
ioctx->mapped_sg_count = count;
|
ioctx->mapped_sg_count = count;
|
||||||
|
|
||||||
if (ioctx->rdma_ius && ioctx->n_rdma_ius)
|
if (ioctx->rdma_wrs && ioctx->n_rdma_wrs)
|
||||||
nrdma = ioctx->n_rdma_ius;
|
nrdma = ioctx->n_rdma_wrs;
|
||||||
else {
|
else {
|
||||||
nrdma = (count + SRPT_DEF_SG_PER_WQE - 1) / SRPT_DEF_SG_PER_WQE
|
nrdma = (count + SRPT_DEF_SG_PER_WQE - 1) / SRPT_DEF_SG_PER_WQE
|
||||||
+ ioctx->n_rbuf;
|
+ ioctx->n_rbuf;
|
||||||
|
|
||||||
ioctx->rdma_ius = kzalloc(nrdma * sizeof *riu, GFP_KERNEL);
|
ioctx->rdma_wrs = kcalloc(nrdma, sizeof(*ioctx->rdma_wrs),
|
||||||
if (!ioctx->rdma_ius)
|
GFP_KERNEL);
|
||||||
|
if (!ioctx->rdma_wrs)
|
||||||
goto free_mem;
|
goto free_mem;
|
||||||
|
|
||||||
ioctx->n_rdma_ius = nrdma;
|
ioctx->n_rdma_wrs = nrdma;
|
||||||
}
|
}
|
||||||
|
|
||||||
db = ioctx->rbufs;
|
db = ioctx->rbufs;
|
||||||
tsize = cmd->data_length;
|
tsize = cmd->data_length;
|
||||||
dma_len = ib_sg_dma_len(dev, &sg[0]);
|
dma_len = ib_sg_dma_len(dev, &sg[0]);
|
||||||
riu = ioctx->rdma_ius;
|
riu = ioctx->rdma_wrs;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* For each remote desc - calculate the #ib_sge.
|
* For each remote desc - calculate the #ib_sge.
|
||||||
|
@ -1139,9 +1144,9 @@ static int srpt_map_sg_to_ib_sge(struct srpt_rdma_ch *ch,
|
||||||
j < count && i < ioctx->n_rbuf && tsize > 0; ++i, ++riu, ++db) {
|
j < count && i < ioctx->n_rbuf && tsize > 0; ++i, ++riu, ++db) {
|
||||||
rsize = be32_to_cpu(db->len);
|
rsize = be32_to_cpu(db->len);
|
||||||
raddr = be64_to_cpu(db->va);
|
raddr = be64_to_cpu(db->va);
|
||||||
riu->raddr = raddr;
|
riu->remote_addr = raddr;
|
||||||
riu->rkey = be32_to_cpu(db->key);
|
riu->rkey = be32_to_cpu(db->key);
|
||||||
riu->sge_cnt = 0;
|
riu->wr.num_sge = 0;
|
||||||
|
|
||||||
/* calculate how many sge required for this remote_buf */
|
/* calculate how many sge required for this remote_buf */
|
||||||
while (rsize > 0 && tsize > 0) {
|
while (rsize > 0 && tsize > 0) {
|
||||||
|
@ -1165,33 +1170,35 @@ static int srpt_map_sg_to_ib_sge(struct srpt_rdma_ch *ch,
|
||||||
rsize = 0;
|
rsize = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
++riu->sge_cnt;
|
++riu->wr.num_sge;
|
||||||
|
|
||||||
if (rsize > 0 && riu->sge_cnt == SRPT_DEF_SG_PER_WQE) {
|
if (rsize > 0 &&
|
||||||
|
riu->wr.num_sge == SRPT_DEF_SG_PER_WQE) {
|
||||||
++ioctx->n_rdma;
|
++ioctx->n_rdma;
|
||||||
riu->sge =
|
riu->wr.sg_list = kmalloc_array(riu->wr.num_sge,
|
||||||
kmalloc(riu->sge_cnt * sizeof *riu->sge,
|
sizeof(*riu->wr.sg_list),
|
||||||
GFP_KERNEL);
|
GFP_KERNEL);
|
||||||
if (!riu->sge)
|
if (!riu->wr.sg_list)
|
||||||
goto free_mem;
|
goto free_mem;
|
||||||
|
|
||||||
++riu;
|
++riu;
|
||||||
riu->sge_cnt = 0;
|
riu->wr.num_sge = 0;
|
||||||
riu->raddr = raddr;
|
riu->remote_addr = raddr;
|
||||||
riu->rkey = be32_to_cpu(db->key);
|
riu->rkey = be32_to_cpu(db->key);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
++ioctx->n_rdma;
|
++ioctx->n_rdma;
|
||||||
riu->sge = kmalloc(riu->sge_cnt * sizeof *riu->sge,
|
riu->wr.sg_list = kmalloc_array(riu->wr.num_sge,
|
||||||
GFP_KERNEL);
|
sizeof(*riu->wr.sg_list),
|
||||||
if (!riu->sge)
|
GFP_KERNEL);
|
||||||
|
if (!riu->wr.sg_list)
|
||||||
goto free_mem;
|
goto free_mem;
|
||||||
}
|
}
|
||||||
|
|
||||||
db = ioctx->rbufs;
|
db = ioctx->rbufs;
|
||||||
tsize = cmd->data_length;
|
tsize = cmd->data_length;
|
||||||
riu = ioctx->rdma_ius;
|
riu = ioctx->rdma_wrs;
|
||||||
sg = sg_orig;
|
sg = sg_orig;
|
||||||
dma_len = ib_sg_dma_len(dev, &sg[0]);
|
dma_len = ib_sg_dma_len(dev, &sg[0]);
|
||||||
dma_addr = ib_sg_dma_address(dev, &sg[0]);
|
dma_addr = ib_sg_dma_address(dev, &sg[0]);
|
||||||
|
@ -1200,7 +1207,7 @@ static int srpt_map_sg_to_ib_sge(struct srpt_rdma_ch *ch,
|
||||||
for (i = 0, j = 0;
|
for (i = 0, j = 0;
|
||||||
j < count && i < ioctx->n_rbuf && tsize > 0; ++i, ++riu, ++db) {
|
j < count && i < ioctx->n_rbuf && tsize > 0; ++i, ++riu, ++db) {
|
||||||
rsize = be32_to_cpu(db->len);
|
rsize = be32_to_cpu(db->len);
|
||||||
sge = riu->sge;
|
sge = riu->wr.sg_list;
|
||||||
k = 0;
|
k = 0;
|
||||||
|
|
||||||
while (rsize > 0 && tsize > 0) {
|
while (rsize > 0 && tsize > 0) {
|
||||||
|
@ -1232,9 +1239,9 @@ static int srpt_map_sg_to_ib_sge(struct srpt_rdma_ch *ch,
|
||||||
}
|
}
|
||||||
|
|
||||||
++k;
|
++k;
|
||||||
if (k == riu->sge_cnt && rsize > 0 && tsize > 0) {
|
if (k == riu->wr.num_sge && rsize > 0 && tsize > 0) {
|
||||||
++riu;
|
++riu;
|
||||||
sge = riu->sge;
|
sge = riu->wr.sg_list;
|
||||||
k = 0;
|
k = 0;
|
||||||
} else if (rsize > 0 && tsize > 0)
|
} else if (rsize > 0 && tsize > 0)
|
||||||
++sge;
|
++sge;
|
||||||
|
@ -1277,8 +1284,8 @@ static struct srpt_send_ioctx *srpt_get_send_ioctx(struct srpt_rdma_ch *ch)
|
||||||
ioctx->n_rbuf = 0;
|
ioctx->n_rbuf = 0;
|
||||||
ioctx->rbufs = NULL;
|
ioctx->rbufs = NULL;
|
||||||
ioctx->n_rdma = 0;
|
ioctx->n_rdma = 0;
|
||||||
ioctx->n_rdma_ius = 0;
|
ioctx->n_rdma_wrs = 0;
|
||||||
ioctx->rdma_ius = NULL;
|
ioctx->rdma_wrs = NULL;
|
||||||
ioctx->mapped_sg_count = 0;
|
ioctx->mapped_sg_count = 0;
|
||||||
init_completion(&ioctx->tx_done);
|
init_completion(&ioctx->tx_done);
|
||||||
ioctx->queue_status_only = false;
|
ioctx->queue_status_only = false;
|
||||||
|
@ -1380,118 +1387,44 @@ static int srpt_abort_cmd(struct srpt_send_ioctx *ioctx)
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* srpt_handle_send_err_comp() - Process an IB_WC_SEND error completion.
|
|
||||||
*/
|
|
||||||
static void srpt_handle_send_err_comp(struct srpt_rdma_ch *ch, u64 wr_id)
|
|
||||||
{
|
|
||||||
struct srpt_send_ioctx *ioctx;
|
|
||||||
enum srpt_command_state state;
|
|
||||||
u32 index;
|
|
||||||
|
|
||||||
atomic_inc(&ch->sq_wr_avail);
|
|
||||||
|
|
||||||
index = idx_from_wr_id(wr_id);
|
|
||||||
ioctx = ch->ioctx_ring[index];
|
|
||||||
state = srpt_get_cmd_state(ioctx);
|
|
||||||
|
|
||||||
WARN_ON(state != SRPT_STATE_CMD_RSP_SENT
|
|
||||||
&& state != SRPT_STATE_MGMT_RSP_SENT
|
|
||||||
&& state != SRPT_STATE_NEED_DATA
|
|
||||||
&& state != SRPT_STATE_DONE);
|
|
||||||
|
|
||||||
/* If SRP_RSP sending failed, undo the ch->req_lim change. */
|
|
||||||
if (state == SRPT_STATE_CMD_RSP_SENT
|
|
||||||
|| state == SRPT_STATE_MGMT_RSP_SENT)
|
|
||||||
atomic_dec(&ch->req_lim);
|
|
||||||
|
|
||||||
srpt_abort_cmd(ioctx);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* srpt_handle_send_comp() - Process an IB send completion notification.
|
|
||||||
*/
|
|
||||||
static void srpt_handle_send_comp(struct srpt_rdma_ch *ch,
|
|
||||||
struct srpt_send_ioctx *ioctx)
|
|
||||||
{
|
|
||||||
enum srpt_command_state state;
|
|
||||||
|
|
||||||
atomic_inc(&ch->sq_wr_avail);
|
|
||||||
|
|
||||||
state = srpt_set_cmd_state(ioctx, SRPT_STATE_DONE);
|
|
||||||
|
|
||||||
if (WARN_ON(state != SRPT_STATE_CMD_RSP_SENT
|
|
||||||
&& state != SRPT_STATE_MGMT_RSP_SENT
|
|
||||||
&& state != SRPT_STATE_DONE))
|
|
||||||
pr_debug("state = %d\n", state);
|
|
||||||
|
|
||||||
if (state != SRPT_STATE_DONE) {
|
|
||||||
srpt_unmap_sg_to_ib_sge(ch, ioctx);
|
|
||||||
transport_generic_free_cmd(&ioctx->cmd, 0);
|
|
||||||
} else {
|
|
||||||
pr_err("IB completion has been received too late for"
|
|
||||||
" wr_id = %u.\n", ioctx->ioctx.index);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* srpt_handle_rdma_comp() - Process an IB RDMA completion notification.
|
|
||||||
*
|
|
||||||
* XXX: what is now target_execute_cmd used to be asynchronous, and unmapping
|
* XXX: what is now target_execute_cmd used to be asynchronous, and unmapping
|
||||||
* the data that has been transferred via IB RDMA had to be postponed until the
|
* the data that has been transferred via IB RDMA had to be postponed until the
|
||||||
* check_stop_free() callback. None of this is necessary anymore and needs to
|
* check_stop_free() callback. None of this is necessary anymore and needs to
|
||||||
* be cleaned up.
|
* be cleaned up.
|
||||||
*/
|
*/
|
||||||
static void srpt_handle_rdma_comp(struct srpt_rdma_ch *ch,
|
static void srpt_rdma_read_done(struct ib_cq *cq, struct ib_wc *wc)
|
||||||
struct srpt_send_ioctx *ioctx,
|
|
||||||
enum srpt_opcode opcode)
|
|
||||||
{
|
{
|
||||||
|
struct srpt_rdma_ch *ch = cq->cq_context;
|
||||||
|
struct srpt_send_ioctx *ioctx =
|
||||||
|
container_of(wc->wr_cqe, struct srpt_send_ioctx, rdma_cqe);
|
||||||
|
|
||||||
WARN_ON(ioctx->n_rdma <= 0);
|
WARN_ON(ioctx->n_rdma <= 0);
|
||||||
atomic_add(ioctx->n_rdma, &ch->sq_wr_avail);
|
atomic_add(ioctx->n_rdma, &ch->sq_wr_avail);
|
||||||
|
|
||||||
if (opcode == SRPT_RDMA_READ_LAST) {
|
if (unlikely(wc->status != IB_WC_SUCCESS)) {
|
||||||
if (srpt_test_and_set_cmd_state(ioctx, SRPT_STATE_NEED_DATA,
|
pr_info("RDMA_READ for ioctx 0x%p failed with status %d\n",
|
||||||
SRPT_STATE_DATA_IN))
|
ioctx, wc->status);
|
||||||
target_execute_cmd(&ioctx->cmd);
|
srpt_abort_cmd(ioctx);
|
||||||
else
|
return;
|
||||||
pr_err("%s[%d]: wrong state = %d\n", __func__,
|
|
||||||
__LINE__, srpt_get_cmd_state(ioctx));
|
|
||||||
} else if (opcode == SRPT_RDMA_ABORT) {
|
|
||||||
ioctx->rdma_aborted = true;
|
|
||||||
} else {
|
|
||||||
WARN(true, "unexpected opcode %d\n", opcode);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (srpt_test_and_set_cmd_state(ioctx, SRPT_STATE_NEED_DATA,
|
||||||
|
SRPT_STATE_DATA_IN))
|
||||||
|
target_execute_cmd(&ioctx->cmd);
|
||||||
|
else
|
||||||
|
pr_err("%s[%d]: wrong state = %d\n", __func__,
|
||||||
|
__LINE__, srpt_get_cmd_state(ioctx));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
static void srpt_rdma_write_done(struct ib_cq *cq, struct ib_wc *wc)
|
||||||
* srpt_handle_rdma_err_comp() - Process an IB RDMA error completion.
|
|
||||||
*/
|
|
||||||
static void srpt_handle_rdma_err_comp(struct srpt_rdma_ch *ch,
|
|
||||||
struct srpt_send_ioctx *ioctx,
|
|
||||||
enum srpt_opcode opcode)
|
|
||||||
{
|
{
|
||||||
enum srpt_command_state state;
|
struct srpt_send_ioctx *ioctx =
|
||||||
|
container_of(wc->wr_cqe, struct srpt_send_ioctx, rdma_cqe);
|
||||||
|
|
||||||
state = srpt_get_cmd_state(ioctx);
|
if (unlikely(wc->status != IB_WC_SUCCESS)) {
|
||||||
switch (opcode) {
|
pr_info("RDMA_WRITE for ioctx 0x%p failed with status %d\n",
|
||||||
case SRPT_RDMA_READ_LAST:
|
ioctx, wc->status);
|
||||||
if (ioctx->n_rdma <= 0) {
|
srpt_abort_cmd(ioctx);
|
||||||
pr_err("Received invalid RDMA read"
|
|
||||||
" error completion with idx %d\n",
|
|
||||||
ioctx->ioctx.index);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
atomic_add(ioctx->n_rdma, &ch->sq_wr_avail);
|
|
||||||
if (state == SRPT_STATE_NEED_DATA)
|
|
||||||
srpt_abort_cmd(ioctx);
|
|
||||||
else
|
|
||||||
pr_err("%s[%d]: wrong state = %d\n",
|
|
||||||
__func__, __LINE__, state);
|
|
||||||
break;
|
|
||||||
case SRPT_RDMA_WRITE_LAST:
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
pr_err("%s[%d]: opcode = %u\n", __func__, __LINE__, opcode);
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1926,32 +1859,26 @@ static void srpt_handle_new_iu(struct srpt_rdma_ch *ch,
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void srpt_process_rcv_completion(struct ib_cq *cq,
|
static void srpt_recv_done(struct ib_cq *cq, struct ib_wc *wc)
|
||||||
struct srpt_rdma_ch *ch,
|
|
||||||
struct ib_wc *wc)
|
|
||||||
{
|
{
|
||||||
struct srpt_device *sdev = ch->sport->sdev;
|
struct srpt_rdma_ch *ch = cq->cq_context;
|
||||||
struct srpt_recv_ioctx *ioctx;
|
struct srpt_recv_ioctx *ioctx =
|
||||||
u32 index;
|
container_of(wc->wr_cqe, struct srpt_recv_ioctx, ioctx.cqe);
|
||||||
|
|
||||||
index = idx_from_wr_id(wc->wr_id);
|
|
||||||
if (wc->status == IB_WC_SUCCESS) {
|
if (wc->status == IB_WC_SUCCESS) {
|
||||||
int req_lim;
|
int req_lim;
|
||||||
|
|
||||||
req_lim = atomic_dec_return(&ch->req_lim);
|
req_lim = atomic_dec_return(&ch->req_lim);
|
||||||
if (unlikely(req_lim < 0))
|
if (unlikely(req_lim < 0))
|
||||||
pr_err("req_lim = %d < 0\n", req_lim);
|
pr_err("req_lim = %d < 0\n", req_lim);
|
||||||
ioctx = sdev->ioctx_ring[index];
|
|
||||||
srpt_handle_new_iu(ch, ioctx, NULL);
|
srpt_handle_new_iu(ch, ioctx, NULL);
|
||||||
} else {
|
} else {
|
||||||
pr_info("receiving failed for idx %u with status %d\n",
|
pr_info("receiving failed for ioctx %p with status %d\n",
|
||||||
index, wc->status);
|
ioctx, wc->status);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* srpt_process_send_completion() - Process an IB send completion.
|
|
||||||
*
|
|
||||||
* Note: Although this has not yet been observed during tests, at least in
|
* Note: Although this has not yet been observed during tests, at least in
|
||||||
* theory it is possible that the srpt_get_send_ioctx() call invoked by
|
* theory it is possible that the srpt_get_send_ioctx() call invoked by
|
||||||
* srpt_handle_new_iu() fails. This is possible because the req_lim_delta
|
* srpt_handle_new_iu() fails. This is possible because the req_lim_delta
|
||||||
|
@ -1964,108 +1891,51 @@ static void srpt_process_rcv_completion(struct ib_cq *cq,
|
||||||
* are queued on cmd_wait_list. The code below processes these delayed
|
* are queued on cmd_wait_list. The code below processes these delayed
|
||||||
* requests one at a time.
|
* requests one at a time.
|
||||||
*/
|
*/
|
||||||
static void srpt_process_send_completion(struct ib_cq *cq,
|
static void srpt_send_done(struct ib_cq *cq, struct ib_wc *wc)
|
||||||
struct srpt_rdma_ch *ch,
|
|
||||||
struct ib_wc *wc)
|
|
||||||
{
|
{
|
||||||
struct srpt_send_ioctx *send_ioctx;
|
struct srpt_rdma_ch *ch = cq->cq_context;
|
||||||
uint32_t index;
|
struct srpt_send_ioctx *ioctx =
|
||||||
enum srpt_opcode opcode;
|
container_of(wc->wr_cqe, struct srpt_send_ioctx, ioctx.cqe);
|
||||||
|
enum srpt_command_state state;
|
||||||
|
|
||||||
index = idx_from_wr_id(wc->wr_id);
|
state = srpt_set_cmd_state(ioctx, SRPT_STATE_DONE);
|
||||||
opcode = opcode_from_wr_id(wc->wr_id);
|
|
||||||
send_ioctx = ch->ioctx_ring[index];
|
WARN_ON(state != SRPT_STATE_CMD_RSP_SENT &&
|
||||||
if (wc->status == IB_WC_SUCCESS) {
|
state != SRPT_STATE_MGMT_RSP_SENT);
|
||||||
if (opcode == SRPT_SEND)
|
|
||||||
srpt_handle_send_comp(ch, send_ioctx);
|
atomic_inc(&ch->sq_wr_avail);
|
||||||
else {
|
|
||||||
WARN_ON(opcode != SRPT_RDMA_ABORT &&
|
if (wc->status != IB_WC_SUCCESS) {
|
||||||
wc->opcode != IB_WC_RDMA_READ);
|
pr_info("sending response for ioctx 0x%p failed"
|
||||||
srpt_handle_rdma_comp(ch, send_ioctx, opcode);
|
" with status %d\n", ioctx, wc->status);
|
||||||
}
|
|
||||||
} else {
|
atomic_dec(&ch->req_lim);
|
||||||
if (opcode == SRPT_SEND) {
|
srpt_abort_cmd(ioctx);
|
||||||
pr_info("sending response for idx %u failed"
|
goto out;
|
||||||
" with status %d\n", index, wc->status);
|
|
||||||
srpt_handle_send_err_comp(ch, wc->wr_id);
|
|
||||||
} else if (opcode != SRPT_RDMA_MID) {
|
|
||||||
pr_info("RDMA t %d for idx %u failed with"
|
|
||||||
" status %d\n", opcode, index, wc->status);
|
|
||||||
srpt_handle_rdma_err_comp(ch, send_ioctx, opcode);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
while (unlikely(opcode == SRPT_SEND
|
if (state != SRPT_STATE_DONE) {
|
||||||
&& !list_empty(&ch->cmd_wait_list)
|
srpt_unmap_sg_to_ib_sge(ch, ioctx);
|
||||||
&& srpt_get_ch_state(ch) == CH_LIVE
|
transport_generic_free_cmd(&ioctx->cmd, 0);
|
||||||
&& (send_ioctx = srpt_get_send_ioctx(ch)) != NULL)) {
|
} else {
|
||||||
|
pr_err("IB completion has been received too late for"
|
||||||
|
" wr_id = %u.\n", ioctx->ioctx.index);
|
||||||
|
}
|
||||||
|
|
||||||
|
out:
|
||||||
|
while (!list_empty(&ch->cmd_wait_list) &&
|
||||||
|
srpt_get_ch_state(ch) == CH_LIVE &&
|
||||||
|
(ioctx = srpt_get_send_ioctx(ch)) != NULL) {
|
||||||
struct srpt_recv_ioctx *recv_ioctx;
|
struct srpt_recv_ioctx *recv_ioctx;
|
||||||
|
|
||||||
recv_ioctx = list_first_entry(&ch->cmd_wait_list,
|
recv_ioctx = list_first_entry(&ch->cmd_wait_list,
|
||||||
struct srpt_recv_ioctx,
|
struct srpt_recv_ioctx,
|
||||||
wait_list);
|
wait_list);
|
||||||
list_del(&recv_ioctx->wait_list);
|
list_del(&recv_ioctx->wait_list);
|
||||||
srpt_handle_new_iu(ch, recv_ioctx, send_ioctx);
|
srpt_handle_new_iu(ch, recv_ioctx, ioctx);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void srpt_process_completion(struct ib_cq *cq, struct srpt_rdma_ch *ch)
|
|
||||||
{
|
|
||||||
struct ib_wc *const wc = ch->wc;
|
|
||||||
int i, n;
|
|
||||||
|
|
||||||
WARN_ON(cq != ch->cq);
|
|
||||||
|
|
||||||
ib_req_notify_cq(cq, IB_CQ_NEXT_COMP);
|
|
||||||
while ((n = ib_poll_cq(cq, ARRAY_SIZE(ch->wc), wc)) > 0) {
|
|
||||||
for (i = 0; i < n; i++) {
|
|
||||||
if (opcode_from_wr_id(wc[i].wr_id) == SRPT_RECV)
|
|
||||||
srpt_process_rcv_completion(cq, ch, &wc[i]);
|
|
||||||
else
|
|
||||||
srpt_process_send_completion(cq, ch, &wc[i]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* srpt_completion() - IB completion queue callback function.
|
|
||||||
*
|
|
||||||
* Notes:
|
|
||||||
* - It is guaranteed that a completion handler will never be invoked
|
|
||||||
* concurrently on two different CPUs for the same completion queue. See also
|
|
||||||
* Documentation/infiniband/core_locking.txt and the implementation of
|
|
||||||
* handle_edge_irq() in kernel/irq/chip.c.
|
|
||||||
* - When threaded IRQs are enabled, completion handlers are invoked in thread
|
|
||||||
* context instead of interrupt context.
|
|
||||||
*/
|
|
||||||
static void srpt_completion(struct ib_cq *cq, void *ctx)
|
|
||||||
{
|
|
||||||
struct srpt_rdma_ch *ch = ctx;
|
|
||||||
|
|
||||||
wake_up_interruptible(&ch->wait_queue);
|
|
||||||
}
|
|
||||||
|
|
||||||
static int srpt_compl_thread(void *arg)
|
|
||||||
{
|
|
||||||
struct srpt_rdma_ch *ch;
|
|
||||||
|
|
||||||
/* Hibernation / freezing of the SRPT kernel thread is not supported. */
|
|
||||||
current->flags |= PF_NOFREEZE;
|
|
||||||
|
|
||||||
ch = arg;
|
|
||||||
BUG_ON(!ch);
|
|
||||||
pr_info("Session %s: kernel thread %s (PID %d) started\n",
|
|
||||||
ch->sess_name, ch->thread->comm, current->pid);
|
|
||||||
while (!kthread_should_stop()) {
|
|
||||||
wait_event_interruptible(ch->wait_queue,
|
|
||||||
(srpt_process_completion(ch->cq, ch),
|
|
||||||
kthread_should_stop()));
|
|
||||||
}
|
|
||||||
pr_info("Session %s: kernel thread %s (PID %d) stopped\n",
|
|
||||||
ch->sess_name, ch->thread->comm, current->pid);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* srpt_create_ch_ib() - Create receive and send completion queues.
|
* srpt_create_ch_ib() - Create receive and send completion queues.
|
||||||
*/
|
*/
|
||||||
|
@ -2075,7 +1945,6 @@ static int srpt_create_ch_ib(struct srpt_rdma_ch *ch)
|
||||||
struct srpt_port *sport = ch->sport;
|
struct srpt_port *sport = ch->sport;
|
||||||
struct srpt_device *sdev = sport->sdev;
|
struct srpt_device *sdev = sport->sdev;
|
||||||
u32 srp_sq_size = sport->port_attrib.srp_sq_size;
|
u32 srp_sq_size = sport->port_attrib.srp_sq_size;
|
||||||
struct ib_cq_init_attr cq_attr = {};
|
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
WARN_ON(ch->rq_size < 1);
|
WARN_ON(ch->rq_size < 1);
|
||||||
|
@ -2086,9 +1955,8 @@ static int srpt_create_ch_ib(struct srpt_rdma_ch *ch)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
retry:
|
retry:
|
||||||
cq_attr.cqe = ch->rq_size + srp_sq_size;
|
ch->cq = ib_alloc_cq(sdev->device, ch, ch->rq_size + srp_sq_size,
|
||||||
ch->cq = ib_create_cq(sdev->device, srpt_completion, NULL, ch,
|
0 /* XXX: spread CQs */, IB_POLL_WORKQUEUE);
|
||||||
&cq_attr);
|
|
||||||
if (IS_ERR(ch->cq)) {
|
if (IS_ERR(ch->cq)) {
|
||||||
ret = PTR_ERR(ch->cq);
|
ret = PTR_ERR(ch->cq);
|
||||||
pr_err("failed to create CQ cqe= %d ret= %d\n",
|
pr_err("failed to create CQ cqe= %d ret= %d\n",
|
||||||
|
@ -2131,18 +1999,6 @@ static int srpt_create_ch_ib(struct srpt_rdma_ch *ch)
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err_destroy_qp;
|
goto err_destroy_qp;
|
||||||
|
|
||||||
init_waitqueue_head(&ch->wait_queue);
|
|
||||||
|
|
||||||
pr_debug("creating thread for session %s\n", ch->sess_name);
|
|
||||||
|
|
||||||
ch->thread = kthread_run(srpt_compl_thread, ch, "ib_srpt_compl");
|
|
||||||
if (IS_ERR(ch->thread)) {
|
|
||||||
pr_err("failed to create kernel thread %ld\n",
|
|
||||||
PTR_ERR(ch->thread));
|
|
||||||
ch->thread = NULL;
|
|
||||||
goto err_destroy_qp;
|
|
||||||
}
|
|
||||||
|
|
||||||
out:
|
out:
|
||||||
kfree(qp_init);
|
kfree(qp_init);
|
||||||
return ret;
|
return ret;
|
||||||
|
@ -2150,17 +2006,14 @@ static int srpt_create_ch_ib(struct srpt_rdma_ch *ch)
|
||||||
err_destroy_qp:
|
err_destroy_qp:
|
||||||
ib_destroy_qp(ch->qp);
|
ib_destroy_qp(ch->qp);
|
||||||
err_destroy_cq:
|
err_destroy_cq:
|
||||||
ib_destroy_cq(ch->cq);
|
ib_free_cq(ch->cq);
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void srpt_destroy_ch_ib(struct srpt_rdma_ch *ch)
|
static void srpt_destroy_ch_ib(struct srpt_rdma_ch *ch)
|
||||||
{
|
{
|
||||||
if (ch->thread)
|
|
||||||
kthread_stop(ch->thread);
|
|
||||||
|
|
||||||
ib_destroy_qp(ch->qp);
|
ib_destroy_qp(ch->qp);
|
||||||
ib_destroy_cq(ch->cq);
|
ib_free_cq(ch->cq);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -2808,12 +2661,8 @@ static int srpt_cm_handler(struct ib_cm_id *cm_id, struct ib_cm_event *event)
|
||||||
static int srpt_perform_rdmas(struct srpt_rdma_ch *ch,
|
static int srpt_perform_rdmas(struct srpt_rdma_ch *ch,
|
||||||
struct srpt_send_ioctx *ioctx)
|
struct srpt_send_ioctx *ioctx)
|
||||||
{
|
{
|
||||||
struct ib_rdma_wr wr;
|
|
||||||
struct ib_send_wr *bad_wr;
|
struct ib_send_wr *bad_wr;
|
||||||
struct rdma_iu *riu;
|
int sq_wr_avail, ret, i;
|
||||||
int i;
|
|
||||||
int ret;
|
|
||||||
int sq_wr_avail;
|
|
||||||
enum dma_data_direction dir;
|
enum dma_data_direction dir;
|
||||||
const int n_rdma = ioctx->n_rdma;
|
const int n_rdma = ioctx->n_rdma;
|
||||||
|
|
||||||
|
@ -2829,59 +2678,32 @@ static int srpt_perform_rdmas(struct srpt_rdma_ch *ch,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ioctx->rdma_aborted = false;
|
for (i = 0; i < n_rdma; i++) {
|
||||||
ret = 0;
|
struct ib_send_wr *wr = &ioctx->rdma_wrs[i].wr;
|
||||||
riu = ioctx->rdma_ius;
|
|
||||||
memset(&wr, 0, sizeof wr);
|
|
||||||
|
|
||||||
for (i = 0; i < n_rdma; ++i, ++riu) {
|
wr->opcode = (dir == DMA_FROM_DEVICE) ?
|
||||||
if (dir == DMA_FROM_DEVICE) {
|
IB_WR_RDMA_WRITE : IB_WR_RDMA_READ;
|
||||||
wr.wr.opcode = IB_WR_RDMA_WRITE;
|
|
||||||
wr.wr.wr_id = encode_wr_id(i == n_rdma - 1 ?
|
if (i == n_rdma - 1) {
|
||||||
SRPT_RDMA_WRITE_LAST :
|
/* only get completion event for the last rdma read */
|
||||||
SRPT_RDMA_MID,
|
if (dir == DMA_TO_DEVICE) {
|
||||||
ioctx->ioctx.index);
|
wr->send_flags = IB_SEND_SIGNALED;
|
||||||
|
ioctx->rdma_cqe.done = srpt_rdma_read_done;
|
||||||
|
} else {
|
||||||
|
ioctx->rdma_cqe.done = srpt_rdma_write_done;
|
||||||
|
}
|
||||||
|
wr->wr_cqe = &ioctx->rdma_cqe;
|
||||||
|
wr->next = NULL;
|
||||||
} else {
|
} else {
|
||||||
wr.wr.opcode = IB_WR_RDMA_READ;
|
wr->wr_cqe = NULL;
|
||||||
wr.wr.wr_id = encode_wr_id(i == n_rdma - 1 ?
|
wr->next = &ioctx->rdma_wrs[i + 1].wr;
|
||||||
SRPT_RDMA_READ_LAST :
|
|
||||||
SRPT_RDMA_MID,
|
|
||||||
ioctx->ioctx.index);
|
|
||||||
}
|
}
|
||||||
wr.wr.next = NULL;
|
|
||||||
wr.remote_addr = riu->raddr;
|
|
||||||
wr.rkey = riu->rkey;
|
|
||||||
wr.wr.num_sge = riu->sge_cnt;
|
|
||||||
wr.wr.sg_list = riu->sge;
|
|
||||||
|
|
||||||
/* only get completion event for the last rdma write */
|
|
||||||
if (i == (n_rdma - 1) && dir == DMA_TO_DEVICE)
|
|
||||||
wr.wr.send_flags = IB_SEND_SIGNALED;
|
|
||||||
|
|
||||||
ret = ib_post_send(ch->qp, &wr.wr, &bad_wr);
|
|
||||||
if (ret)
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ret = ib_post_send(ch->qp, &ioctx->rdma_wrs->wr, &bad_wr);
|
||||||
if (ret)
|
if (ret)
|
||||||
pr_err("%s[%d]: ib_post_send() returned %d for %d/%d\n",
|
pr_err("%s[%d]: ib_post_send() returned %d for %d/%d\n",
|
||||||
__func__, __LINE__, ret, i, n_rdma);
|
__func__, __LINE__, ret, i, n_rdma);
|
||||||
if (ret && i > 0) {
|
|
||||||
wr.wr.num_sge = 0;
|
|
||||||
wr.wr.wr_id = encode_wr_id(SRPT_RDMA_ABORT, ioctx->ioctx.index);
|
|
||||||
wr.wr.send_flags = IB_SEND_SIGNALED;
|
|
||||||
while (ch->state == CH_LIVE &&
|
|
||||||
ib_post_send(ch->qp, &wr.wr, &bad_wr) != 0) {
|
|
||||||
pr_info("Trying to abort failed RDMA transfer [%d]\n",
|
|
||||||
ioctx->ioctx.index);
|
|
||||||
msleep(1000);
|
|
||||||
}
|
|
||||||
while (ch->state != CH_RELEASING && !ioctx->rdma_aborted) {
|
|
||||||
pr_info("Waiting until RDMA abort finished [%d]\n",
|
|
||||||
ioctx->ioctx.index);
|
|
||||||
msleep(1000);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
out:
|
out:
|
||||||
if (unlikely(dir == DMA_TO_DEVICE && ret < 0))
|
if (unlikely(dir == DMA_TO_DEVICE && ret < 0))
|
||||||
atomic_add(n_rdma, &ch->sq_wr_avail);
|
atomic_add(n_rdma, &ch->sq_wr_avail);
|
||||||
|
@ -3190,14 +3012,11 @@ static void srpt_add_one(struct ib_device *device)
|
||||||
init_waitqueue_head(&sdev->ch_releaseQ);
|
init_waitqueue_head(&sdev->ch_releaseQ);
|
||||||
spin_lock_init(&sdev->spinlock);
|
spin_lock_init(&sdev->spinlock);
|
||||||
|
|
||||||
if (ib_query_device(device, &sdev->dev_attr))
|
|
||||||
goto free_dev;
|
|
||||||
|
|
||||||
sdev->pd = ib_alloc_pd(device);
|
sdev->pd = ib_alloc_pd(device);
|
||||||
if (IS_ERR(sdev->pd))
|
if (IS_ERR(sdev->pd))
|
||||||
goto free_dev;
|
goto free_dev;
|
||||||
|
|
||||||
sdev->srq_size = min(srpt_srq_size, sdev->dev_attr.max_srq_wr);
|
sdev->srq_size = min(srpt_srq_size, sdev->device->attrs.max_srq_wr);
|
||||||
|
|
||||||
srq_attr.event_handler = srpt_srq_event;
|
srq_attr.event_handler = srpt_srq_event;
|
||||||
srq_attr.srq_context = (void *)sdev;
|
srq_attr.srq_context = (void *)sdev;
|
||||||
|
@ -3211,7 +3030,7 @@ static void srpt_add_one(struct ib_device *device)
|
||||||
goto err_pd;
|
goto err_pd;
|
||||||
|
|
||||||
pr_debug("%s: create SRQ #wr= %d max_allow=%d dev= %s\n",
|
pr_debug("%s: create SRQ #wr= %d max_allow=%d dev= %s\n",
|
||||||
__func__, sdev->srq_size, sdev->dev_attr.max_srq_wr,
|
__func__, sdev->srq_size, sdev->device->attrs.max_srq_wr,
|
||||||
device->name);
|
device->name);
|
||||||
|
|
||||||
if (!srpt_service_guid)
|
if (!srpt_service_guid)
|
||||||
|
|
|
@ -128,36 +128,6 @@ enum {
|
||||||
DEFAULT_MAX_RDMA_SIZE = 65536,
|
DEFAULT_MAX_RDMA_SIZE = 65536,
|
||||||
};
|
};
|
||||||
|
|
||||||
enum srpt_opcode {
|
|
||||||
SRPT_RECV,
|
|
||||||
SRPT_SEND,
|
|
||||||
SRPT_RDMA_MID,
|
|
||||||
SRPT_RDMA_ABORT,
|
|
||||||
SRPT_RDMA_READ_LAST,
|
|
||||||
SRPT_RDMA_WRITE_LAST,
|
|
||||||
};
|
|
||||||
|
|
||||||
static inline u64 encode_wr_id(u8 opcode, u32 idx)
|
|
||||||
{
|
|
||||||
return ((u64)opcode << 32) | idx;
|
|
||||||
}
|
|
||||||
static inline enum srpt_opcode opcode_from_wr_id(u64 wr_id)
|
|
||||||
{
|
|
||||||
return wr_id >> 32;
|
|
||||||
}
|
|
||||||
static inline u32 idx_from_wr_id(u64 wr_id)
|
|
||||||
{
|
|
||||||
return (u32)wr_id;
|
|
||||||
}
|
|
||||||
|
|
||||||
struct rdma_iu {
|
|
||||||
u64 raddr;
|
|
||||||
u32 rkey;
|
|
||||||
struct ib_sge *sge;
|
|
||||||
u32 sge_cnt;
|
|
||||||
int mem_id;
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* enum srpt_command_state - SCSI command state managed by SRPT.
|
* enum srpt_command_state - SCSI command state managed by SRPT.
|
||||||
* @SRPT_STATE_NEW: New command arrived and is being processed.
|
* @SRPT_STATE_NEW: New command arrived and is being processed.
|
||||||
|
@ -189,6 +159,7 @@ enum srpt_command_state {
|
||||||
* @index: Index of the I/O context in its ioctx_ring array.
|
* @index: Index of the I/O context in its ioctx_ring array.
|
||||||
*/
|
*/
|
||||||
struct srpt_ioctx {
|
struct srpt_ioctx {
|
||||||
|
struct ib_cqe cqe;
|
||||||
void *buf;
|
void *buf;
|
||||||
dma_addr_t dma;
|
dma_addr_t dma;
|
||||||
uint32_t index;
|
uint32_t index;
|
||||||
|
@ -215,32 +186,30 @@ struct srpt_recv_ioctx {
|
||||||
* @sg: Pointer to sg-list associated with this I/O context.
|
* @sg: Pointer to sg-list associated with this I/O context.
|
||||||
* @sg_cnt: SG-list size.
|
* @sg_cnt: SG-list size.
|
||||||
* @mapped_sg_count: ib_dma_map_sg() return value.
|
* @mapped_sg_count: ib_dma_map_sg() return value.
|
||||||
* @n_rdma_ius: Number of elements in the rdma_ius array.
|
* @n_rdma_wrs: Number of elements in the rdma_wrs array.
|
||||||
* @rdma_ius: Array with information about the RDMA mapping.
|
* @rdma_wrs: Array with information about the RDMA mapping.
|
||||||
* @tag: Tag of the received SRP information unit.
|
* @tag: Tag of the received SRP information unit.
|
||||||
* @spinlock: Protects 'state'.
|
* @spinlock: Protects 'state'.
|
||||||
* @state: I/O context state.
|
* @state: I/O context state.
|
||||||
* @rdma_aborted: If initiating a multipart RDMA transfer failed, whether
|
|
||||||
* the already initiated transfers have finished.
|
|
||||||
* @cmd: Target core command data structure.
|
* @cmd: Target core command data structure.
|
||||||
* @sense_data: SCSI sense data.
|
* @sense_data: SCSI sense data.
|
||||||
*/
|
*/
|
||||||
struct srpt_send_ioctx {
|
struct srpt_send_ioctx {
|
||||||
struct srpt_ioctx ioctx;
|
struct srpt_ioctx ioctx;
|
||||||
struct srpt_rdma_ch *ch;
|
struct srpt_rdma_ch *ch;
|
||||||
struct rdma_iu *rdma_ius;
|
struct ib_rdma_wr *rdma_wrs;
|
||||||
|
struct ib_cqe rdma_cqe;
|
||||||
struct srp_direct_buf *rbufs;
|
struct srp_direct_buf *rbufs;
|
||||||
struct srp_direct_buf single_rbuf;
|
struct srp_direct_buf single_rbuf;
|
||||||
struct scatterlist *sg;
|
struct scatterlist *sg;
|
||||||
struct list_head free_list;
|
struct list_head free_list;
|
||||||
spinlock_t spinlock;
|
spinlock_t spinlock;
|
||||||
enum srpt_command_state state;
|
enum srpt_command_state state;
|
||||||
bool rdma_aborted;
|
|
||||||
struct se_cmd cmd;
|
struct se_cmd cmd;
|
||||||
struct completion tx_done;
|
struct completion tx_done;
|
||||||
int sg_cnt;
|
int sg_cnt;
|
||||||
int mapped_sg_count;
|
int mapped_sg_count;
|
||||||
u16 n_rdma_ius;
|
u16 n_rdma_wrs;
|
||||||
u8 n_rdma;
|
u8 n_rdma;
|
||||||
u8 n_rbuf;
|
u8 n_rbuf;
|
||||||
bool queue_status_only;
|
bool queue_status_only;
|
||||||
|
@ -267,9 +236,6 @@ enum rdma_ch_state {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* struct srpt_rdma_ch - RDMA channel.
|
* struct srpt_rdma_ch - RDMA channel.
|
||||||
* @wait_queue: Allows the kernel thread to wait for more work.
|
|
||||||
* @thread: Kernel thread that processes the IB queues associated with
|
|
||||||
* the channel.
|
|
||||||
* @cm_id: IB CM ID associated with the channel.
|
* @cm_id: IB CM ID associated with the channel.
|
||||||
* @qp: IB queue pair used for communicating over this channel.
|
* @qp: IB queue pair used for communicating over this channel.
|
||||||
* @cq: IB completion queue for this channel.
|
* @cq: IB completion queue for this channel.
|
||||||
|
@ -288,7 +254,6 @@ enum rdma_ch_state {
|
||||||
* @free_list: Head of list with free send I/O contexts.
|
* @free_list: Head of list with free send I/O contexts.
|
||||||
* @state: channel state. See also enum rdma_ch_state.
|
* @state: channel state. See also enum rdma_ch_state.
|
||||||
* @ioctx_ring: Send ring.
|
* @ioctx_ring: Send ring.
|
||||||
* @wc: IB work completion array for srpt_process_completion().
|
|
||||||
* @list: Node for insertion in the srpt_device.rch_list list.
|
* @list: Node for insertion in the srpt_device.rch_list list.
|
||||||
* @cmd_wait_list: List of SCSI commands that arrived before the RTU event. This
|
* @cmd_wait_list: List of SCSI commands that arrived before the RTU event. This
|
||||||
* list contains struct srpt_ioctx elements and is protected
|
* list contains struct srpt_ioctx elements and is protected
|
||||||
|
@ -299,8 +264,6 @@ enum rdma_ch_state {
|
||||||
* @release_done: Enables waiting for srpt_release_channel() completion.
|
* @release_done: Enables waiting for srpt_release_channel() completion.
|
||||||
*/
|
*/
|
||||||
struct srpt_rdma_ch {
|
struct srpt_rdma_ch {
|
||||||
wait_queue_head_t wait_queue;
|
|
||||||
struct task_struct *thread;
|
|
||||||
struct ib_cm_id *cm_id;
|
struct ib_cm_id *cm_id;
|
||||||
struct ib_qp *qp;
|
struct ib_qp *qp;
|
||||||
struct ib_cq *cq;
|
struct ib_cq *cq;
|
||||||
|
@ -317,7 +280,6 @@ struct srpt_rdma_ch {
|
||||||
struct list_head free_list;
|
struct list_head free_list;
|
||||||
enum rdma_ch_state state;
|
enum rdma_ch_state state;
|
||||||
struct srpt_send_ioctx **ioctx_ring;
|
struct srpt_send_ioctx **ioctx_ring;
|
||||||
struct ib_wc wc[16];
|
|
||||||
struct list_head list;
|
struct list_head list;
|
||||||
struct list_head cmd_wait_list;
|
struct list_head cmd_wait_list;
|
||||||
struct se_session *sess;
|
struct se_session *sess;
|
||||||
|
@ -377,8 +339,6 @@ struct srpt_port {
|
||||||
* @mr: L_Key (local key) with write access to all local memory.
|
* @mr: L_Key (local key) with write access to all local memory.
|
||||||
* @srq: Per-HCA SRQ (shared receive queue).
|
* @srq: Per-HCA SRQ (shared receive queue).
|
||||||
* @cm_id: Connection identifier.
|
* @cm_id: Connection identifier.
|
||||||
* @dev_attr: Attributes of the InfiniBand device as obtained during the
|
|
||||||
* ib_client.add() callback.
|
|
||||||
* @srq_size: SRQ size.
|
* @srq_size: SRQ size.
|
||||||
* @ioctx_ring: Per-HCA SRQ.
|
* @ioctx_ring: Per-HCA SRQ.
|
||||||
* @rch_list: Per-device channel list -- see also srpt_rdma_ch.list.
|
* @rch_list: Per-device channel list -- see also srpt_rdma_ch.list.
|
||||||
|
@ -393,7 +353,6 @@ struct srpt_device {
|
||||||
struct ib_pd *pd;
|
struct ib_pd *pd;
|
||||||
struct ib_srq *srq;
|
struct ib_srq *srq;
|
||||||
struct ib_cm_id *cm_id;
|
struct ib_cm_id *cm_id;
|
||||||
struct ib_device_attr dev_attr;
|
|
||||||
int srq_size;
|
int srq_size;
|
||||||
struct srpt_recv_ioctx **ioctx_ring;
|
struct srpt_recv_ioctx **ioctx_ring;
|
||||||
struct list_head rch_list;
|
struct list_head rch_list;
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Reference in a new issue