Add support for prio-tagged traffic for RDMA in ibcore.

When receiving a PCP change all GID entries are reloaded.
This ensures the relevant GID entries use prio tagging,
by setting VLAN present and VLAN ID to zero.

The priority for prio tagged traffic is set using the regular
rdma_set_service_type() function.

Fake the real network device to have a VLAN ID of zero
when prio tagging is enabled. This is logic is hidden inside
the rdma_vlan_dev_vlan_id() function which must always be used
to retrieve the VLAN ID throughout all of ibcore and the
infiniband network drivers.

The VLAN presence information then propagates through all
of ibcore and so incoming connections will have the VLAN
bit set. The incoming VLAN ID is then checked against the
return value of rdma_vlan_dev_vlan_id().

MFC after:		1 week
Sponsored by:		Mellanox Technologies
This commit is contained in:
hselasky 2018-07-17 09:11:53 +00:00
parent 22ad2d6d8d
commit ecd38d68da
5 changed files with 32 additions and 11 deletions

View File

@ -96,8 +96,7 @@ static struct ib_ah *create_iboe_ah(struct ib_pd *pd, struct ib_ah_attr *ah_attr
return ERR_PTR(ret); return ERR_PTR(ret);
eth_zero_addr(ah->av.eth.s_mac); eth_zero_addr(ah->av.eth.s_mac);
if (gid_attr.ndev) { if (gid_attr.ndev) {
if (is_vlan_dev(gid_attr.ndev)) vlan_tag = rdma_vlan_dev_vlan_id(gid_attr.ndev);
vlan_tag = vlan_dev_vlan_id(gid_attr.ndev);
memcpy(ah->av.eth.s_mac, IF_LLADDR(gid_attr.ndev), ETH_ALEN); memcpy(ah->av.eth.s_mac, IF_LLADDR(gid_attr.ndev), ETH_ALEN);
dev_put(gid_attr.ndev); dev_put(gid_attr.ndev);
} }

View File

@ -285,14 +285,16 @@ static void ib_gid_to_mlx5_roce_addr(const union ib_gid *gid,
source_l3_address); source_l3_address);
void *mlx5_addr_mac = MLX5_ADDR_OF(roce_addr_layout, mlx5_addr, void *mlx5_addr_mac = MLX5_ADDR_OF(roce_addr_layout, mlx5_addr,
source_mac_47_32); source_mac_47_32);
u16 vlan_id;
if (!gid) if (!gid)
return; return;
ether_addr_copy(mlx5_addr_mac, IF_LLADDR(attr->ndev)); ether_addr_copy(mlx5_addr_mac, IF_LLADDR(attr->ndev));
if (is_vlan_dev(attr->ndev)) { vlan_id = rdma_vlan_dev_vlan_id(attr->ndev);
if (vlan_id != 0xffff) {
MLX5_SET_RA(mlx5_addr, vlan_valid, 1); MLX5_SET_RA(mlx5_addr, vlan_valid, 1);
MLX5_SET_RA(mlx5_addr, vlan_id, vlan_dev_vlan_id(attr->ndev)); MLX5_SET_RA(mlx5_addr, vlan_id, vlan_id);
} }
switch (attr->gid_type) { switch (attr->gid_type) {

View File

@ -402,6 +402,19 @@ static struct notifier_block nb_inetaddr = {
.notifier_call = inetaddr_event .notifier_call = inetaddr_event
}; };
static eventhandler_tag eh_ifnet_event;
static void
roce_ifnet_event(void *arg, struct ifnet *ifp, int event)
{
if (event != IFNET_EVENT_PCP || is_vlan_dev(ifp))
return;
/* make sure GID table is reloaded */
roce_gid_delete_all_event(ifp);
roce_gid_queue_scan_event(ifp);
}
static void static void
roce_rescan_device_handler(struct work_struct *_work) roce_rescan_device_handler(struct work_struct *_work)
{ {
@ -445,11 +458,18 @@ int __init roce_gid_mgmt_init(void)
*/ */
register_netdevice_notifier(&nb_inetaddr); register_netdevice_notifier(&nb_inetaddr);
eh_ifnet_event = EVENTHANDLER_REGISTER(ifnet_event,
roce_ifnet_event, NULL, EVENTHANDLER_PRI_ANY);
return 0; return 0;
} }
void __exit roce_gid_mgmt_cleanup(void) void __exit roce_gid_mgmt_cleanup(void)
{ {
if (eh_ifnet_event != NULL)
EVENTHANDLER_DEREGISTER(ifnet_event, eh_ifnet_event);
unregister_inetaddr_notifier(&nb_inetaddr); unregister_inetaddr_notifier(&nb_inetaddr);
unregister_netdevice_notifier(&nb_inetaddr); unregister_netdevice_notifier(&nb_inetaddr);

View File

@ -402,12 +402,8 @@ static bool find_gid_index(const union ib_gid *gid,
if (ctx->gid_type != gid_attr->gid_type) if (ctx->gid_type != gid_attr->gid_type)
return false; return false;
if (rdma_vlan_dev_vlan_id(gid_attr->ndev) != ctx->vlan_id)
if ((!!(ctx->vlan_id != 0xffff) == !is_vlan_dev(gid_attr->ndev)) ||
(is_vlan_dev(gid_attr->ndev) &&
vlan_dev_vlan_id(gid_attr->ndev) != ctx->vlan_id))
return false; return false;
return true; return true;
} }
@ -484,7 +480,7 @@ int ib_init_ah_from_wc(struct ib_device *device, u8 port_num,
if (rdma_protocol_roce(device, port_num)) { if (rdma_protocol_roce(device, port_num)) {
struct ib_gid_attr dgid_attr; struct ib_gid_attr dgid_attr;
const u16 vlan_id = wc->wc_flags & IB_WC_WITH_VLAN ? const u16 vlan_id = (wc->wc_flags & IB_WC_WITH_VLAN) ?
wc->vlan_id : 0xffff; wc->vlan_id : 0xffff;
if (!(wc->wc_flags & IB_WC_GRH)) if (!(wc->wc_flags & IB_WC_GRH))

View File

@ -166,6 +166,8 @@ static inline u16 rdma_vlan_dev_vlan_id(const struct net_device *dev)
{ {
uint16_t tag; uint16_t tag;
if (dev->if_pcp != IFNET_PCP_NONE)
return 0x0000; /* prio-tagged traffic */
if (VLAN_TAG(__DECONST(struct ifnet *, dev), &tag) != 0) if (VLAN_TAG(__DECONST(struct ifnet *, dev), &tag) != 0)
return 0xffff; return 0xffff;
return tag; return tag;
@ -345,8 +347,10 @@ static inline u16 rdma_get_vlan_id(union ib_gid *dgid)
return vid < 0x1000 ? vid : 0xffff; return vid < 0x1000 ? vid : 0xffff;
} }
static inline struct net_device *rdma_vlan_dev_real_dev(const struct net_device *dev) static inline struct net_device *rdma_vlan_dev_real_dev(struct net_device *dev)
{ {
if (dev->if_pcp != IFNET_PCP_NONE)
return dev; /* prio-tagged traffic */
return VLAN_TRUNKDEV(__DECONST(struct ifnet *, dev)); return VLAN_TRUNKDEV(__DECONST(struct ifnet *, dev));
} }