scsi: qedf: Use granted MAC from the FCF for the FCoE source address if it is available.
authorChad Dupuis <chad.dupuis@cavium.com>
Tue, 15 Aug 2017 17:08:17 +0000 (10:08 -0700)
committerMartin K. Petersen <martin.petersen@oracle.com>
Fri, 25 Aug 2017 02:29:02 +0000 (22:29 -0400)
Currently in the driver we've been using the fc_fcoe_set_mac() function to
set the source MAC for FCoE traffic.  This works well in most cases as it
uses the spec. default FCF-MAC.  However, if the administrator changes the
FCF-MAC switch, then any FCoE traffic we send will be dropped by the
switch.

Instead we should check the granted MAC from the FLOGI payload and use that
address if it is present.  Otherwise, fall back to using the the default
FCF-MAC and the fabric ID of the port as the FCoE MAC address.

Once this address is known we need to set it when doing non-offload
traffic, offload traffic and setting the data_src_address libfcoe uses for
FIP keep alive messages.

Signed-off-by: Chad Dupuis <chad.dupuis@cavium.com>
Signed-off-by: Martin K. Petersen <martin.petersen@oracle.com>
drivers/scsi/qedf/qedf.h
drivers/scsi/qedf/qedf_fip.c
drivers/scsi/qedf/qedf_main.c

index 4d038926a4558c45b8685ca08c4976233a7ed5e7..26ff212eda7e5c12a4ee83e92eac52380a954252 100644 (file)
@@ -443,7 +443,6 @@ extern void qedf_cmd_mgr_free(struct qedf_cmd_mgr *cmgr);
 extern int qedf_queuecommand(struct Scsi_Host *host,
        struct scsi_cmnd *sc_cmd);
 extern void qedf_fip_send(struct fcoe_ctlr *fip, struct sk_buff *skb);
-extern void qedf_update_src_mac(struct fc_lport *lport, u8 *addr);
 extern u8 *qedf_get_src_mac(struct fc_lport *lport);
 extern void qedf_fip_recv(struct qedf_ctx *qedf, struct sk_buff *skb);
 extern void qedf_fcoe_send_vlan_req(struct qedf_ctx *qedf);
index aefd24ca960423f3fed540e58da894b7c94f3d54..28ce0f7ffc7b15fefc7da31a418bdbd7597e2d31 100644 (file)
@@ -242,26 +242,9 @@ void qedf_fip_recv(struct qedf_ctx *qedf, struct sk_buff *skb)
        }
 }
 
-void qedf_update_src_mac(struct fc_lport *lport, u8 *addr)
-{
-       struct qedf_ctx *qedf = lport_priv(lport);
-
-       QEDF_INFO(&(qedf->dbg_ctx), QEDF_LOG_DISC,
-           "Setting data_src_addr=%pM.\n", addr);
-       ether_addr_copy(qedf->data_src_addr, addr);
-}
-
 u8 *qedf_get_src_mac(struct fc_lport *lport)
 {
-       u8 mac[ETH_ALEN];
-       u8 port_id[3];
        struct qedf_ctx *qedf = lport_priv(lport);
 
-       /* We need to use the lport port_id to create the data_src_addr */
-       if (is_zero_ether_addr(qedf->data_src_addr)) {
-               hton24(port_id, lport->port_id);
-               fc_fcoe_set_mac(mac, port_id);
-               qedf->ctlr.update_mac(lport, mac);
-       }
        return qedf->data_src_addr;
 }
index e17e5a7fc73eee4ed385cb3f897d90a21e3c4d0f..c61c1352c364fbc67fc0202bbf0652357f7223e0 100644 (file)
@@ -18,6 +18,7 @@
 #include <linux/kthread.h>
 #include <scsi/libfc.h>
 #include <scsi/scsi_host.h>
+#include <scsi/fc_frame.h>
 #include <linux/if_ether.h>
 #include <linux/if_vlan.h>
 #include <linux/cpu.h>
@@ -187,6 +188,50 @@ static void qedf_handle_link_update(struct work_struct *work)
        }
 }
 
+#define        QEDF_FCOE_MAC_METHOD_GRANGED_MAC                1
+#define QEDF_FCOE_MAC_METHOD_FCF_MAP                   2
+#define QEDF_FCOE_MAC_METHOD_FCOE_SET_MAC              3
+static void qedf_set_data_src_addr(struct qedf_ctx *qedf, struct fc_frame *fp)
+{
+       u8 *granted_mac;
+       struct fc_frame_header *fh = fc_frame_header_get(fp);
+       u8 fc_map[3];
+       int method = 0;
+
+       /* Get granted MAC address from FIP FLOGI payload */
+       granted_mac = fr_cb(fp)->granted_mac;
+
+       /*
+        * We set the source MAC for FCoE traffic based on the Granted MAC
+        * address from the switch.
+        *
+        * If granted_mac is non-zero, we used that.
+        * If the granted_mac is zeroed out, created the FCoE MAC based on
+        * the sel_fcf->fc_map and the d_id fo the FLOGI frame.
+        * If sel_fcf->fc_map is 0 then we use the default FCF-MAC plus the
+        * d_id of the FLOGI frame.
+        */
+       if (!is_zero_ether_addr(granted_mac)) {
+               ether_addr_copy(qedf->data_src_addr, granted_mac);
+               method = QEDF_FCOE_MAC_METHOD_GRANGED_MAC;
+       } else if (qedf->ctlr.sel_fcf->fc_map != 0) {
+               hton24(fc_map, qedf->ctlr.sel_fcf->fc_map);
+               qedf->data_src_addr[0] = fc_map[0];
+               qedf->data_src_addr[1] = fc_map[1];
+               qedf->data_src_addr[2] = fc_map[2];
+               qedf->data_src_addr[3] = fh->fh_d_id[0];
+               qedf->data_src_addr[4] = fh->fh_d_id[1];
+               qedf->data_src_addr[5] = fh->fh_d_id[2];
+               method = QEDF_FCOE_MAC_METHOD_FCF_MAP;
+       } else {
+               fc_fcoe_set_mac(qedf->data_src_addr, fh->fh_d_id);
+               method = QEDF_FCOE_MAC_METHOD_FCOE_SET_MAC;
+       }
+
+       QEDF_INFO(&(qedf->dbg_ctx), QEDF_LOG_DISC,
+           "QEDF data_src_mac=%pM method=%d.\n", qedf->data_src_addr, method);
+}
+
 static void qedf_flogi_resp(struct fc_seq *seq, struct fc_frame *fp,
        void *arg)
 {
@@ -212,6 +257,10 @@ static void qedf_flogi_resp(struct fc_seq *seq, struct fc_frame *fp,
        /* Log stats for FLOGI reject */
        if (fc_frame_payload_op(fp) == ELS_LS_RJT)
                qedf->flogi_failed++;
+       else if (fc_frame_payload_op(fp) == ELS_LS_ACC) {
+               /* Set the source MAC we will use for FCoE traffic */
+               qedf_set_data_src_addr(qedf, fp);
+       }
 
        /* Complete flogi_compl so we can proceed to sending ADISCs */
        complete(&qedf->flogi_compl);
@@ -927,7 +976,7 @@ static int qedf_xmit(struct fc_lport *lport, struct fc_frame *fp)
                ether_addr_copy(eh->h_dest, qedf->ctlr.dest_addr);
 
        /* Set the source MAC address */
-       fc_fcoe_set_mac(eh->h_source, fh->fh_s_id);
+       ether_addr_copy(eh->h_source, qedf->data_src_addr);
 
        hp = (struct fcoe_hdr *)(eh + 1);
        memset(hp, 0, sizeof(*hp));
@@ -1025,7 +1074,6 @@ static int qedf_offload_connection(struct qedf_ctx *qedf,
 {
        struct qed_fcoe_params_offload conn_info;
        u32 port_id;
-       u8 lport_src_id[3];
        int rval;
        uint16_t total_sqe = (fcport->sq_mem_size / sizeof(struct fcoe_wqe));
 
@@ -1054,11 +1102,7 @@ static int qedf_offload_connection(struct qedf_ctx *qedf,
            (dma_addr_t)(*(u64 *)(fcport->sq_pbl + 8));
 
        /* Need to use our FCoE MAC for the offload session */
-       port_id = fc_host_port_id(qedf->lport->host);
-       lport_src_id[2] = (port_id & 0x000000FF);
-       lport_src_id[1] = (port_id & 0x0000FF00) >> 8;
-       lport_src_id[0] = (port_id & 0x00FF0000) >> 16;
-       fc_fcoe_set_mac(conn_info.src_mac, lport_src_id);
+       ether_addr_copy(conn_info.src_mac, qedf->data_src_addr);
 
        ether_addr_copy(conn_info.dst_mac, qedf->ctlr.dest_addr);
 
@@ -1347,7 +1391,6 @@ static void qedf_fcoe_ctlr_setup(struct qedf_ctx *qedf)
        fcoe_ctlr_init(&qedf->ctlr, FIP_ST_AUTO);
 
        qedf->ctlr.send = qedf_fip_send;
-       qedf->ctlr.update_mac = qedf_update_src_mac;
        qedf->ctlr.get_src_addr = qedf_get_src_mac;
        ether_addr_copy(qedf->ctlr.ctl_src_addr, qedf->mac);
 }