aboutsummaryrefslogtreecommitdiffstats
path: root/drivers
diff options
context:
space:
mode:
Diffstat (limited to 'drivers')
-rw-r--r--drivers/scsi/bnx2fc/bnx2fc_fcoe.c168
1 files changed, 139 insertions, 29 deletions
diff --git a/drivers/scsi/bnx2fc/bnx2fc_fcoe.c b/drivers/scsi/bnx2fc/bnx2fc_fcoe.c
index 9b38794d5665..8d975ef4eb69 100644
--- a/drivers/scsi/bnx2fc/bnx2fc_fcoe.c
+++ b/drivers/scsi/bnx2fc/bnx2fc_fcoe.c
@@ -62,6 +62,10 @@ static int bnx2fc_destroy(struct net_device *net_device);
62static int bnx2fc_enable(struct net_device *netdev); 62static int bnx2fc_enable(struct net_device *netdev);
63static int bnx2fc_disable(struct net_device *netdev); 63static int bnx2fc_disable(struct net_device *netdev);
64 64
65/* fcoe_syfs control interface handlers */
66static int bnx2fc_ctlr_alloc(struct net_device *netdev);
67static int bnx2fc_ctlr_enabled(struct fcoe_ctlr_device *cdev);
68
65static void bnx2fc_recv_frame(struct sk_buff *skb); 69static void bnx2fc_recv_frame(struct sk_buff *skb);
66 70
67static void bnx2fc_start_disc(struct bnx2fc_interface *interface); 71static void bnx2fc_start_disc(struct bnx2fc_interface *interface);
@@ -864,6 +868,7 @@ static void bnx2fc_indicate_netevent(void *context, unsigned long event,
864 u16 vlan_id) 868 u16 vlan_id)
865{ 869{
866 struct bnx2fc_hba *hba = (struct bnx2fc_hba *)context; 870 struct bnx2fc_hba *hba = (struct bnx2fc_hba *)context;
871 struct fcoe_ctlr_device *cdev;
867 struct fc_lport *lport; 872 struct fc_lport *lport;
868 struct fc_lport *vport; 873 struct fc_lport *vport;
869 struct bnx2fc_interface *interface, *tmp; 874 struct bnx2fc_interface *interface, *tmp;
@@ -925,28 +930,45 @@ static void bnx2fc_indicate_netevent(void *context, unsigned long event,
925 930
926 bnx2fc_link_speed_update(lport); 931 bnx2fc_link_speed_update(lport);
927 932
933 cdev = fcoe_ctlr_to_ctlr_dev(ctlr);
934
928 if (link_possible && !bnx2fc_link_ok(lport)) { 935 if (link_possible && !bnx2fc_link_ok(lport)) {
929 /* Reset max recv frame size to default */ 936 switch (cdev->enabled) {
930 fc_set_mfs(lport, BNX2FC_MFS); 937 case FCOE_CTLR_DISABLED:
931 /* 938 pr_info("Link up while interface is disabled.\n");
932 * ctlr link up will only be handled during 939 break;
933 * enable to avoid sending discovery solicitation 940 case FCOE_CTLR_ENABLED:
934 * on a stale vlan 941 case FCOE_CTLR_UNUSED:
935 */ 942 /* Reset max recv frame size to default */
936 if (interface->enabled) 943 fc_set_mfs(lport, BNX2FC_MFS);
937 fcoe_ctlr_link_up(ctlr); 944 /*
945 * ctlr link up will only be handled during
946 * enable to avoid sending discovery
947 * solicitation on a stale vlan
948 */
949 if (interface->enabled)
950 fcoe_ctlr_link_up(ctlr);
951 };
938 } else if (fcoe_ctlr_link_down(ctlr)) { 952 } else if (fcoe_ctlr_link_down(ctlr)) {
939 mutex_lock(&lport->lp_mutex); 953 switch (cdev->enabled) {
940 list_for_each_entry(vport, &lport->vports, list) 954 case FCOE_CTLR_DISABLED:
941 fc_host_port_type(vport->host) = 955 pr_info("Link down while interface is disabled.\n");
942 FC_PORTTYPE_UNKNOWN; 956 break;
943 mutex_unlock(&lport->lp_mutex); 957 case FCOE_CTLR_ENABLED:
944 fc_host_port_type(lport->host) = FC_PORTTYPE_UNKNOWN; 958 case FCOE_CTLR_UNUSED:
945 per_cpu_ptr(lport->stats, 959 mutex_lock(&lport->lp_mutex);
946 get_cpu())->LinkFailureCount++; 960 list_for_each_entry(vport, &lport->vports, list)
947 put_cpu(); 961 fc_host_port_type(vport->host) =
948 fcoe_clean_pending_queue(lport); 962 FC_PORTTYPE_UNKNOWN;
949 wait_for_upload = 1; 963 mutex_unlock(&lport->lp_mutex);
964 fc_host_port_type(lport->host) =
965 FC_PORTTYPE_UNKNOWN;
966 per_cpu_ptr(lport->stats,
967 get_cpu())->LinkFailureCount++;
968 put_cpu();
969 fcoe_clean_pending_queue(lport);
970 wait_for_upload = 1;
971 };
950 } 972 }
951 } 973 }
952 mutex_unlock(&bnx2fc_dev_lock); 974 mutex_unlock(&bnx2fc_dev_lock);
@@ -1996,7 +2018,9 @@ static void bnx2fc_ulp_init(struct cnic_dev *dev)
1996 set_bit(BNX2FC_CNIC_REGISTERED, &hba->reg_with_cnic); 2018 set_bit(BNX2FC_CNIC_REGISTERED, &hba->reg_with_cnic);
1997} 2019}
1998 2020
1999 2021/**
2022 * Deperecated: Use bnx2fc_enabled()
2023 */
2000static int bnx2fc_disable(struct net_device *netdev) 2024static int bnx2fc_disable(struct net_device *netdev)
2001{ 2025{
2002 struct bnx2fc_interface *interface; 2026 struct bnx2fc_interface *interface;
@@ -2022,7 +2046,9 @@ static int bnx2fc_disable(struct net_device *netdev)
2022 return rc; 2046 return rc;
2023} 2047}
2024 2048
2025 2049/**
2050 * Deprecated: Use bnx2fc_enabled()
2051 */
2026static int bnx2fc_enable(struct net_device *netdev) 2052static int bnx2fc_enable(struct net_device *netdev)
2027{ 2053{
2028 struct bnx2fc_interface *interface; 2054 struct bnx2fc_interface *interface;
@@ -2048,17 +2074,57 @@ static int bnx2fc_enable(struct net_device *netdev)
2048} 2074}
2049 2075
2050/** 2076/**
2051 * bnx2fc_create - Create bnx2fc FCoE interface 2077 * bnx2fc_ctlr_enabled() - Enable or disable an FCoE Controller
2078 * @cdev: The FCoE Controller that is being enabled or disabled
2052 * 2079 *
2053 * @buffer: The name of Ethernet interface to create on 2080 * fcoe_sysfs will ensure that the state of 'enabled' has
2054 * @kp: The associated kernel param 2081 * changed, so no checking is necessary here. This routine simply
2082 * calls fcoe_enable or fcoe_disable, both of which are deprecated.
2083 * When those routines are removed the functionality can be merged
2084 * here.
2085 */
2086static int bnx2fc_ctlr_enabled(struct fcoe_ctlr_device *cdev)
2087{
2088 struct fcoe_ctlr *ctlr = fcoe_ctlr_device_priv(cdev);
2089 struct fc_lport *lport = ctlr->lp;
2090 struct net_device *netdev = bnx2fc_netdev(lport);
2091
2092 switch (cdev->enabled) {
2093 case FCOE_CTLR_ENABLED:
2094 return bnx2fc_enable(netdev);
2095 case FCOE_CTLR_DISABLED:
2096 return bnx2fc_disable(netdev);
2097 case FCOE_CTLR_UNUSED:
2098 default:
2099 return -ENOTSUPP;
2100 };
2101}
2102
2103enum bnx2fc_create_link_state {
2104 BNX2FC_CREATE_LINK_DOWN,
2105 BNX2FC_CREATE_LINK_UP,
2106};
2107
2108/**
2109 * _bnx2fc_create() - Create bnx2fc FCoE interface
2110 * @netdev : The net_device object the Ethernet interface to create on
2111 * @fip_mode: The FIP mode for this creation
2112 * @link_state: The ctlr link state on creation
2055 * 2113 *
2056 * Called from sysfs. 2114 * Called from either the libfcoe 'create' module parameter
2115 * via fcoe_create or from fcoe_syfs's ctlr_create file.
2116 *
2117 * libfcoe's 'create' module parameter is deprecated so some
2118 * consolidation of code can be done when that interface is
2119 * removed.
2057 * 2120 *
2058 * Returns: 0 for success 2121 * Returns: 0 for success
2059 */ 2122 */
2060static int bnx2fc_create(struct net_device *netdev, enum fip_state fip_mode) 2123static int _bnx2fc_create(struct net_device *netdev,
2124 enum fip_state fip_mode,
2125 enum bnx2fc_create_link_state link_state)
2061{ 2126{
2127 struct fcoe_ctlr_device *cdev;
2062 struct fcoe_ctlr *ctlr; 2128 struct fcoe_ctlr *ctlr;
2063 struct bnx2fc_interface *interface; 2129 struct bnx2fc_interface *interface;
2064 struct bnx2fc_hba *hba; 2130 struct bnx2fc_hba *hba;
@@ -2153,7 +2219,15 @@ static int bnx2fc_create(struct net_device *netdev, enum fip_state fip_mode)
2153 /* Make this master N_port */ 2219 /* Make this master N_port */
2154 ctlr->lp = lport; 2220 ctlr->lp = lport;
2155 2221
2156 if (!bnx2fc_link_ok(lport)) { 2222 cdev = fcoe_ctlr_to_ctlr_dev(ctlr);
2223
2224 if (link_state == BNX2FC_CREATE_LINK_UP)
2225 cdev->enabled = FCOE_CTLR_ENABLED;
2226 else
2227 cdev->enabled = FCOE_CTLR_DISABLED;
2228
2229 if (link_state == BNX2FC_CREATE_LINK_UP &&
2230 !bnx2fc_link_ok(lport)) {
2157 fcoe_ctlr_link_up(ctlr); 2231 fcoe_ctlr_link_up(ctlr);
2158 fc_host_port_type(lport->host) = FC_PORTTYPE_NPORT; 2232 fc_host_port_type(lport->host) = FC_PORTTYPE_NPORT;
2159 set_bit(ADAPTER_STATE_READY, &interface->hba->adapter_state); 2233 set_bit(ADAPTER_STATE_READY, &interface->hba->adapter_state);
@@ -2161,7 +2235,10 @@ static int bnx2fc_create(struct net_device *netdev, enum fip_state fip_mode)
2161 2235
2162 BNX2FC_HBA_DBG(lport, "create: START DISC\n"); 2236 BNX2FC_HBA_DBG(lport, "create: START DISC\n");
2163 bnx2fc_start_disc(interface); 2237 bnx2fc_start_disc(interface);
2164 interface->enabled = true; 2238
2239 if (link_state == BNX2FC_CREATE_LINK_UP)
2240 interface->enabled = true;
2241
2165 /* 2242 /*
2166 * Release from kref_init in bnx2fc_interface_setup, on success 2243 * Release from kref_init in bnx2fc_interface_setup, on success
2167 * lport should be holding a reference taken in bnx2fc_if_create 2244 * lport should be holding a reference taken in bnx2fc_if_create
@@ -2187,6 +2264,37 @@ mod_err:
2187} 2264}
2188 2265
2189/** 2266/**
2267 * bnx2fc_create() - Create a bnx2fc interface
2268 * @netdev : The net_device object the Ethernet interface to create on
2269 * @fip_mode: The FIP mode for this creation
2270 *
2271 * Called from fcoe transport
2272 *
2273 * Returns: 0 for success
2274 */
2275static int bnx2fc_create(struct net_device *netdev, enum fip_state fip_mode)
2276{
2277 return _bnx2fc_create(netdev, fip_mode, BNX2FC_CREATE_LINK_UP);
2278}
2279
2280/**
2281 * bnx2fc_ctlr_alloc() - Allocate a bnx2fc interface from fcoe_sysfs
2282 * @netdev: The net_device to be used by the allocated FCoE Controller
2283 *
2284 * This routine is called from fcoe_sysfs. It will start the fcoe_ctlr
2285 * in a link_down state. The allows the user an opportunity to configure
2286 * the FCoE Controller from sysfs before enabling the FCoE Controller.
2287 *
2288 * Creating in with this routine starts the FCoE Controller in Fabric
2289 * mode. The user can change to VN2VN or another mode before enabling.
2290 */
2291static int bnx2fc_ctlr_alloc(struct net_device *netdev)
2292{
2293 return _bnx2fc_create(netdev, FIP_MODE_FABRIC,
2294 BNX2FC_CREATE_LINK_DOWN);
2295}
2296
2297/**
2190 * bnx2fc_find_hba_for_cnic - maps cnic instance to bnx2fc hba instance 2298 * bnx2fc_find_hba_for_cnic - maps cnic instance to bnx2fc hba instance
2191 * 2299 *
2192 * @cnic: Pointer to cnic device instance 2300 * @cnic: Pointer to cnic device instance
@@ -2311,6 +2419,7 @@ static struct fcoe_transport bnx2fc_transport = {
2311 .name = {"bnx2fc"}, 2419 .name = {"bnx2fc"},
2312 .attached = false, 2420 .attached = false,
2313 .list = LIST_HEAD_INIT(bnx2fc_transport.list), 2421 .list = LIST_HEAD_INIT(bnx2fc_transport.list),
2422 .alloc = bnx2fc_ctlr_alloc,
2314 .match = bnx2fc_match, 2423 .match = bnx2fc_match,
2315 .create = bnx2fc_create, 2424 .create = bnx2fc_create,
2316 .destroy = bnx2fc_destroy, 2425 .destroy = bnx2fc_destroy,
@@ -2555,6 +2664,7 @@ module_init(bnx2fc_mod_init);
2555module_exit(bnx2fc_mod_exit); 2664module_exit(bnx2fc_mod_exit);
2556 2665
2557static struct fcoe_sysfs_function_template bnx2fc_fcoe_sysfs_templ = { 2666static struct fcoe_sysfs_function_template bnx2fc_fcoe_sysfs_templ = {
2667 .set_fcoe_ctlr_enabled = bnx2fc_ctlr_enabled,
2558 .get_fcoe_ctlr_link_fail = bnx2fc_ctlr_get_lesb, 2668 .get_fcoe_ctlr_link_fail = bnx2fc_ctlr_get_lesb,
2559 .get_fcoe_ctlr_vlink_fail = bnx2fc_ctlr_get_lesb, 2669 .get_fcoe_ctlr_vlink_fail = bnx2fc_ctlr_get_lesb,
2560 .get_fcoe_ctlr_miss_fka = bnx2fc_ctlr_get_lesb, 2670 .get_fcoe_ctlr_miss_fka = bnx2fc_ctlr_get_lesb,