mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-15 01:24:33 +00:00
RDMA/cma: Listen on specific address family
The rdma_cm maps IPv4 and IPv6 addresses to the same service ID. This prevents apps from listening only for IPv4 or IPv6 addresses. It also results in an app binding to an IPv4 address receiving connection requests for an IPv6 address. Change this to match socket behavior: restrict listens on IPv4 addresses to only IPv4 addresses, and if a listen is on an IPv6 address, allow it to receive either IPv4 or IPv6 addresses, based on its address family binding. Signed-off-by: Sean Hefty <sean.hefty@intel.com> Signed-off-by: Roland Dreier <roland@purestorage.com>
This commit is contained in:
parent
5b0ec991c0
commit
406b6a25f8
@ -1298,8 +1298,10 @@ static void cma_set_compare_data(enum rdma_port_space ps, struct sockaddr *addr,
|
|||||||
} else {
|
} else {
|
||||||
cma_set_ip_ver(cma_data, 4);
|
cma_set_ip_ver(cma_data, 4);
|
||||||
cma_set_ip_ver(cma_mask, 0xF);
|
cma_set_ip_ver(cma_mask, 0xF);
|
||||||
cma_data->dst_addr.ip4.addr = ip4_addr;
|
if (!cma_any_addr(addr)) {
|
||||||
cma_mask->dst_addr.ip4.addr = htonl(~0);
|
cma_data->dst_addr.ip4.addr = ip4_addr;
|
||||||
|
cma_mask->dst_addr.ip4.addr = htonl(~0);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case AF_INET6:
|
case AF_INET6:
|
||||||
@ -1313,9 +1315,11 @@ static void cma_set_compare_data(enum rdma_port_space ps, struct sockaddr *addr,
|
|||||||
} else {
|
} else {
|
||||||
cma_set_ip_ver(cma_data, 6);
|
cma_set_ip_ver(cma_data, 6);
|
||||||
cma_set_ip_ver(cma_mask, 0xF);
|
cma_set_ip_ver(cma_mask, 0xF);
|
||||||
cma_data->dst_addr.ip6 = ip6_addr;
|
if (!cma_any_addr(addr)) {
|
||||||
memset(&cma_mask->dst_addr.ip6, 0xFF,
|
cma_data->dst_addr.ip6 = ip6_addr;
|
||||||
sizeof cma_mask->dst_addr.ip6);
|
memset(&cma_mask->dst_addr.ip6, 0xFF,
|
||||||
|
sizeof cma_mask->dst_addr.ip6);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
@ -1500,7 +1504,7 @@ static int cma_ib_listen(struct rdma_id_private *id_priv)
|
|||||||
|
|
||||||
addr = (struct sockaddr *) &id_priv->id.route.addr.src_addr;
|
addr = (struct sockaddr *) &id_priv->id.route.addr.src_addr;
|
||||||
svc_id = cma_get_service_id(id_priv->id.ps, addr);
|
svc_id = cma_get_service_id(id_priv->id.ps, addr);
|
||||||
if (cma_any_addr(addr))
|
if (cma_any_addr(addr) && !id_priv->afonly)
|
||||||
ret = ib_cm_listen(id_priv->cm_id.ib, svc_id, 0, NULL);
|
ret = ib_cm_listen(id_priv->cm_id.ib, svc_id, 0, NULL);
|
||||||
else {
|
else {
|
||||||
cma_set_compare_data(id_priv->id.ps, addr, &compare_data);
|
cma_set_compare_data(id_priv->id.ps, addr, &compare_data);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user