提交 a9e527e3 编写于 作者: R Rolf Manderscheid 提交者: Roland Dreier

IPoIB: improve IPv4/IPv6 to IB mcast mapping functions

An IPoIB subnet on an IB fabric that spans multiple IB subnets can't
use link-local scope in multicast GIDs.  The existing routines that
map IP/IPv6 multicast addresses into IB link-level addresses hard-code
the scope to link-local, and they also leave the partition key field
uninitialised.  This patch adds a parameter (the link-level broadcast
address) to the mapping routines, allowing them to initialise both the
scope and the P_Key appropriately, and fixes up the call sites.

The next step will be to add a way to configure the scope for an IPoIB
interface.
Signed-off-by: NRolf Manderscheid <rvm@obsidianresearch.com>
Signed-off-by: NRoland Dreier <rolandd@cisco.com>
上级 755807a2
...@@ -2610,11 +2610,9 @@ static void cma_set_mgid(struct rdma_id_private *id_priv, ...@@ -2610,11 +2610,9 @@ static void cma_set_mgid(struct rdma_id_private *id_priv,
/* IPv6 address is an SA assigned MGID. */ /* IPv6 address is an SA assigned MGID. */
memcpy(mgid, &sin6->sin6_addr, sizeof *mgid); memcpy(mgid, &sin6->sin6_addr, sizeof *mgid);
} else { } else {
ip_ib_mc_map(sin->sin_addr.s_addr, mc_map); ip_ib_mc_map(sin->sin_addr.s_addr, dev_addr->broadcast, mc_map);
if (id_priv->id.ps == RDMA_PS_UDP) if (id_priv->id.ps == RDMA_PS_UDP)
mc_map[7] = 0x01; /* Use RDMA CM signature */ mc_map[7] = 0x01; /* Use RDMA CM signature */
mc_map[8] = ib_addr_get_pkey(dev_addr) >> 8;
mc_map[9] = (unsigned char) ib_addr_get_pkey(dev_addr);
*mgid = *(union ib_gid *) (mc_map + 4); *mgid = *(union ib_gid *) (mc_map + 4);
} }
} }
......
...@@ -788,10 +788,6 @@ void ipoib_mcast_restart_task(struct work_struct *work) ...@@ -788,10 +788,6 @@ void ipoib_mcast_restart_task(struct work_struct *work)
memcpy(mgid.raw, mclist->dmi_addr + 4, sizeof mgid); memcpy(mgid.raw, mclist->dmi_addr + 4, sizeof mgid);
/* Add in the P_Key */
mgid.raw[4] = (priv->pkey >> 8) & 0xff;
mgid.raw[5] = priv->pkey & 0xff;
mcast = __ipoib_mcast_find(dev, &mgid); mcast = __ipoib_mcast_find(dev, &mgid);
if (!mcast || test_bit(IPOIB_MCAST_FLAG_SENDONLY, &mcast->flags)) { if (!mcast || test_bit(IPOIB_MCAST_FLAG_SENDONLY, &mcast->flags)) {
struct ipoib_mcast *nmcast; struct ipoib_mcast *nmcast;
......
...@@ -269,18 +269,21 @@ static inline void ipv6_arcnet_mc_map(const struct in6_addr *addr, char *buf) ...@@ -269,18 +269,21 @@ static inline void ipv6_arcnet_mc_map(const struct in6_addr *addr, char *buf)
buf[0] = 0x00; buf[0] = 0x00;
} }
static inline void ipv6_ib_mc_map(struct in6_addr *addr, char *buf) static inline void ipv6_ib_mc_map(const struct in6_addr *addr,
const unsigned char *broadcast, char *buf)
{ {
unsigned char scope = broadcast[5] & 0xF;
buf[0] = 0; /* Reserved */ buf[0] = 0; /* Reserved */
buf[1] = 0xff; /* Multicast QPN */ buf[1] = 0xff; /* Multicast QPN */
buf[2] = 0xff; buf[2] = 0xff;
buf[3] = 0xff; buf[3] = 0xff;
buf[4] = 0xff; buf[4] = 0xff;
buf[5] = 0x12; /* link local scope */ buf[5] = 0x10 | scope; /* scope from broadcast address */
buf[6] = 0x60; /* IPv6 signature */ buf[6] = 0x60; /* IPv6 signature */
buf[7] = 0x1b; buf[7] = 0x1b;
buf[8] = 0; /* P_Key */ buf[8] = broadcast[8]; /* P_Key */
buf[9] = 0; buf[9] = broadcast[9];
memcpy(buf + 10, addr->s6_addr + 6, 10); memcpy(buf + 10, addr->s6_addr + 6, 10);
} }
#endif #endif
......
...@@ -266,20 +266,22 @@ static inline void ip_eth_mc_map(__be32 naddr, char *buf) ...@@ -266,20 +266,22 @@ static inline void ip_eth_mc_map(__be32 naddr, char *buf)
* Leave P_Key as 0 to be filled in by driver. * Leave P_Key as 0 to be filled in by driver.
*/ */
static inline void ip_ib_mc_map(__be32 naddr, char *buf) static inline void ip_ib_mc_map(__be32 naddr, const unsigned char *broadcast, char *buf)
{ {
__u32 addr; __u32 addr;
unsigned char scope = broadcast[5] & 0xF;
buf[0] = 0; /* Reserved */ buf[0] = 0; /* Reserved */
buf[1] = 0xff; /* Multicast QPN */ buf[1] = 0xff; /* Multicast QPN */
buf[2] = 0xff; buf[2] = 0xff;
buf[3] = 0xff; buf[3] = 0xff;
addr = ntohl(naddr); addr = ntohl(naddr);
buf[4] = 0xff; buf[4] = 0xff;
buf[5] = 0x12; /* link local scope */ buf[5] = 0x10 | scope; /* scope from broadcast address */
buf[6] = 0x40; /* IPv4 signature */ buf[6] = 0x40; /* IPv4 signature */
buf[7] = 0x1b; buf[7] = 0x1b;
buf[8] = 0; /* P_Key */ buf[8] = broadcast[8]; /* P_Key */
buf[9] = 0; buf[9] = broadcast[9];
buf[10] = 0; buf[10] = 0;
buf[11] = 0; buf[11] = 0;
buf[12] = 0; buf[12] = 0;
......
...@@ -211,7 +211,7 @@ int arp_mc_map(__be32 addr, u8 *haddr, struct net_device *dev, int dir) ...@@ -211,7 +211,7 @@ int arp_mc_map(__be32 addr, u8 *haddr, struct net_device *dev, int dir)
ip_tr_mc_map(addr, haddr); ip_tr_mc_map(addr, haddr);
return 0; return 0;
case ARPHRD_INFINIBAND: case ARPHRD_INFINIBAND:
ip_ib_mc_map(addr, haddr); ip_ib_mc_map(addr, dev->broadcast, haddr);
return 0; return 0;
default: default:
if (dir) { if (dir) {
......
...@@ -337,7 +337,7 @@ int ndisc_mc_map(struct in6_addr *addr, char *buf, struct net_device *dev, int d ...@@ -337,7 +337,7 @@ int ndisc_mc_map(struct in6_addr *addr, char *buf, struct net_device *dev, int d
ipv6_arcnet_mc_map(addr, buf); ipv6_arcnet_mc_map(addr, buf);
return 0; return 0;
case ARPHRD_INFINIBAND: case ARPHRD_INFINIBAND:
ipv6_ib_mc_map(addr, buf); ipv6_ib_mc_map(addr, dev->broadcast, buf);
return 0; return 0;
default: default:
if (dir) { if (dir) {
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册