+ rtnl_unlock();
+ spin_unlock_bh(&sk->sk_lock.slock);
+ if (err && err != -EADDRINUSE)
+ goto err_out;
ipv6_eth_mc_map((struct in6_addr *)mgid->raw, ll_addr);
+ err = dev_mc_add(rxe->ndev, ll_addr);
+ if (err)
+ goto err_drop;
+
+ return 0;
- return dev_mc_add(rxe->ndev, ll_addr);
+err_drop:
+ spin_lock_bh(&sk->sk_lock.slock);
+ rtnl_lock();
+ ipv6_sock_mc_drop(sk, rxe->ndev->ifindex, addr6);
+ rtnl_unlock();
+ spin_unlock_bh(&sk->sk_lock.slock);
+err_out:
+ return err;
}
-/**
- * rxe_mcast_del - delete multicast address from rxe device
- * @rxe: rxe device object
- * @mgid: multicast address as a gid
- *
- * Returns 0 on success else an error
- */
-static int rxe_mcast_del(struct rxe_dev *rxe, union ib_gid *mgid)
+static int rxe_mcast_add(struct rxe_mcg *mcg)
{
+ struct rxe_dev *rxe = mcg->rxe;
+ union ib_gid *mgid = &mcg->mgid;
unsigned char ll_addr[ETH_ALEN];
+ struct ip_mreqn imr = {};
+ int err;
+
+ if (mcg->is_ipv6)
+ return rxe_mcast_add6(rxe, mgid);
+
+ imr.imr_multiaddr = *(struct in_addr *)(mgid->raw + 12);
+ imr.imr_ifindex = rxe->ndev->ifindex;
+ rtnl_lock();
+ err = ip_mc_join_group(recv_sockets.sk4->sk, &imr);
+ rtnl_unlock();
+ if (err && err != -EADDRINUSE)
+ goto err_out;
+
+ ip_eth_mc_map(imr.imr_multiaddr.s_addr, ll_addr);
+ err = dev_mc_add(rxe->ndev, ll_addr);
+ if (err)
+ goto err_leave;
+
+ return 0;
+
+err_leave:
+ rtnl_lock();
+ ip_mc_leave_group(recv_sockets.sk4->sk, &imr);
+ rtnl_unlock();
+err_out:
+ return err;
+}
+
+static int rxe_mcast_del6(struct rxe_dev *rxe, union ib_gid *mgid)
+{
+ struct sock *sk = recv_sockets.sk6->sk;
+ unsigned char ll_addr[ETH_ALEN];
+ int err, err2;
ipv6_eth_mc_map((struct in6_addr *)mgid->raw, ll_addr);
+ err = dev_mc_del(rxe->ndev, ll_addr);
+
+ spin_lock_bh(&sk->sk_lock.slock);
+ rtnl_lock();
+ err2 = ipv6_sock_mc_drop(sk, rxe->ndev->ifindex,
+ (struct in6_addr *)mgid);
+ rtnl_unlock();
+ spin_unlock_bh(&sk->sk_lock.slock);
+
+ return err ?: err2;
+}
+
+static int rxe_mcast_del(struct rxe_mcg *mcg)
+{
+ struct rxe_dev *rxe = mcg->rxe;
+ union ib_gid *mgid = &mcg->mgid;
+ unsigned char ll_addr[ETH_ALEN];
+ struct ip_mreqn imr = {};
+ int err, err2;
+
+ if (mcg->is_ipv6)
+ return rxe_mcast_del6(rxe, mgid);
+
+ imr.imr_multiaddr = *(struct in_addr *)(mgid->raw + 12);
+ imr.imr_ifindex = rxe->ndev->ifindex;
+ ip_eth_mc_map(imr.imr_multiaddr.s_addr, ll_addr);
+ err = dev_mc_del(rxe->ndev, ll_addr);
+
+ rtnl_lock();
+ err2 = ip_mc_leave_group(recv_sockets.sk4->sk, &imr);
+ rtnl_unlock();
- return dev_mc_del(rxe->ndev, ll_addr);
+ return err ?: err2;
}
/**
@@ -164,6 +242,7 @@ static void __rxe_init_mcg(struct rxe_dev *rxe,
union ib_gid *mgid,
{
kref_init(&mcg->ref_cnt);
memcpy(&mcg->mgid, mgid, sizeof(mcg->mgid));
+ mcg->is_ipv6 = !ipv6_addr_v4mapped((struct in6_addr *)mgid);
INIT_LIST_HEAD(&mcg->qp_list);
mcg->rxe = rxe;
@@ -225,7 +304,7 @@ static struct rxe_mcg *rxe_get_mcg(struct
rxe_dev *rxe, union ib_gid *mgid)
spin_unlock_bh(&rxe->mcg_lock);
/* add mcast address outside of lock */
- err = rxe_mcast_add(rxe, mgid);
+ err = rxe_mcast_add(mcg);
if (!err)
return mcg;
@@ -273,7 +352,7 @@ static void __rxe_destroy_mcg(struct rxe_mcg *mcg)
static void rxe_destroy_mcg(struct rxe_mcg *mcg)
{
/* delete mcast address outside of lock */
- rxe_mcast_del(mcg->rxe, &mcg->mgid);
+ rxe_mcast_del(mcg);
spin_lock_bh(&mcg->rxe->mcg_lock);
__rxe_destroy_mcg(mcg);
diff --git a/drivers/infiniband/sw/rxe/rxe_net.c
b/drivers/infiniband/sw/rxe/rxe_net.c
index 58c3f3759bf0..b481f8da2002 100644
--- a/drivers/infiniband/sw/rxe/rxe_net.c
+++ b/drivers/infiniband/sw/rxe/rxe_net.c
@@ -18,7 +18,7 @@
#include "rxe_net.h"
#include "rxe_loc.h"
-static struct rxe_recv_sockets recv_sockets;
+struct rxe_recv_sockets recv_sockets;
static struct dst_entry *rxe_find_route4(struct rxe_qp *qp,
struct net_device *ndev,
diff --git a/drivers/infiniband/sw/rxe/rxe_net.h
b/drivers/infiniband/sw/rxe/rxe_net.h
index 45d80d00f86b..89cee7d5340f 100644
--- a/drivers/infiniband/sw/rxe/rxe_net.h
+++ b/drivers/infiniband/sw/rxe/rxe_net.h
@@ -15,6 +15,7 @@ struct rxe_recv_sockets {
struct socket *sk4;
struct socket *sk6;
};
+extern struct rxe_recv_sockets recv_sockets;
int rxe_net_add(const char *ibdev_name, struct net_device *ndev);
diff --git a/drivers/infiniband/sw/rxe/rxe_verbs.h
b/drivers/infiniband/sw/rxe/rxe_verbs.h
index ccb9d19ffe8a..7be9e6232dd9 100644
--- a/drivers/infiniband/sw/rxe/rxe_verbs.h
+++ b/drivers/infiniband/sw/rxe/rxe_verbs.h
@@ -352,6 +352,7 @@ struct rxe_mcg {
atomic_t qp_num;
u32 qkey;
u16 pkey;
+ bool is_ipv6;
};
struct rxe_mca {