Commit 6e89ea3f authored by Robert Love's avatar Robert Love

bnx2fc: Use the fcoe_sysfs control interface

This patch adds support for the new fcoe_sysfs
control interface to bnx2fc.ko. It keeps the deprecated
interface in tact and therefore either the legacy
or the new control interfaces can be used. A mixed mode
is not supported. A user must either use the new
interfaces or the old ones, but not both.

The fcoe_ctlr's link state is now driven by both the
netdev link state as well as the fcoe_ctlr_device's
enabled attribute. The link must be up and the
fcoe_ctlr_device must be enabled before the FCoE
Controller starts discovery or login.
Signed-off-by: default avatarRobert Love <robert.w.love@intel.com>
Acked-by: default avatarNeil Horman <nhorman@tuxdriver.com>
parent 435c8667
...@@ -62,6 +62,10 @@ static int bnx2fc_destroy(struct net_device *net_device); ...@@ -62,6 +62,10 @@ static int bnx2fc_destroy(struct net_device *net_device);
static int bnx2fc_enable(struct net_device *netdev); static int bnx2fc_enable(struct net_device *netdev);
static int bnx2fc_disable(struct net_device *netdev); static int bnx2fc_disable(struct net_device *netdev);
/* fcoe_syfs control interface handlers */
static int bnx2fc_ctlr_alloc(struct net_device *netdev);
static int bnx2fc_ctlr_enabled(struct fcoe_ctlr_device *cdev);
static void bnx2fc_recv_frame(struct sk_buff *skb); static void bnx2fc_recv_frame(struct sk_buff *skb);
static void bnx2fc_start_disc(struct bnx2fc_interface *interface); static void bnx2fc_start_disc(struct bnx2fc_interface *interface);
...@@ -864,6 +868,7 @@ static void bnx2fc_indicate_netevent(void *context, unsigned long event, ...@@ -864,6 +868,7 @@ static void bnx2fc_indicate_netevent(void *context, unsigned long event,
u16 vlan_id) u16 vlan_id)
{ {
struct bnx2fc_hba *hba = (struct bnx2fc_hba *)context; struct bnx2fc_hba *hba = (struct bnx2fc_hba *)context;
struct fcoe_ctlr_device *cdev;
struct fc_lport *lport; struct fc_lport *lport;
struct fc_lport *vport; struct fc_lport *vport;
struct bnx2fc_interface *interface, *tmp; struct bnx2fc_interface *interface, *tmp;
...@@ -925,28 +930,45 @@ static void bnx2fc_indicate_netevent(void *context, unsigned long event, ...@@ -925,28 +930,45 @@ static void bnx2fc_indicate_netevent(void *context, unsigned long event,
bnx2fc_link_speed_update(lport); bnx2fc_link_speed_update(lport);
cdev = fcoe_ctlr_to_ctlr_dev(ctlr);
if (link_possible && !bnx2fc_link_ok(lport)) { if (link_possible && !bnx2fc_link_ok(lport)) {
/* Reset max recv frame size to default */ switch (cdev->enabled) {
fc_set_mfs(lport, BNX2FC_MFS); case FCOE_CTLR_DISABLED:
/* pr_info("Link up while interface is disabled.\n");
* ctlr link up will only be handled during break;
* enable to avoid sending discovery solicitation case FCOE_CTLR_ENABLED:
* on a stale vlan case FCOE_CTLR_UNUSED:
*/ /* Reset max recv frame size to default */
if (interface->enabled) fc_set_mfs(lport, BNX2FC_MFS);
fcoe_ctlr_link_up(ctlr); /*
* ctlr link up will only be handled during
* enable to avoid sending discovery
* solicitation on a stale vlan
*/
if (interface->enabled)
fcoe_ctlr_link_up(ctlr);
};
} else if (fcoe_ctlr_link_down(ctlr)) { } else if (fcoe_ctlr_link_down(ctlr)) {
mutex_lock(&lport->lp_mutex); switch (cdev->enabled) {
list_for_each_entry(vport, &lport->vports, list) case FCOE_CTLR_DISABLED:
fc_host_port_type(vport->host) = pr_info("Link down while interface is disabled.\n");
FC_PORTTYPE_UNKNOWN; break;
mutex_unlock(&lport->lp_mutex); case FCOE_CTLR_ENABLED:
fc_host_port_type(lport->host) = FC_PORTTYPE_UNKNOWN; case FCOE_CTLR_UNUSED:
per_cpu_ptr(lport->stats, mutex_lock(&lport->lp_mutex);
get_cpu())->LinkFailureCount++; list_for_each_entry(vport, &lport->vports, list)
put_cpu(); fc_host_port_type(vport->host) =
fcoe_clean_pending_queue(lport); FC_PORTTYPE_UNKNOWN;
wait_for_upload = 1; mutex_unlock(&lport->lp_mutex);
fc_host_port_type(lport->host) =
FC_PORTTYPE_UNKNOWN;
per_cpu_ptr(lport->stats,
get_cpu())->LinkFailureCount++;
put_cpu();
fcoe_clean_pending_queue(lport);
wait_for_upload = 1;
};
} }
} }
mutex_unlock(&bnx2fc_dev_lock); mutex_unlock(&bnx2fc_dev_lock);
...@@ -1996,7 +2018,9 @@ static void bnx2fc_ulp_init(struct cnic_dev *dev) ...@@ -1996,7 +2018,9 @@ static void bnx2fc_ulp_init(struct cnic_dev *dev)
set_bit(BNX2FC_CNIC_REGISTERED, &hba->reg_with_cnic); set_bit(BNX2FC_CNIC_REGISTERED, &hba->reg_with_cnic);
} }
/**
* Deperecated: Use bnx2fc_enabled()
*/
static int bnx2fc_disable(struct net_device *netdev) static int bnx2fc_disable(struct net_device *netdev)
{ {
struct bnx2fc_interface *interface; struct bnx2fc_interface *interface;
...@@ -2022,7 +2046,9 @@ static int bnx2fc_disable(struct net_device *netdev) ...@@ -2022,7 +2046,9 @@ static int bnx2fc_disable(struct net_device *netdev)
return rc; return rc;
} }
/**
* Deprecated: Use bnx2fc_enabled()
*/
static int bnx2fc_enable(struct net_device *netdev) static int bnx2fc_enable(struct net_device *netdev)
{ {
struct bnx2fc_interface *interface; struct bnx2fc_interface *interface;
...@@ -2048,17 +2074,57 @@ static int bnx2fc_enable(struct net_device *netdev) ...@@ -2048,17 +2074,57 @@ static int bnx2fc_enable(struct net_device *netdev)
} }
/** /**
* bnx2fc_create - Create bnx2fc FCoE interface * bnx2fc_ctlr_enabled() - Enable or disable an FCoE Controller
* @cdev: The FCoE Controller that is being enabled or disabled
* *
* @buffer: The name of Ethernet interface to create on * fcoe_sysfs will ensure that the state of 'enabled' has
* @kp: The associated kernel param * changed, so no checking is necessary here. This routine simply
* calls fcoe_enable or fcoe_disable, both of which are deprecated.
* When those routines are removed the functionality can be merged
* here.
*/
static int bnx2fc_ctlr_enabled(struct fcoe_ctlr_device *cdev)
{
struct fcoe_ctlr *ctlr = fcoe_ctlr_device_priv(cdev);
struct fc_lport *lport = ctlr->lp;
struct net_device *netdev = bnx2fc_netdev(lport);
switch (cdev->enabled) {
case FCOE_CTLR_ENABLED:
return bnx2fc_enable(netdev);
case FCOE_CTLR_DISABLED:
return bnx2fc_disable(netdev);
case FCOE_CTLR_UNUSED:
default:
return -ENOTSUPP;
};
}
enum bnx2fc_create_link_state {
BNX2FC_CREATE_LINK_DOWN,
BNX2FC_CREATE_LINK_UP,
};
/**
* _bnx2fc_create() - Create bnx2fc FCoE interface
* @netdev : The net_device object the Ethernet interface to create on
* @fip_mode: The FIP mode for this creation
* @link_state: The ctlr link state on creation
* *
* Called from sysfs. * Called from either the libfcoe 'create' module parameter
* via fcoe_create or from fcoe_syfs's ctlr_create file.
*
* libfcoe's 'create' module parameter is deprecated so some
* consolidation of code can be done when that interface is
* removed.
* *
* Returns: 0 for success * Returns: 0 for success
*/ */
static int bnx2fc_create(struct net_device *netdev, enum fip_state fip_mode) static int _bnx2fc_create(struct net_device *netdev,
enum fip_state fip_mode,
enum bnx2fc_create_link_state link_state)
{ {
struct fcoe_ctlr_device *cdev;
struct fcoe_ctlr *ctlr; struct fcoe_ctlr *ctlr;
struct bnx2fc_interface *interface; struct bnx2fc_interface *interface;
struct bnx2fc_hba *hba; struct bnx2fc_hba *hba;
...@@ -2153,7 +2219,15 @@ static int bnx2fc_create(struct net_device *netdev, enum fip_state fip_mode) ...@@ -2153,7 +2219,15 @@ static int bnx2fc_create(struct net_device *netdev, enum fip_state fip_mode)
/* Make this master N_port */ /* Make this master N_port */
ctlr->lp = lport; ctlr->lp = lport;
if (!bnx2fc_link_ok(lport)) { cdev = fcoe_ctlr_to_ctlr_dev(ctlr);
if (link_state == BNX2FC_CREATE_LINK_UP)
cdev->enabled = FCOE_CTLR_ENABLED;
else
cdev->enabled = FCOE_CTLR_DISABLED;
if (link_state == BNX2FC_CREATE_LINK_UP &&
!bnx2fc_link_ok(lport)) {
fcoe_ctlr_link_up(ctlr); fcoe_ctlr_link_up(ctlr);
fc_host_port_type(lport->host) = FC_PORTTYPE_NPORT; fc_host_port_type(lport->host) = FC_PORTTYPE_NPORT;
set_bit(ADAPTER_STATE_READY, &interface->hba->adapter_state); 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,7 +2235,10 @@ static int bnx2fc_create(struct net_device *netdev, enum fip_state fip_mode)
BNX2FC_HBA_DBG(lport, "create: START DISC\n"); BNX2FC_HBA_DBG(lport, "create: START DISC\n");
bnx2fc_start_disc(interface); bnx2fc_start_disc(interface);
interface->enabled = true;
if (link_state == BNX2FC_CREATE_LINK_UP)
interface->enabled = true;
/* /*
* Release from kref_init in bnx2fc_interface_setup, on success * Release from kref_init in bnx2fc_interface_setup, on success
* lport should be holding a reference taken in bnx2fc_if_create * lport should be holding a reference taken in bnx2fc_if_create
...@@ -2186,6 +2263,37 @@ static int bnx2fc_create(struct net_device *netdev, enum fip_state fip_mode) ...@@ -2186,6 +2263,37 @@ static int bnx2fc_create(struct net_device *netdev, enum fip_state fip_mode)
return rc; return rc;
} }
/**
* bnx2fc_create() - Create a bnx2fc interface
* @netdev : The net_device object the Ethernet interface to create on
* @fip_mode: The FIP mode for this creation
*
* Called from fcoe transport
*
* Returns: 0 for success
*/
static int bnx2fc_create(struct net_device *netdev, enum fip_state fip_mode)
{
return _bnx2fc_create(netdev, fip_mode, BNX2FC_CREATE_LINK_UP);
}
/**
* bnx2fc_ctlr_alloc() - Allocate a bnx2fc interface from fcoe_sysfs
* @netdev: The net_device to be used by the allocated FCoE Controller
*
* This routine is called from fcoe_sysfs. It will start the fcoe_ctlr
* in a link_down state. The allows the user an opportunity to configure
* the FCoE Controller from sysfs before enabling the FCoE Controller.
*
* Creating in with this routine starts the FCoE Controller in Fabric
* mode. The user can change to VN2VN or another mode before enabling.
*/
static int bnx2fc_ctlr_alloc(struct net_device *netdev)
{
return _bnx2fc_create(netdev, FIP_MODE_FABRIC,
BNX2FC_CREATE_LINK_DOWN);
}
/** /**
* bnx2fc_find_hba_for_cnic - maps cnic instance to bnx2fc hba instance * bnx2fc_find_hba_for_cnic - maps cnic instance to bnx2fc hba instance
* *
...@@ -2311,6 +2419,7 @@ static struct fcoe_transport bnx2fc_transport = { ...@@ -2311,6 +2419,7 @@ static struct fcoe_transport bnx2fc_transport = {
.name = {"bnx2fc"}, .name = {"bnx2fc"},
.attached = false, .attached = false,
.list = LIST_HEAD_INIT(bnx2fc_transport.list), .list = LIST_HEAD_INIT(bnx2fc_transport.list),
.alloc = bnx2fc_ctlr_alloc,
.match = bnx2fc_match, .match = bnx2fc_match,
.create = bnx2fc_create, .create = bnx2fc_create,
.destroy = bnx2fc_destroy, .destroy = bnx2fc_destroy,
...@@ -2555,6 +2664,7 @@ module_init(bnx2fc_mod_init); ...@@ -2555,6 +2664,7 @@ module_init(bnx2fc_mod_init);
module_exit(bnx2fc_mod_exit); module_exit(bnx2fc_mod_exit);
static struct fcoe_sysfs_function_template bnx2fc_fcoe_sysfs_templ = { static struct fcoe_sysfs_function_template bnx2fc_fcoe_sysfs_templ = {
.set_fcoe_ctlr_enabled = bnx2fc_ctlr_enabled,
.get_fcoe_ctlr_link_fail = bnx2fc_ctlr_get_lesb, .get_fcoe_ctlr_link_fail = bnx2fc_ctlr_get_lesb,
.get_fcoe_ctlr_vlink_fail = bnx2fc_ctlr_get_lesb, .get_fcoe_ctlr_vlink_fail = bnx2fc_ctlr_get_lesb,
.get_fcoe_ctlr_miss_fka = bnx2fc_ctlr_get_lesb, .get_fcoe_ctlr_miss_fka = bnx2fc_ctlr_get_lesb,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment