2 * Copyright (c) 2013, Cisco Systems, Inc. All rights reserved.
4 * This program is free software; you may redistribute it and/or modify
5 * it under the terms of the GNU General Public License as published by
6 * the Free Software Foundation; version 2 of the License.
8 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
9 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
10 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
11 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
12 * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
13 * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
14 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
18 #include <linux/netdevice.h>
19 #include <linux/pci.h>
22 #include "usnic_common_pkt_hdr.h"
23 #include "usnic_fwd.h"
24 #include "usnic_log.h"
26 static int usnic_fwd_devcmd_locked(struct usnic_fwd_dev *ufdev, int vnic_idx,
27 enum vnic_devcmd_cmd cmd, u64 *a0,
31 struct net_device *netdev = ufdev->netdev;
33 lockdep_assert_held(&ufdev->lock);
35 status = enic_api_devcmd_proxy_by_index(netdev,
41 if (status == ERR_EINVAL && cmd == CMD_DEL_FILTER) {
42 usnic_dbg("Dev %s vnic idx %u cmd %u already deleted",
43 ufdev->name, vnic_idx, cmd);
45 usnic_err("Dev %s vnic idx %u cmd %u failed with status %d\n",
46 ufdev->name, vnic_idx, cmd,
50 usnic_dbg("Dev %s vnic idx %u cmd %u success",
51 ufdev->name, vnic_idx, cmd);
57 static int usnic_fwd_devcmd(struct usnic_fwd_dev *ufdev, int vnic_idx,
58 enum vnic_devcmd_cmd cmd, u64 *a0, u64 *a1)
62 spin_lock(&ufdev->lock);
63 status = usnic_fwd_devcmd_locked(ufdev, vnic_idx, cmd, a0, a1);
64 spin_unlock(&ufdev->lock);
69 struct usnic_fwd_dev *usnic_fwd_dev_alloc(struct pci_dev *pdev)
71 struct usnic_fwd_dev *ufdev;
73 ufdev = kzalloc(sizeof(*ufdev), GFP_KERNEL);
78 ufdev->netdev = pci_get_drvdata(pdev);
79 spin_lock_init(&ufdev->lock);
80 strncpy(ufdev->name, netdev_name(ufdev->netdev),
81 sizeof(ufdev->name) - 1);
86 void usnic_fwd_dev_free(struct usnic_fwd_dev *ufdev)
91 void usnic_fwd_set_mac(struct usnic_fwd_dev *ufdev, char mac[ETH_ALEN])
93 spin_lock(&ufdev->lock);
94 memcpy(&ufdev->mac, mac, sizeof(ufdev->mac));
95 spin_unlock(&ufdev->lock);
98 void usnic_fwd_add_ipaddr(struct usnic_fwd_dev *ufdev, __be32 inaddr)
100 spin_lock(&ufdev->lock);
102 ufdev->inaddr = inaddr;
103 spin_unlock(&ufdev->lock);
106 void usnic_fwd_del_ipaddr(struct usnic_fwd_dev *ufdev)
108 spin_lock(&ufdev->lock);
110 spin_unlock(&ufdev->lock);
113 void usnic_fwd_carrier_up(struct usnic_fwd_dev *ufdev)
115 spin_lock(&ufdev->lock);
117 spin_unlock(&ufdev->lock);
120 void usnic_fwd_carrier_down(struct usnic_fwd_dev *ufdev)
122 spin_lock(&ufdev->lock);
124 spin_unlock(&ufdev->lock);
127 void usnic_fwd_set_mtu(struct usnic_fwd_dev *ufdev, unsigned int mtu)
129 spin_lock(&ufdev->lock);
131 spin_unlock(&ufdev->lock);
134 static int usnic_fwd_dev_ready_locked(struct usnic_fwd_dev *ufdev)
136 lockdep_assert_held(&ufdev->lock);
144 static int validate_filter_locked(struct usnic_fwd_dev *ufdev,
145 struct filter *filter)
148 lockdep_assert_held(&ufdev->lock);
150 if (filter->type == FILTER_IPV4_5TUPLE) {
151 if (!(filter->u.ipv4.flags & FILTER_FIELD_5TUP_DST_AD))
153 if (!(filter->u.ipv4.flags & FILTER_FIELD_5TUP_DST_PT))
155 else if (ufdev->inaddr == 0)
157 else if (filter->u.ipv4.dst_port == 0)
159 else if (ntohl(ufdev->inaddr) != filter->u.ipv4.dst_addr)
168 static void fill_tlv(struct filter_tlv *tlv, struct filter *filter,
169 struct filter_action *action)
171 tlv->type = CLSF_TLV_FILTER;
172 tlv->length = sizeof(struct filter);
173 *((struct filter *)&tlv->val) = *filter;
175 tlv = (struct filter_tlv *)((char *)tlv + sizeof(struct filter_tlv) +
176 sizeof(struct filter));
177 tlv->type = CLSF_TLV_ACTION;
178 tlv->length = sizeof(struct filter_action);
179 *((struct filter_action *)&tlv->val) = *action;
182 struct usnic_fwd_flow*
183 usnic_fwd_alloc_flow(struct usnic_fwd_dev *ufdev, struct filter *filter,
184 struct usnic_filter_action *uaction)
186 struct filter_tlv *tlv;
187 struct pci_dev *pdev;
188 struct usnic_fwd_flow *flow;
195 tlv_size = (2*sizeof(struct filter_tlv) + sizeof(struct filter) +
196 sizeof(struct filter_action));
198 flow = kzalloc(sizeof(*flow), GFP_ATOMIC);
200 return ERR_PTR(-ENOMEM);
202 tlv = pci_alloc_consistent(pdev, tlv_size, &tlv_pa);
204 usnic_err("Failed to allocate memory\n");
209 fill_tlv(tlv, filter, &uaction->action);
211 spin_lock(&ufdev->lock);
212 status = usnic_fwd_dev_ready_locked(ufdev);
214 usnic_err("Forwarding dev %s not ready with status %d\n",
215 ufdev->name, status);
219 status = validate_filter_locked(ufdev, filter);
221 usnic_err("Failed to validate filter with status %d\n",
229 status = usnic_fwd_devcmd_locked(ufdev, uaction->vnic_idx,
230 CMD_ADD_FILTER, &a0, &a1);
232 usnic_err("VF %s Filter add failed with status:%d",
233 ufdev->name, status);
237 usnic_dbg("VF %s FILTER ID:%llu", ufdev->name, a0);
240 flow->flow_id = (uint32_t) a0;
241 flow->vnic_idx = uaction->vnic_idx;
245 spin_unlock(&ufdev->lock);
246 pci_free_consistent(pdev, tlv_size, tlv, tlv_pa);
251 return ERR_PTR(status);
254 int usnic_fwd_dealloc_flow(struct usnic_fwd_flow *flow)
261 status = usnic_fwd_devcmd(flow->ufdev, flow->vnic_idx,
262 CMD_DEL_FILTER, &a0, &a1);
264 if (status == ERR_EINVAL) {
265 usnic_dbg("Filter %u already deleted for VF Idx %u pf: %s status: %d",
266 flow->flow_id, flow->vnic_idx,
267 flow->ufdev->name, status);
269 usnic_err("PF %s VF Idx %u Filter: %u FILTER DELETE failed with status %d",
270 flow->ufdev->name, flow->vnic_idx,
271 flow->flow_id, status);
275 * Log the error and fake success to the caller because if
276 * a flow fails to be deleted in the firmware, it is an
277 * unrecoverable error.
280 usnic_dbg("PF %s VF Idx %u Filter: %u FILTER DELETED",
281 flow->ufdev->name, flow->vnic_idx,
289 int usnic_fwd_enable_qp(struct usnic_fwd_dev *ufdev, int vnic_idx, int qp_idx)
292 struct net_device *pf_netdev;
295 pf_netdev = ufdev->netdev;
299 status = usnic_fwd_devcmd(ufdev, vnic_idx, CMD_QP_ENABLE,
302 usnic_err("PF %s VNIC Index %u RQ Index: %u ENABLE Failed with status %d",
303 netdev_name(pf_netdev),
308 usnic_dbg("PF %s VNIC Index %u RQ Index: %u ENABLED",
309 netdev_name(pf_netdev),
316 int usnic_fwd_disable_qp(struct usnic_fwd_dev *ufdev, int vnic_idx, int qp_idx)
320 struct net_device *pf_netdev;
322 pf_netdev = ufdev->netdev;
326 status = usnic_fwd_devcmd(ufdev, vnic_idx, CMD_QP_DISABLE,
329 usnic_err("PF %s VNIC Index %u RQ Index: %u DISABLE Failed with status %d",
330 netdev_name(pf_netdev),
335 usnic_dbg("PF %s VNIC Index %u RQ Index: %u DISABLED",
336 netdev_name(pf_netdev),