aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorHiral Patel <hiralpat@cisco.com>2013-02-25 19:18:36 -0500
committerJames Bottomley <JBottomley@Parallels.com>2013-05-02 10:30:40 -0400
commitd3c995f1dcf938f1084388d92b8fb97bec366566 (patch)
tree75c3763c926d5c2f87046c0ab1eee83e3305bdb0
parent73287a43cc79ca06629a88d1a199cd283f42456a (diff)
[SCSI] fnic: FIP VLAN Discovery Feature Support
FIP VLAN discovery discovers the FCoE VLAN that will be used by all other FIP protocols as well as by the FCoE encapsulation for Fibre Channel payloads on the established virtual link. One of the goals of FC-BB-5 was to be as nonintrusive as possible on initiators and targets, and therefore FIP VLAN discovery occurs in the native VLAN used by the initiator or target to exchange Ethernet traffic. The FIP VLAN discovery protocol is the only FIP protocol running on the native VLAN; all other FIP protocols run on the discovered FCoE VLANs. If an administrator has manually configured FCoE VLANs on ENodes and FCFs, there is no need to use this protocol. FIP and FCoE will run over the configured VLANs. An ENode without FCoE VLANs configuration would use this automated discovery protocol to discover over which VLANs FCoE is running. The ENode sends a FIP VLAN discovery request to a multicast MAC address called All-FCF-MACs, which is a multicast MAC address to which all FCFs listen. All FCFs that can be reached in the native VLAN of the ENode are expected to respond on the same VLAN with a response that lists one or more FCoE VLANs that are available for the ENode's VN_Port login. This protocol has the sole purpose of allowing the ENode to discover all the available FCoE VLANs. Now the ENode may enable a subset of these VLANs for FCoE Running the FIP protocol in these VLANs on a per VLAN basis. And FCoE data transactions also would occur on this VLAN. Hence, Except for FIP VLAN discovery, all other FIP and FCoE traffic runs on the selected FCoE VLAN. Its only the FIP VLAN Discovery protocol that is permitted to run on the Default native VLAN of the system. [**** NOTE ****] We are working on moving this feature definitions and functionality to libfcoe module. We need this patch to be approved, as Suse is looking forward to merge this feature in SLES 11 SP3 release. Once this patch is approved, we will submit patch which should move vlan discovery feature to libfoce. [Fengguang Wu <fengguang.wu@intel.com>: kmalloc cast removal] Signed-off-by: Anantha Prakash T <atungara@cisco.com> Signed-off-by: Hiral Patel <hiralpat@cisco.com> Signed-off-by: James Bottomley <JBottomley@Parallels.com>
-rw-r--r--drivers/scsi/fnic/fnic.h32
-rw-r--r--drivers/scsi/fnic/fnic_fcs.c558
-rw-r--r--drivers/scsi/fnic/fnic_fip.h68
-rw-r--r--drivers/scsi/fnic/fnic_main.c51
-rw-r--r--drivers/scsi/fnic/vnic_dev.c10
-rw-r--r--drivers/scsi/fnic/vnic_dev.h2
-rw-r--r--drivers/scsi/fnic/vnic_devcmd.h67
7 files changed, 784 insertions, 4 deletions
diff --git a/drivers/scsi/fnic/fnic.h b/drivers/scsi/fnic/fnic.h
index 98436c363035..acec42a78aef 100644
--- a/drivers/scsi/fnic/fnic.h
+++ b/drivers/scsi/fnic/fnic.h
@@ -192,6 +192,18 @@ enum fnic_state {
192 192
193struct mempool; 193struct mempool;
194 194
195enum fnic_evt {
196 FNIC_EVT_START_VLAN_DISC = 1,
197 FNIC_EVT_START_FCF_DISC = 2,
198 FNIC_EVT_MAX,
199};
200
201struct fnic_event {
202 struct list_head list;
203 struct fnic *fnic;
204 enum fnic_evt event;
205};
206
195/* Per-instance private data structure */ 207/* Per-instance private data structure */
196struct fnic { 208struct fnic {
197 struct fc_lport *lport; 209 struct fc_lport *lport;
@@ -254,6 +266,18 @@ struct fnic {
254 struct sk_buff_head frame_queue; 266 struct sk_buff_head frame_queue;
255 struct sk_buff_head tx_queue; 267 struct sk_buff_head tx_queue;
256 268
269 /*** FIP related data members -- start ***/
270 void (*set_vlan)(struct fnic *, u16 vlan);
271 struct work_struct fip_frame_work;
272 struct sk_buff_head fip_frame_queue;
273 struct timer_list fip_timer;
274 struct list_head vlans;
275 spinlock_t vlans_lock;
276
277 struct work_struct event_work;
278 struct list_head evlist;
279 /*** FIP related data members -- end ***/
280
257 /* copy work queue cache line section */ 281 /* copy work queue cache line section */
258 ____cacheline_aligned struct vnic_wq_copy wq_copy[FNIC_WQ_COPY_MAX]; 282 ____cacheline_aligned struct vnic_wq_copy wq_copy[FNIC_WQ_COPY_MAX];
259 /* completion queue cache line section */ 283 /* completion queue cache line section */
@@ -278,6 +302,7 @@ static inline struct fnic *fnic_from_ctlr(struct fcoe_ctlr *fip)
278} 302}
279 303
280extern struct workqueue_struct *fnic_event_queue; 304extern struct workqueue_struct *fnic_event_queue;
305extern struct workqueue_struct *fnic_fip_queue;
281extern struct device_attribute *fnic_attrs[]; 306extern struct device_attribute *fnic_attrs[];
282 307
283void fnic_clear_intr_mode(struct fnic *fnic); 308void fnic_clear_intr_mode(struct fnic *fnic);
@@ -289,6 +314,7 @@ int fnic_send(struct fc_lport *, struct fc_frame *);
289void fnic_free_wq_buf(struct vnic_wq *wq, struct vnic_wq_buf *buf); 314void fnic_free_wq_buf(struct vnic_wq *wq, struct vnic_wq_buf *buf);
290void fnic_handle_frame(struct work_struct *work); 315void fnic_handle_frame(struct work_struct *work);
291void fnic_handle_link(struct work_struct *work); 316void fnic_handle_link(struct work_struct *work);
317void fnic_handle_event(struct work_struct *work);
292int fnic_rq_cmpl_handler(struct fnic *fnic, int); 318int fnic_rq_cmpl_handler(struct fnic *fnic, int);
293int fnic_alloc_rq_frame(struct vnic_rq *rq); 319int fnic_alloc_rq_frame(struct vnic_rq *rq);
294void fnic_free_rq_buf(struct vnic_rq *rq, struct vnic_rq_buf *buf); 320void fnic_free_rq_buf(struct vnic_rq *rq, struct vnic_rq_buf *buf);
@@ -321,6 +347,12 @@ void fnic_handle_link_event(struct fnic *fnic);
321 347
322int fnic_is_abts_pending(struct fnic *, struct scsi_cmnd *); 348int fnic_is_abts_pending(struct fnic *, struct scsi_cmnd *);
323 349
350void fnic_handle_fip_frame(struct work_struct *work);
351void fnic_handle_fip_event(struct fnic *fnic);
352void fnic_fcoe_reset_vlans(struct fnic *fnic);
353void fnic_fcoe_evlist_free(struct fnic *fnic);
354extern void fnic_handle_fip_timer(struct fnic *fnic);
355
324static inline int 356static inline int
325fnic_chk_state_flags_locked(struct fnic *fnic, unsigned long st_flags) 357fnic_chk_state_flags_locked(struct fnic *fnic, unsigned long st_flags)
326{ 358{
diff --git a/drivers/scsi/fnic/fnic_fcs.c b/drivers/scsi/fnic/fnic_fcs.c
index 483eb9dbe663..34a0b7ddb688 100644
--- a/drivers/scsi/fnic/fnic_fcs.c
+++ b/drivers/scsi/fnic/fnic_fcs.c
@@ -31,12 +31,20 @@
31#include <scsi/libfc.h> 31#include <scsi/libfc.h>
32#include "fnic_io.h" 32#include "fnic_io.h"
33#include "fnic.h" 33#include "fnic.h"
34#include "fnic_fip.h"
34#include "cq_enet_desc.h" 35#include "cq_enet_desc.h"
35#include "cq_exch_desc.h" 36#include "cq_exch_desc.h"
36 37
38static u8 fcoe_all_fcfs[ETH_ALEN];
39struct workqueue_struct *fnic_fip_queue;
37struct workqueue_struct *fnic_event_queue; 40struct workqueue_struct *fnic_event_queue;
38 41
39static void fnic_set_eth_mode(struct fnic *); 42static void fnic_set_eth_mode(struct fnic *);
43static void fnic_fcoe_send_vlan_req(struct fnic *fnic);
44static void fnic_fcoe_start_fcf_disc(struct fnic *fnic);
45static void fnic_fcoe_process_vlan_resp(struct fnic *fnic, struct sk_buff *);
46static int fnic_fcoe_vlan_check(struct fnic *fnic, u16 flag);
47static int fnic_fcoe_handle_fip_frame(struct fnic *fnic, struct sk_buff *skb);
40 48
41void fnic_handle_link(struct work_struct *work) 49void fnic_handle_link(struct work_struct *work)
42{ 50{
@@ -69,6 +77,11 @@ void fnic_handle_link(struct work_struct *work)
69 FNIC_FCS_DBG(KERN_DEBUG, fnic->lport->host, 77 FNIC_FCS_DBG(KERN_DEBUG, fnic->lport->host,
70 "link down\n"); 78 "link down\n");
71 fcoe_ctlr_link_down(&fnic->ctlr); 79 fcoe_ctlr_link_down(&fnic->ctlr);
80 if (fnic->config.flags & VFCF_FIP_CAPABLE) {
81 /* start FCoE VLAN discovery */
82 fnic_fcoe_send_vlan_req(fnic);
83 return;
84 }
72 FNIC_FCS_DBG(KERN_DEBUG, fnic->lport->host, 85 FNIC_FCS_DBG(KERN_DEBUG, fnic->lport->host,
73 "link up\n"); 86 "link up\n");
74 fcoe_ctlr_link_up(&fnic->ctlr); 87 fcoe_ctlr_link_up(&fnic->ctlr);
@@ -79,6 +92,11 @@ void fnic_handle_link(struct work_struct *work)
79 } else if (fnic->link_status) { 92 } else if (fnic->link_status) {
80 /* DOWN -> UP */ 93 /* DOWN -> UP */
81 spin_unlock_irqrestore(&fnic->fnic_lock, flags); 94 spin_unlock_irqrestore(&fnic->fnic_lock, flags);
95 if (fnic->config.flags & VFCF_FIP_CAPABLE) {
96 /* start FCoE VLAN discovery */
97 fnic_fcoe_send_vlan_req(fnic);
98 return;
99 }
82 FNIC_FCS_DBG(KERN_DEBUG, fnic->lport->host, "link up\n"); 100 FNIC_FCS_DBG(KERN_DEBUG, fnic->lport->host, "link up\n");
83 fcoe_ctlr_link_up(&fnic->ctlr); 101 fcoe_ctlr_link_up(&fnic->ctlr);
84 } else { 102 } else {
@@ -128,6 +146,441 @@ void fnic_handle_frame(struct work_struct *work)
128 } 146 }
129} 147}
130 148
149void fnic_fcoe_evlist_free(struct fnic *fnic)
150{
151 struct fnic_event *fevt = NULL;
152 struct fnic_event *next = NULL;
153 unsigned long flags;
154
155 spin_lock_irqsave(&fnic->fnic_lock, flags);
156 if (list_empty(&fnic->evlist)) {
157 spin_unlock_irqrestore(&fnic->fnic_lock, flags);
158 return;
159 }
160
161 list_for_each_entry_safe(fevt, next, &fnic->evlist, list) {
162 list_del(&fevt->list);
163 kfree(fevt);
164 }
165 spin_unlock_irqrestore(&fnic->fnic_lock, flags);
166}
167
168void fnic_handle_event(struct work_struct *work)
169{
170 struct fnic *fnic = container_of(work, struct fnic, event_work);
171 struct fnic_event *fevt = NULL;
172 struct fnic_event *next = NULL;
173 unsigned long flags;
174
175 spin_lock_irqsave(&fnic->fnic_lock, flags);
176 if (list_empty(&fnic->evlist)) {
177 spin_unlock_irqrestore(&fnic->fnic_lock, flags);
178 return;
179 }
180
181 list_for_each_entry_safe(fevt, next, &fnic->evlist, list) {
182 if (fnic->stop_rx_link_events) {
183 list_del(&fevt->list);
184 kfree(fevt);
185 spin_unlock_irqrestore(&fnic->fnic_lock, flags);
186 return;
187 }
188 /*
189 * If we're in a transitional state, just re-queue and return.
190 * The queue will be serviced when we get to a stable state.
191 */
192 if (fnic->state != FNIC_IN_FC_MODE &&
193 fnic->state != FNIC_IN_ETH_MODE) {
194 spin_unlock_irqrestore(&fnic->fnic_lock, flags);
195 return;
196 }
197
198 list_del(&fevt->list);
199 switch (fevt->event) {
200 case FNIC_EVT_START_VLAN_DISC:
201 spin_unlock_irqrestore(&fnic->fnic_lock, flags);
202 fnic_fcoe_send_vlan_req(fnic);
203 spin_lock_irqsave(&fnic->fnic_lock, flags);
204 break;
205 case FNIC_EVT_START_FCF_DISC:
206 FNIC_FCS_DBG(KERN_DEBUG, fnic->lport->host,
207 "Start FCF Discovery\n");
208 fnic_fcoe_start_fcf_disc(fnic);
209 break;
210 default:
211 FNIC_FCS_DBG(KERN_DEBUG, fnic->lport->host,
212 "Unknown event 0x%x\n", fevt->event);
213 break;
214 }
215 kfree(fevt);
216 }
217 spin_unlock_irqrestore(&fnic->fnic_lock, flags);
218}
219
220/**
221 * Check if the Received FIP FLOGI frame is rejected
222 * @fip: The FCoE controller that received the frame
223 * @skb: The received FIP frame
224 *
225 * Returns non-zero if the frame is rejected with unsupported cmd with
226 * insufficient resource els explanation.
227 */
228static inline int is_fnic_fip_flogi_reject(struct fcoe_ctlr *fip,
229 struct sk_buff *skb)
230{
231 struct fc_lport *lport = fip->lp;
232 struct fip_header *fiph;
233 struct fc_frame_header *fh = NULL;
234 struct fip_desc *desc;
235 struct fip_encaps *els;
236 enum fip_desc_type els_dtype = 0;
237 u16 op;
238 u8 els_op;
239 u8 sub;
240
241 size_t els_len = 0;
242 size_t rlen;
243 size_t dlen = 0;
244
245 if (skb_linearize(skb))
246 return 0;
247
248 if (skb->len < sizeof(*fiph))
249 return 0;
250
251 fiph = (struct fip_header *)skb->data;
252 op = ntohs(fiph->fip_op);
253 sub = fiph->fip_subcode;
254
255 if (op != FIP_OP_LS)
256 return 0;
257
258 if (sub != FIP_SC_REP)
259 return 0;
260
261 rlen = ntohs(fiph->fip_dl_len) * 4;
262 if (rlen + sizeof(*fiph) > skb->len)
263 return 0;
264
265 desc = (struct fip_desc *)(fiph + 1);
266 dlen = desc->fip_dlen * FIP_BPW;
267
268 if (desc->fip_dtype == FIP_DT_FLOGI) {
269
270 shost_printk(KERN_DEBUG, lport->host,
271 " FIP TYPE FLOGI: fab name:%llx "
272 "vfid:%d map:%x\n",
273 fip->sel_fcf->fabric_name, fip->sel_fcf->vfid,
274 fip->sel_fcf->fc_map);
275 if (dlen < sizeof(*els) + sizeof(*fh) + 1)
276 return 0;
277
278 els_len = dlen - sizeof(*els);
279 els = (struct fip_encaps *)desc;
280 fh = (struct fc_frame_header *)(els + 1);
281 els_dtype = desc->fip_dtype;
282
283 if (!fh)
284 return 0;
285
286 /*
287 * ELS command code, reason and explanation should be = Reject,
288 * unsupported command and insufficient resource
289 */
290 els_op = *(u8 *)(fh + 1);
291 if (els_op == ELS_LS_RJT) {
292 shost_printk(KERN_INFO, lport->host,
293 "Flogi Request Rejected by Switch\n");
294 return 1;
295 }
296 shost_printk(KERN_INFO, lport->host,
297 "Flogi Request Accepted by Switch\n");
298 }
299 return 0;
300}
301
302static void fnic_fcoe_send_vlan_req(struct fnic *fnic)
303{
304 struct fcoe_ctlr *fip = &fnic->ctlr;
305 struct sk_buff *skb;
306 char *eth_fr;
307 int fr_len;
308 struct fip_vlan *vlan;
309 u64 vlan_tov;
310
311 fnic_fcoe_reset_vlans(fnic);
312 fnic->set_vlan(fnic, 0);
313 FNIC_FCS_DBG(KERN_INFO, fnic->lport->host,
314 "Sending VLAN request...\n");
315 skb = dev_alloc_skb(sizeof(struct fip_vlan));
316 if (!skb)
317 return;
318
319 fr_len = sizeof(*vlan);
320 eth_fr = (char *)skb->data;
321 vlan = (struct fip_vlan *)eth_fr;
322
323 memset(vlan, 0, sizeof(*vlan));
324 memcpy(vlan->eth.h_source, fip->ctl_src_addr, ETH_ALEN);
325 memcpy(vlan->eth.h_dest, fcoe_all_fcfs, ETH_ALEN);
326 vlan->eth.h_proto = htons(ETH_P_FIP);
327
328 vlan->fip.fip_ver = FIP_VER_ENCAPS(FIP_VER);
329 vlan->fip.fip_op = htons(FIP_OP_VLAN);
330 vlan->fip.fip_subcode = FIP_SC_VL_REQ;
331 vlan->fip.fip_dl_len = htons(sizeof(vlan->desc) / FIP_BPW);
332
333 vlan->desc.mac.fd_desc.fip_dtype = FIP_DT_MAC;
334 vlan->desc.mac.fd_desc.fip_dlen = sizeof(vlan->desc.mac) / FIP_BPW;
335 memcpy(&vlan->desc.mac.fd_mac, fip->ctl_src_addr, ETH_ALEN);
336
337 vlan->desc.wwnn.fd_desc.fip_dtype = FIP_DT_NAME;
338 vlan->desc.wwnn.fd_desc.fip_dlen = sizeof(vlan->desc.wwnn) / FIP_BPW;
339 put_unaligned_be64(fip->lp->wwnn, &vlan->desc.wwnn.fd_wwn);
340
341 skb_put(skb, sizeof(*vlan));
342 skb->protocol = htons(ETH_P_FIP);
343 skb_reset_mac_header(skb);
344 skb_reset_network_header(skb);
345 fip->send(fip, skb);
346
347 /* set a timer so that we can retry if there no response */
348 vlan_tov = jiffies + msecs_to_jiffies(FCOE_CTLR_FIPVLAN_TOV);
349 mod_timer(&fnic->fip_timer, round_jiffies(vlan_tov));
350}
351
352static void fnic_fcoe_process_vlan_resp(struct fnic *fnic, struct sk_buff *skb)
353{
354 struct fcoe_ctlr *fip = &fnic->ctlr;
355 struct fip_header *fiph;
356 struct fip_desc *desc;
357 u16 vid;
358 size_t rlen;
359 size_t dlen;
360 struct fcoe_vlan *vlan;
361 u64 sol_time;
362 unsigned long flags;
363
364 FNIC_FCS_DBG(KERN_INFO, fnic->lport->host,
365 "Received VLAN response...\n");
366
367 fiph = (struct fip_header *) skb->data;
368
369 FNIC_FCS_DBG(KERN_INFO, fnic->lport->host,
370 "Received VLAN response... OP 0x%x SUB_OP 0x%x\n",
371 ntohs(fiph->fip_op), fiph->fip_subcode);
372
373 rlen = ntohs(fiph->fip_dl_len) * 4;
374 fnic_fcoe_reset_vlans(fnic);
375 spin_lock_irqsave(&fnic->vlans_lock, flags);
376 desc = (struct fip_desc *)(fiph + 1);
377 while (rlen > 0) {
378 dlen = desc->fip_dlen * FIP_BPW;
379 switch (desc->fip_dtype) {
380 case FIP_DT_VLAN:
381 vid = ntohs(((struct fip_vlan_desc *)desc)->fd_vlan);
382 shost_printk(KERN_INFO, fnic->lport->host,
383 "process_vlan_resp: FIP VLAN %d\n", vid);
384 vlan = kmalloc(sizeof(*vlan),
385 GFP_ATOMIC);
386 if (!vlan) {
387 /* retry from timer */
388 spin_unlock_irqrestore(&fnic->vlans_lock,
389 flags);
390 goto out;
391 }
392 memset(vlan, 0, sizeof(struct fcoe_vlan));
393 vlan->vid = vid & 0x0fff;
394 vlan->state = FIP_VLAN_AVAIL;
395 list_add_tail(&vlan->list, &fnic->vlans);
396 break;
397 }
398 desc = (struct fip_desc *)((char *)desc + dlen);
399 rlen -= dlen;
400 }
401
402 /* any VLAN descriptors present ? */
403 if (list_empty(&fnic->vlans)) {
404 /* retry from timer */
405 FNIC_FCS_DBG(KERN_INFO, fnic->lport->host,
406 "No VLAN descriptors in FIP VLAN response\n");
407 spin_unlock_irqrestore(&fnic->vlans_lock, flags);
408 goto out;
409 }
410
411 vlan = list_first_entry(&fnic->vlans, struct fcoe_vlan, list);
412 fnic->set_vlan(fnic, vlan->vid);
413 vlan->state = FIP_VLAN_SENT; /* sent now */
414 vlan->sol_count++;
415 spin_unlock_irqrestore(&fnic->vlans_lock, flags);
416
417 /* start the solicitation */
418 fcoe_ctlr_link_up(fip);
419
420 sol_time = jiffies + msecs_to_jiffies(FCOE_CTLR_START_DELAY);
421 mod_timer(&fnic->fip_timer, round_jiffies(sol_time));
422out:
423 return;
424}
425
426static void fnic_fcoe_start_fcf_disc(struct fnic *fnic)
427{
428 unsigned long flags;
429 struct fcoe_vlan *vlan;
430 u64 sol_time;
431
432 spin_lock_irqsave(&fnic->vlans_lock, flags);
433 vlan = list_first_entry(&fnic->vlans, struct fcoe_vlan, list);
434 fnic->set_vlan(fnic, vlan->vid);
435 vlan->state = FIP_VLAN_SENT; /* sent now */
436 vlan->sol_count = 1;
437 spin_unlock_irqrestore(&fnic->vlans_lock, flags);
438
439 /* start the solicitation */
440 fcoe_ctlr_link_up(&fnic->ctlr);
441
442 sol_time = jiffies + msecs_to_jiffies(FCOE_CTLR_START_DELAY);
443 mod_timer(&fnic->fip_timer, round_jiffies(sol_time));
444}
445
446static int fnic_fcoe_vlan_check(struct fnic *fnic, u16 flag)
447{
448 unsigned long flags;
449 struct fcoe_vlan *fvlan;
450
451 spin_lock_irqsave(&fnic->vlans_lock, flags);
452 if (list_empty(&fnic->vlans)) {
453 spin_unlock_irqrestore(&fnic->vlans_lock, flags);
454 return -EINVAL;
455 }
456
457 fvlan = list_first_entry(&fnic->vlans, struct fcoe_vlan, list);
458 if (fvlan->state == FIP_VLAN_USED) {
459 spin_unlock_irqrestore(&fnic->vlans_lock, flags);
460 return 0;
461 }
462
463 if (fvlan->state == FIP_VLAN_SENT) {
464 fvlan->state = FIP_VLAN_USED;
465 spin_unlock_irqrestore(&fnic->vlans_lock, flags);
466 return 0;
467 }
468 spin_unlock_irqrestore(&fnic->vlans_lock, flags);
469 return -EINVAL;
470}
471
472static void fnic_event_enq(struct fnic *fnic, enum fnic_evt ev)
473{
474 struct fnic_event *fevt;
475 unsigned long flags;
476
477 fevt = kmalloc(sizeof(*fevt), GFP_ATOMIC);
478 if (!fevt)
479 return;
480
481 fevt->fnic = fnic;
482 fevt->event = ev;
483
484 spin_lock_irqsave(&fnic->fnic_lock, flags);
485 list_add_tail(&fevt->list, &fnic->evlist);
486 spin_unlock_irqrestore(&fnic->fnic_lock, flags);
487
488 schedule_work(&fnic->event_work);
489}
490
491static int fnic_fcoe_handle_fip_frame(struct fnic *fnic, struct sk_buff *skb)
492{
493 struct fip_header *fiph;
494 int ret = 1;
495 u16 op;
496 u8 sub;
497
498 if (!skb || !(skb->data))
499 return -1;
500
501 if (skb_linearize(skb))
502 goto drop;
503
504 fiph = (struct fip_header *)skb->data;
505 op = ntohs(fiph->fip_op);
506 sub = fiph->fip_subcode;
507
508 if (FIP_VER_DECAPS(fiph->fip_ver) != FIP_VER)
509 goto drop;
510
511 if (ntohs(fiph->fip_dl_len) * FIP_BPW + sizeof(*fiph) > skb->len)
512 goto drop;
513
514 if (op == FIP_OP_DISC && sub == FIP_SC_ADV) {
515 if (fnic_fcoe_vlan_check(fnic, ntohs(fiph->fip_flags)))
516 goto drop;
517 /* pass it on to fcoe */
518 ret = 1;
519 } else if (op == FIP_OP_VLAN && sub == FIP_SC_VL_REP) {
520 /* set the vlan as used */
521 fnic_fcoe_process_vlan_resp(fnic, skb);
522 ret = 0;
523 } else if (op == FIP_OP_CTRL && sub == FIP_SC_CLR_VLINK) {
524 /* received CVL request, restart vlan disc */
525 fnic_event_enq(fnic, FNIC_EVT_START_VLAN_DISC);
526 /* pass it on to fcoe */
527 ret = 1;
528 }
529drop:
530 return ret;
531}
532
533void fnic_handle_fip_frame(struct work_struct *work)
534{
535 struct fnic *fnic = container_of(work, struct fnic, fip_frame_work);
536 unsigned long flags;
537 struct sk_buff *skb;
538 struct ethhdr *eh;
539
540 while ((skb = skb_dequeue(&fnic->fip_frame_queue))) {
541 spin_lock_irqsave(&fnic->fnic_lock, flags);
542 if (fnic->stop_rx_link_events) {
543 spin_unlock_irqrestore(&fnic->fnic_lock, flags);
544 dev_kfree_skb(skb);
545 return;
546 }
547 /*
548 * If we're in a transitional state, just re-queue and return.
549 * The queue will be serviced when we get to a stable state.
550 */
551 if (fnic->state != FNIC_IN_FC_MODE &&
552 fnic->state != FNIC_IN_ETH_MODE) {
553 skb_queue_head(&fnic->fip_frame_queue, skb);
554 spin_unlock_irqrestore(&fnic->fnic_lock, flags);
555 return;
556 }
557 spin_unlock_irqrestore(&fnic->fnic_lock, flags);
558 eh = (struct ethhdr *)skb->data;
559 if (eh->h_proto == htons(ETH_P_FIP)) {
560 skb_pull(skb, sizeof(*eh));
561 if (fnic_fcoe_handle_fip_frame(fnic, skb) <= 0) {
562 dev_kfree_skb(skb);
563 continue;
564 }
565 /*
566 * If there's FLOGI rejects - clear all
567 * fcf's & restart from scratch
568 */
569 if (is_fnic_fip_flogi_reject(&fnic->ctlr, skb)) {
570 shost_printk(KERN_INFO, fnic->lport->host,
571 "Trigger a Link down - VLAN Disc\n");
572 fcoe_ctlr_link_down(&fnic->ctlr);
573 /* start FCoE VLAN discovery */
574 fnic_fcoe_send_vlan_req(fnic);
575 dev_kfree_skb(skb);
576 continue;
577 }
578 fcoe_ctlr_recv(&fnic->ctlr, skb);
579 continue;
580 }
581 }
582}
583
131/** 584/**
132 * fnic_import_rq_eth_pkt() - handle received FCoE or FIP frame. 585 * fnic_import_rq_eth_pkt() - handle received FCoE or FIP frame.
133 * @fnic: fnic instance. 586 * @fnic: fnic instance.
@@ -150,8 +603,8 @@ static inline int fnic_import_rq_eth_pkt(struct fnic *fnic, struct sk_buff *skb)
150 skb_reset_mac_header(skb); 603 skb_reset_mac_header(skb);
151 } 604 }
152 if (eh->h_proto == htons(ETH_P_FIP)) { 605 if (eh->h_proto == htons(ETH_P_FIP)) {
153 skb_pull(skb, sizeof(*eh)); 606 skb_queue_tail(&fnic->fip_frame_queue, skb);
154 fcoe_ctlr_recv(&fnic->ctlr, skb); 607 queue_work(fnic_fip_queue, &fnic->fip_frame_work);
155 return 1; /* let caller know packet was used */ 608 return 1; /* let caller know packet was used */
156 } 609 }
157 if (eh->h_proto != htons(ETH_P_FCOE)) 610 if (eh->h_proto != htons(ETH_P_FCOE))
@@ -720,3 +1173,104 @@ void fnic_free_wq_buf(struct vnic_wq *wq, struct vnic_wq_buf *buf)
720 dev_kfree_skb(fp_skb(fp)); 1173 dev_kfree_skb(fp_skb(fp));
721 buf->os_buf = NULL; 1174 buf->os_buf = NULL;
722} 1175}
1176
1177void fnic_fcoe_reset_vlans(struct fnic *fnic)
1178{
1179 unsigned long flags;
1180 struct fcoe_vlan *vlan;
1181 struct fcoe_vlan *next;
1182
1183 /*
1184 * indicate a link down to fcoe so that all fcf's are free'd
1185 * might not be required since we did this before sending vlan
1186 * discovery request
1187 */
1188 spin_lock_irqsave(&fnic->vlans_lock, flags);
1189 if (!list_empty(&fnic->vlans)) {
1190 list_for_each_entry_safe(vlan, next, &fnic->vlans, list) {
1191 list_del(&vlan->list);
1192 kfree(vlan);
1193 }
1194 }
1195 spin_unlock_irqrestore(&fnic->vlans_lock, flags);
1196}
1197
1198void fnic_handle_fip_timer(struct fnic *fnic)
1199{
1200 unsigned long flags;
1201 struct fcoe_vlan *vlan;
1202 u64 sol_time;
1203
1204 spin_lock_irqsave(&fnic->fnic_lock, flags);
1205 if (fnic->stop_rx_link_events) {
1206 spin_unlock_irqrestore(&fnic->fnic_lock, flags);
1207 return;
1208 }
1209 spin_unlock_irqrestore(&fnic->fnic_lock, flags);
1210
1211 if (fnic->ctlr.mode == FIP_ST_NON_FIP)
1212 return;
1213
1214 spin_lock_irqsave(&fnic->vlans_lock, flags);
1215 if (list_empty(&fnic->vlans)) {
1216 /* no vlans available, try again */
1217 FNIC_FCS_DBG(KERN_DEBUG, fnic->lport->host,
1218 "Start VLAN Discovery\n");
1219 spin_unlock_irqrestore(&fnic->vlans_lock, flags);
1220 fnic_event_enq(fnic, FNIC_EVT_START_VLAN_DISC);
1221 return;
1222 }
1223
1224 vlan = list_first_entry(&fnic->vlans, struct fcoe_vlan, list);
1225 shost_printk(KERN_DEBUG, fnic->lport->host,
1226 "fip_timer: vlan %d state %d sol_count %d\n",
1227 vlan->vid, vlan->state, vlan->sol_count);
1228 switch (vlan->state) {
1229 case FIP_VLAN_USED:
1230 FNIC_FCS_DBG(KERN_DEBUG, fnic->lport->host,
1231 "FIP VLAN is selected for FC transaction\n");
1232 spin_unlock_irqrestore(&fnic->vlans_lock, flags);
1233 break;
1234 case FIP_VLAN_FAILED:
1235 /* if all vlans are in failed state, restart vlan disc */
1236 FNIC_FCS_DBG(KERN_DEBUG, fnic->lport->host,
1237 "Start VLAN Discovery\n");
1238 spin_unlock_irqrestore(&fnic->vlans_lock, flags);
1239 fnic_event_enq(fnic, FNIC_EVT_START_VLAN_DISC);
1240 break;
1241 case FIP_VLAN_SENT:
1242 if (vlan->sol_count >= FCOE_CTLR_MAX_SOL) {
1243 /*
1244 * no response on this vlan, remove from the list.
1245 * Try the next vlan
1246 */
1247 shost_printk(KERN_INFO, fnic->lport->host,
1248 "Dequeue this VLAN ID %d from list\n",
1249 vlan->vid);
1250 list_del(&vlan->list);
1251 kfree(vlan);
1252 vlan = NULL;
1253 if (list_empty(&fnic->vlans)) {
1254 /* we exhausted all vlans, restart vlan disc */
1255 spin_unlock_irqrestore(&fnic->vlans_lock,
1256 flags);
1257 shost_printk(KERN_INFO, fnic->lport->host,
1258 "fip_timer: vlan list empty, "
1259 "trigger vlan disc\n");
1260 fnic_event_enq(fnic, FNIC_EVT_START_VLAN_DISC);
1261 return;
1262 }
1263 /* check the next vlan */
1264 vlan = list_first_entry(&fnic->vlans, struct fcoe_vlan,
1265 list);
1266 fnic->set_vlan(fnic, vlan->vid);
1267 vlan->state = FIP_VLAN_SENT; /* sent now */
1268 }
1269 spin_unlock_irqrestore(&fnic->vlans_lock, flags);
1270 vlan->sol_count++;
1271 sol_time = jiffies + msecs_to_jiffies
1272 (FCOE_CTLR_START_DELAY);
1273 mod_timer(&fnic->fip_timer, round_jiffies(sol_time));
1274 break;
1275 }
1276}
diff --git a/drivers/scsi/fnic/fnic_fip.h b/drivers/scsi/fnic/fnic_fip.h
new file mode 100644
index 000000000000..87e74c2ab971
--- /dev/null
+++ b/drivers/scsi/fnic/fnic_fip.h
@@ -0,0 +1,68 @@
1/*
2 * Copyright 2008 Cisco Systems, Inc. All rights reserved.
3 * Copyright 2007 Nuova Systems, Inc. All rights reserved.
4 *
5 * This program is free software; you may redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; version 2 of the License.
8 *
9 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
10 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
11 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
12 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
13 * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
14 * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
15 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
16 * SOFTWARE.
17 */
18
19#ifndef _FNIC_FIP_H_
20#define _FNIC_FIP_H_
21
22
23#define FCOE_CTLR_START_DELAY 2000 /* ms after first adv. to choose FCF */
24#define FCOE_CTLR_FIPVLAN_TOV 2000 /* ms after FIP VLAN disc */
25#define FCOE_CTLR_MAX_SOL 8
26
27#define FINC_MAX_FLOGI_REJECTS 8
28
29/*
30 * FIP_DT_VLAN descriptor.
31 */
32struct fip_vlan_desc {
33 struct fip_desc fd_desc;
34 __be16 fd_vlan;
35} __attribute__((packed));
36
37struct vlan {
38 __be16 vid;
39 __be16 type;
40};
41
42/*
43 * VLAN entry.
44 */
45struct fcoe_vlan {
46 struct list_head list;
47 u16 vid; /* vlan ID */
48 u16 sol_count; /* no. of sols sent */
49 u16 state; /* state */
50};
51
52enum fip_vlan_state {
53 FIP_VLAN_AVAIL = 0, /* don't do anything */
54 FIP_VLAN_SENT = 1, /* sent */
55 FIP_VLAN_USED = 2, /* succeed */
56 FIP_VLAN_FAILED = 3, /* failed to response */
57};
58
59struct fip_vlan {
60 struct ethhdr eth;
61 struct fip_header fip;
62 struct {
63 struct fip_mac_desc mac;
64 struct fip_wwn_desc wwnn;
65 } desc;
66};
67
68#endif /* __FINC_FIP_H_ */
diff --git a/drivers/scsi/fnic/fnic_main.c b/drivers/scsi/fnic/fnic_main.c
index d601ac543c52..5f09d1814d26 100644
--- a/drivers/scsi/fnic/fnic_main.c
+++ b/drivers/scsi/fnic/fnic_main.c
@@ -39,6 +39,7 @@
39#include "vnic_intr.h" 39#include "vnic_intr.h"
40#include "vnic_stats.h" 40#include "vnic_stats.h"
41#include "fnic_io.h" 41#include "fnic_io.h"
42#include "fnic_fip.h"
42#include "fnic.h" 43#include "fnic.h"
43 44
44#define PCI_DEVICE_ID_CISCO_FNIC 0x0045 45#define PCI_DEVICE_ID_CISCO_FNIC 0x0045
@@ -292,6 +293,13 @@ static void fnic_notify_timer(unsigned long data)
292 round_jiffies(jiffies + FNIC_NOTIFY_TIMER_PERIOD)); 293 round_jiffies(jiffies + FNIC_NOTIFY_TIMER_PERIOD));
293} 294}
294 295
296static void fnic_fip_notify_timer(unsigned long data)
297{
298 struct fnic *fnic = (struct fnic *)data;
299
300 fnic_handle_fip_timer(fnic);
301}
302
295static void fnic_notify_timer_start(struct fnic *fnic) 303static void fnic_notify_timer_start(struct fnic *fnic)
296{ 304{
297 switch (vnic_dev_get_intr_mode(fnic->vdev)) { 305 switch (vnic_dev_get_intr_mode(fnic->vdev)) {
@@ -403,6 +411,12 @@ static u8 *fnic_get_mac(struct fc_lport *lport)
403 return fnic->data_src_addr; 411 return fnic->data_src_addr;
404} 412}
405 413
414static void fnic_set_vlan(struct fnic *fnic, u16 vlan_id)
415{
416 u16 old_vlan;
417 old_vlan = vnic_dev_set_default_vlan(fnic->vdev, vlan_id);
418}
419
406static int fnic_probe(struct pci_dev *pdev, const struct pci_device_id *ent) 420static int fnic_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
407{ 421{
408 struct Scsi_Host *host; 422 struct Scsi_Host *host;
@@ -620,7 +634,29 @@ static int fnic_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
620 vnic_dev_packet_filter(fnic->vdev, 1, 1, 0, 0, 0); 634 vnic_dev_packet_filter(fnic->vdev, 1, 1, 0, 0, 0);
621 vnic_dev_add_addr(fnic->vdev, FIP_ALL_ENODE_MACS); 635 vnic_dev_add_addr(fnic->vdev, FIP_ALL_ENODE_MACS);
622 vnic_dev_add_addr(fnic->vdev, fnic->ctlr.ctl_src_addr); 636 vnic_dev_add_addr(fnic->vdev, fnic->ctlr.ctl_src_addr);
637 fnic->set_vlan = fnic_set_vlan;
623 fcoe_ctlr_init(&fnic->ctlr, FIP_MODE_AUTO); 638 fcoe_ctlr_init(&fnic->ctlr, FIP_MODE_AUTO);
639 setup_timer(&fnic->fip_timer, fnic_fip_notify_timer,
640 (unsigned long)fnic);
641 spin_lock_init(&fnic->vlans_lock);
642 INIT_WORK(&fnic->fip_frame_work, fnic_handle_fip_frame);
643 INIT_WORK(&fnic->event_work, fnic_handle_event);
644 skb_queue_head_init(&fnic->fip_frame_queue);
645 spin_lock_irqsave(&fnic_list_lock, flags);
646 if (!fnic_fip_queue) {
647 fnic_fip_queue =
648 create_singlethread_workqueue("fnic_fip_q");
649 if (!fnic_fip_queue) {
650 spin_unlock_irqrestore(&fnic_list_lock, flags);
651 printk(KERN_ERR PFX "fnic FIP work queue "
652 "create failed\n");
653 err = -ENOMEM;
654 goto err_out_free_max_pool;
655 }
656 }
657 spin_unlock_irqrestore(&fnic_list_lock, flags);
658 INIT_LIST_HEAD(&fnic->evlist);
659 INIT_LIST_HEAD(&fnic->vlans);
624 } else { 660 } else {
625 shost_printk(KERN_INFO, fnic->lport->host, 661 shost_printk(KERN_INFO, fnic->lport->host,
626 "firmware uses non-FIP mode\n"); 662 "firmware uses non-FIP mode\n");
@@ -807,6 +843,13 @@ static void fnic_remove(struct pci_dev *pdev)
807 skb_queue_purge(&fnic->frame_queue); 843 skb_queue_purge(&fnic->frame_queue);
808 skb_queue_purge(&fnic->tx_queue); 844 skb_queue_purge(&fnic->tx_queue);
809 845
846 if (fnic->config.flags & VFCF_FIP_CAPABLE) {
847 del_timer_sync(&fnic->fip_timer);
848 skb_queue_purge(&fnic->fip_frame_queue);
849 fnic_fcoe_reset_vlans(fnic);
850 fnic_fcoe_evlist_free(fnic);
851 }
852
810 /* 853 /*
811 * Log off the fabric. This stops all remote ports, dns port, 854 * Log off the fabric. This stops all remote ports, dns port,
812 * logs off the fabric. This flushes all rport, disc, lport work 855 * logs off the fabric. This flushes all rport, disc, lport work
@@ -889,8 +932,8 @@ static int __init fnic_init_module(void)
889 len = sizeof(struct fnic_sgl_list); 932 len = sizeof(struct fnic_sgl_list);
890 fnic_sgl_cache[FNIC_SGL_CACHE_MAX] = kmem_cache_create 933 fnic_sgl_cache[FNIC_SGL_CACHE_MAX] = kmem_cache_create
891 ("fnic_sgl_max", len + FNIC_SG_DESC_ALIGN, FNIC_SG_DESC_ALIGN, 934 ("fnic_sgl_max", len + FNIC_SG_DESC_ALIGN, FNIC_SG_DESC_ALIGN,
892 SLAB_HWCACHE_ALIGN, 935 SLAB_HWCACHE_ALIGN,
893 NULL); 936 NULL);
894 if (!fnic_sgl_cache[FNIC_SGL_CACHE_MAX]) { 937 if (!fnic_sgl_cache[FNIC_SGL_CACHE_MAX]) {
895 printk(KERN_ERR PFX "failed to create fnic max sgl slab\n"); 938 printk(KERN_ERR PFX "failed to create fnic max sgl slab\n");
896 err = -ENOMEM; 939 err = -ENOMEM;
@@ -951,6 +994,10 @@ static void __exit fnic_cleanup_module(void)
951{ 994{
952 pci_unregister_driver(&fnic_driver); 995 pci_unregister_driver(&fnic_driver);
953 destroy_workqueue(fnic_event_queue); 996 destroy_workqueue(fnic_event_queue);
997 if (fnic_fip_queue) {
998 flush_workqueue(fnic_fip_queue);
999 destroy_workqueue(fnic_fip_queue);
1000 }
954 kmem_cache_destroy(fnic_sgl_cache[FNIC_SGL_CACHE_MAX]); 1001 kmem_cache_destroy(fnic_sgl_cache[FNIC_SGL_CACHE_MAX]);
955 kmem_cache_destroy(fnic_sgl_cache[FNIC_SGL_CACHE_DFLT]); 1002 kmem_cache_destroy(fnic_sgl_cache[FNIC_SGL_CACHE_DFLT]);
956 kmem_cache_destroy(fnic_io_req_cache); 1003 kmem_cache_destroy(fnic_io_req_cache);
diff --git a/drivers/scsi/fnic/vnic_dev.c b/drivers/scsi/fnic/vnic_dev.c
index b576be734e2e..9795d6f3e197 100644
--- a/drivers/scsi/fnic/vnic_dev.c
+++ b/drivers/scsi/fnic/vnic_dev.c
@@ -584,6 +584,16 @@ int vnic_dev_init(struct vnic_dev *vdev, int arg)
584 return vnic_dev_cmd(vdev, CMD_INIT, &a0, &a1, wait); 584 return vnic_dev_cmd(vdev, CMD_INIT, &a0, &a1, wait);
585} 585}
586 586
587u16 vnic_dev_set_default_vlan(struct vnic_dev *vdev, u16 new_default_vlan)
588{
589 u64 a0 = new_default_vlan, a1 = 0;
590 int wait = 1000;
591 int old_vlan = 0;
592
593 old_vlan = vnic_dev_cmd(vdev, CMD_SET_DEFAULT_VLAN, &a0, &a1, wait);
594 return (u16)old_vlan;
595}
596
587int vnic_dev_link_status(struct vnic_dev *vdev) 597int vnic_dev_link_status(struct vnic_dev *vdev)
588{ 598{
589 if (vdev->linkstatus) 599 if (vdev->linkstatus)
diff --git a/drivers/scsi/fnic/vnic_dev.h b/drivers/scsi/fnic/vnic_dev.h
index f9935a8a5a09..40d4195f562b 100644
--- a/drivers/scsi/fnic/vnic_dev.h
+++ b/drivers/scsi/fnic/vnic_dev.h
@@ -148,6 +148,8 @@ int vnic_dev_disable(struct vnic_dev *vdev);
148int vnic_dev_open(struct vnic_dev *vdev, int arg); 148int vnic_dev_open(struct vnic_dev *vdev, int arg);
149int vnic_dev_open_done(struct vnic_dev *vdev, int *done); 149int vnic_dev_open_done(struct vnic_dev *vdev, int *done);
150int vnic_dev_init(struct vnic_dev *vdev, int arg); 150int vnic_dev_init(struct vnic_dev *vdev, int arg);
151u16 vnic_dev_set_default_vlan(struct vnic_dev *vdev,
152 u16 new_default_vlan);
151int vnic_dev_soft_reset(struct vnic_dev *vdev, int arg); 153int vnic_dev_soft_reset(struct vnic_dev *vdev, int arg);
152int vnic_dev_soft_reset_done(struct vnic_dev *vdev, int *done); 154int vnic_dev_soft_reset_done(struct vnic_dev *vdev, int *done);
153void vnic_dev_set_intr_mode(struct vnic_dev *vdev, 155void vnic_dev_set_intr_mode(struct vnic_dev *vdev,
diff --git a/drivers/scsi/fnic/vnic_devcmd.h b/drivers/scsi/fnic/vnic_devcmd.h
index 7c9ccbd4134b..3e2fcbda6aed 100644
--- a/drivers/scsi/fnic/vnic_devcmd.h
+++ b/drivers/scsi/fnic/vnic_devcmd.h
@@ -196,6 +196,73 @@ enum vnic_devcmd_cmd {
196 196
197 /* undo initialize of virtual link */ 197 /* undo initialize of virtual link */
198 CMD_DEINIT = _CMDCNW(_CMD_DIR_NONE, _CMD_VTYPE_ALL, 34), 198 CMD_DEINIT = _CMDCNW(_CMD_DIR_NONE, _CMD_VTYPE_ALL, 34),
199
200 /* check fw capability of a cmd:
201 * in: (u32)a0=cmd
202 * out: (u32)a0=errno, 0:valid cmd, a1=supported VNIC_STF_* bits */
203 CMD_CAPABILITY = _CMDC(_CMD_DIR_RW, _CMD_VTYPE_ALL, 36),
204
205 /* persistent binding info
206 * in: (u64)a0=paddr of arg
207 * (u32)a1=CMD_PERBI_XXX */
208 CMD_PERBI = _CMDC(_CMD_DIR_RW, _CMD_VTYPE_FC, 37),
209
210 /* Interrupt Assert Register functionality
211 * in: (u16)a0=interrupt number to assert
212 */
213 CMD_IAR = _CMDCNW(_CMD_DIR_WRITE, _CMD_VTYPE_ALL, 38),
214
215 /* initiate hangreset, like softreset after hang detected */
216 CMD_HANG_RESET = _CMDC(_CMD_DIR_NONE, _CMD_VTYPE_ALL, 39),
217
218 /* hangreset status:
219 * out: a0=0 reset complete, a0=1 reset in progress */
220 CMD_HANG_RESET_STATUS = _CMDC(_CMD_DIR_READ, _CMD_VTYPE_ALL, 40),
221
222 /*
223 * Set hw ingress packet vlan rewrite mode:
224 * in: (u32)a0=new vlan rewrite mode
225 * out: (u32)a0=old vlan rewrite mode */
226 CMD_IG_VLAN_REWRITE_MODE = _CMDC(_CMD_DIR_RW, _CMD_VTYPE_ENET, 41),
227
228 /*
229 * in: (u16)a0=bdf of target vnic
230 * (u32)a1=cmd to proxy
231 * a2-a15=args to cmd in a1
232 * out: (u32)a0=status of proxied cmd
233 * a1-a15=out args of proxied cmd */
234 CMD_PROXY_BY_BDF = _CMDC(_CMD_DIR_RW, _CMD_VTYPE_ALL, 42),
235
236 /*
237 * As for BY_BDF except a0 is index of hvnlink subordinate vnic
238 * or SR-IOV virtual vnic
239 */
240 CMD_PROXY_BY_INDEX = _CMDC(_CMD_DIR_RW, _CMD_VTYPE_ALL, 43),
241
242 /*
243 * For HPP toggle:
244 * adapter-info-get
245 * in: (u64)a0=phsical address of buffer passed in from caller.
246 * (u16)a1=size of buffer specified in a0.
247 * out: (u64)a0=phsical address of buffer passed in from caller.
248 * (u16)a1=actual bytes from VIF-CONFIG-INFO TLV, or
249 * 0 if no VIF-CONFIG-INFO TLV was ever received. */
250 CMD_CONFIG_INFO_GET = _CMDC(_CMD_DIR_RW, _CMD_VTYPE_ALL, 44),
251
252 /*
253 * INT13 API: (u64)a0=paddr to vnic_int13_params struct
254 * (u32)a1=INT13_CMD_xxx
255 */
256 CMD_INT13_ALL = _CMDC(_CMD_DIR_WRITE, _CMD_VTYPE_ALL, 45),
257
258 /*
259 * Set default vlan:
260 * in: (u16)a0=new default vlan
261 * (u16)a1=zero for overriding vlan with param a0,
262 * non-zero for resetting vlan to the default
263 * out: (u16)a0=old default vlan
264 */
265 CMD_SET_DEFAULT_VLAN = _CMDC(_CMD_DIR_RW, _CMD_VTYPE_ALL, 46)
199}; 266};
200 267
201/* flags for CMD_OPEN */ 268/* flags for CMD_OPEN */