2009-06-09 05:14:44 +04:00
/* bnx2i.c: Broadcom NetXtreme II iSCSI driver.
*
2011-06-24 02:51:37 +04:00
* Copyright ( c ) 2006 - 2011 Broadcom Corporation
2009-06-09 05:14:44 +04:00
* Copyright ( c ) 2007 , 2008 Red Hat , Inc . All rights reserved .
* Copyright ( c ) 2007 , 2008 Mike Christie
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation .
*
* Written by : Anil Veerabhadrappa ( anilgv @ broadcom . com )
2010-11-24 02:29:31 +03:00
* Maintained by : Eddie Wai ( eddie . wai @ broadcom . com )
2009-06-09 05:14:44 +04:00
*/
# include "bnx2i.h"
static struct list_head adapter_list = LIST_HEAD_INIT ( adapter_list ) ;
static u32 adapter_count ;
# define DRV_MODULE_NAME "bnx2i"
2011-06-24 02:51:37 +04:00
# define DRV_MODULE_VERSION "2.7.0.3"
# define DRV_MODULE_RELDATE "Jun 15, 2011"
2009-06-09 05:14:44 +04:00
static char version [ ] __devinitdata =
" Broadcom NetXtreme II iSCSI Driver " DRV_MODULE_NAME \
" v " DRV_MODULE_VERSION " ( " DRV_MODULE_RELDATE " ) \n " ;
2010-08-13 03:44:31 +04:00
MODULE_AUTHOR ( " Anil Veerabhadrappa <anilgv@broadcom.com> and "
" Eddie Wai <eddie.wai@broadcom.com> " ) ;
2011-02-17 00:04:28 +03:00
MODULE_DESCRIPTION ( " Broadcom NetXtreme II BCM5706/5708/5709/57710/57711/57712 "
2011-06-20 19:15:56 +04:00
" /57800/57810/57840 iSCSI Driver " ) ;
2009-06-09 05:14:44 +04:00
MODULE_LICENSE ( " GPL " ) ;
MODULE_VERSION ( DRV_MODULE_VERSION ) ;
2009-06-24 01:07:40 +04:00
static DEFINE_MUTEX ( bnx2i_dev_lock ) ;
2009-06-09 05:14:44 +04:00
2009-12-07 22:40:18 +03:00
unsigned int event_coal_min = 24 ;
module_param ( event_coal_min , int , 0664 ) ;
MODULE_PARM_DESC ( event_coal_min , " Event Coalescing Minimum Commands " ) ;
2011-06-24 02:51:34 +04:00
unsigned int event_coal_div = 2 ;
2009-06-09 05:14:44 +04:00
module_param ( event_coal_div , int , 0664 ) ;
MODULE_PARM_DESC ( event_coal_div , " Event Coalescing Divide Factor " ) ;
unsigned int en_tcp_dack = 1 ;
module_param ( en_tcp_dack , int , 0664 ) ;
MODULE_PARM_DESC ( en_tcp_dack , " Enable TCP Delayed ACK " ) ;
unsigned int error_mask1 = 0x00 ;
module_param ( error_mask1 , int , 0664 ) ;
MODULE_PARM_DESC ( error_mask1 , " Config FW iSCSI Error Mask #1 " ) ;
unsigned int error_mask2 = 0x00 ;
module_param ( error_mask2 , int , 0664 ) ;
MODULE_PARM_DESC ( error_mask2 , " Config FW iSCSI Error Mask #2 " ) ;
unsigned int sq_size ;
module_param ( sq_size , int , 0664 ) ;
MODULE_PARM_DESC ( sq_size , " Configure SQ size " ) ;
unsigned int rq_size = BNX2I_RQ_WQES_DEFAULT ;
module_param ( rq_size , int , 0664 ) ;
MODULE_PARM_DESC ( rq_size , " Configure RQ size " ) ;
u64 iscsi_error_mask = 0x00 ;
2011-06-24 02:51:34 +04:00
DEFINE_PER_CPU ( struct bnx2i_percpu_s , bnx2i_percpu ) ;
static int bnx2i_cpu_callback ( struct notifier_block * nfb ,
unsigned long action , void * hcpu ) ;
/* notification function for CPU hotplug events */
static struct notifier_block bnx2i_cpu_notifier = {
. notifier_call = bnx2i_cpu_callback ,
} ;
2009-06-09 05:14:44 +04:00
/**
* bnx2i_identify_device - identifies NetXtreme II device type
* @ hba : Adapter structure pointer
*
* This function identifies the NX2 device type and sets appropriate
* queue mailbox register access method , 5709 requires driver to
* access MBOX regs using * bin * mode
*/
void bnx2i_identify_device ( struct bnx2i_hba * hba )
{
hba - > cnic_dev_type = 0 ;
if ( ( hba - > pci_did = = PCI_DEVICE_ID_NX2_5706 ) | |
( hba - > pci_did = = PCI_DEVICE_ID_NX2_5706S ) )
set_bit ( BNX2I_NX2_DEV_5706 , & hba - > cnic_dev_type ) ;
else if ( ( hba - > pci_did = = PCI_DEVICE_ID_NX2_5708 ) | |
( hba - > pci_did = = PCI_DEVICE_ID_NX2_5708S ) )
set_bit ( BNX2I_NX2_DEV_5708 , & hba - > cnic_dev_type ) ;
else if ( ( hba - > pci_did = = PCI_DEVICE_ID_NX2_5709 ) | |
( hba - > pci_did = = PCI_DEVICE_ID_NX2_5709S ) ) {
set_bit ( BNX2I_NX2_DEV_5709 , & hba - > cnic_dev_type ) ;
hba - > mail_queue_access = BNX2I_MQ_BIN_MODE ;
2011-06-20 19:15:56 +04:00
} else if ( hba - > pci_did = = PCI_DEVICE_ID_NX2_57710 | |
hba - > pci_did = = PCI_DEVICE_ID_NX2_57711 | |
hba - > pci_did = = PCI_DEVICE_ID_NX2_57711E | |
hba - > pci_did = = PCI_DEVICE_ID_NX2_57712 | |
hba - > pci_did = = PCI_DEVICE_ID_NX2_57712E | |
hba - > pci_did = = PCI_DEVICE_ID_NX2_57800 | |
hba - > pci_did = = PCI_DEVICE_ID_NX2_57800_MF | |
hba - > pci_did = = PCI_DEVICE_ID_NX2_57800_VF | |
hba - > pci_did = = PCI_DEVICE_ID_NX2_57810 | |
hba - > pci_did = = PCI_DEVICE_ID_NX2_57810_MF | |
hba - > pci_did = = PCI_DEVICE_ID_NX2_57810_VF | |
hba - > pci_did = = PCI_DEVICE_ID_NX2_57840 | |
hba - > pci_did = = PCI_DEVICE_ID_NX2_57840_MF | |
hba - > pci_did = = PCI_DEVICE_ID_NX2_57840_VF )
2009-06-09 05:14:44 +04:00
set_bit ( BNX2I_NX2_DEV_57710 , & hba - > cnic_dev_type ) ;
2009-12-07 22:39:33 +03:00
else
printk ( KERN_ALERT " bnx2i: unknown device, 0x%x \n " ,
hba - > pci_did ) ;
2009-06-09 05:14:44 +04:00
}
/**
* get_adapter_list_head - returns head of adapter list
*/
struct bnx2i_hba * get_adapter_list_head ( void )
{
struct bnx2i_hba * hba = NULL ;
struct bnx2i_hba * tmp_hba ;
if ( ! adapter_count )
goto hba_not_found ;
2009-06-24 01:07:40 +04:00
mutex_lock ( & bnx2i_dev_lock ) ;
2009-06-09 05:14:44 +04:00
list_for_each_entry ( tmp_hba , & adapter_list , link ) {
if ( tmp_hba - > cnic & & tmp_hba - > cnic - > cm_select_dev ) {
hba = tmp_hba ;
break ;
}
}
2009-06-24 01:07:40 +04:00
mutex_unlock ( & bnx2i_dev_lock ) ;
2009-06-09 05:14:44 +04:00
hba_not_found :
return hba ;
}
/**
* bnx2i_find_hba_for_cnic - maps cnic device instance to bnx2i adapter instance
* @ cnic : pointer to cnic device instance
*
*/
struct bnx2i_hba * bnx2i_find_hba_for_cnic ( struct cnic_dev * cnic )
{
struct bnx2i_hba * hba , * temp ;
2009-06-24 01:07:40 +04:00
mutex_lock ( & bnx2i_dev_lock ) ;
2009-06-09 05:14:44 +04:00
list_for_each_entry_safe ( hba , temp , & adapter_list , link ) {
if ( hba - > cnic = = cnic ) {
2009-06-24 01:07:40 +04:00
mutex_unlock ( & bnx2i_dev_lock ) ;
2009-06-09 05:14:44 +04:00
return hba ;
}
}
2009-06-24 01:07:40 +04:00
mutex_unlock ( & bnx2i_dev_lock ) ;
2009-06-09 05:14:44 +04:00
return NULL ;
}
/**
* bnx2i_start - cnic callback to initialize & start adapter instance
* @ handle : transparent handle pointing to adapter structure
*
* This function maps adapter structure to pcidev structure and initiates
* firmware handshake to enable / initialize on chip iscsi components
* This bnx2i - cnic interface api callback is issued after following
* 2 conditions are met -
* a ) underlying network interface is up ( marked by event ' NETDEV_UP '
* from netdev
* b ) bnx2i adapter instance is registered
*/
void bnx2i_start ( void * handle )
{
# define BNX2I_INIT_POLL_TIME (1000 / HZ)
struct bnx2i_hba * hba = handle ;
int i = HZ ;
2011-06-24 02:51:35 +04:00
/*
* We should never register devices that don ' t support iSCSI
* ( see bnx2i_init_one ) , so something is wrong if we try to
* start a iSCSI adapter on hardware with 0 supported iSCSI
* connections
*/
BUG_ON ( ! hba - > cnic - > max_iscsi_conn ) ;
2011-02-17 00:04:27 +03:00
2009-06-09 05:14:44 +04:00
bnx2i_send_fw_iscsi_init_msg ( hba ) ;
while ( ! test_bit ( ADAPTER_STATE_UP , & hba - > adapter_state ) & & i - - )
msleep ( BNX2I_INIT_POLL_TIME ) ;
}
2010-08-13 03:44:30 +04:00
/**
* bnx2i_chip_cleanup - local routine to handle chip cleanup
* @ hba : Adapter instance to register
*
* Driver checks if adapter still has any active connections before
* executing the cleanup process
*/
static void bnx2i_chip_cleanup ( struct bnx2i_hba * hba )
{
struct bnx2i_endpoint * bnx2i_ep ;
struct list_head * pos , * tmp ;
if ( hba - > ofld_conns_active ) {
/* Stage to force the disconnection
* This is the case where the daemon is either slow or
* not present
*/
printk ( KERN_ALERT " bnx2i: (%s) chip cleanup for %d active "
" connections \n " , hba - > netdev - > name ,
hba - > ofld_conns_active ) ;
mutex_lock ( & hba - > net_dev_lock ) ;
list_for_each_safe ( pos , tmp , & hba - > ep_active_list ) {
bnx2i_ep = list_entry ( pos , struct bnx2i_endpoint , link ) ;
/* Clean up the chip only */
bnx2i_hw_ep_disconnect ( bnx2i_ep ) ;
bnx2i_ep - > cm_sk = NULL ;
}
mutex_unlock ( & hba - > net_dev_lock ) ;
}
}
2009-06-09 05:14:44 +04:00
/**
* bnx2i_stop - cnic callback to shutdown adapter instance
* @ handle : transparent handle pointing to adapter structure
*
* driver checks if adapter is already in shutdown mode , if not start
* the shutdown process
*/
void bnx2i_stop ( void * handle )
{
struct bnx2i_hba * hba = handle ;
2010-07-02 02:34:52 +04:00
int conns_active ;
2010-11-24 02:29:28 +03:00
int wait_delay = 1 * HZ ;
2009-06-09 05:14:44 +04:00
/* check if cleanup happened in GOING_DOWN context */
2010-11-24 02:29:28 +03:00
if ( ! test_and_set_bit ( ADAPTER_STATE_GOING_DOWN ,
& hba - > adapter_state ) ) {
2009-06-09 05:14:44 +04:00
iscsi_host_for_each_session ( hba - > shost ,
bnx2i_drop_session ) ;
2010-11-24 02:29:28 +03:00
wait_delay = hba - > hba_shutdown_tmo ;
}
/* Wait for inflight offload connection tasks to complete before
* proceeding . Forcefully terminate all connection recovery in
* progress at the earliest , either in bind ( ) , send_pdu ( LOGIN ) ,
* or conn_start ( )
*/
wait_event_interruptible_timeout ( hba - > eh_wait ,
( list_empty ( & hba - > ep_ofld_list ) & &
list_empty ( & hba - > ep_destroy_list ) ) ,
2011-05-16 22:13:19 +04:00
2 * HZ ) ;
2010-04-09 02:59:15 +04:00
/* Wait for all endpoints to be torn down, Chip will be reset once
* control returns to network driver . So it is required to cleanup and
* release all connection resources before returning from this routine .
*/
2010-07-02 02:34:52 +04:00
while ( hba - > ofld_conns_active ) {
conns_active = hba - > ofld_conns_active ;
wait_event_interruptible_timeout ( hba - > eh_wait ,
( hba - > ofld_conns_active ! = conns_active ) ,
2010-11-24 02:29:28 +03:00
wait_delay ) ;
2010-07-02 02:34:52 +04:00
if ( hba - > ofld_conns_active = = conns_active )
break ;
}
2010-08-13 03:44:30 +04:00
bnx2i_chip_cleanup ( hba ) ;
2010-07-02 02:34:52 +04:00
2010-04-09 02:59:15 +04:00
/* This flag should be cleared last so that ep_disconnect() gracefully
* cleans up connection context
*/
2010-11-24 02:29:28 +03:00
clear_bit ( ADAPTER_STATE_GOING_DOWN , & hba - > adapter_state ) ;
2010-04-09 02:59:15 +04:00
clear_bit ( ADAPTER_STATE_UP , & hba - > adapter_state ) ;
2009-06-09 05:14:44 +04:00
}
2010-11-24 02:29:28 +03:00
2009-06-09 05:14:44 +04:00
/**
* bnx2i_init_one - initialize an adapter instance and allocate memory resources
* @ hba : bnx2i adapter instance
* @ cnic : cnic device handle
*
2009-06-24 01:07:40 +04:00
* Global resource lock is held during critical sections below . This routine is
* called from either cnic_register_driver ( ) or device hot plug context and
* and does majority of device specific initialization
2009-06-09 05:14:44 +04:00
*/
static int bnx2i_init_one ( struct bnx2i_hba * hba , struct cnic_dev * cnic )
{
int rc ;
2009-06-24 01:07:40 +04:00
mutex_lock ( & bnx2i_dev_lock ) ;
2011-06-24 02:51:35 +04:00
if ( ! cnic - > max_iscsi_conn ) {
printk ( KERN_ALERT " bnx2i: dev %s does not support "
" iSCSI \n " , hba - > netdev - > name ) ;
rc = - EOPNOTSUPP ;
goto out ;
}
2010-03-25 20:54:42 +03:00
hba - > cnic = cnic ;
2009-06-24 01:04:23 +04:00
rc = cnic - > register_device ( cnic , CNIC_ULP_ISCSI , hba ) ;
if ( ! rc ) {
2009-06-09 05:14:44 +04:00
hba - > age + + ;
set_bit ( BNX2I_CNIC_REGISTERED , & hba - > reg_with_cnic ) ;
2009-06-24 01:07:40 +04:00
list_add_tail ( & hba - > link , & adapter_list ) ;
adapter_count + + ;
2009-06-24 01:04:23 +04:00
} else if ( rc = = - EBUSY ) /* duplicate registration */
printk ( KERN_ALERT " bnx2i, duplicate registration "
" hba=%p, cnic=%p \n " , hba , cnic ) ;
else if ( rc = = - EAGAIN )
printk ( KERN_ERR " bnx2i, driver not registered \n " ) ;
else if ( rc = = - EINVAL )
printk ( KERN_ERR " bnx2i, invalid type %d \n " , CNIC_ULP_ISCSI ) ;
else
printk ( KERN_ERR " bnx2i dev reg, unknown error, %d \n " , rc ) ;
2009-06-09 05:14:44 +04:00
2011-06-24 02:51:35 +04:00
out :
2009-06-24 01:07:40 +04:00
mutex_unlock ( & bnx2i_dev_lock ) ;
2009-06-24 01:04:23 +04:00
return rc ;
2009-06-09 05:14:44 +04:00
}
/**
* bnx2i_ulp_init - initialize an adapter instance
* @ dev : cnic device handle
*
* Called from cnic_register_driver ( ) context to initialize all enumerated
* cnic devices . This routine allocate adapter structure and other
* device specific resources .
*/
void bnx2i_ulp_init ( struct cnic_dev * dev )
{
struct bnx2i_hba * hba ;
/* Allocate a HBA structure for this device */
hba = bnx2i_alloc_hba ( dev ) ;
if ( ! hba ) {
printk ( KERN_ERR " bnx2i init: hba initialization failed \n " ) ;
return ;
}
/* Get PCI related information and update hba struct members */
clear_bit ( BNX2I_CNIC_REGISTERED , & hba - > reg_with_cnic ) ;
if ( bnx2i_init_one ( hba , dev ) ) {
printk ( KERN_ERR " bnx2i - hba %p init failed \n " , hba ) ;
bnx2i_free_hba ( hba ) ;
2010-03-25 20:54:42 +03:00
}
2009-06-09 05:14:44 +04:00
}
/**
* bnx2i_ulp_exit - shuts down adapter instance and frees all resources
* @ dev : cnic device handle
*
*/
void bnx2i_ulp_exit ( struct cnic_dev * dev )
{
struct bnx2i_hba * hba ;
hba = bnx2i_find_hba_for_cnic ( dev ) ;
if ( ! hba ) {
printk ( KERN_INFO " bnx2i_ulp_exit: hba not "
" found, dev 0x%p \n " , dev ) ;
return ;
}
2009-06-24 01:07:40 +04:00
mutex_lock ( & bnx2i_dev_lock ) ;
2009-06-09 05:14:44 +04:00
list_del_init ( & hba - > link ) ;
adapter_count - - ;
if ( test_bit ( BNX2I_CNIC_REGISTERED , & hba - > reg_with_cnic ) ) {
hba - > cnic - > unregister_device ( hba - > cnic , CNIC_ULP_ISCSI ) ;
clear_bit ( BNX2I_CNIC_REGISTERED , & hba - > reg_with_cnic ) ;
}
2009-06-24 01:07:40 +04:00
mutex_unlock ( & bnx2i_dev_lock ) ;
2009-06-09 05:14:44 +04:00
bnx2i_free_hba ( hba ) ;
}
2011-06-24 02:51:34 +04:00
/**
* bnx2i_percpu_thread_create - Create a receive thread for an
* online CPU
*
* @ cpu : cpu index for the online cpu
*/
static void bnx2i_percpu_thread_create ( unsigned int cpu )
{
struct bnx2i_percpu_s * p ;
struct task_struct * thread ;
p = & per_cpu ( bnx2i_percpu , cpu ) ;
thread = kthread_create ( bnx2i_percpu_io_thread , ( void * ) p ,
" bnx2i_thread/%d " , cpu ) ;
/* bind thread to the cpu */
if ( likely ( ! IS_ERR ( thread ) ) ) {
kthread_bind ( thread , cpu ) ;
p - > iothread = thread ;
wake_up_process ( thread ) ;
}
}
static void bnx2i_percpu_thread_destroy ( unsigned int cpu )
{
struct bnx2i_percpu_s * p ;
struct task_struct * thread ;
struct bnx2i_work * work , * tmp ;
/* Prevent any new work from being queued for this CPU */
p = & per_cpu ( bnx2i_percpu , cpu ) ;
spin_lock_bh ( & p - > p_work_lock ) ;
thread = p - > iothread ;
p - > iothread = NULL ;
/* Free all work in the list */
list_for_each_entry_safe ( work , tmp , & p - > work_list , list ) {
list_del_init ( & work - > list ) ;
bnx2i_process_scsi_cmd_resp ( work - > session ,
work - > bnx2i_conn , & work - > cqe ) ;
kfree ( work ) ;
}
spin_unlock_bh ( & p - > p_work_lock ) ;
if ( thread )
kthread_stop ( thread ) ;
}
/**
* bnx2i_cpu_callback - Handler for CPU hotplug events
*
* @ nfb : The callback data block
* @ action : The event triggering the callback
* @ hcpu : The index of the CPU that the event is for
*
* This creates or destroys per - CPU data for iSCSI
*
* Returns NOTIFY_OK always .
*/
static int bnx2i_cpu_callback ( struct notifier_block * nfb ,
unsigned long action , void * hcpu )
{
unsigned cpu = ( unsigned long ) hcpu ;
switch ( action ) {
case CPU_ONLINE :
case CPU_ONLINE_FROZEN :
printk ( KERN_INFO " bnx2i: CPU %x online: Create Rx thread \n " ,
cpu ) ;
bnx2i_percpu_thread_create ( cpu ) ;
break ;
case CPU_DEAD :
case CPU_DEAD_FROZEN :
printk ( KERN_INFO " CPU %x offline: Remove Rx thread \n " , cpu ) ;
bnx2i_percpu_thread_destroy ( cpu ) ;
break ;
default :
break ;
}
return NOTIFY_OK ;
}
2009-06-09 05:14:44 +04:00
/**
* bnx2i_mod_init - module init entry point
*
* initialize any driver wide global data structures such as endpoint pool ,
* tcp port manager / queue , sysfs . finally driver will register itself
* with the cnic module
*/
static int __init bnx2i_mod_init ( void )
{
int err ;
2011-06-24 02:51:34 +04:00
unsigned cpu = 0 ;
struct bnx2i_percpu_s * p ;
2009-06-09 05:14:44 +04:00
printk ( KERN_INFO " %s " , version ) ;
2009-12-07 22:39:54 +03:00
if ( sq_size & & ! is_power_of_2 ( sq_size ) )
2009-06-09 05:14:44 +04:00
sq_size = roundup_pow_of_two ( sq_size ) ;
2009-06-24 01:07:40 +04:00
mutex_init ( & bnx2i_dev_lock ) ;
2009-06-09 05:14:44 +04:00
bnx2i_scsi_xport_template =
iscsi_register_transport ( & bnx2i_iscsi_transport ) ;
if ( ! bnx2i_scsi_xport_template ) {
printk ( KERN_ERR " Could not register bnx2i transport. \n " ) ;
err = - ENOMEM ;
goto out ;
}
err = cnic_register_driver ( CNIC_ULP_ISCSI , & bnx2i_cnic_cb ) ;
if ( err ) {
printk ( KERN_ERR " Could not register bnx2i cnic driver. \n " ) ;
goto unreg_xport ;
}
2011-06-24 02:51:34 +04:00
/* Create percpu kernel threads to handle iSCSI I/O completions */
for_each_possible_cpu ( cpu ) {
p = & per_cpu ( bnx2i_percpu , cpu ) ;
INIT_LIST_HEAD ( & p - > work_list ) ;
spin_lock_init ( & p - > p_work_lock ) ;
p - > iothread = NULL ;
}
for_each_online_cpu ( cpu )
bnx2i_percpu_thread_create ( cpu ) ;
/* Initialize per CPU interrupt thread */
register_hotcpu_notifier ( & bnx2i_cpu_notifier ) ;
2009-06-09 05:14:44 +04:00
return 0 ;
unreg_xport :
iscsi_unregister_transport ( & bnx2i_iscsi_transport ) ;
out :
return err ;
}
/**
* bnx2i_mod_exit - module cleanup / exit entry point
*
* Global resource lock and host adapter lock is held during critical sections
* in this function . Driver will browse through the adapter list , cleans - up
* each instance , unregisters iscsi transport name and finally driver will
* unregister itself with the cnic module
*/
static void __exit bnx2i_mod_exit ( void )
{
struct bnx2i_hba * hba ;
2011-06-24 02:51:34 +04:00
unsigned cpu = 0 ;
2009-06-09 05:14:44 +04:00
2009-06-24 01:07:40 +04:00
mutex_lock ( & bnx2i_dev_lock ) ;
2009-06-09 05:14:44 +04:00
while ( ! list_empty ( & adapter_list ) ) {
hba = list_entry ( adapter_list . next , struct bnx2i_hba , link ) ;
list_del ( & hba - > link ) ;
adapter_count - - ;
if ( test_bit ( BNX2I_CNIC_REGISTERED , & hba - > reg_with_cnic ) ) {
2010-08-13 03:44:30 +04:00
bnx2i_chip_cleanup ( hba ) ;
2009-06-09 05:14:44 +04:00
hba - > cnic - > unregister_device ( hba - > cnic , CNIC_ULP_ISCSI ) ;
clear_bit ( BNX2I_CNIC_REGISTERED , & hba - > reg_with_cnic ) ;
}
bnx2i_free_hba ( hba ) ;
}
2009-06-24 01:07:40 +04:00
mutex_unlock ( & bnx2i_dev_lock ) ;
2009-06-09 05:14:44 +04:00
2011-06-24 02:51:34 +04:00
unregister_hotcpu_notifier ( & bnx2i_cpu_notifier ) ;
for_each_online_cpu ( cpu )
bnx2i_percpu_thread_destroy ( cpu ) ;
2009-06-09 05:14:44 +04:00
iscsi_unregister_transport ( & bnx2i_iscsi_transport ) ;
cnic_unregister_driver ( CNIC_ULP_ISCSI ) ;
}
module_init ( bnx2i_mod_init ) ;
module_exit ( bnx2i_mod_exit ) ;