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(ð, 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(ð.smac[0], &dev->nic_info.mac_addr[0], ETH_ALEN);
120*4882a593Smuzhiyun status = ocrdma_resolve_dmac(dev, attr, ð.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_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