xref: /OK3568_Linux_fs/kernel/drivers/infiniband/hw/ocrdma/ocrdma_ah.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun /* This file is part of the Emulex RoCE Device Driver for
2*4882a593Smuzhiyun  * RoCE (RDMA over Converged Ethernet) adapters.
3*4882a593Smuzhiyun  * Copyright (C) 2012-2015 Emulex. All rights reserved.
4*4882a593Smuzhiyun  * EMULEX and SLI are trademarks of Emulex.
5*4882a593Smuzhiyun  * www.emulex.com
6*4882a593Smuzhiyun  *
7*4882a593Smuzhiyun  * This software is available to you under a choice of one of two licenses.
8*4882a593Smuzhiyun  * You may choose to be licensed under the terms of the GNU General Public
9*4882a593Smuzhiyun  * License (GPL) Version 2, available from the file COPYING in the main
10*4882a593Smuzhiyun  * directory of this source tree, or the BSD license below:
11*4882a593Smuzhiyun  *
12*4882a593Smuzhiyun  * Redistribution and use in source and binary forms, with or without
13*4882a593Smuzhiyun  * modification, are permitted provided that the following conditions
14*4882a593Smuzhiyun  * are met:
15*4882a593Smuzhiyun  *
16*4882a593Smuzhiyun  * - Redistributions of source code must retain the above copyright notice,
17*4882a593Smuzhiyun  *   this list of conditions and the following disclaimer.
18*4882a593Smuzhiyun  *
19*4882a593Smuzhiyun  * - Redistributions in binary form must reproduce the above copyright
20*4882a593Smuzhiyun  *   notice, this list of conditions and the following disclaimer in
21*4882a593Smuzhiyun  *   the documentation and/or other materials provided with the distribution.
22*4882a593Smuzhiyun  *
23*4882a593Smuzhiyun  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
24*4882a593Smuzhiyun  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,THE
25*4882a593Smuzhiyun  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
26*4882a593Smuzhiyun  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
27*4882a593Smuzhiyun  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
28*4882a593Smuzhiyun  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
29*4882a593Smuzhiyun  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
30*4882a593Smuzhiyun  * BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
31*4882a593Smuzhiyun  * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
32*4882a593Smuzhiyun  * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
33*4882a593Smuzhiyun  * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
34*4882a593Smuzhiyun  *
35*4882a593Smuzhiyun  * Contact Information:
36*4882a593Smuzhiyun  * linux-drivers@emulex.com
37*4882a593Smuzhiyun  *
38*4882a593Smuzhiyun  * Emulex
39*4882a593Smuzhiyun  * 3333 Susan Street
40*4882a593Smuzhiyun  * Costa Mesa, CA 92626
41*4882a593Smuzhiyun  */
42*4882a593Smuzhiyun 
43*4882a593Smuzhiyun #include <net/neighbour.h>
44*4882a593Smuzhiyun #include <net/netevent.h>
45*4882a593Smuzhiyun 
46*4882a593Smuzhiyun #include <rdma/ib_addr.h>
47*4882a593Smuzhiyun #include <rdma/ib_mad.h>
48*4882a593Smuzhiyun #include <rdma/ib_cache.h>
49*4882a593Smuzhiyun 
50*4882a593Smuzhiyun #include "ocrdma.h"
51*4882a593Smuzhiyun #include "ocrdma_verbs.h"
52*4882a593Smuzhiyun #include "ocrdma_ah.h"
53*4882a593Smuzhiyun #include "ocrdma_hw.h"
54*4882a593Smuzhiyun #include "ocrdma_stats.h"
55*4882a593Smuzhiyun 
56*4882a593Smuzhiyun #define OCRDMA_VID_PCP_SHIFT	0xD
57*4882a593Smuzhiyun 
ocrdma_hdr_type_to_proto_num(int devid,u8 hdr_type)58*4882a593Smuzhiyun static u16 ocrdma_hdr_type_to_proto_num(int devid, u8 hdr_type)
59*4882a593Smuzhiyun {
60*4882a593Smuzhiyun 	switch (hdr_type) {
61*4882a593Smuzhiyun 	case OCRDMA_L3_TYPE_IB_GRH:
62*4882a593Smuzhiyun 		return (u16)ETH_P_IBOE;
63*4882a593Smuzhiyun 	case OCRDMA_L3_TYPE_IPV4:
64*4882a593Smuzhiyun 		return (u16)0x0800;
65*4882a593Smuzhiyun 	case OCRDMA_L3_TYPE_IPV6:
66*4882a593Smuzhiyun 		return (u16)0x86dd;
67*4882a593Smuzhiyun 	default:
68*4882a593Smuzhiyun 		pr_err("ocrdma%d: Invalid network header\n", devid);
69*4882a593Smuzhiyun 		return 0;
70*4882a593Smuzhiyun 	}
71*4882a593Smuzhiyun }
72*4882a593Smuzhiyun 
set_av_attr(struct ocrdma_dev * dev,struct ocrdma_ah * ah,struct rdma_ah_attr * attr,const union ib_gid * sgid,int pdid,bool * isvlan,u16 vlan_tag)73*4882a593Smuzhiyun static inline int set_av_attr(struct ocrdma_dev *dev, struct ocrdma_ah *ah,
74*4882a593Smuzhiyun 			struct rdma_ah_attr *attr, const union ib_gid *sgid,
75*4882a593Smuzhiyun 			int pdid, bool *isvlan, u16 vlan_tag)
76*4882a593Smuzhiyun {
77*4882a593Smuzhiyun 	int status;
78*4882a593Smuzhiyun 	struct ocrdma_eth_vlan eth;
79*4882a593Smuzhiyun 	struct ocrdma_grh grh;
80*4882a593Smuzhiyun 	int eth_sz;
81*4882a593Smuzhiyun 	u16 proto_num = 0;
82*4882a593Smuzhiyun 	u8 nxthdr = 0x11;
83*4882a593Smuzhiyun 	struct iphdr ipv4;
84*4882a593Smuzhiyun 	const struct ib_global_route *ib_grh;
85*4882a593Smuzhiyun 	union {
86*4882a593Smuzhiyun 		struct sockaddr_in  _sockaddr_in;
87*4882a593Smuzhiyun 		struct sockaddr_in6 _sockaddr_in6;
88*4882a593Smuzhiyun 	} sgid_addr, dgid_addr;
89*4882a593Smuzhiyun 
90*4882a593Smuzhiyun 	memset(&eth, 0, sizeof(eth));
91*4882a593Smuzhiyun 	memset(&grh, 0, sizeof(grh));
92*4882a593Smuzhiyun 
93*4882a593Smuzhiyun 	/* Protocol Number */
94*4882a593Smuzhiyun 	proto_num = ocrdma_hdr_type_to_proto_num(dev->id, ah->hdr_type);
95*4882a593Smuzhiyun 	if (!proto_num)
96*4882a593Smuzhiyun 		return -EINVAL;
97*4882a593Smuzhiyun 	nxthdr = (proto_num == ETH_P_IBOE) ? 0x1b : 0x11;
98*4882a593Smuzhiyun 	/* VLAN */
99*4882a593Smuzhiyun 	if (!vlan_tag || (vlan_tag > 0xFFF))
100*4882a593Smuzhiyun 		vlan_tag = dev->pvid;
101*4882a593Smuzhiyun 	if (vlan_tag || dev->pfc_state) {
102*4882a593Smuzhiyun 		if (!vlan_tag) {
103*4882a593Smuzhiyun 			pr_err("ocrdma%d:Using VLAN with PFC is recommended\n",
104*4882a593Smuzhiyun 				dev->id);
105*4882a593Smuzhiyun 			pr_err("ocrdma%d:Using VLAN 0 for this connection\n",
106*4882a593Smuzhiyun 				dev->id);
107*4882a593Smuzhiyun 		}
108*4882a593Smuzhiyun 		eth.eth_type = cpu_to_be16(0x8100);
109*4882a593Smuzhiyun 		eth.roce_eth_type = cpu_to_be16(proto_num);
110*4882a593Smuzhiyun 		vlan_tag |= (dev->sl & 0x07) << OCRDMA_VID_PCP_SHIFT;
111*4882a593Smuzhiyun 		eth.vlan_tag = cpu_to_be16(vlan_tag);
112*4882a593Smuzhiyun 		eth_sz = sizeof(struct ocrdma_eth_vlan);
113*4882a593Smuzhiyun 		*isvlan = true;
114*4882a593Smuzhiyun 	} else {
115*4882a593Smuzhiyun 		eth.eth_type = cpu_to_be16(proto_num);
116*4882a593Smuzhiyun 		eth_sz = sizeof(struct ocrdma_eth_basic);
117*4882a593Smuzhiyun 	}
118*4882a593Smuzhiyun 	/* MAC */
119*4882a593Smuzhiyun 	memcpy(&eth.smac[0], &dev->nic_info.mac_addr[0], ETH_ALEN);
120*4882a593Smuzhiyun 	status = ocrdma_resolve_dmac(dev, attr, &eth.dmac[0]);
121*4882a593Smuzhiyun 	if (status)
122*4882a593Smuzhiyun 		return status;
123*4882a593Smuzhiyun 	ib_grh = rdma_ah_read_grh(attr);
124*4882a593Smuzhiyun 	ah->sgid_index = ib_grh->sgid_index;
125*4882a593Smuzhiyun 	/* Eth HDR */
126*4882a593Smuzhiyun 	memcpy(&ah->av->eth_hdr, &eth, eth_sz);
127*4882a593Smuzhiyun 	if (ah->hdr_type == RDMA_NETWORK_IPV4) {
128*4882a593Smuzhiyun 		*((__be16 *)&ipv4) = htons((4 << 12) | (5 << 8) |
129*4882a593Smuzhiyun 					   ib_grh->traffic_class);
130*4882a593Smuzhiyun 		ipv4.id = cpu_to_be16(pdid);
131*4882a593Smuzhiyun 		ipv4.frag_off = htons(IP_DF);
132*4882a593Smuzhiyun 		ipv4.tot_len = htons(0);
133*4882a593Smuzhiyun 		ipv4.ttl = ib_grh->hop_limit;
134*4882a593Smuzhiyun 		ipv4.protocol = nxthdr;
135*4882a593Smuzhiyun 		rdma_gid2ip((struct sockaddr *)&sgid_addr, sgid);
136*4882a593Smuzhiyun 		ipv4.saddr = sgid_addr._sockaddr_in.sin_addr.s_addr;
137*4882a593Smuzhiyun 		rdma_gid2ip((struct sockaddr*)&dgid_addr, &ib_grh->dgid);
138*4882a593Smuzhiyun 		ipv4.daddr = dgid_addr._sockaddr_in.sin_addr.s_addr;
139*4882a593Smuzhiyun 		memcpy((u8 *)ah->av + eth_sz, &ipv4, sizeof(struct iphdr));
140*4882a593Smuzhiyun 	} else {
141*4882a593Smuzhiyun 		memcpy(&grh.sgid[0], sgid->raw, sizeof(union ib_gid));
142*4882a593Smuzhiyun 		grh.tclass_flow = cpu_to_be32((6 << 28) |
143*4882a593Smuzhiyun 					      (ib_grh->traffic_class << 24) |
144*4882a593Smuzhiyun 					      ib_grh->flow_label);
145*4882a593Smuzhiyun 		memcpy(&grh.dgid[0], ib_grh->dgid.raw,
146*4882a593Smuzhiyun 		       sizeof(ib_grh->dgid.raw));
147*4882a593Smuzhiyun 		grh.pdid_hoplimit = cpu_to_be32((pdid << 16) |
148*4882a593Smuzhiyun 						(nxthdr << 8) |
149*4882a593Smuzhiyun 						ib_grh->hop_limit);
150*4882a593Smuzhiyun 		memcpy((u8 *)ah->av + eth_sz, &grh, sizeof(struct ocrdma_grh));
151*4882a593Smuzhiyun 	}
152*4882a593Smuzhiyun 	if (*isvlan)
153*4882a593Smuzhiyun 		ah->av->valid |= OCRDMA_AV_VLAN_VALID;
154*4882a593Smuzhiyun 	ah->av->valid = cpu_to_le32(ah->av->valid);
155*4882a593Smuzhiyun 	return status;
156*4882a593Smuzhiyun }
157*4882a593Smuzhiyun 
ocrdma_create_ah(struct ib_ah * ibah,struct rdma_ah_init_attr * init_attr,struct ib_udata * udata)158*4882a593Smuzhiyun int ocrdma_create_ah(struct ib_ah *ibah, struct rdma_ah_init_attr *init_attr,
159*4882a593Smuzhiyun 		     struct ib_udata *udata)
160*4882a593Smuzhiyun {
161*4882a593Smuzhiyun 	u32 *ahid_addr;
162*4882a593Smuzhiyun 	int status;
163*4882a593Smuzhiyun 	struct ocrdma_ah *ah = get_ocrdma_ah(ibah);
164*4882a593Smuzhiyun 	bool isvlan = false;
165*4882a593Smuzhiyun 	u16 vlan_tag = 0xffff;
166*4882a593Smuzhiyun 	const struct ib_gid_attr *sgid_attr;
167*4882a593Smuzhiyun 	struct ocrdma_pd *pd = get_ocrdma_pd(ibah->pd);
168*4882a593Smuzhiyun 	struct rdma_ah_attr *attr = init_attr->ah_attr;
169*4882a593Smuzhiyun 	struct ocrdma_dev *dev = get_ocrdma_dev(ibah->device);
170*4882a593Smuzhiyun 
171*4882a593Smuzhiyun 	if ((attr->type != RDMA_AH_ATTR_TYPE_ROCE) ||
172*4882a593Smuzhiyun 	    !(rdma_ah_get_ah_flags(attr) & IB_AH_GRH))
173*4882a593Smuzhiyun 		return -EINVAL;
174*4882a593Smuzhiyun 
175*4882a593Smuzhiyun 	if (atomic_cmpxchg(&dev->update_sl, 1, 0))
176*4882a593Smuzhiyun 		ocrdma_init_service_level(dev);
177*4882a593Smuzhiyun 
178*4882a593Smuzhiyun 	sgid_attr = attr->grh.sgid_attr;
179*4882a593Smuzhiyun 	status = rdma_read_gid_l2_fields(sgid_attr, &vlan_tag, NULL);
180*4882a593Smuzhiyun 	if (status)
181*4882a593Smuzhiyun 		return status;
182*4882a593Smuzhiyun 
183*4882a593Smuzhiyun 	status = ocrdma_alloc_av(dev, ah);
184*4882a593Smuzhiyun 	if (status)
185*4882a593Smuzhiyun 		goto av_err;
186*4882a593Smuzhiyun 
187*4882a593Smuzhiyun 	/* Get network header type for this GID */
188*4882a593Smuzhiyun 	ah->hdr_type = rdma_gid_attr_network_type(sgid_attr);
189*4882a593Smuzhiyun 
190*4882a593Smuzhiyun 	status = set_av_attr(dev, ah, attr, &sgid_attr->gid, pd->id,
191*4882a593Smuzhiyun 			     &isvlan, vlan_tag);
192*4882a593Smuzhiyun 	if (status)
193*4882a593Smuzhiyun 		goto av_conf_err;
194*4882a593Smuzhiyun 
195*4882a593Smuzhiyun 	/* if pd is for the user process, pass the ah_id to user space */
196*4882a593Smuzhiyun 	if ((pd->uctx) && (pd->uctx->ah_tbl.va)) {
197*4882a593Smuzhiyun 		ahid_addr = pd->uctx->ah_tbl.va + rdma_ah_get_dlid(attr);
198*4882a593Smuzhiyun 		*ahid_addr = 0;
199*4882a593Smuzhiyun 		*ahid_addr |= ah->id & OCRDMA_AH_ID_MASK;
200*4882a593Smuzhiyun 		if (ocrdma_is_udp_encap_supported(dev)) {
201*4882a593Smuzhiyun 			*ahid_addr |= ((u32)ah->hdr_type &
202*4882a593Smuzhiyun 				       OCRDMA_AH_L3_TYPE_MASK) <<
203*4882a593Smuzhiyun 				       OCRDMA_AH_L3_TYPE_SHIFT;
204*4882a593Smuzhiyun 		}
205*4882a593Smuzhiyun 		if (isvlan)
206*4882a593Smuzhiyun 			*ahid_addr |= (OCRDMA_AH_VLAN_VALID_MASK <<
207*4882a593Smuzhiyun 				       OCRDMA_AH_VLAN_VALID_SHIFT);
208*4882a593Smuzhiyun 	}
209*4882a593Smuzhiyun 
210*4882a593Smuzhiyun 	return 0;
211*4882a593Smuzhiyun 
212*4882a593Smuzhiyun av_conf_err:
213*4882a593Smuzhiyun 	ocrdma_free_av(dev, ah);
214*4882a593Smuzhiyun av_err:
215*4882a593Smuzhiyun 	return status;
216*4882a593Smuzhiyun }
217*4882a593Smuzhiyun 
ocrdma_destroy_ah(struct ib_ah * ibah,u32 flags)218*4882a593Smuzhiyun int ocrdma_destroy_ah(struct ib_ah *ibah, u32 flags)
219*4882a593Smuzhiyun {
220*4882a593Smuzhiyun 	struct ocrdma_ah *ah = get_ocrdma_ah(ibah);
221*4882a593Smuzhiyun 	struct ocrdma_dev *dev = get_ocrdma_dev(ibah->device);
222*4882a593Smuzhiyun 
223*4882a593Smuzhiyun 	ocrdma_free_av(dev, ah);
224*4882a593Smuzhiyun 	return 0;
225*4882a593Smuzhiyun }
226*4882a593Smuzhiyun 
ocrdma_query_ah(struct ib_ah * ibah,struct rdma_ah_attr * attr)227*4882a593Smuzhiyun int ocrdma_query_ah(struct ib_ah *ibah, struct rdma_ah_attr *attr)
228*4882a593Smuzhiyun {
229*4882a593Smuzhiyun 	struct ocrdma_ah *ah = get_ocrdma_ah(ibah);
230*4882a593Smuzhiyun 	struct ocrdma_av *av = ah->av;
231*4882a593Smuzhiyun 	struct ocrdma_grh *grh;
232*4882a593Smuzhiyun 
233*4882a593Smuzhiyun 	attr->type = ibah->type;
234*4882a593Smuzhiyun 	if (ah->av->valid & OCRDMA_AV_VALID) {
235*4882a593Smuzhiyun 		grh = (struct ocrdma_grh *)((u8 *)ah->av +
236*4882a593Smuzhiyun 				sizeof(struct ocrdma_eth_vlan));
237*4882a593Smuzhiyun 		rdma_ah_set_sl(attr, be16_to_cpu(av->eth_hdr.vlan_tag) >> 13);
238*4882a593Smuzhiyun 	} else {
239*4882a593Smuzhiyun 		grh = (struct ocrdma_grh *)((u8 *)ah->av +
240*4882a593Smuzhiyun 					sizeof(struct ocrdma_eth_basic));
241*4882a593Smuzhiyun 		rdma_ah_set_sl(attr, 0);
242*4882a593Smuzhiyun 	}
243*4882a593Smuzhiyun 	rdma_ah_set_grh(attr, NULL,
244*4882a593Smuzhiyun 			be32_to_cpu(grh->tclass_flow) & 0xffffffff,
245*4882a593Smuzhiyun 			ah->sgid_index,
246*4882a593Smuzhiyun 			be32_to_cpu(grh->pdid_hoplimit) & 0xff,
247*4882a593Smuzhiyun 			be32_to_cpu(grh->tclass_flow) >> 24);
248*4882a593Smuzhiyun 	rdma_ah_set_dgid_raw(attr, &grh->dgid[0]);
249*4882a593Smuzhiyun 	return 0;
250*4882a593Smuzhiyun }
251*4882a593Smuzhiyun 
ocrdma_process_mad(struct ib_device * ibdev,int process_mad_flags,u8 port_num,const struct ib_wc * in_wc,const struct ib_grh * in_grh,const struct ib_mad * in,struct ib_mad * out,size_t * out_mad_size,u16 * out_mad_pkey_index)252*4882a593Smuzhiyun int ocrdma_process_mad(struct ib_device *ibdev, int process_mad_flags,
253*4882a593Smuzhiyun 		       u8 port_num, const struct ib_wc *in_wc,
254*4882a593Smuzhiyun 		       const struct ib_grh *in_grh, const struct ib_mad *in,
255*4882a593Smuzhiyun 		       struct ib_mad *out, size_t *out_mad_size,
256*4882a593Smuzhiyun 		       u16 *out_mad_pkey_index)
257*4882a593Smuzhiyun {
258*4882a593Smuzhiyun 	int status = IB_MAD_RESULT_SUCCESS;
259*4882a593Smuzhiyun 	struct ocrdma_dev *dev;
260*4882a593Smuzhiyun 
261*4882a593Smuzhiyun 	if (in->mad_hdr.mgmt_class == IB_MGMT_CLASS_PERF_MGMT) {
262*4882a593Smuzhiyun 		dev = get_ocrdma_dev(ibdev);
263*4882a593Smuzhiyun 		ocrdma_pma_counters(dev, out);
264*4882a593Smuzhiyun 		status |= IB_MAD_RESULT_REPLY;
265*4882a593Smuzhiyun 	}
266*4882a593Smuzhiyun 
267*4882a593Smuzhiyun 	return status;
268*4882a593Smuzhiyun }
269