2009-09-24 04:46:15 +04:00
/*
2010-09-15 22:50:55 +04:00
* Copyright ( c ) 2005 - 2010 Brocade Communications Systems , Inc .
2009-09-24 04:46:15 +04:00
* All rights reserved
* www . brocade . com
*
* Linux driver for Brocade Fibre Channel Host Bus Adapter .
*
* This program is free software ; you can redistribute it and / or modify it
* under the terms of the GNU General Public License ( GPL ) Version 2 as
* published by the Free Software Foundation
*
* This program is distributed in the hope that it will be useful , but
* WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the GNU
* General Public License for more details .
*/
2010-12-10 06:12:32 +03:00
# include "bfad_drv.h"
2010-09-15 22:50:55 +04:00
# include "bfa_modules.h"
2009-09-24 04:46:15 +04:00
BFA_TRC_FILE ( HAL , FCPIM ) ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* BFA ITNIM Related definitions
*/
static void bfa_itnim_update_del_itn_stats ( struct bfa_itnim_s * itnim ) ;
2011-07-21 04:04:03 +04:00
static bfa_boolean_t bfa_ioim_lm_proc_rpl_data ( struct bfa_ioim_s * ioim ) ;
static bfa_boolean_t bfa_ioim_lm_proc_inq_data ( struct bfa_ioim_s * ioim ) ;
static void bfa_ioim_lm_init ( struct bfa_s * bfa ) ;
2010-09-15 22:50:55 +04:00
# define BFA_ITNIM_FROM_TAG(_fcpim, _tag) \
( ( ( _fcpim ) - > itnim_arr + ( ( _tag ) & ( ( _fcpim ) - > num_itnims - 1 ) ) ) )
# define bfa_fcpim_additn(__itnim) \
list_add_tail ( & ( __itnim ) - > qe , & ( __itnim ) - > fcpim - > itnim_q )
# define bfa_fcpim_delitn(__itnim) do { \
2010-12-27 08:46:35 +03:00
WARN_ON ( ! bfa_q_is_on_q ( & ( __itnim ) - > fcpim - > itnim_q , __itnim ) ) ; \
2010-09-15 22:50:55 +04:00
bfa_itnim_update_del_itn_stats ( __itnim ) ; \
list_del ( & ( __itnim ) - > qe ) ; \
2010-12-27 08:46:35 +03:00
WARN_ON ( ! list_empty ( & ( __itnim ) - > io_q ) ) ; \
WARN_ON ( ! list_empty ( & ( __itnim ) - > io_cleanup_q ) ) ; \
WARN_ON ( ! list_empty ( & ( __itnim ) - > pending_q ) ) ; \
2010-09-15 22:50:55 +04:00
} while ( 0 )
# define bfa_itnim_online_cb(__itnim) do { \
if ( ( __itnim ) - > bfa - > fcs ) \
bfa_cb_itnim_online ( ( __itnim ) - > ditn ) ; \
else { \
bfa_cb_queue ( ( __itnim ) - > bfa , & ( __itnim ) - > hcb_qe , \
__bfa_cb_itnim_online , ( __itnim ) ) ; \
} \
} while ( 0 )
# define bfa_itnim_offline_cb(__itnim) do { \
if ( ( __itnim ) - > bfa - > fcs ) \
bfa_cb_itnim_offline ( ( __itnim ) - > ditn ) ; \
else { \
bfa_cb_queue ( ( __itnim ) - > bfa , & ( __itnim ) - > hcb_qe , \
__bfa_cb_itnim_offline , ( __itnim ) ) ; \
} \
} while ( 0 )
2011-07-21 04:04:03 +04:00
# define bfa_ioim_rp_wwn(__ioim) \
( ( ( struct bfa_fcs_rport_s * ) \
( __ioim ) - > itnim - > rport - > rport_drv ) - > pwwn )
# define bfa_ioim_lp_wwn(__ioim) \
( ( BFA_LPS_FROM_TAG ( BFA_LPS_MOD ( ( __ioim ) - > bfa ) , \
( __ioim ) - > itnim - > rport - > rport_info . lp_tag ) ) - > pwwn ) \
2010-09-15 22:50:55 +04:00
# define bfa_itnim_sler_cb(__itnim) do { \
if ( ( __itnim ) - > bfa - > fcs ) \
bfa_cb_itnim_sler ( ( __itnim ) - > ditn ) ; \
else { \
bfa_cb_queue ( ( __itnim ) - > bfa , & ( __itnim ) - > hcb_qe , \
__bfa_cb_itnim_sler , ( __itnim ) ) ; \
} \
} while ( 0 )
2011-07-21 04:04:03 +04:00
enum bfa_ioim_lm_status {
BFA_IOIM_LM_PRESENT = 1 ,
BFA_IOIM_LM_LUN_NOT_SUP = 2 ,
BFA_IOIM_LM_RPL_DATA_CHANGED = 3 ,
BFA_IOIM_LM_LUN_NOT_RDY = 4 ,
} ;
enum bfa_ioim_lm_ua_status {
BFA_IOIM_LM_UA_RESET = 0 ,
BFA_IOIM_LM_UA_SET = 1 ,
} ;
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* itnim state machine event
2010-09-15 22:50:55 +04:00
*/
enum bfa_itnim_event {
BFA_ITNIM_SM_CREATE = 1 , /* itnim is created */
BFA_ITNIM_SM_ONLINE = 2 , /* itnim is online */
BFA_ITNIM_SM_OFFLINE = 3 , /* itnim is offline */
BFA_ITNIM_SM_FWRSP = 4 , /* firmware response */
BFA_ITNIM_SM_DELETE = 5 , /* deleting an existing itnim */
BFA_ITNIM_SM_CLEANUP = 6 , /* IO cleanup completion */
BFA_ITNIM_SM_SLER = 7 , /* second level error recovery */
BFA_ITNIM_SM_HWFAIL = 8 , /* IOC h/w failure event */
BFA_ITNIM_SM_QRESUME = 9 , /* queue space available */
} ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* BFA IOIM related definitions
*/
# define bfa_ioim_move_to_comp_q(__ioim) do { \
list_del ( & ( __ioim ) - > qe ) ; \
list_add_tail ( & ( __ioim ) - > qe , & ( __ioim ) - > fcpim - > ioim_comp_q ) ; \
} while ( 0 )
# define bfa_ioim_cb_profile_comp(__fcpim, __ioim) do { \
if ( ( __fcpim ) - > profile_comp ) \
( __fcpim ) - > profile_comp ( __ioim ) ; \
} while ( 0 )
# define bfa_ioim_cb_profile_start(__fcpim, __ioim) do { \
if ( ( __fcpim ) - > profile_start ) \
( __fcpim ) - > profile_start ( __ioim ) ; \
} while ( 0 )
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* IO state machine events
*/
enum bfa_ioim_event {
BFA_IOIM_SM_START = 1 , /* io start request from host */
BFA_IOIM_SM_COMP_GOOD = 2 , /* io good comp, resource free */
BFA_IOIM_SM_COMP = 3 , /* io comp, resource is free */
BFA_IOIM_SM_COMP_UTAG = 4 , /* io comp, resource is free */
BFA_IOIM_SM_DONE = 5 , /* io comp, resource not free */
BFA_IOIM_SM_FREE = 6 , /* io resource is freed */
BFA_IOIM_SM_ABORT = 7 , /* abort request from scsi stack */
BFA_IOIM_SM_ABORT_COMP = 8 , /* abort from f/w */
BFA_IOIM_SM_ABORT_DONE = 9 , /* abort completion from f/w */
BFA_IOIM_SM_QRESUME = 10 , /* CQ space available to queue IO */
BFA_IOIM_SM_SGALLOCED = 11 , /* SG page allocation successful */
BFA_IOIM_SM_SQRETRY = 12 , /* sequence recovery retry */
BFA_IOIM_SM_HCB = 13 , /* bfa callback complete */
BFA_IOIM_SM_CLEANUP = 14 , /* IO cleanup from itnim */
BFA_IOIM_SM_TMSTART = 15 , /* IO cleanup from tskim */
BFA_IOIM_SM_TMDONE = 16 , /* IO cleanup from tskim */
BFA_IOIM_SM_HWFAIL = 17 , /* IOC h/w failure event */
BFA_IOIM_SM_IOTOV = 18 , /* ITN offline TOV */
2011-07-21 04:04:03 +04:00
BFA_IOIM_SM_LM_LUN_NOT_SUP = 19 , /* lunmask lun not supported */
BFA_IOIM_SM_LM_RPL_DC = 20 , /* lunmask report-lun data changed */
BFA_IOIM_SM_LM_LUN_NOT_RDY = 21 , /* lunmask lun not ready */
2010-09-15 22:50:55 +04:00
} ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* BFA TSKIM related definitions
*/
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* task management completion handling
*/
# define bfa_tskim_qcomp(__tskim, __cbfn) do { \
bfa_cb_queue ( ( __tskim ) - > bfa , & ( __tskim ) - > hcb_qe , __cbfn , ( __tskim ) ) ; \
bfa_tskim_notify_comp ( __tskim ) ; \
} while ( 0 )
# define bfa_tskim_notify_comp(__tskim) do { \
if ( ( __tskim ) - > notify ) \
bfa_itnim_tskdone ( ( __tskim ) - > itnim ) ; \
} while ( 0 )
enum bfa_tskim_event {
BFA_TSKIM_SM_START = 1 , /* TM command start */
BFA_TSKIM_SM_DONE = 2 , /* TM completion */
BFA_TSKIM_SM_QRESUME = 3 , /* resume after qfull */
BFA_TSKIM_SM_HWFAIL = 5 , /* IOC h/w failure event */
BFA_TSKIM_SM_HCB = 6 , /* BFA callback completion */
BFA_TSKIM_SM_IOS_DONE = 7 , /* IO and sub TM completions */
BFA_TSKIM_SM_CLEANUP = 8 , /* TM cleanup on ITN offline */
BFA_TSKIM_SM_CLEANUP_DONE = 9 , /* TM abort completion */
} ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* forward declaration for BFA ITNIM functions
*/
static void bfa_itnim_iocdisable_cleanup ( struct bfa_itnim_s * itnim ) ;
static bfa_boolean_t bfa_itnim_send_fwcreate ( struct bfa_itnim_s * itnim ) ;
static bfa_boolean_t bfa_itnim_send_fwdelete ( struct bfa_itnim_s * itnim ) ;
static void bfa_itnim_cleanp_comp ( void * itnim_cbarg ) ;
static void bfa_itnim_cleanup ( struct bfa_itnim_s * itnim ) ;
static void __bfa_cb_itnim_online ( void * cbarg , bfa_boolean_t complete ) ;
static void __bfa_cb_itnim_offline ( void * cbarg , bfa_boolean_t complete ) ;
static void __bfa_cb_itnim_sler ( void * cbarg , bfa_boolean_t complete ) ;
static void bfa_itnim_iotov_online ( struct bfa_itnim_s * itnim ) ;
static void bfa_itnim_iotov_cleanup ( struct bfa_itnim_s * itnim ) ;
static void bfa_itnim_iotov ( void * itnim_arg ) ;
static void bfa_itnim_iotov_start ( struct bfa_itnim_s * itnim ) ;
static void bfa_itnim_iotov_stop ( struct bfa_itnim_s * itnim ) ;
static void bfa_itnim_iotov_delete ( struct bfa_itnim_s * itnim ) ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* forward declaration of ITNIM state machine
*/
static void bfa_itnim_sm_uninit ( struct bfa_itnim_s * itnim ,
enum bfa_itnim_event event ) ;
static void bfa_itnim_sm_created ( struct bfa_itnim_s * itnim ,
enum bfa_itnim_event event ) ;
static void bfa_itnim_sm_fwcreate ( struct bfa_itnim_s * itnim ,
enum bfa_itnim_event event ) ;
static void bfa_itnim_sm_delete_pending ( struct bfa_itnim_s * itnim ,
enum bfa_itnim_event event ) ;
static void bfa_itnim_sm_online ( struct bfa_itnim_s * itnim ,
enum bfa_itnim_event event ) ;
static void bfa_itnim_sm_sler ( struct bfa_itnim_s * itnim ,
enum bfa_itnim_event event ) ;
static void bfa_itnim_sm_cleanup_offline ( struct bfa_itnim_s * itnim ,
enum bfa_itnim_event event ) ;
static void bfa_itnim_sm_cleanup_delete ( struct bfa_itnim_s * itnim ,
enum bfa_itnim_event event ) ;
static void bfa_itnim_sm_fwdelete ( struct bfa_itnim_s * itnim ,
enum bfa_itnim_event event ) ;
static void bfa_itnim_sm_offline ( struct bfa_itnim_s * itnim ,
enum bfa_itnim_event event ) ;
static void bfa_itnim_sm_iocdisable ( struct bfa_itnim_s * itnim ,
enum bfa_itnim_event event ) ;
static void bfa_itnim_sm_deleting ( struct bfa_itnim_s * itnim ,
enum bfa_itnim_event event ) ;
static void bfa_itnim_sm_fwcreate_qfull ( struct bfa_itnim_s * itnim ,
enum bfa_itnim_event event ) ;
static void bfa_itnim_sm_fwdelete_qfull ( struct bfa_itnim_s * itnim ,
enum bfa_itnim_event event ) ;
static void bfa_itnim_sm_deleting_qfull ( struct bfa_itnim_s * itnim ,
enum bfa_itnim_event event ) ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* forward declaration for BFA IOIM functions
*/
static bfa_boolean_t bfa_ioim_send_ioreq ( struct bfa_ioim_s * ioim ) ;
2010-12-10 06:10:27 +03:00
static bfa_boolean_t bfa_ioim_sgpg_alloc ( struct bfa_ioim_s * ioim ) ;
2010-09-15 22:50:55 +04:00
static bfa_boolean_t bfa_ioim_send_abort ( struct bfa_ioim_s * ioim ) ;
static void bfa_ioim_notify_cleanup ( struct bfa_ioim_s * ioim ) ;
static void __bfa_cb_ioim_good_comp ( void * cbarg , bfa_boolean_t complete ) ;
static void __bfa_cb_ioim_comp ( void * cbarg , bfa_boolean_t complete ) ;
static void __bfa_cb_ioim_abort ( void * cbarg , bfa_boolean_t complete ) ;
static void __bfa_cb_ioim_failed ( void * cbarg , bfa_boolean_t complete ) ;
static void __bfa_cb_ioim_pathtov ( void * cbarg , bfa_boolean_t complete ) ;
static bfa_boolean_t bfa_ioim_is_abortable ( struct bfa_ioim_s * ioim ) ;
2011-07-21 04:04:03 +04:00
static void __bfa_cb_ioim_lm_lun_not_sup ( void * cbarg , bfa_boolean_t complete ) ;
static void __bfa_cb_ioim_lm_rpl_dc ( void * cbarg , bfa_boolean_t complete ) ;
static void __bfa_cb_ioim_lm_lun_not_rdy ( void * cbarg , bfa_boolean_t complete ) ;
2010-09-15 22:50:55 +04:00
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* forward declaration of BFA IO state machine
*/
static void bfa_ioim_sm_uninit ( struct bfa_ioim_s * ioim ,
enum bfa_ioim_event event ) ;
static void bfa_ioim_sm_sgalloc ( struct bfa_ioim_s * ioim ,
enum bfa_ioim_event event ) ;
static void bfa_ioim_sm_active ( struct bfa_ioim_s * ioim ,
enum bfa_ioim_event event ) ;
static void bfa_ioim_sm_abort ( struct bfa_ioim_s * ioim ,
enum bfa_ioim_event event ) ;
static void bfa_ioim_sm_cleanup ( struct bfa_ioim_s * ioim ,
enum bfa_ioim_event event ) ;
static void bfa_ioim_sm_qfull ( struct bfa_ioim_s * ioim ,
enum bfa_ioim_event event ) ;
static void bfa_ioim_sm_abort_qfull ( struct bfa_ioim_s * ioim ,
enum bfa_ioim_event event ) ;
static void bfa_ioim_sm_cleanup_qfull ( struct bfa_ioim_s * ioim ,
enum bfa_ioim_event event ) ;
static void bfa_ioim_sm_hcb ( struct bfa_ioim_s * ioim ,
enum bfa_ioim_event event ) ;
static void bfa_ioim_sm_hcb_free ( struct bfa_ioim_s * ioim ,
enum bfa_ioim_event event ) ;
static void bfa_ioim_sm_resfree ( struct bfa_ioim_s * ioim ,
enum bfa_ioim_event event ) ;
static void bfa_ioim_sm_cmnd_retry ( struct bfa_ioim_s * ioim ,
enum bfa_ioim_event event ) ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* forward declaration for BFA TSKIM functions
*/
static void __bfa_cb_tskim_done ( void * cbarg , bfa_boolean_t complete ) ;
static void __bfa_cb_tskim_failed ( void * cbarg , bfa_boolean_t complete ) ;
static bfa_boolean_t bfa_tskim_match_scope ( struct bfa_tskim_s * tskim ,
2010-12-10 06:11:39 +03:00
struct scsi_lun lun ) ;
2010-09-15 22:50:55 +04:00
static void bfa_tskim_gather_ios ( struct bfa_tskim_s * tskim ) ;
static void bfa_tskim_cleanp_comp ( void * tskim_cbarg ) ;
static void bfa_tskim_cleanup_ios ( struct bfa_tskim_s * tskim ) ;
static bfa_boolean_t bfa_tskim_send ( struct bfa_tskim_s * tskim ) ;
static bfa_boolean_t bfa_tskim_send_abort ( struct bfa_tskim_s * tskim ) ;
static void bfa_tskim_iocdisable_ios ( struct bfa_tskim_s * tskim ) ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* forward declaration of BFA TSKIM state machine
*/
static void bfa_tskim_sm_uninit ( struct bfa_tskim_s * tskim ,
enum bfa_tskim_event event ) ;
static void bfa_tskim_sm_active ( struct bfa_tskim_s * tskim ,
enum bfa_tskim_event event ) ;
static void bfa_tskim_sm_cleanup ( struct bfa_tskim_s * tskim ,
enum bfa_tskim_event event ) ;
static void bfa_tskim_sm_iocleanup ( struct bfa_tskim_s * tskim ,
enum bfa_tskim_event event ) ;
static void bfa_tskim_sm_qfull ( struct bfa_tskim_s * tskim ,
enum bfa_tskim_event event ) ;
static void bfa_tskim_sm_cleanup_qfull ( struct bfa_tskim_s * tskim ,
enum bfa_tskim_event event ) ;
static void bfa_tskim_sm_hcb ( struct bfa_tskim_s * tskim ,
enum bfa_tskim_event event ) ;
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:07:46 +03:00
* BFA FCP Initiator Mode module
2009-09-24 04:46:15 +04:00
*/
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* Compute and return memory needed by FCP ( im ) module .
2009-09-24 04:46:15 +04:00
*/
static void
2011-06-25 07:24:29 +04:00
bfa_fcpim_meminfo ( struct bfa_iocfc_cfg_s * cfg , u32 * km_len )
2009-09-24 04:46:15 +04:00
{
2011-06-25 07:24:29 +04:00
bfa_itnim_meminfo ( cfg , km_len ) ;
2009-09-24 04:46:15 +04:00
2010-10-19 04:17:23 +04:00
/*
2009-09-24 04:46:15 +04:00
* IO memory
*/
* km_len + = cfg - > fwcfg . num_ioim_reqs *
( sizeof ( struct bfa_ioim_s ) + sizeof ( struct bfa_ioim_sp_s ) ) ;
2010-10-19 04:17:23 +04:00
/*
2009-09-24 04:46:15 +04:00
* task management command memory
*/
if ( cfg - > fwcfg . num_tskim_reqs < BFA_TSKIM_MIN )
cfg - > fwcfg . num_tskim_reqs = BFA_TSKIM_MIN ;
* km_len + = cfg - > fwcfg . num_tskim_reqs * sizeof ( struct bfa_tskim_s ) ;
}
static void
2011-06-14 02:53:58 +04:00
bfa_fcpim_attach ( struct bfa_fcp_mod_s * fcp , void * bfad ,
2011-06-25 07:24:29 +04:00
struct bfa_iocfc_cfg_s * cfg , struct bfa_pcidev_s * pcidev )
2009-09-24 04:46:15 +04:00
{
2011-06-14 02:53:58 +04:00
struct bfa_fcpim_s * fcpim = & fcp - > fcpim ;
struct bfa_s * bfa = fcp - > bfa ;
2009-09-24 04:46:15 +04:00
bfa_trc ( bfa , cfg - > drvcfg . path_tov ) ;
bfa_trc ( bfa , cfg - > fwcfg . num_rports ) ;
bfa_trc ( bfa , cfg - > fwcfg . num_ioim_reqs ) ;
bfa_trc ( bfa , cfg - > fwcfg . num_tskim_reqs ) ;
2011-06-14 02:53:58 +04:00
fcpim - > fcp = fcp ;
2010-09-15 22:50:55 +04:00
fcpim - > bfa = bfa ;
fcpim - > num_itnims = cfg - > fwcfg . num_rports ;
2009-09-24 04:46:15 +04:00
fcpim - > num_tskim_reqs = cfg - > fwcfg . num_tskim_reqs ;
2010-09-15 22:50:55 +04:00
fcpim - > path_tov = cfg - > drvcfg . path_tov ;
fcpim - > delay_comp = cfg - > drvcfg . delay_comp ;
fcpim - > profile_comp = NULL ;
fcpim - > profile_start = NULL ;
2009-09-24 04:46:15 +04:00
2011-06-25 07:24:29 +04:00
bfa_itnim_attach ( fcpim ) ;
bfa_tskim_attach ( fcpim ) ;
bfa_ioim_attach ( fcpim ) ;
2009-09-24 04:46:15 +04:00
}
static void
2011-06-14 02:53:58 +04:00
bfa_fcpim_iocdisable ( struct bfa_fcp_mod_s * fcp )
2009-09-24 04:46:15 +04:00
{
2011-06-14 02:53:58 +04:00
struct bfa_fcpim_s * fcpim = & fcp - > fcpim ;
2009-09-24 04:46:15 +04:00
struct bfa_itnim_s * itnim ;
2010-09-15 22:50:55 +04:00
struct list_head * qe , * qen ;
2009-09-24 04:46:15 +04:00
2011-06-25 07:24:08 +04:00
/* Enqueue unused ioim resources to free_q */
list_splice_tail_init ( & fcpim - > tskim_unused_q , & fcpim - > tskim_free_q ) ;
2009-09-24 04:46:15 +04:00
list_for_each_safe ( qe , qen , & fcpim - > itnim_q ) {
itnim = ( struct bfa_itnim_s * ) qe ;
bfa_itnim_iocdisable ( itnim ) ;
}
}
void
bfa_fcpim_path_tov_set ( struct bfa_s * bfa , u16 path_tov )
{
2011-06-14 02:53:58 +04:00
struct bfa_fcpim_s * fcpim = BFA_FCPIM ( bfa ) ;
2009-09-24 04:46:15 +04:00
fcpim - > path_tov = path_tov * 1000 ;
if ( fcpim - > path_tov > BFA_FCPIM_PATHTOV_MAX )
fcpim - > path_tov = BFA_FCPIM_PATHTOV_MAX ;
}
u16
bfa_fcpim_path_tov_get ( struct bfa_s * bfa )
{
2011-06-14 02:53:58 +04:00
struct bfa_fcpim_s * fcpim = BFA_FCPIM ( bfa ) ;
2009-09-24 04:46:15 +04:00
2009-09-25 23:29:54 +04:00
return fcpim - > path_tov / 1000 ;
2009-09-24 04:46:15 +04:00
}
2011-06-25 07:25:15 +04:00
# define bfa_fcpim_add_iostats(__l, __r, __stats) \
( __l - > __stats + = __r - > __stats )
void
bfa_fcpim_add_stats ( struct bfa_itnim_iostats_s * lstats ,
struct bfa_itnim_iostats_s * rstats )
{
bfa_fcpim_add_iostats ( lstats , rstats , total_ios ) ;
bfa_fcpim_add_iostats ( lstats , rstats , qresumes ) ;
bfa_fcpim_add_iostats ( lstats , rstats , no_iotags ) ;
bfa_fcpim_add_iostats ( lstats , rstats , io_aborts ) ;
bfa_fcpim_add_iostats ( lstats , rstats , no_tskims ) ;
bfa_fcpim_add_iostats ( lstats , rstats , iocomp_ok ) ;
bfa_fcpim_add_iostats ( lstats , rstats , iocomp_underrun ) ;
bfa_fcpim_add_iostats ( lstats , rstats , iocomp_overrun ) ;
bfa_fcpim_add_iostats ( lstats , rstats , iocomp_aborted ) ;
bfa_fcpim_add_iostats ( lstats , rstats , iocomp_timedout ) ;
bfa_fcpim_add_iostats ( lstats , rstats , iocom_nexus_abort ) ;
bfa_fcpim_add_iostats ( lstats , rstats , iocom_proto_err ) ;
bfa_fcpim_add_iostats ( lstats , rstats , iocom_dif_err ) ;
bfa_fcpim_add_iostats ( lstats , rstats , iocom_sqer_needed ) ;
bfa_fcpim_add_iostats ( lstats , rstats , iocom_res_free ) ;
bfa_fcpim_add_iostats ( lstats , rstats , iocom_hostabrts ) ;
bfa_fcpim_add_iostats ( lstats , rstats , iocom_utags ) ;
bfa_fcpim_add_iostats ( lstats , rstats , io_cleanups ) ;
bfa_fcpim_add_iostats ( lstats , rstats , io_tmaborts ) ;
bfa_fcpim_add_iostats ( lstats , rstats , onlines ) ;
bfa_fcpim_add_iostats ( lstats , rstats , offlines ) ;
bfa_fcpim_add_iostats ( lstats , rstats , creates ) ;
bfa_fcpim_add_iostats ( lstats , rstats , deletes ) ;
bfa_fcpim_add_iostats ( lstats , rstats , create_comps ) ;
bfa_fcpim_add_iostats ( lstats , rstats , delete_comps ) ;
bfa_fcpim_add_iostats ( lstats , rstats , sler_events ) ;
bfa_fcpim_add_iostats ( lstats , rstats , fw_create ) ;
bfa_fcpim_add_iostats ( lstats , rstats , fw_delete ) ;
bfa_fcpim_add_iostats ( lstats , rstats , ioc_disabled ) ;
bfa_fcpim_add_iostats ( lstats , rstats , cleanup_comps ) ;
bfa_fcpim_add_iostats ( lstats , rstats , tm_cmnds ) ;
bfa_fcpim_add_iostats ( lstats , rstats , tm_fw_rsps ) ;
bfa_fcpim_add_iostats ( lstats , rstats , tm_success ) ;
bfa_fcpim_add_iostats ( lstats , rstats , tm_failures ) ;
bfa_fcpim_add_iostats ( lstats , rstats , tm_io_comps ) ;
bfa_fcpim_add_iostats ( lstats , rstats , tm_qresumes ) ;
bfa_fcpim_add_iostats ( lstats , rstats , tm_iocdowns ) ;
bfa_fcpim_add_iostats ( lstats , rstats , tm_cleanups ) ;
bfa_fcpim_add_iostats ( lstats , rstats , tm_cleanup_comps ) ;
bfa_fcpim_add_iostats ( lstats , rstats , io_comps ) ;
bfa_fcpim_add_iostats ( lstats , rstats , input_reqs ) ;
bfa_fcpim_add_iostats ( lstats , rstats , output_reqs ) ;
bfa_fcpim_add_iostats ( lstats , rstats , rd_throughput ) ;
bfa_fcpim_add_iostats ( lstats , rstats , wr_throughput ) ;
2011-07-21 04:04:03 +04:00
bfa_fcpim_add_iostats ( lstats , rstats , lm_lun_across_sg ) ;
bfa_fcpim_add_iostats ( lstats , rstats , lm_lun_not_sup ) ;
bfa_fcpim_add_iostats ( lstats , rstats , lm_rpl_data_changed ) ;
bfa_fcpim_add_iostats ( lstats , rstats , lm_wire_residue_changed ) ;
bfa_fcpim_add_iostats ( lstats , rstats , lm_small_buf_addresidue ) ;
bfa_fcpim_add_iostats ( lstats , rstats , lm_lun_not_rdy ) ;
2011-06-25 07:25:15 +04:00
}
bfa_status_t
bfa_fcpim_port_iostats ( struct bfa_s * bfa ,
struct bfa_itnim_iostats_s * stats , u8 lp_tag )
{
struct bfa_fcpim_s * fcpim = BFA_FCPIM ( bfa ) ;
struct list_head * qe , * qen ;
struct bfa_itnim_s * itnim ;
/* accumulate IO stats from itnim */
memset ( stats , 0 , sizeof ( struct bfa_itnim_iostats_s ) ) ;
list_for_each_safe ( qe , qen , & fcpim - > itnim_q ) {
itnim = ( struct bfa_itnim_s * ) qe ;
if ( itnim - > rport - > rport_info . lp_tag ! = lp_tag )
continue ;
bfa_fcpim_add_stats ( stats , & ( itnim - > stats ) ) ;
}
return BFA_STATUS_OK ;
}
2011-07-21 04:01:52 +04:00
void
bfa_ioim_profile_comp ( struct bfa_ioim_s * ioim )
{
struct bfa_itnim_latency_s * io_lat =
& ( ioim - > itnim - > ioprofile . io_latency ) ;
u32 val , idx ;
val = ( u32 ) ( jiffies - ioim - > start_time ) ;
idx = bfa_ioim_get_index ( scsi_bufflen ( ( struct scsi_cmnd * ) ioim - > dio ) ) ;
bfa_itnim_ioprofile_update ( ioim - > itnim , idx ) ;
io_lat - > count [ idx ] + + ;
io_lat - > min [ idx ] = ( io_lat - > min [ idx ] < val ) ? io_lat - > min [ idx ] : val ;
io_lat - > max [ idx ] = ( io_lat - > max [ idx ] > val ) ? io_lat - > max [ idx ] : val ;
io_lat - > avg [ idx ] + = val ;
}
void
bfa_ioim_profile_start ( struct bfa_ioim_s * ioim )
{
ioim - > start_time = jiffies ;
}
bfa_status_t
bfa_fcpim_profile_on ( struct bfa_s * bfa , u32 time )
{
struct bfa_itnim_s * itnim ;
struct bfa_fcpim_s * fcpim = BFA_FCPIM ( bfa ) ;
struct list_head * qe , * qen ;
/* accumulate IO stats from itnim */
list_for_each_safe ( qe , qen , & fcpim - > itnim_q ) {
itnim = ( struct bfa_itnim_s * ) qe ;
bfa_itnim_clear_stats ( itnim ) ;
}
fcpim - > io_profile = BFA_TRUE ;
fcpim - > io_profile_start_time = time ;
fcpim - > profile_comp = bfa_ioim_profile_comp ;
fcpim - > profile_start = bfa_ioim_profile_start ;
return BFA_STATUS_OK ;
}
bfa_status_t
bfa_fcpim_profile_off ( struct bfa_s * bfa )
{
struct bfa_fcpim_s * fcpim = BFA_FCPIM ( bfa ) ;
fcpim - > io_profile = BFA_FALSE ;
fcpim - > io_profile_start_time = 0 ;
fcpim - > profile_comp = NULL ;
fcpim - > profile_start = NULL ;
return BFA_STATUS_OK ;
}
2009-09-24 04:46:15 +04:00
u16
bfa_fcpim_qdepth_get ( struct bfa_s * bfa )
{
2011-06-14 02:53:58 +04:00
struct bfa_fcpim_s * fcpim = BFA_FCPIM ( bfa ) ;
2009-09-24 04:46:15 +04:00
2009-09-25 23:29:54 +04:00
return fcpim - > q_depth ;
2009-09-24 04:46:15 +04:00
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* BFA ITNIM module state machine functions
*/
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* Beginning / unallocated state - no events expected .
2010-09-15 22:50:55 +04:00
*/
static void
bfa_itnim_sm_uninit ( struct bfa_itnim_s * itnim , enum bfa_itnim_event event )
{
bfa_trc ( itnim - > bfa , itnim - > rport - > rport_tag ) ;
bfa_trc ( itnim - > bfa , event ) ;
switch ( event ) {
case BFA_ITNIM_SM_CREATE :
bfa_sm_set_state ( itnim , bfa_itnim_sm_created ) ;
itnim - > is_online = BFA_FALSE ;
bfa_fcpim_additn ( itnim ) ;
break ;
default :
bfa_sm_fault ( itnim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* Beginning state , only online event expected .
2010-09-15 22:50:55 +04:00
*/
static void
bfa_itnim_sm_created ( struct bfa_itnim_s * itnim , enum bfa_itnim_event event )
{
bfa_trc ( itnim - > bfa , itnim - > rport - > rport_tag ) ;
bfa_trc ( itnim - > bfa , event ) ;
switch ( event ) {
case BFA_ITNIM_SM_ONLINE :
if ( bfa_itnim_send_fwcreate ( itnim ) )
bfa_sm_set_state ( itnim , bfa_itnim_sm_fwcreate ) ;
else
bfa_sm_set_state ( itnim , bfa_itnim_sm_fwcreate_qfull ) ;
break ;
case BFA_ITNIM_SM_DELETE :
bfa_sm_set_state ( itnim , bfa_itnim_sm_uninit ) ;
bfa_fcpim_delitn ( itnim ) ;
break ;
case BFA_ITNIM_SM_HWFAIL :
bfa_sm_set_state ( itnim , bfa_itnim_sm_iocdisable ) ;
break ;
default :
bfa_sm_fault ( itnim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Waiting for itnim create response from firmware .
*/
static void
bfa_itnim_sm_fwcreate ( struct bfa_itnim_s * itnim , enum bfa_itnim_event event )
{
bfa_trc ( itnim - > bfa , itnim - > rport - > rport_tag ) ;
bfa_trc ( itnim - > bfa , event ) ;
switch ( event ) {
case BFA_ITNIM_SM_FWRSP :
bfa_sm_set_state ( itnim , bfa_itnim_sm_online ) ;
itnim - > is_online = BFA_TRUE ;
bfa_itnim_iotov_online ( itnim ) ;
bfa_itnim_online_cb ( itnim ) ;
break ;
case BFA_ITNIM_SM_DELETE :
bfa_sm_set_state ( itnim , bfa_itnim_sm_delete_pending ) ;
break ;
case BFA_ITNIM_SM_OFFLINE :
if ( bfa_itnim_send_fwdelete ( itnim ) )
bfa_sm_set_state ( itnim , bfa_itnim_sm_fwdelete ) ;
else
bfa_sm_set_state ( itnim , bfa_itnim_sm_fwdelete_qfull ) ;
break ;
case BFA_ITNIM_SM_HWFAIL :
bfa_sm_set_state ( itnim , bfa_itnim_sm_iocdisable ) ;
break ;
default :
bfa_sm_fault ( itnim - > bfa , event ) ;
}
}
static void
bfa_itnim_sm_fwcreate_qfull ( struct bfa_itnim_s * itnim ,
enum bfa_itnim_event event )
{
bfa_trc ( itnim - > bfa , itnim - > rport - > rport_tag ) ;
bfa_trc ( itnim - > bfa , event ) ;
switch ( event ) {
case BFA_ITNIM_SM_QRESUME :
bfa_sm_set_state ( itnim , bfa_itnim_sm_fwcreate ) ;
bfa_itnim_send_fwcreate ( itnim ) ;
break ;
case BFA_ITNIM_SM_DELETE :
bfa_sm_set_state ( itnim , bfa_itnim_sm_uninit ) ;
bfa_reqq_wcancel ( & itnim - > reqq_wait ) ;
bfa_fcpim_delitn ( itnim ) ;
break ;
case BFA_ITNIM_SM_OFFLINE :
bfa_sm_set_state ( itnim , bfa_itnim_sm_offline ) ;
bfa_reqq_wcancel ( & itnim - > reqq_wait ) ;
bfa_itnim_offline_cb ( itnim ) ;
break ;
case BFA_ITNIM_SM_HWFAIL :
bfa_sm_set_state ( itnim , bfa_itnim_sm_iocdisable ) ;
bfa_reqq_wcancel ( & itnim - > reqq_wait ) ;
break ;
default :
bfa_sm_fault ( itnim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* Waiting for itnim create response from firmware , a delete is pending .
2010-09-15 22:50:55 +04:00
*/
static void
bfa_itnim_sm_delete_pending ( struct bfa_itnim_s * itnim ,
enum bfa_itnim_event event )
{
bfa_trc ( itnim - > bfa , itnim - > rport - > rport_tag ) ;
bfa_trc ( itnim - > bfa , event ) ;
switch ( event ) {
case BFA_ITNIM_SM_FWRSP :
if ( bfa_itnim_send_fwdelete ( itnim ) )
bfa_sm_set_state ( itnim , bfa_itnim_sm_deleting ) ;
else
bfa_sm_set_state ( itnim , bfa_itnim_sm_deleting_qfull ) ;
break ;
case BFA_ITNIM_SM_HWFAIL :
bfa_sm_set_state ( itnim , bfa_itnim_sm_uninit ) ;
bfa_fcpim_delitn ( itnim ) ;
break ;
default :
bfa_sm_fault ( itnim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* Online state - normal parking state .
2010-09-15 22:50:55 +04:00
*/
static void
bfa_itnim_sm_online ( struct bfa_itnim_s * itnim , enum bfa_itnim_event event )
{
bfa_trc ( itnim - > bfa , itnim - > rport - > rport_tag ) ;
bfa_trc ( itnim - > bfa , event ) ;
switch ( event ) {
case BFA_ITNIM_SM_OFFLINE :
bfa_sm_set_state ( itnim , bfa_itnim_sm_cleanup_offline ) ;
itnim - > is_online = BFA_FALSE ;
bfa_itnim_iotov_start ( itnim ) ;
bfa_itnim_cleanup ( itnim ) ;
break ;
case BFA_ITNIM_SM_DELETE :
bfa_sm_set_state ( itnim , bfa_itnim_sm_cleanup_delete ) ;
itnim - > is_online = BFA_FALSE ;
bfa_itnim_cleanup ( itnim ) ;
break ;
case BFA_ITNIM_SM_SLER :
bfa_sm_set_state ( itnim , bfa_itnim_sm_sler ) ;
itnim - > is_online = BFA_FALSE ;
bfa_itnim_iotov_start ( itnim ) ;
bfa_itnim_sler_cb ( itnim ) ;
break ;
case BFA_ITNIM_SM_HWFAIL :
bfa_sm_set_state ( itnim , bfa_itnim_sm_iocdisable ) ;
itnim - > is_online = BFA_FALSE ;
bfa_itnim_iotov_start ( itnim ) ;
bfa_itnim_iocdisable_cleanup ( itnim ) ;
break ;
default :
bfa_sm_fault ( itnim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* Second level error recovery need .
2010-09-15 22:50:55 +04:00
*/
static void
bfa_itnim_sm_sler ( struct bfa_itnim_s * itnim , enum bfa_itnim_event event )
{
bfa_trc ( itnim - > bfa , itnim - > rport - > rport_tag ) ;
bfa_trc ( itnim - > bfa , event ) ;
switch ( event ) {
case BFA_ITNIM_SM_OFFLINE :
bfa_sm_set_state ( itnim , bfa_itnim_sm_cleanup_offline ) ;
bfa_itnim_cleanup ( itnim ) ;
break ;
case BFA_ITNIM_SM_DELETE :
bfa_sm_set_state ( itnim , bfa_itnim_sm_cleanup_delete ) ;
bfa_itnim_cleanup ( itnim ) ;
bfa_itnim_iotov_delete ( itnim ) ;
break ;
case BFA_ITNIM_SM_HWFAIL :
bfa_sm_set_state ( itnim , bfa_itnim_sm_iocdisable ) ;
bfa_itnim_iocdisable_cleanup ( itnim ) ;
break ;
default :
bfa_sm_fault ( itnim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* Going offline . Waiting for active IO cleanup .
2010-09-15 22:50:55 +04:00
*/
static void
bfa_itnim_sm_cleanup_offline ( struct bfa_itnim_s * itnim ,
enum bfa_itnim_event event )
{
bfa_trc ( itnim - > bfa , itnim - > rport - > rport_tag ) ;
bfa_trc ( itnim - > bfa , event ) ;
switch ( event ) {
case BFA_ITNIM_SM_CLEANUP :
if ( bfa_itnim_send_fwdelete ( itnim ) )
bfa_sm_set_state ( itnim , bfa_itnim_sm_fwdelete ) ;
else
bfa_sm_set_state ( itnim , bfa_itnim_sm_fwdelete_qfull ) ;
break ;
case BFA_ITNIM_SM_DELETE :
bfa_sm_set_state ( itnim , bfa_itnim_sm_cleanup_delete ) ;
bfa_itnim_iotov_delete ( itnim ) ;
break ;
case BFA_ITNIM_SM_HWFAIL :
bfa_sm_set_state ( itnim , bfa_itnim_sm_iocdisable ) ;
bfa_itnim_iocdisable_cleanup ( itnim ) ;
bfa_itnim_offline_cb ( itnim ) ;
break ;
case BFA_ITNIM_SM_SLER :
break ;
default :
bfa_sm_fault ( itnim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* Deleting itnim . Waiting for active IO cleanup .
2010-09-15 22:50:55 +04:00
*/
static void
bfa_itnim_sm_cleanup_delete ( struct bfa_itnim_s * itnim ,
enum bfa_itnim_event event )
{
bfa_trc ( itnim - > bfa , itnim - > rport - > rport_tag ) ;
bfa_trc ( itnim - > bfa , event ) ;
switch ( event ) {
case BFA_ITNIM_SM_CLEANUP :
if ( bfa_itnim_send_fwdelete ( itnim ) )
bfa_sm_set_state ( itnim , bfa_itnim_sm_deleting ) ;
else
bfa_sm_set_state ( itnim , bfa_itnim_sm_deleting_qfull ) ;
break ;
case BFA_ITNIM_SM_HWFAIL :
bfa_sm_set_state ( itnim , bfa_itnim_sm_iocdisable ) ;
bfa_itnim_iocdisable_cleanup ( itnim ) ;
break ;
default :
bfa_sm_fault ( itnim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Rport offline . Fimrware itnim is being deleted - awaiting f / w response .
*/
static void
bfa_itnim_sm_fwdelete ( struct bfa_itnim_s * itnim , enum bfa_itnim_event event )
{
bfa_trc ( itnim - > bfa , itnim - > rport - > rport_tag ) ;
bfa_trc ( itnim - > bfa , event ) ;
switch ( event ) {
case BFA_ITNIM_SM_FWRSP :
bfa_sm_set_state ( itnim , bfa_itnim_sm_offline ) ;
bfa_itnim_offline_cb ( itnim ) ;
break ;
case BFA_ITNIM_SM_DELETE :
bfa_sm_set_state ( itnim , bfa_itnim_sm_deleting ) ;
break ;
case BFA_ITNIM_SM_HWFAIL :
bfa_sm_set_state ( itnim , bfa_itnim_sm_iocdisable ) ;
bfa_itnim_offline_cb ( itnim ) ;
break ;
default :
bfa_sm_fault ( itnim - > bfa , event ) ;
}
}
static void
bfa_itnim_sm_fwdelete_qfull ( struct bfa_itnim_s * itnim ,
enum bfa_itnim_event event )
{
bfa_trc ( itnim - > bfa , itnim - > rport - > rport_tag ) ;
bfa_trc ( itnim - > bfa , event ) ;
switch ( event ) {
case BFA_ITNIM_SM_QRESUME :
bfa_sm_set_state ( itnim , bfa_itnim_sm_fwdelete ) ;
bfa_itnim_send_fwdelete ( itnim ) ;
break ;
case BFA_ITNIM_SM_DELETE :
bfa_sm_set_state ( itnim , bfa_itnim_sm_deleting_qfull ) ;
break ;
case BFA_ITNIM_SM_HWFAIL :
bfa_sm_set_state ( itnim , bfa_itnim_sm_iocdisable ) ;
bfa_reqq_wcancel ( & itnim - > reqq_wait ) ;
bfa_itnim_offline_cb ( itnim ) ;
break ;
default :
bfa_sm_fault ( itnim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* Offline state .
2010-09-15 22:50:55 +04:00
*/
static void
bfa_itnim_sm_offline ( struct bfa_itnim_s * itnim , enum bfa_itnim_event event )
{
bfa_trc ( itnim - > bfa , itnim - > rport - > rport_tag ) ;
bfa_trc ( itnim - > bfa , event ) ;
switch ( event ) {
case BFA_ITNIM_SM_DELETE :
bfa_sm_set_state ( itnim , bfa_itnim_sm_uninit ) ;
bfa_itnim_iotov_delete ( itnim ) ;
bfa_fcpim_delitn ( itnim ) ;
break ;
case BFA_ITNIM_SM_ONLINE :
if ( bfa_itnim_send_fwcreate ( itnim ) )
bfa_sm_set_state ( itnim , bfa_itnim_sm_fwcreate ) ;
else
bfa_sm_set_state ( itnim , bfa_itnim_sm_fwcreate_qfull ) ;
break ;
case BFA_ITNIM_SM_HWFAIL :
bfa_sm_set_state ( itnim , bfa_itnim_sm_iocdisable ) ;
break ;
default :
bfa_sm_fault ( itnim - > bfa , event ) ;
}
}
static void
bfa_itnim_sm_iocdisable ( struct bfa_itnim_s * itnim ,
enum bfa_itnim_event event )
{
bfa_trc ( itnim - > bfa , itnim - > rport - > rport_tag ) ;
bfa_trc ( itnim - > bfa , event ) ;
switch ( event ) {
case BFA_ITNIM_SM_DELETE :
bfa_sm_set_state ( itnim , bfa_itnim_sm_uninit ) ;
bfa_itnim_iotov_delete ( itnim ) ;
bfa_fcpim_delitn ( itnim ) ;
break ;
case BFA_ITNIM_SM_OFFLINE :
bfa_itnim_offline_cb ( itnim ) ;
break ;
case BFA_ITNIM_SM_ONLINE :
if ( bfa_itnim_send_fwcreate ( itnim ) )
bfa_sm_set_state ( itnim , bfa_itnim_sm_fwcreate ) ;
else
bfa_sm_set_state ( itnim , bfa_itnim_sm_fwcreate_qfull ) ;
break ;
case BFA_ITNIM_SM_HWFAIL :
break ;
default :
bfa_sm_fault ( itnim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* Itnim is deleted , waiting for firmware response to delete .
2010-09-15 22:50:55 +04:00
*/
static void
bfa_itnim_sm_deleting ( struct bfa_itnim_s * itnim , enum bfa_itnim_event event )
{
bfa_trc ( itnim - > bfa , itnim - > rport - > rport_tag ) ;
bfa_trc ( itnim - > bfa , event ) ;
switch ( event ) {
case BFA_ITNIM_SM_FWRSP :
case BFA_ITNIM_SM_HWFAIL :
bfa_sm_set_state ( itnim , bfa_itnim_sm_uninit ) ;
bfa_fcpim_delitn ( itnim ) ;
break ;
default :
bfa_sm_fault ( itnim - > bfa , event ) ;
}
}
static void
bfa_itnim_sm_deleting_qfull ( struct bfa_itnim_s * itnim ,
enum bfa_itnim_event event )
{
bfa_trc ( itnim - > bfa , itnim - > rport - > rport_tag ) ;
bfa_trc ( itnim - > bfa , event ) ;
switch ( event ) {
case BFA_ITNIM_SM_QRESUME :
bfa_sm_set_state ( itnim , bfa_itnim_sm_deleting ) ;
bfa_itnim_send_fwdelete ( itnim ) ;
break ;
case BFA_ITNIM_SM_HWFAIL :
bfa_sm_set_state ( itnim , bfa_itnim_sm_uninit ) ;
bfa_reqq_wcancel ( & itnim - > reqq_wait ) ;
bfa_fcpim_delitn ( itnim ) ;
break ;
default :
bfa_sm_fault ( itnim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* Initiate cleanup of all IOs on an IOC failure .
2010-09-15 22:50:55 +04:00
*/
static void
bfa_itnim_iocdisable_cleanup ( struct bfa_itnim_s * itnim )
{
struct bfa_tskim_s * tskim ;
struct bfa_ioim_s * ioim ;
struct list_head * qe , * qen ;
list_for_each_safe ( qe , qen , & itnim - > tsk_q ) {
tskim = ( struct bfa_tskim_s * ) qe ;
bfa_tskim_iocdisable ( tskim ) ;
}
list_for_each_safe ( qe , qen , & itnim - > io_q ) {
ioim = ( struct bfa_ioim_s * ) qe ;
bfa_ioim_iocdisable ( ioim ) ;
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* For IO request in pending queue , we pretend an early timeout .
*/
list_for_each_safe ( qe , qen , & itnim - > pending_q ) {
ioim = ( struct bfa_ioim_s * ) qe ;
bfa_ioim_tov ( ioim ) ;
}
list_for_each_safe ( qe , qen , & itnim - > io_cleanup_q ) {
ioim = ( struct bfa_ioim_s * ) qe ;
bfa_ioim_iocdisable ( ioim ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* IO cleanup completion
2010-09-15 22:50:55 +04:00
*/
static void
bfa_itnim_cleanp_comp ( void * itnim_cbarg )
{
struct bfa_itnim_s * itnim = itnim_cbarg ;
bfa_stats ( itnim , cleanup_comps ) ;
bfa_sm_send_event ( itnim , BFA_ITNIM_SM_CLEANUP ) ;
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* Initiate cleanup of all IOs .
2010-09-15 22:50:55 +04:00
*/
static void
bfa_itnim_cleanup ( struct bfa_itnim_s * itnim )
{
struct bfa_ioim_s * ioim ;
struct bfa_tskim_s * tskim ;
struct list_head * qe , * qen ;
bfa_wc_init ( & itnim - > wc , bfa_itnim_cleanp_comp , itnim ) ;
list_for_each_safe ( qe , qen , & itnim - > io_q ) {
ioim = ( struct bfa_ioim_s * ) qe ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Move IO to a cleanup queue from active queue so that a later
* TM will not pickup this IO .
*/
list_del ( & ioim - > qe ) ;
list_add_tail ( & ioim - > qe , & itnim - > io_cleanup_q ) ;
bfa_wc_up ( & itnim - > wc ) ;
bfa_ioim_cleanup ( ioim ) ;
}
list_for_each_safe ( qe , qen , & itnim - > tsk_q ) {
tskim = ( struct bfa_tskim_s * ) qe ;
bfa_wc_up ( & itnim - > wc ) ;
bfa_tskim_cleanup ( tskim ) ;
}
bfa_wc_wait ( & itnim - > wc ) ;
}
static void
__bfa_cb_itnim_online ( void * cbarg , bfa_boolean_t complete )
{
struct bfa_itnim_s * itnim = cbarg ;
if ( complete )
bfa_cb_itnim_online ( itnim - > ditn ) ;
}
static void
__bfa_cb_itnim_offline ( void * cbarg , bfa_boolean_t complete )
{
struct bfa_itnim_s * itnim = cbarg ;
if ( complete )
bfa_cb_itnim_offline ( itnim - > ditn ) ;
}
static void
__bfa_cb_itnim_sler ( void * cbarg , bfa_boolean_t complete )
{
struct bfa_itnim_s * itnim = cbarg ;
if ( complete )
bfa_cb_itnim_sler ( itnim - > ditn ) ;
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Call to resume any I / O requests waiting for room in request queue .
*/
static void
bfa_itnim_qresume ( void * cbarg )
{
struct bfa_itnim_s * itnim = cbarg ;
bfa_sm_send_event ( itnim , BFA_ITNIM_SM_QRESUME ) ;
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* bfa_itnim_public
*/
void
bfa_itnim_iodone ( struct bfa_itnim_s * itnim )
{
bfa_wc_down ( & itnim - > wc ) ;
}
void
bfa_itnim_tskdone ( struct bfa_itnim_s * itnim )
{
bfa_wc_down ( & itnim - > wc ) ;
}
void
2011-06-25 07:24:29 +04:00
bfa_itnim_meminfo ( struct bfa_iocfc_cfg_s * cfg , u32 * km_len )
2010-09-15 22:50:55 +04:00
{
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* ITN memory
*/
* km_len + = cfg - > fwcfg . num_rports * sizeof ( struct bfa_itnim_s ) ;
}
void
2011-06-25 07:24:29 +04:00
bfa_itnim_attach ( struct bfa_fcpim_s * fcpim )
2010-09-15 22:50:55 +04:00
{
struct bfa_s * bfa = fcpim - > bfa ;
2011-06-25 07:24:29 +04:00
struct bfa_fcp_mod_s * fcp = fcpim - > fcp ;
2010-09-15 22:50:55 +04:00
struct bfa_itnim_s * itnim ;
int i , j ;
INIT_LIST_HEAD ( & fcpim - > itnim_q ) ;
2011-06-25 07:24:29 +04:00
itnim = ( struct bfa_itnim_s * ) bfa_mem_kva_curp ( fcp ) ;
2010-09-15 22:50:55 +04:00
fcpim - > itnim_arr = itnim ;
for ( i = 0 ; i < fcpim - > num_itnims ; i + + , itnim + + ) {
2010-10-19 04:08:54 +04:00
memset ( itnim , 0 , sizeof ( struct bfa_itnim_s ) ) ;
2010-09-15 22:50:55 +04:00
itnim - > bfa = bfa ;
itnim - > fcpim = fcpim ;
itnim - > reqq = BFA_REQQ_QOS_LO ;
itnim - > rport = BFA_RPORT_FROM_TAG ( bfa , i ) ;
itnim - > iotov_active = BFA_FALSE ;
bfa_reqq_winit ( & itnim - > reqq_wait , bfa_itnim_qresume , itnim ) ;
INIT_LIST_HEAD ( & itnim - > io_q ) ;
INIT_LIST_HEAD ( & itnim - > io_cleanup_q ) ;
INIT_LIST_HEAD ( & itnim - > pending_q ) ;
INIT_LIST_HEAD ( & itnim - > tsk_q ) ;
INIT_LIST_HEAD ( & itnim - > delay_comp_q ) ;
for ( j = 0 ; j < BFA_IOBUCKET_MAX ; j + + )
itnim - > ioprofile . io_latency . min [ j ] = ~ 0 ;
bfa_sm_set_state ( itnim , bfa_itnim_sm_uninit ) ;
}
2011-06-25 07:24:29 +04:00
bfa_mem_kva_curp ( fcp ) = ( u8 * ) itnim ;
2010-09-15 22:50:55 +04:00
}
void
bfa_itnim_iocdisable ( struct bfa_itnim_s * itnim )
{
bfa_stats ( itnim , ioc_disabled ) ;
bfa_sm_send_event ( itnim , BFA_ITNIM_SM_HWFAIL ) ;
}
static bfa_boolean_t
bfa_itnim_send_fwcreate ( struct bfa_itnim_s * itnim )
{
2011-06-14 02:51:24 +04:00
struct bfi_itn_create_req_s * m ;
2010-09-15 22:50:55 +04:00
itnim - > msg_no + + ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* check for room in queue to send request now
*/
m = bfa_reqq_next ( itnim - > bfa , itnim - > reqq ) ;
if ( ! m ) {
bfa_reqq_wait ( itnim - > bfa , itnim - > reqq , & itnim - > reqq_wait ) ;
return BFA_FALSE ;
}
2011-06-14 02:51:24 +04:00
bfi_h2i_set ( m - > mh , BFI_MC_ITN , BFI_ITN_H2I_CREATE_REQ ,
2011-06-25 07:24:08 +04:00
bfa_fn_lpu ( itnim - > bfa ) ) ;
2010-09-15 22:50:55 +04:00
m - > fw_handle = itnim - > rport - > fw_handle ;
m - > class = FC_CLASS_3 ;
m - > seq_rec = itnim - > seq_rec ;
m - > msg_no = itnim - > msg_no ;
bfa_stats ( itnim , fw_create ) ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* queue I / O message to firmware
*/
2011-06-25 07:24:08 +04:00
bfa_reqq_produce ( itnim - > bfa , itnim - > reqq , m - > mh ) ;
2010-09-15 22:50:55 +04:00
return BFA_TRUE ;
}
static bfa_boolean_t
bfa_itnim_send_fwdelete ( struct bfa_itnim_s * itnim )
{
2011-06-14 02:51:24 +04:00
struct bfi_itn_delete_req_s * m ;
2010-09-15 22:50:55 +04:00
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* check for room in queue to send request now
*/
m = bfa_reqq_next ( itnim - > bfa , itnim - > reqq ) ;
if ( ! m ) {
bfa_reqq_wait ( itnim - > bfa , itnim - > reqq , & itnim - > reqq_wait ) ;
return BFA_FALSE ;
}
2011-06-14 02:51:24 +04:00
bfi_h2i_set ( m - > mh , BFI_MC_ITN , BFI_ITN_H2I_DELETE_REQ ,
2011-06-25 07:24:08 +04:00
bfa_fn_lpu ( itnim - > bfa ) ) ;
2010-09-15 22:50:55 +04:00
m - > fw_handle = itnim - > rport - > fw_handle ;
bfa_stats ( itnim , fw_delete ) ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* queue I / O message to firmware
*/
2011-06-25 07:24:08 +04:00
bfa_reqq_produce ( itnim - > bfa , itnim - > reqq , m - > mh ) ;
2010-09-15 22:50:55 +04:00
return BFA_TRUE ;
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Cleanup all pending failed inflight requests .
*/
static void
bfa_itnim_delayed_comp ( struct bfa_itnim_s * itnim , bfa_boolean_t iotov )
{
struct bfa_ioim_s * ioim ;
struct list_head * qe , * qen ;
list_for_each_safe ( qe , qen , & itnim - > delay_comp_q ) {
ioim = ( struct bfa_ioim_s * ) qe ;
bfa_ioim_delayed_comp ( ioim , iotov ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Start all pending IO requests .
*/
static void
bfa_itnim_iotov_online ( struct bfa_itnim_s * itnim )
{
struct bfa_ioim_s * ioim ;
bfa_itnim_iotov_stop ( itnim ) ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Abort all inflight IO requests in the queue
*/
bfa_itnim_delayed_comp ( itnim , BFA_FALSE ) ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Start all pending IO requests .
*/
while ( ! list_empty ( & itnim - > pending_q ) ) {
bfa_q_deq ( & itnim - > pending_q , & ioim ) ;
list_add_tail ( & ioim - > qe , & itnim - > io_q ) ;
bfa_ioim_start ( ioim ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Fail all pending IO requests
*/
static void
bfa_itnim_iotov_cleanup ( struct bfa_itnim_s * itnim )
{
struct bfa_ioim_s * ioim ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Fail all inflight IO requests in the queue
*/
bfa_itnim_delayed_comp ( itnim , BFA_TRUE ) ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Fail any pending IO requests .
*/
while ( ! list_empty ( & itnim - > pending_q ) ) {
bfa_q_deq ( & itnim - > pending_q , & ioim ) ;
list_add_tail ( & ioim - > qe , & ioim - > fcpim - > ioim_comp_q ) ;
bfa_ioim_tov ( ioim ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* IO TOV timer callback . Fail any pending IO requests .
*/
static void
bfa_itnim_iotov ( void * itnim_arg )
{
struct bfa_itnim_s * itnim = itnim_arg ;
itnim - > iotov_active = BFA_FALSE ;
bfa_cb_itnim_tov_begin ( itnim - > ditn ) ;
bfa_itnim_iotov_cleanup ( itnim ) ;
bfa_cb_itnim_tov ( itnim - > ditn ) ;
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Start IO TOV timer for failing back pending IO requests in offline state .
*/
static void
bfa_itnim_iotov_start ( struct bfa_itnim_s * itnim )
{
if ( itnim - > fcpim - > path_tov > 0 ) {
itnim - > iotov_active = BFA_TRUE ;
2010-12-27 08:46:35 +03:00
WARN_ON ( ! bfa_itnim_hold_io ( itnim ) ) ;
2010-09-15 22:50:55 +04:00
bfa_timer_start ( itnim - > bfa , & itnim - > timer ,
bfa_itnim_iotov , itnim , itnim - > fcpim - > path_tov ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Stop IO TOV timer .
*/
static void
bfa_itnim_iotov_stop ( struct bfa_itnim_s * itnim )
{
if ( itnim - > iotov_active ) {
itnim - > iotov_active = BFA_FALSE ;
bfa_timer_stop ( & itnim - > timer ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Stop IO TOV timer .
*/
static void
bfa_itnim_iotov_delete ( struct bfa_itnim_s * itnim )
{
bfa_boolean_t pathtov_active = BFA_FALSE ;
if ( itnim - > iotov_active )
pathtov_active = BFA_TRUE ;
bfa_itnim_iotov_stop ( itnim ) ;
if ( pathtov_active )
bfa_cb_itnim_tov_begin ( itnim - > ditn ) ;
bfa_itnim_iotov_cleanup ( itnim ) ;
if ( pathtov_active )
bfa_cb_itnim_tov ( itnim - > ditn ) ;
}
static void
bfa_itnim_update_del_itn_stats ( struct bfa_itnim_s * itnim )
{
2011-06-14 02:53:58 +04:00
struct bfa_fcpim_s * fcpim = BFA_FCPIM ( itnim - > bfa ) ;
2010-09-15 22:50:55 +04:00
fcpim - > del_itn_stats . del_itn_iocomp_aborted + =
itnim - > stats . iocomp_aborted ;
fcpim - > del_itn_stats . del_itn_iocomp_timedout + =
itnim - > stats . iocomp_timedout ;
fcpim - > del_itn_stats . del_itn_iocom_sqer_needed + =
itnim - > stats . iocom_sqer_needed ;
fcpim - > del_itn_stats . del_itn_iocom_res_free + =
itnim - > stats . iocom_res_free ;
fcpim - > del_itn_stats . del_itn_iocom_hostabrts + =
itnim - > stats . iocom_hostabrts ;
fcpim - > del_itn_stats . del_itn_total_ios + = itnim - > stats . total_ios ;
fcpim - > del_itn_stats . del_io_iocdowns + = itnim - > stats . io_iocdowns ;
fcpim - > del_itn_stats . del_tm_iocdowns + = itnim - > stats . tm_iocdowns ;
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* bfa_itnim_public
2010-09-15 22:50:55 +04:00
*/
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* Itnim interrupt processing .
2010-09-15 22:50:55 +04:00
*/
void
bfa_itnim_isr ( struct bfa_s * bfa , struct bfi_msg_s * m )
{
2011-06-14 02:53:58 +04:00
struct bfa_fcpim_s * fcpim = BFA_FCPIM ( bfa ) ;
2011-06-14 02:51:24 +04:00
union bfi_itn_i2h_msg_u msg ;
2010-09-15 22:50:55 +04:00
struct bfa_itnim_s * itnim ;
bfa_trc ( bfa , m - > mhdr . msg_id ) ;
msg . msg = m ;
switch ( m - > mhdr . msg_id ) {
2011-06-14 02:51:24 +04:00
case BFI_ITN_I2H_CREATE_RSP :
2010-09-15 22:50:55 +04:00
itnim = BFA_ITNIM_FROM_TAG ( fcpim ,
msg . create_rsp - > bfa_handle ) ;
2010-12-27 08:46:35 +03:00
WARN_ON ( msg . create_rsp - > status ! = BFA_STATUS_OK ) ;
2010-09-15 22:50:55 +04:00
bfa_stats ( itnim , create_comps ) ;
bfa_sm_send_event ( itnim , BFA_ITNIM_SM_FWRSP ) ;
break ;
2011-06-14 02:51:24 +04:00
case BFI_ITN_I2H_DELETE_RSP :
2010-09-15 22:50:55 +04:00
itnim = BFA_ITNIM_FROM_TAG ( fcpim ,
msg . delete_rsp - > bfa_handle ) ;
2010-12-27 08:46:35 +03:00
WARN_ON ( msg . delete_rsp - > status ! = BFA_STATUS_OK ) ;
2010-09-15 22:50:55 +04:00
bfa_stats ( itnim , delete_comps ) ;
bfa_sm_send_event ( itnim , BFA_ITNIM_SM_FWRSP ) ;
break ;
2011-06-14 02:51:24 +04:00
case BFI_ITN_I2H_SLER_EVENT :
2010-09-15 22:50:55 +04:00
itnim = BFA_ITNIM_FROM_TAG ( fcpim ,
msg . sler_event - > bfa_handle ) ;
bfa_stats ( itnim , sler_events ) ;
bfa_sm_send_event ( itnim , BFA_ITNIM_SM_SLER ) ;
break ;
default :
bfa_trc ( bfa , m - > mhdr . msg_id ) ;
2010-12-27 08:46:35 +03:00
WARN_ON ( 1 ) ;
2010-09-15 22:50:55 +04:00
}
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* bfa_itnim_api
2010-09-15 22:50:55 +04:00
*/
struct bfa_itnim_s *
bfa_itnim_create ( struct bfa_s * bfa , struct bfa_rport_s * rport , void * ditn )
{
2011-06-14 02:53:58 +04:00
struct bfa_fcpim_s * fcpim = BFA_FCPIM ( bfa ) ;
2010-09-15 22:50:55 +04:00
struct bfa_itnim_s * itnim ;
2011-06-14 02:53:58 +04:00
bfa_itn_create ( bfa , rport , bfa_itnim_isr ) ;
2010-09-15 22:50:55 +04:00
itnim = BFA_ITNIM_FROM_TAG ( fcpim , rport - > rport_tag ) ;
2010-12-27 08:46:35 +03:00
WARN_ON ( itnim - > rport ! = rport ) ;
2010-09-15 22:50:55 +04:00
itnim - > ditn = ditn ;
bfa_stats ( itnim , creates ) ;
bfa_sm_send_event ( itnim , BFA_ITNIM_SM_CREATE ) ;
return itnim ;
}
void
bfa_itnim_delete ( struct bfa_itnim_s * itnim )
{
bfa_stats ( itnim , deletes ) ;
bfa_sm_send_event ( itnim , BFA_ITNIM_SM_DELETE ) ;
}
void
bfa_itnim_online ( struct bfa_itnim_s * itnim , bfa_boolean_t seq_rec )
{
itnim - > seq_rec = seq_rec ;
bfa_stats ( itnim , onlines ) ;
bfa_sm_send_event ( itnim , BFA_ITNIM_SM_ONLINE ) ;
}
void
bfa_itnim_offline ( struct bfa_itnim_s * itnim )
{
bfa_stats ( itnim , offlines ) ;
bfa_sm_send_event ( itnim , BFA_ITNIM_SM_OFFLINE ) ;
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Return true if itnim is considered offline for holding off IO request .
* IO is not held if itnim is being deleted .
*/
bfa_boolean_t
bfa_itnim_hold_io ( struct bfa_itnim_s * itnim )
{
return itnim - > fcpim - > path_tov & & itnim - > iotov_active & &
( bfa_sm_cmp_state ( itnim , bfa_itnim_sm_fwcreate ) | |
bfa_sm_cmp_state ( itnim , bfa_itnim_sm_sler ) | |
bfa_sm_cmp_state ( itnim , bfa_itnim_sm_cleanup_offline ) | |
bfa_sm_cmp_state ( itnim , bfa_itnim_sm_fwdelete ) | |
bfa_sm_cmp_state ( itnim , bfa_itnim_sm_offline ) | |
bfa_sm_cmp_state ( itnim , bfa_itnim_sm_iocdisable ) ) ;
}
2011-07-21 04:01:52 +04:00
# define bfa_io_lat_clock_res_div HZ
# define bfa_io_lat_clock_res_mul 1000
bfa_status_t
bfa_itnim_get_ioprofile ( struct bfa_itnim_s * itnim ,
struct bfa_itnim_ioprofile_s * ioprofile )
{
struct bfa_fcpim_s * fcpim = BFA_FCPIM ( itnim - > bfa ) ;
if ( ! fcpim - > io_profile )
return BFA_STATUS_IOPROFILE_OFF ;
itnim - > ioprofile . index = BFA_IOBUCKET_MAX ;
itnim - > ioprofile . io_profile_start_time =
bfa_io_profile_start_time ( itnim - > bfa ) ;
itnim - > ioprofile . clock_res_mul = bfa_io_lat_clock_res_mul ;
itnim - > ioprofile . clock_res_div = bfa_io_lat_clock_res_div ;
* ioprofile = itnim - > ioprofile ;
return BFA_STATUS_OK ;
}
2010-09-15 22:50:55 +04:00
void
bfa_itnim_clear_stats ( struct bfa_itnim_s * itnim )
{
int j ;
2010-10-19 04:08:54 +04:00
memset ( & itnim - > stats , 0 , sizeof ( itnim - > stats ) ) ;
memset ( & itnim - > ioprofile , 0 , sizeof ( itnim - > ioprofile ) ) ;
2010-09-15 22:50:55 +04:00
for ( j = 0 ; j < BFA_IOBUCKET_MAX ; j + + )
itnim - > ioprofile . io_latency . min [ j ] = ~ 0 ;
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* BFA IO module state machine functions
*/
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* IO is not started ( unallocated ) .
2010-09-15 22:50:55 +04:00
*/
static void
bfa_ioim_sm_uninit ( struct bfa_ioim_s * ioim , enum bfa_ioim_event event )
{
switch ( event ) {
case BFA_IOIM_SM_START :
if ( ! bfa_itnim_is_online ( ioim - > itnim ) ) {
if ( ! bfa_itnim_hold_io ( ioim - > itnim ) ) {
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
list_del ( & ioim - > qe ) ;
list_add_tail ( & ioim - > qe ,
& ioim - > fcpim - > ioim_comp_q ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe ,
__bfa_cb_ioim_pathtov , ioim ) ;
} else {
list_del ( & ioim - > qe ) ;
list_add_tail ( & ioim - > qe ,
& ioim - > itnim - > pending_q ) ;
}
break ;
}
if ( ioim - > nsges > BFI_SGE_INLINE ) {
2010-12-10 06:10:27 +03:00
if ( ! bfa_ioim_sgpg_alloc ( ioim ) ) {
2010-09-15 22:50:55 +04:00
bfa_sm_set_state ( ioim , bfa_ioim_sm_sgalloc ) ;
return ;
}
}
if ( ! bfa_ioim_send_ioreq ( ioim ) ) {
bfa_sm_set_state ( ioim , bfa_ioim_sm_qfull ) ;
break ;
}
bfa_sm_set_state ( ioim , bfa_ioim_sm_active ) ;
break ;
case BFA_IOIM_SM_IOTOV :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
bfa_ioim_move_to_comp_q ( ioim ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe ,
__bfa_cb_ioim_pathtov , ioim ) ;
break ;
case BFA_IOIM_SM_ABORT :
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* IO in pending queue can get abort requests . Complete abort
* requests immediately .
*/
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
2010-12-27 08:46:35 +03:00
WARN_ON ( ! bfa_q_is_on_q ( & ioim - > itnim - > pending_q , ioim ) ) ;
2010-09-15 22:50:55 +04:00
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe ,
2011-07-21 04:04:03 +04:00
__bfa_cb_ioim_abort , ioim ) ;
break ;
case BFA_IOIM_SM_LM_LUN_NOT_SUP :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
bfa_ioim_move_to_comp_q ( ioim ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe ,
__bfa_cb_ioim_lm_lun_not_sup , ioim ) ;
break ;
case BFA_IOIM_SM_LM_RPL_DC :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
bfa_ioim_move_to_comp_q ( ioim ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe ,
__bfa_cb_ioim_lm_rpl_dc , ioim ) ;
break ;
case BFA_IOIM_SM_LM_LUN_NOT_RDY :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
bfa_ioim_move_to_comp_q ( ioim ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe ,
__bfa_cb_ioim_lm_lun_not_rdy , ioim ) ;
2010-09-15 22:50:55 +04:00
break ;
default :
bfa_sm_fault ( ioim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* IO is waiting for SG pages .
2010-09-15 22:50:55 +04:00
*/
static void
bfa_ioim_sm_sgalloc ( struct bfa_ioim_s * ioim , enum bfa_ioim_event event )
{
bfa_trc ( ioim - > bfa , ioim - > iotag ) ;
bfa_trc ( ioim - > bfa , event ) ;
switch ( event ) {
case BFA_IOIM_SM_SGALLOCED :
if ( ! bfa_ioim_send_ioreq ( ioim ) ) {
bfa_sm_set_state ( ioim , bfa_ioim_sm_qfull ) ;
break ;
}
bfa_sm_set_state ( ioim , bfa_ioim_sm_active ) ;
break ;
case BFA_IOIM_SM_CLEANUP :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
bfa_sgpg_wcancel ( ioim - > bfa , & ioim - > iosp - > sgpg_wqe ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe , __bfa_cb_ioim_failed ,
ioim ) ;
bfa_ioim_notify_cleanup ( ioim ) ;
break ;
case BFA_IOIM_SM_ABORT :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
bfa_sgpg_wcancel ( ioim - > bfa , & ioim - > iosp - > sgpg_wqe ) ;
bfa_ioim_move_to_comp_q ( ioim ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe , __bfa_cb_ioim_abort ,
ioim ) ;
break ;
case BFA_IOIM_SM_HWFAIL :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
bfa_sgpg_wcancel ( ioim - > bfa , & ioim - > iosp - > sgpg_wqe ) ;
bfa_ioim_move_to_comp_q ( ioim ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe , __bfa_cb_ioim_failed ,
ioim ) ;
break ;
default :
bfa_sm_fault ( ioim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* IO is active .
2010-09-15 22:50:55 +04:00
*/
static void
bfa_ioim_sm_active ( struct bfa_ioim_s * ioim , enum bfa_ioim_event event )
{
switch ( event ) {
case BFA_IOIM_SM_COMP_GOOD :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
bfa_ioim_move_to_comp_q ( ioim ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe ,
__bfa_cb_ioim_good_comp , ioim ) ;
break ;
case BFA_IOIM_SM_COMP :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
bfa_ioim_move_to_comp_q ( ioim ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe , __bfa_cb_ioim_comp ,
ioim ) ;
break ;
case BFA_IOIM_SM_DONE :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb_free ) ;
bfa_ioim_move_to_comp_q ( ioim ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe , __bfa_cb_ioim_comp ,
ioim ) ;
break ;
case BFA_IOIM_SM_ABORT :
ioim - > iosp - > abort_explicit = BFA_TRUE ;
ioim - > io_cbfn = __bfa_cb_ioim_abort ;
if ( bfa_ioim_send_abort ( ioim ) )
bfa_sm_set_state ( ioim , bfa_ioim_sm_abort ) ;
else {
bfa_sm_set_state ( ioim , bfa_ioim_sm_abort_qfull ) ;
bfa_stats ( ioim - > itnim , qwait ) ;
bfa_reqq_wait ( ioim - > bfa , ioim - > reqq ,
& ioim - > iosp - > reqq_wait ) ;
}
break ;
case BFA_IOIM_SM_CLEANUP :
ioim - > iosp - > abort_explicit = BFA_FALSE ;
ioim - > io_cbfn = __bfa_cb_ioim_failed ;
if ( bfa_ioim_send_abort ( ioim ) )
bfa_sm_set_state ( ioim , bfa_ioim_sm_cleanup ) ;
else {
bfa_sm_set_state ( ioim , bfa_ioim_sm_cleanup_qfull ) ;
bfa_stats ( ioim - > itnim , qwait ) ;
bfa_reqq_wait ( ioim - > bfa , ioim - > reqq ,
& ioim - > iosp - > reqq_wait ) ;
}
break ;
case BFA_IOIM_SM_HWFAIL :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
bfa_ioim_move_to_comp_q ( ioim ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe , __bfa_cb_ioim_failed ,
ioim ) ;
break ;
case BFA_IOIM_SM_SQRETRY :
2010-12-14 03:23:27 +03:00
if ( bfa_ioim_maxretry_reached ( ioim ) ) {
/* max retry reached, free IO */
2010-09-15 22:50:55 +04:00
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb_free ) ;
bfa_ioim_move_to_comp_q ( ioim ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe ,
__bfa_cb_ioim_failed , ioim ) ;
break ;
}
/* waiting for IO tag resource free */
bfa_sm_set_state ( ioim , bfa_ioim_sm_cmnd_retry ) ;
break ;
default :
bfa_sm_fault ( ioim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* IO is retried with new tag .
*/
2010-09-15 22:50:55 +04:00
static void
bfa_ioim_sm_cmnd_retry ( struct bfa_ioim_s * ioim , enum bfa_ioim_event event )
{
switch ( event ) {
case BFA_IOIM_SM_FREE :
/* abts and rrq done. Now retry the IO with new tag */
2010-12-14 03:23:27 +03:00
bfa_ioim_update_iotag ( ioim ) ;
2010-09-15 22:50:55 +04:00
if ( ! bfa_ioim_send_ioreq ( ioim ) ) {
bfa_sm_set_state ( ioim , bfa_ioim_sm_qfull ) ;
break ;
}
bfa_sm_set_state ( ioim , bfa_ioim_sm_active ) ;
break ;
case BFA_IOIM_SM_CLEANUP :
ioim - > iosp - > abort_explicit = BFA_FALSE ;
ioim - > io_cbfn = __bfa_cb_ioim_failed ;
if ( bfa_ioim_send_abort ( ioim ) )
bfa_sm_set_state ( ioim , bfa_ioim_sm_cleanup ) ;
else {
bfa_sm_set_state ( ioim , bfa_ioim_sm_cleanup_qfull ) ;
bfa_stats ( ioim - > itnim , qwait ) ;
bfa_reqq_wait ( ioim - > bfa , ioim - > reqq ,
& ioim - > iosp - > reqq_wait ) ;
}
break ;
case BFA_IOIM_SM_HWFAIL :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
bfa_ioim_move_to_comp_q ( ioim ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe ,
__bfa_cb_ioim_failed , ioim ) ;
break ;
case BFA_IOIM_SM_ABORT :
2010-10-19 04:17:23 +04:00
/* in this state IO abort is done.
2010-09-15 22:50:55 +04:00
* Waiting for IO tag resource free .
*/
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb_free ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe , __bfa_cb_ioim_abort ,
ioim ) ;
break ;
default :
bfa_sm_fault ( ioim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* IO is being aborted , waiting for completion from firmware .
2010-09-15 22:50:55 +04:00
*/
static void
bfa_ioim_sm_abort ( struct bfa_ioim_s * ioim , enum bfa_ioim_event event )
{
bfa_trc ( ioim - > bfa , ioim - > iotag ) ;
bfa_trc ( ioim - > bfa , event ) ;
switch ( event ) {
case BFA_IOIM_SM_COMP_GOOD :
case BFA_IOIM_SM_COMP :
case BFA_IOIM_SM_DONE :
case BFA_IOIM_SM_FREE :
break ;
case BFA_IOIM_SM_ABORT_DONE :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb_free ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe , __bfa_cb_ioim_abort ,
ioim ) ;
break ;
case BFA_IOIM_SM_ABORT_COMP :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
bfa_ioim_move_to_comp_q ( ioim ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe , __bfa_cb_ioim_abort ,
ioim ) ;
break ;
case BFA_IOIM_SM_COMP_UTAG :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
bfa_ioim_move_to_comp_q ( ioim ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe , __bfa_cb_ioim_abort ,
ioim ) ;
break ;
case BFA_IOIM_SM_CLEANUP :
2010-12-27 08:46:35 +03:00
WARN_ON ( ioim - > iosp - > abort_explicit ! = BFA_TRUE ) ;
2010-09-15 22:50:55 +04:00
ioim - > iosp - > abort_explicit = BFA_FALSE ;
if ( bfa_ioim_send_abort ( ioim ) )
bfa_sm_set_state ( ioim , bfa_ioim_sm_cleanup ) ;
else {
bfa_sm_set_state ( ioim , bfa_ioim_sm_cleanup_qfull ) ;
bfa_stats ( ioim - > itnim , qwait ) ;
bfa_reqq_wait ( ioim - > bfa , ioim - > reqq ,
& ioim - > iosp - > reqq_wait ) ;
}
break ;
case BFA_IOIM_SM_HWFAIL :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
bfa_ioim_move_to_comp_q ( ioim ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe , __bfa_cb_ioim_failed ,
ioim ) ;
break ;
default :
bfa_sm_fault ( ioim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* IO is being cleaned up ( implicit abort ) , waiting for completion from
* firmware .
*/
static void
bfa_ioim_sm_cleanup ( struct bfa_ioim_s * ioim , enum bfa_ioim_event event )
{
bfa_trc ( ioim - > bfa , ioim - > iotag ) ;
bfa_trc ( ioim - > bfa , event ) ;
switch ( event ) {
case BFA_IOIM_SM_COMP_GOOD :
case BFA_IOIM_SM_COMP :
case BFA_IOIM_SM_DONE :
case BFA_IOIM_SM_FREE :
break ;
case BFA_IOIM_SM_ABORT :
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* IO is already being aborted implicitly
*/
ioim - > io_cbfn = __bfa_cb_ioim_abort ;
break ;
case BFA_IOIM_SM_ABORT_DONE :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb_free ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe , ioim - > io_cbfn , ioim ) ;
bfa_ioim_notify_cleanup ( ioim ) ;
break ;
case BFA_IOIM_SM_ABORT_COMP :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe , ioim - > io_cbfn , ioim ) ;
bfa_ioim_notify_cleanup ( ioim ) ;
break ;
case BFA_IOIM_SM_COMP_UTAG :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe , ioim - > io_cbfn , ioim ) ;
bfa_ioim_notify_cleanup ( ioim ) ;
break ;
case BFA_IOIM_SM_HWFAIL :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
bfa_ioim_move_to_comp_q ( ioim ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe , __bfa_cb_ioim_failed ,
ioim ) ;
break ;
case BFA_IOIM_SM_CLEANUP :
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* IO can be in cleanup state already due to TM command .
* 2 nd cleanup request comes from ITN offline event .
*/
break ;
default :
bfa_sm_fault ( ioim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* IO is waiting for room in request CQ
2010-09-15 22:50:55 +04:00
*/
static void
bfa_ioim_sm_qfull ( struct bfa_ioim_s * ioim , enum bfa_ioim_event event )
{
bfa_trc ( ioim - > bfa , ioim - > iotag ) ;
bfa_trc ( ioim - > bfa , event ) ;
switch ( event ) {
case BFA_IOIM_SM_QRESUME :
bfa_sm_set_state ( ioim , bfa_ioim_sm_active ) ;
bfa_ioim_send_ioreq ( ioim ) ;
break ;
case BFA_IOIM_SM_ABORT :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
bfa_reqq_wcancel ( & ioim - > iosp - > reqq_wait ) ;
bfa_ioim_move_to_comp_q ( ioim ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe , __bfa_cb_ioim_abort ,
ioim ) ;
break ;
case BFA_IOIM_SM_CLEANUP :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
bfa_reqq_wcancel ( & ioim - > iosp - > reqq_wait ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe , __bfa_cb_ioim_failed ,
ioim ) ;
bfa_ioim_notify_cleanup ( ioim ) ;
break ;
case BFA_IOIM_SM_HWFAIL :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
bfa_reqq_wcancel ( & ioim - > iosp - > reqq_wait ) ;
bfa_ioim_move_to_comp_q ( ioim ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe , __bfa_cb_ioim_failed ,
ioim ) ;
break ;
default :
bfa_sm_fault ( ioim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* Active IO is being aborted , waiting for room in request CQ .
2010-09-15 22:50:55 +04:00
*/
static void
bfa_ioim_sm_abort_qfull ( struct bfa_ioim_s * ioim , enum bfa_ioim_event event )
{
bfa_trc ( ioim - > bfa , ioim - > iotag ) ;
bfa_trc ( ioim - > bfa , event ) ;
switch ( event ) {
case BFA_IOIM_SM_QRESUME :
bfa_sm_set_state ( ioim , bfa_ioim_sm_abort ) ;
bfa_ioim_send_abort ( ioim ) ;
break ;
case BFA_IOIM_SM_CLEANUP :
2010-12-27 08:46:35 +03:00
WARN_ON ( ioim - > iosp - > abort_explicit ! = BFA_TRUE ) ;
2010-09-15 22:50:55 +04:00
ioim - > iosp - > abort_explicit = BFA_FALSE ;
bfa_sm_set_state ( ioim , bfa_ioim_sm_cleanup_qfull ) ;
break ;
case BFA_IOIM_SM_COMP_GOOD :
case BFA_IOIM_SM_COMP :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
bfa_reqq_wcancel ( & ioim - > iosp - > reqq_wait ) ;
bfa_ioim_move_to_comp_q ( ioim ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe , __bfa_cb_ioim_abort ,
ioim ) ;
break ;
case BFA_IOIM_SM_DONE :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb_free ) ;
bfa_reqq_wcancel ( & ioim - > iosp - > reqq_wait ) ;
bfa_ioim_move_to_comp_q ( ioim ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe , __bfa_cb_ioim_abort ,
ioim ) ;
break ;
case BFA_IOIM_SM_HWFAIL :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
bfa_reqq_wcancel ( & ioim - > iosp - > reqq_wait ) ;
bfa_ioim_move_to_comp_q ( ioim ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe , __bfa_cb_ioim_failed ,
ioim ) ;
break ;
default :
bfa_sm_fault ( ioim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* Active IO is being cleaned up , waiting for room in request CQ .
2010-09-15 22:50:55 +04:00
*/
static void
bfa_ioim_sm_cleanup_qfull ( struct bfa_ioim_s * ioim , enum bfa_ioim_event event )
{
bfa_trc ( ioim - > bfa , ioim - > iotag ) ;
bfa_trc ( ioim - > bfa , event ) ;
switch ( event ) {
case BFA_IOIM_SM_QRESUME :
bfa_sm_set_state ( ioim , bfa_ioim_sm_cleanup ) ;
bfa_ioim_send_abort ( ioim ) ;
break ;
case BFA_IOIM_SM_ABORT :
2010-10-19 04:17:23 +04:00
/*
tree-wide: fix comment/printk typos
"gadget", "through", "command", "maintain", "maintain", "controller", "address",
"between", "initiali[zs]e", "instead", "function", "select", "already",
"equal", "access", "management", "hierarchy", "registration", "interest",
"relative", "memory", "offset", "already",
Signed-off-by: Uwe Kleine-König <u.kleine-koenig@pengutronix.de>
Signed-off-by: Jiri Kosina <jkosina@suse.cz>
2010-11-01 22:38:34 +03:00
* IO is already being cleaned up implicitly
2010-09-15 22:50:55 +04:00
*/
ioim - > io_cbfn = __bfa_cb_ioim_abort ;
break ;
case BFA_IOIM_SM_COMP_GOOD :
case BFA_IOIM_SM_COMP :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
bfa_reqq_wcancel ( & ioim - > iosp - > reqq_wait ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe , ioim - > io_cbfn , ioim ) ;
bfa_ioim_notify_cleanup ( ioim ) ;
break ;
case BFA_IOIM_SM_DONE :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb_free ) ;
bfa_reqq_wcancel ( & ioim - > iosp - > reqq_wait ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe , ioim - > io_cbfn , ioim ) ;
bfa_ioim_notify_cleanup ( ioim ) ;
break ;
case BFA_IOIM_SM_HWFAIL :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
bfa_reqq_wcancel ( & ioim - > iosp - > reqq_wait ) ;
bfa_ioim_move_to_comp_q ( ioim ) ;
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe , __bfa_cb_ioim_failed ,
ioim ) ;
break ;
default :
bfa_sm_fault ( ioim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* IO bfa callback is pending .
*/
static void
bfa_ioim_sm_hcb ( struct bfa_ioim_s * ioim , enum bfa_ioim_event event )
{
switch ( event ) {
case BFA_IOIM_SM_HCB :
bfa_sm_set_state ( ioim , bfa_ioim_sm_uninit ) ;
bfa_ioim_free ( ioim ) ;
break ;
case BFA_IOIM_SM_CLEANUP :
bfa_ioim_notify_cleanup ( ioim ) ;
break ;
case BFA_IOIM_SM_HWFAIL :
break ;
default :
bfa_sm_fault ( ioim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* IO bfa callback is pending . IO resource cannot be freed .
*/
static void
bfa_ioim_sm_hcb_free ( struct bfa_ioim_s * ioim , enum bfa_ioim_event event )
{
bfa_trc ( ioim - > bfa , ioim - > iotag ) ;
bfa_trc ( ioim - > bfa , event ) ;
switch ( event ) {
case BFA_IOIM_SM_HCB :
bfa_sm_set_state ( ioim , bfa_ioim_sm_resfree ) ;
list_del ( & ioim - > qe ) ;
list_add_tail ( & ioim - > qe , & ioim - > fcpim - > ioim_resfree_q ) ;
break ;
case BFA_IOIM_SM_FREE :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
break ;
case BFA_IOIM_SM_CLEANUP :
bfa_ioim_notify_cleanup ( ioim ) ;
break ;
case BFA_IOIM_SM_HWFAIL :
bfa_sm_set_state ( ioim , bfa_ioim_sm_hcb ) ;
break ;
default :
bfa_sm_fault ( ioim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* IO is completed , waiting resource free from firmware .
*/
static void
bfa_ioim_sm_resfree ( struct bfa_ioim_s * ioim , enum bfa_ioim_event event )
{
bfa_trc ( ioim - > bfa , ioim - > iotag ) ;
bfa_trc ( ioim - > bfa , event ) ;
switch ( event ) {
case BFA_IOIM_SM_FREE :
bfa_sm_set_state ( ioim , bfa_ioim_sm_uninit ) ;
bfa_ioim_free ( ioim ) ;
break ;
case BFA_IOIM_SM_CLEANUP :
bfa_ioim_notify_cleanup ( ioim ) ;
break ;
case BFA_IOIM_SM_HWFAIL :
break ;
default :
bfa_sm_fault ( ioim - > bfa , event ) ;
}
}
2011-07-21 04:04:03 +04:00
/*
* This is called from bfa_fcpim_start after the bfa_init ( ) with flash read
* is complete by driver . now invalidate the stale content of lun mask
* like unit attention , rp tag and lp tag .
*/
static void
bfa_ioim_lm_init ( struct bfa_s * bfa )
{
struct bfa_lun_mask_s * lunm_list ;
int i ;
if ( bfa_get_lun_mask_status ( bfa ) = = BFA_LUNMASK_MINCFG )
return ;
lunm_list = bfa_get_lun_mask_list ( bfa ) ;
for ( i = 0 ; i < MAX_LUN_MASK_CFG ; i + + ) {
lunm_list [ i ] . ua = BFA_IOIM_LM_UA_RESET ;
lunm_list [ i ] . lp_tag = BFA_LP_TAG_INVALID ;
lunm_list [ i ] . rp_tag = BFA_RPORT_TAG_INVALID ;
}
}
/*
* Validate LUN for LUN masking
*/
static enum bfa_ioim_lm_status
bfa_ioim_lm_check ( struct bfa_ioim_s * ioim , struct bfa_lps_s * lps ,
struct bfa_rport_s * rp , struct scsi_lun lun )
{
u8 i ;
struct bfa_lun_mask_s * lun_list = bfa_get_lun_mask_list ( ioim - > bfa ) ;
struct scsi_cmnd * cmnd = ( struct scsi_cmnd * ) ioim - > dio ;
struct scsi_cdb_s * cdb = ( struct scsi_cdb_s * ) cmnd - > cmnd ;
if ( ( cdb - > scsi_cdb [ 0 ] = = REPORT_LUNS ) & &
( scsilun_to_int ( ( struct scsi_lun * ) & lun ) = = 0 ) ) {
ioim - > proc_rsp_data = bfa_ioim_lm_proc_rpl_data ;
return BFA_IOIM_LM_PRESENT ;
}
for ( i = 0 ; i < MAX_LUN_MASK_CFG ; i + + ) {
if ( lun_list [ i ] . state ! = BFA_IOIM_LUN_MASK_ACTIVE )
continue ;
if ( ( scsilun_to_int ( ( struct scsi_lun * ) & lun_list [ i ] . lun ) = =
scsilun_to_int ( ( struct scsi_lun * ) & lun ) )
& & ( rp - > rport_tag = = lun_list [ i ] . rp_tag )
& & ( ( u8 ) ioim - > itnim - > rport - > rport_info . lp_tag = =
lun_list [ i ] . lp_tag ) ) {
bfa_trc ( ioim - > bfa , lun_list [ i ] . rp_tag ) ;
bfa_trc ( ioim - > bfa , lun_list [ i ] . lp_tag ) ;
bfa_trc ( ioim - > bfa , scsilun_to_int (
( struct scsi_lun * ) & lun_list [ i ] . lun ) ) ;
if ( ( lun_list [ i ] . ua = = BFA_IOIM_LM_UA_SET ) & &
( ( cdb - > scsi_cdb [ 0 ] ! = INQUIRY ) | |
( cdb - > scsi_cdb [ 0 ] ! = REPORT_LUNS ) ) ) {
lun_list [ i ] . ua = BFA_IOIM_LM_UA_RESET ;
return BFA_IOIM_LM_RPL_DATA_CHANGED ;
}
if ( cdb - > scsi_cdb [ 0 ] = = REPORT_LUNS )
ioim - > proc_rsp_data = bfa_ioim_lm_proc_rpl_data ;
return BFA_IOIM_LM_PRESENT ;
}
}
if ( ( cdb - > scsi_cdb [ 0 ] = = INQUIRY ) & &
( scsilun_to_int ( ( struct scsi_lun * ) & lun ) = = 0 ) ) {
ioim - > proc_rsp_data = bfa_ioim_lm_proc_inq_data ;
return BFA_IOIM_LM_PRESENT ;
}
if ( cdb - > scsi_cdb [ 0 ] = = TEST_UNIT_READY )
return BFA_IOIM_LM_LUN_NOT_RDY ;
return BFA_IOIM_LM_LUN_NOT_SUP ;
}
static bfa_boolean_t
bfa_ioim_lm_proc_rsp_data_dummy ( struct bfa_ioim_s * ioim )
{
return BFA_TRUE ;
}
static void
bfa_ioim_lm_fetch_lun ( struct bfa_ioim_s * ioim , u8 * rl_data , int offset ,
int buf_lun_cnt )
{
struct bfa_lun_mask_s * lun_list = bfa_get_lun_mask_list ( ioim - > bfa ) ;
struct scsi_lun * lun_data = ( struct scsi_lun * ) ( rl_data + offset ) ;
struct scsi_lun lun ;
int i , j ;
bfa_trc ( ioim - > bfa , buf_lun_cnt ) ;
for ( j = 0 ; j < buf_lun_cnt ; j + + ) {
lun = * ( ( struct scsi_lun * ) ( lun_data + j ) ) ;
for ( i = 0 ; i < MAX_LUN_MASK_CFG ; i + + ) {
if ( lun_list [ i ] . state ! = BFA_IOIM_LUN_MASK_ACTIVE )
continue ;
if ( ( lun_list [ i ] . rp_wwn = = bfa_ioim_rp_wwn ( ioim ) ) & &
( lun_list [ i ] . lp_wwn = = bfa_ioim_lp_wwn ( ioim ) ) & &
( scsilun_to_int ( ( struct scsi_lun * ) & lun_list [ i ] . lun )
= = scsilun_to_int ( ( struct scsi_lun * ) & lun ) ) ) {
lun_list [ i ] . state = BFA_IOIM_LUN_MASK_FETCHED ;
break ;
}
} /* next lun in mask DB */
} /* next lun in buf */
}
static int
bfa_ioim_lm_update_lun_sg ( struct bfa_ioim_s * ioim , u32 * pgdlen ,
struct scsi_report_luns_data_s * rl )
{
struct scsi_cmnd * cmnd = ( struct scsi_cmnd * ) ioim - > dio ;
struct scatterlist * sg = scsi_sglist ( cmnd ) ;
struct bfa_lun_mask_s * lun_list = bfa_get_lun_mask_list ( ioim - > bfa ) ;
struct scsi_lun * prev_rl_data = NULL , * base_rl_data ;
int i , j , sgeid , lun_fetched_cnt = 0 , prev_sg_len = 0 , base_count ;
int lun_across_sg_bytes , bytes_from_next_buf ;
u64 last_lun , temp_last_lun ;
/* fetch luns from the first sg element */
bfa_ioim_lm_fetch_lun ( ioim , ( u8 * ) ( rl - > lun ) , 0 ,
( sg_dma_len ( sg ) / sizeof ( struct scsi_lun ) ) - 1 ) ;
/* fetch luns from multiple sg elements */
scsi_for_each_sg ( cmnd , sg , scsi_sg_count ( cmnd ) , sgeid ) {
if ( sgeid = = 0 ) {
prev_sg_len = sg_dma_len ( sg ) ;
prev_rl_data = ( struct scsi_lun * )
phys_to_virt ( sg_dma_address ( sg ) ) ;
continue ;
}
/* if the buf is having more data */
lun_across_sg_bytes = prev_sg_len % sizeof ( struct scsi_lun ) ;
if ( lun_across_sg_bytes ) {
bfa_trc ( ioim - > bfa , lun_across_sg_bytes ) ;
bfa_stats ( ioim - > itnim , lm_lun_across_sg ) ;
bytes_from_next_buf = sizeof ( struct scsi_lun ) -
lun_across_sg_bytes ;
/* from next buf take higher bytes */
temp_last_lun = * ( ( u64 * )
phys_to_virt ( sg_dma_address ( sg ) ) ) ;
last_lun | = temp_last_lun > >
( lun_across_sg_bytes * BITS_PER_BYTE ) ;
/* from prev buf take higher bytes */
temp_last_lun = * ( ( u64 * ) ( prev_rl_data +
( prev_sg_len - lun_across_sg_bytes ) ) ) ;
temp_last_lun > > = bytes_from_next_buf * BITS_PER_BYTE ;
last_lun = last_lun | ( temp_last_lun < <
( bytes_from_next_buf * BITS_PER_BYTE ) ) ;
bfa_ioim_lm_fetch_lun ( ioim , ( u8 * ) & last_lun , 0 , 1 ) ;
} else
bytes_from_next_buf = 0 ;
* pgdlen + = sg_dma_len ( sg ) ;
prev_sg_len = sg_dma_len ( sg ) ;
prev_rl_data = ( struct scsi_lun * )
phys_to_virt ( sg_dma_address ( sg ) ) ;
bfa_ioim_lm_fetch_lun ( ioim , ( u8 * ) prev_rl_data ,
bytes_from_next_buf ,
sg_dma_len ( sg ) / sizeof ( struct scsi_lun ) ) ;
}
/* update the report luns data - based on fetched luns */
sg = scsi_sglist ( cmnd ) ;
base_rl_data = ( struct scsi_lun * ) rl - > lun ;
base_count = ( sg_dma_len ( sg ) / sizeof ( struct scsi_lun ) ) - 1 ;
for ( i = 0 , j = 0 ; i < MAX_LUN_MASK_CFG ; i + + ) {
if ( lun_list [ i ] . state = = BFA_IOIM_LUN_MASK_FETCHED ) {
base_rl_data [ j ] = lun_list [ i ] . lun ;
lun_list [ i ] . state = BFA_IOIM_LUN_MASK_ACTIVE ;
j + + ;
lun_fetched_cnt + + ;
}
if ( j > base_count ) {
j = 0 ;
sg = sg_next ( sg ) ;
base_rl_data = ( struct scsi_lun * )
phys_to_virt ( sg_dma_address ( sg ) ) ;
base_count = sg_dma_len ( sg ) / sizeof ( struct scsi_lun ) ;
}
}
bfa_trc ( ioim - > bfa , lun_fetched_cnt ) ;
return lun_fetched_cnt ;
}
static bfa_boolean_t
bfa_ioim_lm_proc_inq_data ( struct bfa_ioim_s * ioim )
{
struct scsi_inquiry_data_s * inq ;
struct scatterlist * sg = scsi_sglist ( ( struct scsi_cmnd * ) ioim - > dio ) ;
ioim - > proc_rsp_data = bfa_ioim_lm_proc_rsp_data_dummy ;
inq = ( struct scsi_inquiry_data_s * ) phys_to_virt ( sg_dma_address ( sg ) ) ;
bfa_trc ( ioim - > bfa , inq - > device_type ) ;
inq - > peripheral_qual = SCSI_INQ_PQ_NOT_CON ;
return 0 ;
}
static bfa_boolean_t
bfa_ioim_lm_proc_rpl_data ( struct bfa_ioim_s * ioim )
{
struct scsi_cmnd * cmnd = ( struct scsi_cmnd * ) ioim - > dio ;
struct scatterlist * sg = scsi_sglist ( cmnd ) ;
struct bfi_ioim_rsp_s * m ;
struct scsi_report_luns_data_s * rl = NULL ;
int lun_count = 0 , lun_fetched_cnt = 0 ;
u32 residue , pgdlen = 0 ;
ioim - > proc_rsp_data = bfa_ioim_lm_proc_rsp_data_dummy ;
if ( bfa_get_lun_mask_status ( ioim - > bfa ) ! = BFA_LUNMASK_ENABLED )
return BFA_TRUE ;
m = ( struct bfi_ioim_rsp_s * ) & ioim - > iosp - > comp_rspmsg ;
if ( m - > scsi_status = = SCSI_STATUS_CHECK_CONDITION )
return BFA_TRUE ;
pgdlen = sg_dma_len ( sg ) ;
bfa_trc ( ioim - > bfa , pgdlen ) ;
rl = ( struct scsi_report_luns_data_s * ) phys_to_virt ( sg_dma_address ( sg ) ) ;
lun_count = cpu_to_be32 ( rl - > lun_list_length ) / sizeof ( struct scsi_lun ) ;
lun_fetched_cnt = bfa_ioim_lm_update_lun_sg ( ioim , & pgdlen , rl ) ;
if ( lun_count = = lun_fetched_cnt )
return BFA_TRUE ;
bfa_trc ( ioim - > bfa , lun_count ) ;
bfa_trc ( ioim - > bfa , lun_fetched_cnt ) ;
bfa_trc ( ioim - > bfa , be32_to_cpu ( rl - > lun_list_length ) ) ;
if ( be32_to_cpu ( rl - > lun_list_length ) < = pgdlen )
rl - > lun_list_length = be32_to_cpu ( lun_fetched_cnt ) *
sizeof ( struct scsi_lun ) ;
else
bfa_stats ( ioim - > itnim , lm_small_buf_addresidue ) ;
bfa_trc ( ioim - > bfa , be32_to_cpu ( rl - > lun_list_length ) ) ;
bfa_trc ( ioim - > bfa , be32_to_cpu ( m - > residue ) ) ;
residue = be32_to_cpu ( m - > residue ) ;
residue + = ( lun_count - lun_fetched_cnt ) * sizeof ( struct scsi_lun ) ;
bfa_stats ( ioim - > itnim , lm_wire_residue_changed ) ;
m - > residue = be32_to_cpu ( residue ) ;
bfa_trc ( ioim - > bfa , ioim - > nsges ) ;
return BFA_FALSE ;
}
2010-09-15 22:50:55 +04:00
static void
__bfa_cb_ioim_good_comp ( void * cbarg , bfa_boolean_t complete )
{
struct bfa_ioim_s * ioim = cbarg ;
if ( ! complete ) {
bfa_sm_send_event ( ioim , BFA_IOIM_SM_HCB ) ;
return ;
}
bfa_cb_ioim_good_comp ( ioim - > bfa - > bfad , ioim - > dio ) ;
}
static void
__bfa_cb_ioim_comp ( void * cbarg , bfa_boolean_t complete )
{
struct bfa_ioim_s * ioim = cbarg ;
struct bfi_ioim_rsp_s * m ;
u8 * snsinfo = NULL ;
u8 sns_len = 0 ;
s32 residue = 0 ;
if ( ! complete ) {
bfa_sm_send_event ( ioim , BFA_IOIM_SM_HCB ) ;
return ;
}
m = ( struct bfi_ioim_rsp_s * ) & ioim - > iosp - > comp_rspmsg ;
if ( m - > io_status = = BFI_IOIM_STS_OK ) {
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* setup sense information , if present
*/
if ( ( m - > scsi_status = = SCSI_STATUS_CHECK_CONDITION ) & &
m - > sns_len ) {
sns_len = m - > sns_len ;
2011-06-14 02:53:58 +04:00
snsinfo = BFA_SNSINFO_FROM_TAG ( ioim - > fcpim - > fcp ,
ioim - > iotag ) ;
2010-09-15 22:50:55 +04:00
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* setup residue value correctly for normal completions
*/
if ( m - > resid_flags = = FCP_RESID_UNDER ) {
2010-10-19 04:10:50 +04:00
residue = be32_to_cpu ( m - > residue ) ;
2010-09-15 22:50:55 +04:00
bfa_stats ( ioim - > itnim , iocomp_underrun ) ;
}
if ( m - > resid_flags = = FCP_RESID_OVER ) {
2010-10-19 04:10:50 +04:00
residue = be32_to_cpu ( m - > residue ) ;
2010-09-15 22:50:55 +04:00
residue = - residue ;
bfa_stats ( ioim - > itnim , iocomp_overrun ) ;
}
}
bfa_cb_ioim_done ( ioim - > bfa - > bfad , ioim - > dio , m - > io_status ,
m - > scsi_status , sns_len , snsinfo , residue ) ;
}
2011-07-21 04:04:03 +04:00
static void
__bfa_cb_ioim_lm_lun_not_sup ( void * cbarg , bfa_boolean_t complete )
{
struct bfa_ioim_s * ioim = cbarg ;
int sns_len = 0xD ;
u32 residue = scsi_bufflen ( ( struct scsi_cmnd * ) ioim - > dio ) ;
struct scsi_sense_s * snsinfo ;
if ( ! complete ) {
bfa_sm_send_event ( ioim , BFA_IOIM_SM_HCB ) ;
return ;
}
snsinfo = ( struct scsi_sense_s * ) BFA_SNSINFO_FROM_TAG (
ioim - > fcpim - > fcp , ioim - > iotag ) ;
snsinfo - > rsp_code = SCSI_SENSE_CUR_ERR ;
snsinfo - > add_sense_length = 0xa ;
snsinfo - > asc = SCSI_ASC_LUN_NOT_SUPPORTED ;
snsinfo - > sense_key = ILLEGAL_REQUEST ;
bfa_trc ( ioim - > bfa , residue ) ;
bfa_cb_ioim_done ( ioim - > bfa - > bfad , ioim - > dio , BFI_IOIM_STS_OK ,
SCSI_STATUS_CHECK_CONDITION , sns_len ,
( u8 * ) snsinfo , residue ) ;
}
static void
__bfa_cb_ioim_lm_rpl_dc ( void * cbarg , bfa_boolean_t complete )
{
struct bfa_ioim_s * ioim = cbarg ;
int sns_len = 0xD ;
u32 residue = scsi_bufflen ( ( struct scsi_cmnd * ) ioim - > dio ) ;
struct scsi_sense_s * snsinfo ;
if ( ! complete ) {
bfa_sm_send_event ( ioim , BFA_IOIM_SM_HCB ) ;
return ;
}
snsinfo = ( struct scsi_sense_s * ) BFA_SNSINFO_FROM_TAG ( ioim - > fcpim - > fcp ,
ioim - > iotag ) ;
snsinfo - > rsp_code = SCSI_SENSE_CUR_ERR ;
snsinfo - > sense_key = SCSI_MP_IEC_UNIT_ATTN ;
snsinfo - > asc = SCSI_ASC_TOCC ;
snsinfo - > add_sense_length = 0x6 ;
snsinfo - > ascq = SCSI_ASCQ_RL_DATA_CHANGED ;
bfa_trc ( ioim - > bfa , residue ) ;
bfa_cb_ioim_done ( ioim - > bfa - > bfad , ioim - > dio , BFI_IOIM_STS_OK ,
SCSI_STATUS_CHECK_CONDITION , sns_len ,
( u8 * ) snsinfo , residue ) ;
}
static void
__bfa_cb_ioim_lm_lun_not_rdy ( void * cbarg , bfa_boolean_t complete )
{
struct bfa_ioim_s * ioim = cbarg ;
int sns_len = 0xD ;
u32 residue = scsi_bufflen ( ( struct scsi_cmnd * ) ioim - > dio ) ;
struct scsi_sense_s * snsinfo ;
if ( ! complete ) {
bfa_sm_send_event ( ioim , BFA_IOIM_SM_HCB ) ;
return ;
}
snsinfo = ( struct scsi_sense_s * ) BFA_SNSINFO_FROM_TAG (
ioim - > fcpim - > fcp , ioim - > iotag ) ;
snsinfo - > rsp_code = SCSI_SENSE_CUR_ERR ;
snsinfo - > add_sense_length = 0xa ;
snsinfo - > sense_key = NOT_READY ;
snsinfo - > asc = SCSI_ASC_LUN_NOT_READY ;
snsinfo - > ascq = SCSI_ASCQ_MAN_INTR_REQ ;
bfa_trc ( ioim - > bfa , residue ) ;
bfa_cb_ioim_done ( ioim - > bfa - > bfad , ioim - > dio , BFI_IOIM_STS_OK ,
SCSI_STATUS_CHECK_CONDITION , sns_len ,
( u8 * ) snsinfo , residue ) ;
}
void
bfa_fcpim_lunmask_rp_update ( struct bfa_s * bfa , wwn_t lp_wwn , wwn_t rp_wwn ,
u16 rp_tag , u8 lp_tag )
{
struct bfa_lun_mask_s * lun_list ;
u8 i ;
if ( bfa_get_lun_mask_status ( bfa ) = = BFA_LUNMASK_MINCFG )
return ;
lun_list = bfa_get_lun_mask_list ( bfa ) ;
for ( i = 0 ; i < MAX_LUN_MASK_CFG ; i + + ) {
if ( lun_list [ i ] . state = = BFA_IOIM_LUN_MASK_ACTIVE ) {
if ( ( lun_list [ i ] . lp_wwn = = lp_wwn ) & &
( lun_list [ i ] . rp_wwn = = rp_wwn ) ) {
lun_list [ i ] . rp_tag = rp_tag ;
lun_list [ i ] . lp_tag = lp_tag ;
}
}
}
}
2011-07-21 04:04:24 +04:00
/*
* set UA for all active luns in LM DB
*/
static void
bfa_ioim_lm_set_ua ( struct bfa_s * bfa )
{
struct bfa_lun_mask_s * lunm_list ;
int i ;
lunm_list = bfa_get_lun_mask_list ( bfa ) ;
for ( i = 0 ; i < MAX_LUN_MASK_CFG ; i + + ) {
if ( lunm_list [ i ] . state ! = BFA_IOIM_LUN_MASK_ACTIVE )
continue ;
lunm_list [ i ] . ua = BFA_IOIM_LM_UA_SET ;
}
}
bfa_status_t
bfa_fcpim_lunmask_update ( struct bfa_s * bfa , u32 update )
{
struct bfa_lunmask_cfg_s * lun_mask ;
bfa_trc ( bfa , bfa_get_lun_mask_status ( bfa ) ) ;
if ( bfa_get_lun_mask_status ( bfa ) = = BFA_LUNMASK_MINCFG )
return BFA_STATUS_FAILED ;
if ( bfa_get_lun_mask_status ( bfa ) = = update )
return BFA_STATUS_NO_CHANGE ;
lun_mask = bfa_get_lun_mask ( bfa ) ;
lun_mask - > status = update ;
if ( bfa_get_lun_mask_status ( bfa ) = = BFA_LUNMASK_ENABLED )
bfa_ioim_lm_set_ua ( bfa ) ;
return bfa_dconf_update ( bfa ) ;
}
bfa_status_t
bfa_fcpim_lunmask_clear ( struct bfa_s * bfa )
{
int i ;
struct bfa_lun_mask_s * lunm_list ;
bfa_trc ( bfa , bfa_get_lun_mask_status ( bfa ) ) ;
if ( bfa_get_lun_mask_status ( bfa ) = = BFA_LUNMASK_MINCFG )
return BFA_STATUS_FAILED ;
lunm_list = bfa_get_lun_mask_list ( bfa ) ;
for ( i = 0 ; i < MAX_LUN_MASK_CFG ; i + + ) {
if ( lunm_list [ i ] . state = = BFA_IOIM_LUN_MASK_ACTIVE ) {
if ( lunm_list [ i ] . rp_tag ! = BFA_RPORT_TAG_INVALID )
bfa_rport_unset_lunmask ( bfa ,
BFA_RPORT_FROM_TAG ( bfa , lunm_list [ i ] . rp_tag ) ) ;
}
}
memset ( lunm_list , 0 , sizeof ( struct bfa_lun_mask_s ) * MAX_LUN_MASK_CFG ) ;
return bfa_dconf_update ( bfa ) ;
}
bfa_status_t
bfa_fcpim_lunmask_query ( struct bfa_s * bfa , void * buf )
{
struct bfa_lunmask_cfg_s * lun_mask ;
bfa_trc ( bfa , bfa_get_lun_mask_status ( bfa ) ) ;
if ( bfa_get_lun_mask_status ( bfa ) = = BFA_LUNMASK_MINCFG )
return BFA_STATUS_FAILED ;
lun_mask = bfa_get_lun_mask ( bfa ) ;
memcpy ( buf , lun_mask , sizeof ( struct bfa_lunmask_cfg_s ) ) ;
return BFA_STATUS_OK ;
}
bfa_status_t
bfa_fcpim_lunmask_add ( struct bfa_s * bfa , u16 vf_id , wwn_t * pwwn ,
wwn_t rpwwn , struct scsi_lun lun )
{
struct bfa_lun_mask_s * lunm_list ;
struct bfa_rport_s * rp = NULL ;
int i , free_index = MAX_LUN_MASK_CFG + 1 ;
struct bfa_fcs_lport_s * port = NULL ;
struct bfa_fcs_rport_s * rp_fcs ;
bfa_trc ( bfa , bfa_get_lun_mask_status ( bfa ) ) ;
if ( bfa_get_lun_mask_status ( bfa ) = = BFA_LUNMASK_MINCFG )
return BFA_STATUS_FAILED ;
port = bfa_fcs_lookup_port ( & ( ( struct bfad_s * ) bfa - > bfad ) - > bfa_fcs ,
vf_id , * pwwn ) ;
if ( port ) {
* pwwn = port - > port_cfg . pwwn ;
rp_fcs = bfa_fcs_lport_get_rport_by_pwwn ( port , rpwwn ) ;
rp = rp_fcs - > bfa_rport ;
}
lunm_list = bfa_get_lun_mask_list ( bfa ) ;
/* if entry exists */
for ( i = 0 ; i < MAX_LUN_MASK_CFG ; i + + ) {
if ( lunm_list [ i ] . state ! = BFA_IOIM_LUN_MASK_ACTIVE )
free_index = i ;
if ( ( lunm_list [ i ] . lp_wwn = = * pwwn ) & &
( lunm_list [ i ] . rp_wwn = = rpwwn ) & &
( scsilun_to_int ( ( struct scsi_lun * ) & lunm_list [ i ] . lun ) = =
scsilun_to_int ( ( struct scsi_lun * ) & lun ) ) )
return BFA_STATUS_ENTRY_EXISTS ;
}
if ( free_index > MAX_LUN_MASK_CFG )
return BFA_STATUS_MAX_ENTRY_REACHED ;
if ( rp ) {
lunm_list [ free_index ] . lp_tag = bfa_lps_get_tag_from_pid ( bfa ,
rp - > rport_info . local_pid ) ;
lunm_list [ free_index ] . rp_tag = rp - > rport_tag ;
} else {
lunm_list [ free_index ] . lp_tag = BFA_LP_TAG_INVALID ;
lunm_list [ free_index ] . rp_tag = BFA_RPORT_TAG_INVALID ;
}
lunm_list [ free_index ] . lp_wwn = * pwwn ;
lunm_list [ free_index ] . rp_wwn = rpwwn ;
lunm_list [ free_index ] . lun = lun ;
lunm_list [ free_index ] . state = BFA_IOIM_LUN_MASK_ACTIVE ;
/* set for all luns in this rp */
for ( i = 0 ; i < MAX_LUN_MASK_CFG ; i + + ) {
if ( ( lunm_list [ i ] . lp_wwn = = * pwwn ) & &
( lunm_list [ i ] . rp_wwn = = rpwwn ) )
lunm_list [ i ] . ua = BFA_IOIM_LM_UA_SET ;
}
return bfa_dconf_update ( bfa ) ;
}
bfa_status_t
bfa_fcpim_lunmask_delete ( struct bfa_s * bfa , u16 vf_id , wwn_t * pwwn ,
wwn_t rpwwn , struct scsi_lun lun )
{
struct bfa_lun_mask_s * lunm_list ;
struct bfa_rport_s * rp = NULL ;
struct bfa_fcs_lport_s * port = NULL ;
struct bfa_fcs_rport_s * rp_fcs ;
int i ;
/* in min cfg lunm_list could be NULL but no commands should run. */
if ( bfa_get_lun_mask_status ( bfa ) = = BFA_LUNMASK_MINCFG )
return BFA_STATUS_FAILED ;
bfa_trc ( bfa , bfa_get_lun_mask_status ( bfa ) ) ;
bfa_trc ( bfa , * pwwn ) ;
bfa_trc ( bfa , rpwwn ) ;
bfa_trc ( bfa , scsilun_to_int ( ( struct scsi_lun * ) & lun ) ) ;
if ( * pwwn = = 0 ) {
port = bfa_fcs_lookup_port (
& ( ( struct bfad_s * ) bfa - > bfad ) - > bfa_fcs ,
vf_id , * pwwn ) ;
if ( port ) {
* pwwn = port - > port_cfg . pwwn ;
rp_fcs = bfa_fcs_lport_get_rport_by_pwwn ( port , rpwwn ) ;
rp = rp_fcs - > bfa_rport ;
}
}
lunm_list = bfa_get_lun_mask_list ( bfa ) ;
for ( i = 0 ; i < MAX_LUN_MASK_CFG ; i + + ) {
if ( ( lunm_list [ i ] . lp_wwn = = * pwwn ) & &
( lunm_list [ i ] . rp_wwn = = rpwwn ) & &
( scsilun_to_int ( ( struct scsi_lun * ) & lunm_list [ i ] . lun ) = =
scsilun_to_int ( ( struct scsi_lun * ) & lun ) ) ) {
lunm_list [ i ] . lp_wwn = 0 ;
lunm_list [ i ] . rp_wwn = 0 ;
int_to_scsilun ( 0 , & lunm_list [ i ] . lun ) ;
lunm_list [ i ] . state = BFA_IOIM_LUN_MASK_INACTIVE ;
if ( lunm_list [ i ] . rp_tag ! = BFA_RPORT_TAG_INVALID ) {
lunm_list [ i ] . rp_tag = BFA_RPORT_TAG_INVALID ;
lunm_list [ i ] . lp_tag = BFA_LP_TAG_INVALID ;
}
return bfa_dconf_update ( bfa ) ;
}
}
/* set for all luns in this rp */
for ( i = 0 ; i < MAX_LUN_MASK_CFG ; i + + ) {
if ( ( lunm_list [ i ] . lp_wwn = = * pwwn ) & &
( lunm_list [ i ] . rp_wwn = = rpwwn ) )
lunm_list [ i ] . ua = BFA_IOIM_LM_UA_SET ;
}
return BFA_STATUS_ENTRY_NOT_EXISTS ;
}
2010-09-15 22:50:55 +04:00
static void
__bfa_cb_ioim_failed ( void * cbarg , bfa_boolean_t complete )
{
struct bfa_ioim_s * ioim = cbarg ;
if ( ! complete ) {
bfa_sm_send_event ( ioim , BFA_IOIM_SM_HCB ) ;
return ;
}
2011-07-21 04:04:03 +04:00
ioim - > proc_rsp_data = bfa_ioim_lm_proc_rsp_data_dummy ;
2010-09-15 22:50:55 +04:00
bfa_cb_ioim_done ( ioim - > bfa - > bfad , ioim - > dio , BFI_IOIM_STS_ABORTED ,
0 , 0 , NULL , 0 ) ;
}
static void
__bfa_cb_ioim_pathtov ( void * cbarg , bfa_boolean_t complete )
{
struct bfa_ioim_s * ioim = cbarg ;
bfa_stats ( ioim - > itnim , path_tov_expired ) ;
if ( ! complete ) {
bfa_sm_send_event ( ioim , BFA_IOIM_SM_HCB ) ;
return ;
}
2011-07-21 04:04:03 +04:00
ioim - > proc_rsp_data = bfa_ioim_lm_proc_rsp_data_dummy ;
2010-09-15 22:50:55 +04:00
bfa_cb_ioim_done ( ioim - > bfa - > bfad , ioim - > dio , BFI_IOIM_STS_PATHTOV ,
0 , 0 , NULL , 0 ) ;
}
static void
__bfa_cb_ioim_abort ( void * cbarg , bfa_boolean_t complete )
{
struct bfa_ioim_s * ioim = cbarg ;
if ( ! complete ) {
bfa_sm_send_event ( ioim , BFA_IOIM_SM_HCB ) ;
return ;
}
2011-07-21 04:04:03 +04:00
ioim - > proc_rsp_data = bfa_ioim_lm_proc_rsp_data_dummy ;
2010-09-15 22:50:55 +04:00
bfa_cb_ioim_abort ( ioim - > bfa - > bfad , ioim - > dio ) ;
}
static void
bfa_ioim_sgpg_alloced ( void * cbarg )
{
struct bfa_ioim_s * ioim = cbarg ;
ioim - > nsgpgs = BFA_SGPG_NPAGE ( ioim - > nsges ) ;
list_splice_tail_init ( & ioim - > iosp - > sgpg_wqe . sgpg_q , & ioim - > sgpg_q ) ;
2010-12-10 06:10:27 +03:00
ioim - > sgpg = bfa_q_first ( & ioim - > sgpg_q ) ;
2010-09-15 22:50:55 +04:00
bfa_sm_send_event ( ioim , BFA_IOIM_SM_SGALLOCED ) ;
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Send I / O request to firmware .
*/
static bfa_boolean_t
bfa_ioim_send_ioreq ( struct bfa_ioim_s * ioim )
{
struct bfa_itnim_s * itnim = ioim - > itnim ;
struct bfi_ioim_req_s * m ;
2010-12-10 06:13:20 +03:00
static struct fcp_cmnd_s cmnd_z0 = { { { 0 } } } ;
2010-12-10 06:10:27 +03:00
struct bfi_sge_s * sge , * sgpge ;
2010-09-15 22:50:55 +04:00
u32 pgdlen = 0 ;
u32 fcp_dl ;
u64 addr ;
struct scatterlist * sg ;
2010-12-10 06:10:27 +03:00
struct bfa_sgpg_s * sgpg ;
2010-09-15 22:50:55 +04:00
struct scsi_cmnd * cmnd = ( struct scsi_cmnd * ) ioim - > dio ;
2010-12-10 06:10:27 +03:00
u32 i , sge_id , pgcumsz ;
2010-12-10 06:11:39 +03:00
enum dma_data_direction dmadir ;
2010-09-15 22:50:55 +04:00
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* check for room in queue to send request now
*/
m = bfa_reqq_next ( ioim - > bfa , ioim - > reqq ) ;
if ( ! m ) {
bfa_stats ( ioim - > itnim , qwait ) ;
bfa_reqq_wait ( ioim - > bfa , ioim - > reqq ,
& ioim - > iosp - > reqq_wait ) ;
return BFA_FALSE ;
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* build i / o request message next
*/
2010-10-19 04:10:50 +04:00
m - > io_tag = cpu_to_be16 ( ioim - > iotag ) ;
2010-09-15 22:50:55 +04:00
m - > rport_hdl = ioim - > itnim - > rport - > fw_handle ;
2010-12-10 06:11:39 +03:00
m - > io_timeout = 0 ;
2010-09-15 22:50:55 +04:00
sge = & m - > sges [ 0 ] ;
2010-12-10 06:10:27 +03:00
sgpg = ioim - > sgpg ;
sge_id = 0 ;
sgpge = NULL ;
pgcumsz = 0 ;
scsi_for_each_sg ( cmnd , sg , ioim - > nsges , i ) {
if ( i = = 0 ) {
/* build inline IO SG element */
2010-12-10 06:12:32 +03:00
addr = bfa_sgaddr_le ( sg_dma_address ( sg ) ) ;
2010-12-10 06:10:27 +03:00
sge - > sga = * ( union bfi_addr_u * ) & addr ;
pgdlen = sg_dma_len ( sg ) ;
sge - > sg_len = pgdlen ;
sge - > flags = ( ioim - > nsges > BFI_SGE_INLINE ) ?
2010-09-15 22:50:55 +04:00
BFI_SGE_DATA_CPL : BFI_SGE_DATA_LAST ;
2010-12-10 06:10:27 +03:00
bfa_sge_to_be ( sge ) ;
sge + + ;
} else {
if ( sge_id = = 0 )
sgpge = sgpg - > sgpg - > sges ;
2010-12-10 06:12:32 +03:00
addr = bfa_sgaddr_le ( sg_dma_address ( sg ) ) ;
2010-12-10 06:10:27 +03:00
sgpge - > sga = * ( union bfi_addr_u * ) & addr ;
sgpge - > sg_len = sg_dma_len ( sg ) ;
pgcumsz + = sgpge - > sg_len ;
/* set flags */
if ( i < ( ioim - > nsges - 1 ) & &
sge_id < ( BFI_SGPG_DATA_SGES - 1 ) )
sgpge - > flags = BFI_SGE_DATA ;
else if ( i < ( ioim - > nsges - 1 ) )
sgpge - > flags = BFI_SGE_DATA_CPL ;
else
sgpge - > flags = BFI_SGE_DATA_LAST ;
bfa_sge_to_le ( sgpge ) ;
sgpge + + ;
if ( i = = ( ioim - > nsges - 1 ) ) {
sgpge - > flags = BFI_SGE_PGDLEN ;
sgpge - > sga . a32 . addr_lo = 0 ;
sgpge - > sga . a32 . addr_hi = 0 ;
sgpge - > sg_len = pgcumsz ;
bfa_sge_to_le ( sgpge ) ;
} else if ( + + sge_id = = BFI_SGPG_DATA_SGES ) {
sgpg = ( struct bfa_sgpg_s * ) bfa_q_next ( sgpg ) ;
sgpge - > flags = BFI_SGE_LINK ;
sgpge - > sga = sgpg - > sgpg_pa ;
sgpge - > sg_len = pgcumsz ;
bfa_sge_to_le ( sgpge ) ;
sge_id = 0 ;
pgcumsz = 0 ;
}
}
2010-09-15 22:50:55 +04:00
}
if ( ioim - > nsges > BFI_SGE_INLINE ) {
sge - > sga = ioim - > sgpg - > sgpg_pa ;
} else {
sge - > sga . a32 . addr_lo = 0 ;
sge - > sga . a32 . addr_hi = 0 ;
}
sge - > sg_len = pgdlen ;
sge - > flags = BFI_SGE_PGDLEN ;
bfa_sge_to_be ( sge ) ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* set up I / O command parameters
*/
2010-10-19 04:08:54 +04:00
m - > cmnd = cmnd_z0 ;
2010-12-10 06:11:39 +03:00
int_to_scsilun ( cmnd - > device - > lun , & m - > cmnd . lun ) ;
dmadir = cmnd - > sc_data_direction ;
if ( dmadir = = DMA_TO_DEVICE )
m - > cmnd . iodir = FCP_IODIR_WRITE ;
else if ( dmadir = = DMA_FROM_DEVICE )
m - > cmnd . iodir = FCP_IODIR_READ ;
else
m - > cmnd . iodir = FCP_IODIR_NONE ;
2010-12-27 08:50:10 +03:00
m - > cmnd . cdb = * ( struct scsi_cdb_s * ) cmnd - > cmnd ;
2010-12-10 06:11:39 +03:00
fcp_dl = scsi_bufflen ( cmnd ) ;
2010-10-19 04:10:50 +04:00
m - > cmnd . fcp_dl = cpu_to_be32 ( fcp_dl ) ;
2010-09-15 22:50:55 +04:00
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* set up I / O message header
*/
switch ( m - > cmnd . iodir ) {
case FCP_IODIR_READ :
2011-06-25 07:24:08 +04:00
bfi_h2i_set ( m - > mh , BFI_MC_IOIM_READ , 0 , bfa_fn_lpu ( ioim - > bfa ) ) ;
2010-09-15 22:50:55 +04:00
bfa_stats ( itnim , input_reqs ) ;
ioim - > itnim - > stats . rd_throughput + = fcp_dl ;
break ;
case FCP_IODIR_WRITE :
2011-06-25 07:24:08 +04:00
bfi_h2i_set ( m - > mh , BFI_MC_IOIM_WRITE , 0 , bfa_fn_lpu ( ioim - > bfa ) ) ;
2010-09-15 22:50:55 +04:00
bfa_stats ( itnim , output_reqs ) ;
ioim - > itnim - > stats . wr_throughput + = fcp_dl ;
break ;
case FCP_IODIR_RW :
bfa_stats ( itnim , input_reqs ) ;
bfa_stats ( itnim , output_reqs ) ;
default :
2011-06-25 07:24:08 +04:00
bfi_h2i_set ( m - > mh , BFI_MC_IOIM_IO , 0 , bfa_fn_lpu ( ioim - > bfa ) ) ;
2010-09-15 22:50:55 +04:00
}
if ( itnim - > seq_rec | |
2010-12-10 06:11:39 +03:00
( scsi_bufflen ( cmnd ) & ( sizeof ( u32 ) - 1 ) ) )
2011-06-25 07:24:08 +04:00
bfi_h2i_set ( m - > mh , BFI_MC_IOIM_IO , 0 , bfa_fn_lpu ( ioim - > bfa ) ) ;
2010-09-15 22:50:55 +04:00
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* queue I / O message to firmware
*/
2011-06-25 07:24:08 +04:00
bfa_reqq_produce ( ioim - > bfa , ioim - > reqq , m - > mh ) ;
2010-09-15 22:50:55 +04:00
return BFA_TRUE ;
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Setup any additional SG pages needed . Inline SG element is setup
* at queuing time .
*/
static bfa_boolean_t
2010-12-10 06:10:27 +03:00
bfa_ioim_sgpg_alloc ( struct bfa_ioim_s * ioim )
2010-09-15 22:50:55 +04:00
{
u16 nsgpgs ;
2010-12-27 08:46:35 +03:00
WARN_ON ( ioim - > nsges < = BFI_SGE_INLINE ) ;
2010-09-15 22:50:55 +04:00
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* allocate SG pages needed
*/
nsgpgs = BFA_SGPG_NPAGE ( ioim - > nsges ) ;
if ( ! nsgpgs )
return BFA_TRUE ;
if ( bfa_sgpg_malloc ( ioim - > bfa , & ioim - > sgpg_q , nsgpgs )
! = BFA_STATUS_OK ) {
bfa_sgpg_wait ( ioim - > bfa , & ioim - > iosp - > sgpg_wqe , nsgpgs ) ;
return BFA_FALSE ;
}
ioim - > nsgpgs = nsgpgs ;
2010-12-10 06:10:27 +03:00
ioim - > sgpg = bfa_q_first ( & ioim - > sgpg_q ) ;
2010-09-15 22:50:55 +04:00
return BFA_TRUE ;
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Send I / O abort request to firmware .
*/
static bfa_boolean_t
bfa_ioim_send_abort ( struct bfa_ioim_s * ioim )
{
struct bfi_ioim_abort_req_s * m ;
enum bfi_ioim_h2i msgop ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* check for room in queue to send request now
*/
m = bfa_reqq_next ( ioim - > bfa , ioim - > reqq ) ;
if ( ! m )
return BFA_FALSE ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* build i / o request message next
*/
if ( ioim - > iosp - > abort_explicit )
msgop = BFI_IOIM_H2I_IOABORT_REQ ;
else
msgop = BFI_IOIM_H2I_IOCLEANUP_REQ ;
2011-06-25 07:24:08 +04:00
bfi_h2i_set ( m - > mh , BFI_MC_IOIM , msgop , bfa_fn_lpu ( ioim - > bfa ) ) ;
2010-10-19 04:10:50 +04:00
m - > io_tag = cpu_to_be16 ( ioim - > iotag ) ;
2010-09-15 22:50:55 +04:00
m - > abort_tag = + + ioim - > abort_tag ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* queue I / O message to firmware
*/
2011-06-25 07:24:08 +04:00
bfa_reqq_produce ( ioim - > bfa , ioim - > reqq , m - > mh ) ;
2010-09-15 22:50:55 +04:00
return BFA_TRUE ;
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Call to resume any I / O requests waiting for room in request queue .
*/
static void
bfa_ioim_qresume ( void * cbarg )
{
struct bfa_ioim_s * ioim = cbarg ;
bfa_stats ( ioim - > itnim , qresumes ) ;
bfa_sm_send_event ( ioim , BFA_IOIM_SM_QRESUME ) ;
}
static void
bfa_ioim_notify_cleanup ( struct bfa_ioim_s * ioim )
{
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Move IO from itnim queue to fcpim global queue since itnim will be
* freed .
*/
list_del ( & ioim - > qe ) ;
list_add_tail ( & ioim - > qe , & ioim - > fcpim - > ioim_comp_q ) ;
if ( ! ioim - > iosp - > tskim ) {
if ( ioim - > fcpim - > delay_comp & & ioim - > itnim - > iotov_active ) {
bfa_cb_dequeue ( & ioim - > hcb_qe ) ;
list_del ( & ioim - > qe ) ;
list_add_tail ( & ioim - > qe , & ioim - > itnim - > delay_comp_q ) ;
}
bfa_itnim_iodone ( ioim - > itnim ) ;
} else
2010-12-10 06:08:43 +03:00
bfa_wc_down ( & ioim - > iosp - > tskim - > wc ) ;
2010-09-15 22:50:55 +04:00
}
static bfa_boolean_t
bfa_ioim_is_abortable ( struct bfa_ioim_s * ioim )
{
if ( ( bfa_sm_cmp_state ( ioim , bfa_ioim_sm_uninit ) & &
( ! bfa_q_is_on_q ( & ioim - > itnim - > pending_q , ioim ) ) ) | |
( bfa_sm_cmp_state ( ioim , bfa_ioim_sm_abort ) ) | |
( bfa_sm_cmp_state ( ioim , bfa_ioim_sm_abort_qfull ) ) | |
( bfa_sm_cmp_state ( ioim , bfa_ioim_sm_hcb ) ) | |
( bfa_sm_cmp_state ( ioim , bfa_ioim_sm_hcb_free ) ) | |
( bfa_sm_cmp_state ( ioim , bfa_ioim_sm_resfree ) ) )
return BFA_FALSE ;
return BFA_TRUE ;
}
void
bfa_ioim_delayed_comp ( struct bfa_ioim_s * ioim , bfa_boolean_t iotov )
{
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* If path tov timer expired , failback with PATHTOV status - these
* IO requests are not normally retried by IO stack .
*
* Otherwise device cameback online and fail it with normal failed
* status so that IO stack retries these failed IO requests .
*/
if ( iotov )
ioim - > io_cbfn = __bfa_cb_ioim_pathtov ;
else {
ioim - > io_cbfn = __bfa_cb_ioim_failed ;
bfa_stats ( ioim - > itnim , iocom_nexus_abort ) ;
}
bfa_cb_queue ( ioim - > bfa , & ioim - > hcb_qe , ioim - > io_cbfn , ioim ) ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Move IO to fcpim global queue since itnim will be
* freed .
*/
list_del ( & ioim - > qe ) ;
list_add_tail ( & ioim - > qe , & ioim - > fcpim - > ioim_comp_q ) ;
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Memory allocation and initialization .
*/
void
2011-06-25 07:24:29 +04:00
bfa_ioim_attach ( struct bfa_fcpim_s * fcpim )
2010-09-15 22:50:55 +04:00
{
struct bfa_ioim_s * ioim ;
2011-06-25 07:24:29 +04:00
struct bfa_fcp_mod_s * fcp = fcpim - > fcp ;
2010-09-15 22:50:55 +04:00
struct bfa_ioim_sp_s * iosp ;
u16 i ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* claim memory first
*/
2011-06-25 07:24:29 +04:00
ioim = ( struct bfa_ioim_s * ) bfa_mem_kva_curp ( fcp ) ;
2010-09-15 22:50:55 +04:00
fcpim - > ioim_arr = ioim ;
2011-06-25 07:24:29 +04:00
bfa_mem_kva_curp ( fcp ) = ( u8 * ) ( ioim + fcpim - > fcp - > num_ioim_reqs ) ;
2010-09-15 22:50:55 +04:00
2011-06-25 07:24:29 +04:00
iosp = ( struct bfa_ioim_sp_s * ) bfa_mem_kva_curp ( fcp ) ;
2010-09-15 22:50:55 +04:00
fcpim - > ioim_sp_arr = iosp ;
2011-06-25 07:24:29 +04:00
bfa_mem_kva_curp ( fcp ) = ( u8 * ) ( iosp + fcpim - > fcp - > num_ioim_reqs ) ;
2010-09-15 22:50:55 +04:00
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Initialize ioim free queues
*/
INIT_LIST_HEAD ( & fcpim - > ioim_resfree_q ) ;
INIT_LIST_HEAD ( & fcpim - > ioim_comp_q ) ;
2011-06-14 02:53:58 +04:00
for ( i = 0 ; i < fcpim - > fcp - > num_ioim_reqs ;
i + + , ioim + + , iosp + + ) {
2010-09-15 22:50:55 +04:00
/*
* initialize IOIM
*/
2010-10-19 04:08:54 +04:00
memset ( ioim , 0 , sizeof ( struct bfa_ioim_s ) ) ;
2010-09-15 22:50:55 +04:00
ioim - > iotag = i ;
ioim - > bfa = fcpim - > bfa ;
ioim - > fcpim = fcpim ;
ioim - > iosp = iosp ;
2011-07-21 04:04:03 +04:00
ioim - > proc_rsp_data = bfa_ioim_lm_proc_rsp_data_dummy ;
2010-09-15 22:50:55 +04:00
INIT_LIST_HEAD ( & ioim - > sgpg_q ) ;
bfa_reqq_winit ( & ioim - > iosp - > reqq_wait ,
bfa_ioim_qresume , ioim ) ;
bfa_sgpg_winit ( & ioim - > iosp - > sgpg_wqe ,
bfa_ioim_sgpg_alloced , ioim ) ;
bfa_sm_set_state ( ioim , bfa_ioim_sm_uninit ) ;
}
}
void
bfa_ioim_isr ( struct bfa_s * bfa , struct bfi_msg_s * m )
{
2011-06-14 02:53:58 +04:00
struct bfa_fcpim_s * fcpim = BFA_FCPIM ( bfa ) ;
2010-09-15 22:50:55 +04:00
struct bfi_ioim_rsp_s * rsp = ( struct bfi_ioim_rsp_s * ) m ;
struct bfa_ioim_s * ioim ;
u16 iotag ;
enum bfa_ioim_event evt = BFA_IOIM_SM_COMP ;
2010-10-19 04:10:50 +04:00
iotag = be16_to_cpu ( rsp - > io_tag ) ;
2010-09-15 22:50:55 +04:00
ioim = BFA_IOIM_FROM_TAG ( fcpim , iotag ) ;
2010-12-27 08:46:35 +03:00
WARN_ON ( ioim - > iotag ! = iotag ) ;
2010-09-15 22:50:55 +04:00
bfa_trc ( ioim - > bfa , ioim - > iotag ) ;
bfa_trc ( ioim - > bfa , rsp - > io_status ) ;
bfa_trc ( ioim - > bfa , rsp - > reuse_io_tag ) ;
if ( bfa_sm_cmp_state ( ioim , bfa_ioim_sm_active ) )
2010-10-19 04:08:54 +04:00
ioim - > iosp - > comp_rspmsg = * m ;
2010-09-15 22:50:55 +04:00
switch ( rsp - > io_status ) {
case BFI_IOIM_STS_OK :
bfa_stats ( ioim - > itnim , iocomp_ok ) ;
if ( rsp - > reuse_io_tag = = 0 )
evt = BFA_IOIM_SM_DONE ;
else
evt = BFA_IOIM_SM_COMP ;
2011-07-21 04:04:03 +04:00
ioim - > proc_rsp_data ( ioim ) ;
2010-09-15 22:50:55 +04:00
break ;
case BFI_IOIM_STS_TIMEDOUT :
bfa_stats ( ioim - > itnim , iocomp_timedout ) ;
case BFI_IOIM_STS_ABORTED :
rsp - > io_status = BFI_IOIM_STS_ABORTED ;
bfa_stats ( ioim - > itnim , iocomp_aborted ) ;
if ( rsp - > reuse_io_tag = = 0 )
evt = BFA_IOIM_SM_DONE ;
else
evt = BFA_IOIM_SM_COMP ;
break ;
case BFI_IOIM_STS_PROTO_ERR :
bfa_stats ( ioim - > itnim , iocom_proto_err ) ;
2010-12-27 08:46:35 +03:00
WARN_ON ( ! rsp - > reuse_io_tag ) ;
2010-09-15 22:50:55 +04:00
evt = BFA_IOIM_SM_COMP ;
break ;
case BFI_IOIM_STS_SQER_NEEDED :
bfa_stats ( ioim - > itnim , iocom_sqer_needed ) ;
2010-12-27 08:46:35 +03:00
WARN_ON ( rsp - > reuse_io_tag ! = 0 ) ;
2010-09-15 22:50:55 +04:00
evt = BFA_IOIM_SM_SQRETRY ;
break ;
case BFI_IOIM_STS_RES_FREE :
bfa_stats ( ioim - > itnim , iocom_res_free ) ;
evt = BFA_IOIM_SM_FREE ;
break ;
case BFI_IOIM_STS_HOST_ABORTED :
bfa_stats ( ioim - > itnim , iocom_hostabrts ) ;
if ( rsp - > abort_tag ! = ioim - > abort_tag ) {
bfa_trc ( ioim - > bfa , rsp - > abort_tag ) ;
bfa_trc ( ioim - > bfa , ioim - > abort_tag ) ;
2011-07-21 04:04:03 +04:00
ioim - > proc_rsp_data = bfa_ioim_lm_proc_rsp_data_dummy ;
2010-09-15 22:50:55 +04:00
return ;
}
if ( rsp - > reuse_io_tag )
evt = BFA_IOIM_SM_ABORT_COMP ;
else
evt = BFA_IOIM_SM_ABORT_DONE ;
break ;
case BFI_IOIM_STS_UTAG :
bfa_stats ( ioim - > itnim , iocom_utags ) ;
evt = BFA_IOIM_SM_COMP_UTAG ;
break ;
default :
2010-12-27 08:46:35 +03:00
WARN_ON ( 1 ) ;
2010-09-15 22:50:55 +04:00
}
2011-07-21 04:04:03 +04:00
ioim - > proc_rsp_data = bfa_ioim_lm_proc_rsp_data_dummy ;
2010-09-15 22:50:55 +04:00
bfa_sm_send_event ( ioim , evt ) ;
}
void
bfa_ioim_good_comp_isr ( struct bfa_s * bfa , struct bfi_msg_s * m )
{
2011-06-14 02:53:58 +04:00
struct bfa_fcpim_s * fcpim = BFA_FCPIM ( bfa ) ;
2010-09-15 22:50:55 +04:00
struct bfi_ioim_rsp_s * rsp = ( struct bfi_ioim_rsp_s * ) m ;
struct bfa_ioim_s * ioim ;
u16 iotag ;
2010-10-19 04:10:50 +04:00
iotag = be16_to_cpu ( rsp - > io_tag ) ;
2010-09-15 22:50:55 +04:00
ioim = BFA_IOIM_FROM_TAG ( fcpim , iotag ) ;
2010-12-27 08:46:35 +03:00
WARN_ON ( BFA_IOIM_TAG_2_ID ( ioim - > iotag ) ! = iotag ) ;
2010-09-15 22:50:55 +04:00
bfa_ioim_cb_profile_comp ( fcpim , ioim ) ;
2011-07-21 04:04:03 +04:00
if ( bfa_get_lun_mask_status ( bfa ) ! = BFA_LUNMASK_ENABLED ) {
bfa_sm_send_event ( ioim , BFA_IOIM_SM_COMP_GOOD ) ;
return ;
}
if ( ioim - > proc_rsp_data ( ioim ) = = BFA_TRUE )
bfa_sm_send_event ( ioim , BFA_IOIM_SM_COMP_GOOD ) ;
else
bfa_sm_send_event ( ioim , BFA_IOIM_SM_COMP ) ;
2010-09-15 22:50:55 +04:00
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Called by itnim to clean up IO while going offline .
*/
void
bfa_ioim_cleanup ( struct bfa_ioim_s * ioim )
{
bfa_trc ( ioim - > bfa , ioim - > iotag ) ;
bfa_stats ( ioim - > itnim , io_cleanups ) ;
ioim - > iosp - > tskim = NULL ;
bfa_sm_send_event ( ioim , BFA_IOIM_SM_CLEANUP ) ;
}
void
bfa_ioim_cleanup_tm ( struct bfa_ioim_s * ioim , struct bfa_tskim_s * tskim )
{
bfa_trc ( ioim - > bfa , ioim - > iotag ) ;
bfa_stats ( ioim - > itnim , io_tmaborts ) ;
ioim - > iosp - > tskim = tskim ;
bfa_sm_send_event ( ioim , BFA_IOIM_SM_CLEANUP ) ;
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* IOC failure handling .
*/
void
bfa_ioim_iocdisable ( struct bfa_ioim_s * ioim )
{
bfa_trc ( ioim - > bfa , ioim - > iotag ) ;
bfa_stats ( ioim - > itnim , io_iocdowns ) ;
bfa_sm_send_event ( ioim , BFA_IOIM_SM_HWFAIL ) ;
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* IO offline TOV popped . Fail the pending IO .
*/
void
bfa_ioim_tov ( struct bfa_ioim_s * ioim )
{
bfa_trc ( ioim - > bfa , ioim - > iotag ) ;
bfa_sm_send_event ( ioim , BFA_IOIM_SM_IOTOV ) ;
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Allocate IOIM resource for initiator mode I / O request .
*/
struct bfa_ioim_s *
bfa_ioim_alloc ( struct bfa_s * bfa , struct bfad_ioim_s * dio ,
struct bfa_itnim_s * itnim , u16 nsges )
{
2011-06-14 02:53:58 +04:00
struct bfa_fcpim_s * fcpim = BFA_FCPIM ( bfa ) ;
2010-09-15 22:50:55 +04:00
struct bfa_ioim_s * ioim ;
2011-06-14 02:53:58 +04:00
struct bfa_iotag_s * iotag = NULL ;
2010-09-15 22:50:55 +04:00
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* alocate IOIM resource
*/
2011-06-14 02:53:58 +04:00
bfa_q_deq ( & fcpim - > fcp - > iotag_ioim_free_q , & iotag ) ;
if ( ! iotag ) {
2010-09-15 22:50:55 +04:00
bfa_stats ( itnim , no_iotags ) ;
return NULL ;
}
2011-06-14 02:53:58 +04:00
ioim = BFA_IOIM_FROM_TAG ( fcpim , iotag - > tag ) ;
2010-09-15 22:50:55 +04:00
ioim - > dio = dio ;
ioim - > itnim = itnim ;
ioim - > nsges = nsges ;
ioim - > nsgpgs = 0 ;
bfa_stats ( itnim , total_ios ) ;
fcpim - > ios_active + + ;
list_add_tail ( & ioim - > qe , & itnim - > io_q ) ;
return ioim ;
}
void
bfa_ioim_free ( struct bfa_ioim_s * ioim )
{
2011-06-14 02:53:58 +04:00
struct bfa_fcpim_s * fcpim = ioim - > fcpim ;
struct bfa_iotag_s * iotag ;
2010-09-15 22:50:55 +04:00
if ( ioim - > nsgpgs > 0 )
bfa_sgpg_mfree ( ioim - > bfa , & ioim - > sgpg_q , ioim - > nsgpgs ) ;
bfa_stats ( ioim - > itnim , io_comps ) ;
fcpim - > ios_active - - ;
2010-12-14 03:23:27 +03:00
ioim - > iotag & = BFA_IOIM_IOTAG_MASK ;
2011-06-14 02:53:58 +04:00
WARN_ON ( ! ( ioim - > iotag <
( fcpim - > fcp - > num_ioim_reqs + fcpim - > fcp - > num_fwtio_reqs ) ) ) ;
iotag = BFA_IOTAG_FROM_TAG ( fcpim - > fcp , ioim - > iotag ) ;
if ( ioim - > iotag < fcpim - > fcp - > num_ioim_reqs )
list_add_tail ( & iotag - > qe , & fcpim - > fcp - > iotag_ioim_free_q ) ;
else
list_add_tail ( & iotag - > qe , & fcpim - > fcp - > iotag_tio_free_q ) ;
2010-09-15 22:50:55 +04:00
list_del ( & ioim - > qe ) ;
}
void
bfa_ioim_start ( struct bfa_ioim_s * ioim )
{
2011-07-21 04:04:03 +04:00
struct scsi_cmnd * cmnd = ( struct scsi_cmnd * ) ioim - > dio ;
struct bfa_lps_s * lps ;
enum bfa_ioim_lm_status status ;
struct scsi_lun scsilun ;
if ( bfa_get_lun_mask_status ( ioim - > bfa ) = = BFA_LUNMASK_ENABLED ) {
lps = BFA_IOIM_TO_LPS ( ioim ) ;
int_to_scsilun ( cmnd - > device - > lun , & scsilun ) ;
status = bfa_ioim_lm_check ( ioim , lps ,
ioim - > itnim - > rport , scsilun ) ;
if ( status = = BFA_IOIM_LM_LUN_NOT_RDY ) {
bfa_sm_send_event ( ioim , BFA_IOIM_SM_LM_LUN_NOT_RDY ) ;
bfa_stats ( ioim - > itnim , lm_lun_not_rdy ) ;
return ;
}
if ( status = = BFA_IOIM_LM_LUN_NOT_SUP ) {
bfa_sm_send_event ( ioim , BFA_IOIM_SM_LM_LUN_NOT_SUP ) ;
bfa_stats ( ioim - > itnim , lm_lun_not_sup ) ;
return ;
}
if ( status = = BFA_IOIM_LM_RPL_DATA_CHANGED ) {
bfa_sm_send_event ( ioim , BFA_IOIM_SM_LM_RPL_DC ) ;
bfa_stats ( ioim - > itnim , lm_rpl_data_changed ) ;
return ;
}
}
2010-09-15 22:50:55 +04:00
bfa_ioim_cb_profile_start ( ioim - > fcpim , ioim ) ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Obtain the queue over which this request has to be issued
*/
ioim - > reqq = bfa_fcpim_ioredirect_enabled ( ioim - > bfa ) ?
2010-12-10 06:11:39 +03:00
BFA_FALSE : bfa_itnim_get_reqq ( ioim ) ;
2010-09-15 22:50:55 +04:00
bfa_sm_send_event ( ioim , BFA_IOIM_SM_START ) ;
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Driver I / O abort request .
*/
bfa_status_t
bfa_ioim_abort ( struct bfa_ioim_s * ioim )
{
bfa_trc ( ioim - > bfa , ioim - > iotag ) ;
if ( ! bfa_ioim_is_abortable ( ioim ) )
return BFA_STATUS_FAILED ;
bfa_stats ( ioim - > itnim , io_aborts ) ;
bfa_sm_send_event ( ioim , BFA_IOIM_SM_ABORT ) ;
return BFA_STATUS_OK ;
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* BFA TSKIM state machine functions
*/
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* Task management command beginning state .
2010-09-15 22:50:55 +04:00
*/
static void
bfa_tskim_sm_uninit ( struct bfa_tskim_s * tskim , enum bfa_tskim_event event )
{
bfa_trc ( tskim - > bfa , event ) ;
switch ( event ) {
case BFA_TSKIM_SM_START :
bfa_sm_set_state ( tskim , bfa_tskim_sm_active ) ;
bfa_tskim_gather_ios ( tskim ) ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* If device is offline , do not send TM on wire . Just cleanup
* any pending IO requests and complete TM request .
*/
if ( ! bfa_itnim_is_online ( tskim - > itnim ) ) {
bfa_sm_set_state ( tskim , bfa_tskim_sm_iocleanup ) ;
tskim - > tsk_status = BFI_TSKIM_STS_OK ;
bfa_tskim_cleanup_ios ( tskim ) ;
return ;
}
if ( ! bfa_tskim_send ( tskim ) ) {
bfa_sm_set_state ( tskim , bfa_tskim_sm_qfull ) ;
bfa_stats ( tskim - > itnim , tm_qwait ) ;
bfa_reqq_wait ( tskim - > bfa , tskim - > itnim - > reqq ,
& tskim - > reqq_wait ) ;
}
break ;
default :
bfa_sm_fault ( tskim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* TM command is active , awaiting completion from firmware to
* cleanup IO requests in TM scope .
2010-09-15 22:50:55 +04:00
*/
static void
bfa_tskim_sm_active ( struct bfa_tskim_s * tskim , enum bfa_tskim_event event )
{
bfa_trc ( tskim - > bfa , event ) ;
switch ( event ) {
case BFA_TSKIM_SM_DONE :
bfa_sm_set_state ( tskim , bfa_tskim_sm_iocleanup ) ;
bfa_tskim_cleanup_ios ( tskim ) ;
break ;
case BFA_TSKIM_SM_CLEANUP :
bfa_sm_set_state ( tskim , bfa_tskim_sm_cleanup ) ;
if ( ! bfa_tskim_send_abort ( tskim ) ) {
bfa_sm_set_state ( tskim , bfa_tskim_sm_cleanup_qfull ) ;
bfa_stats ( tskim - > itnim , tm_qwait ) ;
bfa_reqq_wait ( tskim - > bfa , tskim - > itnim - > reqq ,
& tskim - > reqq_wait ) ;
}
break ;
case BFA_TSKIM_SM_HWFAIL :
bfa_sm_set_state ( tskim , bfa_tskim_sm_hcb ) ;
bfa_tskim_iocdisable_ios ( tskim ) ;
bfa_tskim_qcomp ( tskim , __bfa_cb_tskim_failed ) ;
break ;
default :
bfa_sm_fault ( tskim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* An active TM is being cleaned up since ITN is offline . Awaiting cleanup
* completion event from firmware .
2010-09-15 22:50:55 +04:00
*/
static void
bfa_tskim_sm_cleanup ( struct bfa_tskim_s * tskim , enum bfa_tskim_event event )
{
bfa_trc ( tskim - > bfa , event ) ;
switch ( event ) {
case BFA_TSKIM_SM_DONE :
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Ignore and wait for ABORT completion from firmware .
*/
break ;
case BFA_TSKIM_SM_CLEANUP_DONE :
bfa_sm_set_state ( tskim , bfa_tskim_sm_iocleanup ) ;
bfa_tskim_cleanup_ios ( tskim ) ;
break ;
case BFA_TSKIM_SM_HWFAIL :
bfa_sm_set_state ( tskim , bfa_tskim_sm_hcb ) ;
bfa_tskim_iocdisable_ios ( tskim ) ;
bfa_tskim_qcomp ( tskim , __bfa_cb_tskim_failed ) ;
break ;
default :
bfa_sm_fault ( tskim - > bfa , event ) ;
}
}
static void
bfa_tskim_sm_iocleanup ( struct bfa_tskim_s * tskim , enum bfa_tskim_event event )
{
bfa_trc ( tskim - > bfa , event ) ;
switch ( event ) {
case BFA_TSKIM_SM_IOS_DONE :
bfa_sm_set_state ( tskim , bfa_tskim_sm_hcb ) ;
bfa_tskim_qcomp ( tskim , __bfa_cb_tskim_done ) ;
break ;
case BFA_TSKIM_SM_CLEANUP :
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Ignore , TM command completed on wire .
* Notify TM conmpletion on IO cleanup completion .
*/
break ;
case BFA_TSKIM_SM_HWFAIL :
bfa_sm_set_state ( tskim , bfa_tskim_sm_hcb ) ;
bfa_tskim_iocdisable_ios ( tskim ) ;
bfa_tskim_qcomp ( tskim , __bfa_cb_tskim_failed ) ;
break ;
default :
bfa_sm_fault ( tskim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* Task management command is waiting for room in request CQ
2010-09-15 22:50:55 +04:00
*/
static void
bfa_tskim_sm_qfull ( struct bfa_tskim_s * tskim , enum bfa_tskim_event event )
{
bfa_trc ( tskim - > bfa , event ) ;
switch ( event ) {
case BFA_TSKIM_SM_QRESUME :
bfa_sm_set_state ( tskim , bfa_tskim_sm_active ) ;
bfa_tskim_send ( tskim ) ;
break ;
case BFA_TSKIM_SM_CLEANUP :
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* No need to send TM on wire since ITN is offline .
*/
bfa_sm_set_state ( tskim , bfa_tskim_sm_iocleanup ) ;
bfa_reqq_wcancel ( & tskim - > reqq_wait ) ;
bfa_tskim_cleanup_ios ( tskim ) ;
break ;
case BFA_TSKIM_SM_HWFAIL :
bfa_sm_set_state ( tskim , bfa_tskim_sm_hcb ) ;
bfa_reqq_wcancel ( & tskim - > reqq_wait ) ;
bfa_tskim_iocdisable_ios ( tskim ) ;
bfa_tskim_qcomp ( tskim , __bfa_cb_tskim_failed ) ;
break ;
default :
bfa_sm_fault ( tskim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* Task management command is active , awaiting for room in request CQ
* to send clean up request .
2010-09-15 22:50:55 +04:00
*/
static void
bfa_tskim_sm_cleanup_qfull ( struct bfa_tskim_s * tskim ,
enum bfa_tskim_event event )
{
bfa_trc ( tskim - > bfa , event ) ;
switch ( event ) {
case BFA_TSKIM_SM_DONE :
bfa_reqq_wcancel ( & tskim - > reqq_wait ) ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Fall through ! ! !
*/
case BFA_TSKIM_SM_QRESUME :
bfa_sm_set_state ( tskim , bfa_tskim_sm_cleanup ) ;
bfa_tskim_send_abort ( tskim ) ;
break ;
case BFA_TSKIM_SM_HWFAIL :
bfa_sm_set_state ( tskim , bfa_tskim_sm_hcb ) ;
bfa_reqq_wcancel ( & tskim - > reqq_wait ) ;
bfa_tskim_iocdisable_ios ( tskim ) ;
bfa_tskim_qcomp ( tskim , __bfa_cb_tskim_failed ) ;
break ;
default :
bfa_sm_fault ( tskim - > bfa , event ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* BFA callback is pending
2010-09-15 22:50:55 +04:00
*/
static void
bfa_tskim_sm_hcb ( struct bfa_tskim_s * tskim , enum bfa_tskim_event event )
{
bfa_trc ( tskim - > bfa , event ) ;
switch ( event ) {
case BFA_TSKIM_SM_HCB :
bfa_sm_set_state ( tskim , bfa_tskim_sm_uninit ) ;
bfa_tskim_free ( tskim ) ;
break ;
case BFA_TSKIM_SM_CLEANUP :
bfa_tskim_notify_comp ( tskim ) ;
break ;
case BFA_TSKIM_SM_HWFAIL :
break ;
default :
bfa_sm_fault ( tskim - > bfa , event ) ;
}
}
static void
__bfa_cb_tskim_done ( void * cbarg , bfa_boolean_t complete )
{
struct bfa_tskim_s * tskim = cbarg ;
if ( ! complete ) {
bfa_sm_send_event ( tskim , BFA_TSKIM_SM_HCB ) ;
return ;
}
bfa_stats ( tskim - > itnim , tm_success ) ;
bfa_cb_tskim_done ( tskim - > bfa - > bfad , tskim - > dtsk , tskim - > tsk_status ) ;
}
static void
__bfa_cb_tskim_failed ( void * cbarg , bfa_boolean_t complete )
{
struct bfa_tskim_s * tskim = cbarg ;
if ( ! complete ) {
bfa_sm_send_event ( tskim , BFA_TSKIM_SM_HCB ) ;
return ;
}
bfa_stats ( tskim - > itnim , tm_failures ) ;
bfa_cb_tskim_done ( tskim - > bfa - > bfad , tskim - > dtsk ,
BFI_TSKIM_STS_FAILED ) ;
}
2010-12-10 06:13:20 +03:00
static bfa_boolean_t
2010-12-10 06:11:39 +03:00
bfa_tskim_match_scope ( struct bfa_tskim_s * tskim , struct scsi_lun lun )
2010-09-15 22:50:55 +04:00
{
switch ( tskim - > tm_cmnd ) {
case FCP_TM_TARGET_RESET :
return BFA_TRUE ;
case FCP_TM_ABORT_TASK_SET :
case FCP_TM_CLEAR_TASK_SET :
case FCP_TM_LUN_RESET :
case FCP_TM_CLEAR_ACA :
2010-12-10 06:13:20 +03:00
return ! memcmp ( & tskim - > lun , & lun , sizeof ( lun ) ) ;
2010-09-15 22:50:55 +04:00
default :
2010-12-27 08:46:35 +03:00
WARN_ON ( 1 ) ;
2010-09-15 22:50:55 +04:00
}
return BFA_FALSE ;
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* Gather affected IO requests and task management commands .
2010-09-15 22:50:55 +04:00
*/
static void
bfa_tskim_gather_ios ( struct bfa_tskim_s * tskim )
{
struct bfa_itnim_s * itnim = tskim - > itnim ;
struct bfa_ioim_s * ioim ;
2010-12-10 06:11:39 +03:00
struct list_head * qe , * qen ;
struct scsi_cmnd * cmnd ;
struct scsi_lun scsilun ;
2010-09-15 22:50:55 +04:00
INIT_LIST_HEAD ( & tskim - > io_q ) ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Gather any active IO requests first .
*/
list_for_each_safe ( qe , qen , & itnim - > io_q ) {
ioim = ( struct bfa_ioim_s * ) qe ;
2010-12-10 06:11:39 +03:00
cmnd = ( struct scsi_cmnd * ) ioim - > dio ;
int_to_scsilun ( cmnd - > device - > lun , & scsilun ) ;
if ( bfa_tskim_match_scope ( tskim , scsilun ) ) {
2010-09-15 22:50:55 +04:00
list_del ( & ioim - > qe ) ;
list_add_tail ( & ioim - > qe , & tskim - > io_q ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Failback any pending IO requests immediately .
*/
list_for_each_safe ( qe , qen , & itnim - > pending_q ) {
ioim = ( struct bfa_ioim_s * ) qe ;
2010-12-10 06:11:39 +03:00
cmnd = ( struct scsi_cmnd * ) ioim - > dio ;
int_to_scsilun ( cmnd - > device - > lun , & scsilun ) ;
if ( bfa_tskim_match_scope ( tskim , scsilun ) ) {
2010-09-15 22:50:55 +04:00
list_del ( & ioim - > qe ) ;
list_add_tail ( & ioim - > qe , & ioim - > fcpim - > ioim_comp_q ) ;
bfa_ioim_tov ( ioim ) ;
}
}
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* IO cleanup completion
2010-09-15 22:50:55 +04:00
*/
static void
bfa_tskim_cleanp_comp ( void * tskim_cbarg )
{
struct bfa_tskim_s * tskim = tskim_cbarg ;
bfa_stats ( tskim - > itnim , tm_io_comps ) ;
bfa_sm_send_event ( tskim , BFA_TSKIM_SM_IOS_DONE ) ;
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* Gather affected IO requests and task management commands .
2010-09-15 22:50:55 +04:00
*/
static void
bfa_tskim_cleanup_ios ( struct bfa_tskim_s * tskim )
{
struct bfa_ioim_s * ioim ;
struct list_head * qe , * qen ;
bfa_wc_init ( & tskim - > wc , bfa_tskim_cleanp_comp , tskim ) ;
list_for_each_safe ( qe , qen , & tskim - > io_q ) {
ioim = ( struct bfa_ioim_s * ) qe ;
bfa_wc_up ( & tskim - > wc ) ;
bfa_ioim_cleanup_tm ( ioim , tskim ) ;
}
bfa_wc_wait ( & tskim - > wc ) ;
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* Send task management request to firmware .
2010-09-15 22:50:55 +04:00
*/
static bfa_boolean_t
bfa_tskim_send ( struct bfa_tskim_s * tskim )
{
struct bfa_itnim_s * itnim = tskim - > itnim ;
struct bfi_tskim_req_s * m ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* check for room in queue to send request now
*/
m = bfa_reqq_next ( tskim - > bfa , itnim - > reqq ) ;
if ( ! m )
return BFA_FALSE ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* build i / o request message next
*/
bfi_h2i_set ( m - > mh , BFI_MC_TSKIM , BFI_TSKIM_H2I_TM_REQ ,
2011-06-25 07:24:08 +04:00
bfa_fn_lpu ( tskim - > bfa ) ) ;
2010-09-15 22:50:55 +04:00
2010-10-19 04:10:50 +04:00
m - > tsk_tag = cpu_to_be16 ( tskim - > tsk_tag ) ;
2010-09-15 22:50:55 +04:00
m - > itn_fhdl = tskim - > itnim - > rport - > fw_handle ;
m - > t_secs = tskim - > tsecs ;
m - > lun = tskim - > lun ;
m - > tm_flags = tskim - > tm_cmnd ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* queue I / O message to firmware
*/
2011-06-25 07:24:08 +04:00
bfa_reqq_produce ( tskim - > bfa , itnim - > reqq , m - > mh ) ;
2010-09-15 22:50:55 +04:00
return BFA_TRUE ;
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* Send abort request to cleanup an active TM to firmware .
2010-09-15 22:50:55 +04:00
*/
static bfa_boolean_t
bfa_tskim_send_abort ( struct bfa_tskim_s * tskim )
{
struct bfa_itnim_s * itnim = tskim - > itnim ;
struct bfi_tskim_abortreq_s * m ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* check for room in queue to send request now
*/
m = bfa_reqq_next ( tskim - > bfa , itnim - > reqq ) ;
if ( ! m )
return BFA_FALSE ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* build i / o request message next
*/
bfi_h2i_set ( m - > mh , BFI_MC_TSKIM , BFI_TSKIM_H2I_ABORT_REQ ,
2011-06-25 07:24:08 +04:00
bfa_fn_lpu ( tskim - > bfa ) ) ;
2010-09-15 22:50:55 +04:00
2010-10-19 04:10:50 +04:00
m - > tsk_tag = cpu_to_be16 ( tskim - > tsk_tag ) ;
2010-09-15 22:50:55 +04:00
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* queue I / O message to firmware
*/
2011-06-25 07:24:08 +04:00
bfa_reqq_produce ( tskim - > bfa , itnim - > reqq , m - > mh ) ;
2010-09-15 22:50:55 +04:00
return BFA_TRUE ;
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* Call to resume task management cmnd waiting for room in request queue .
2010-09-15 22:50:55 +04:00
*/
static void
bfa_tskim_qresume ( void * cbarg )
{
struct bfa_tskim_s * tskim = cbarg ;
bfa_stats ( tskim - > itnim , tm_qresumes ) ;
bfa_sm_send_event ( tskim , BFA_TSKIM_SM_QRESUME ) ;
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Cleanup IOs associated with a task mangement command on IOC failures .
*/
static void
bfa_tskim_iocdisable_ios ( struct bfa_tskim_s * tskim )
{
struct bfa_ioim_s * ioim ;
struct list_head * qe , * qen ;
list_for_each_safe ( qe , qen , & tskim - > io_q ) {
ioim = ( struct bfa_ioim_s * ) qe ;
bfa_ioim_iocdisable ( ioim ) ;
}
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Notification on completions from related ioim .
*/
void
bfa_tskim_iodone ( struct bfa_tskim_s * tskim )
{
bfa_wc_down ( & tskim - > wc ) ;
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Handle IOC h / w failure notification from itnim .
*/
void
bfa_tskim_iocdisable ( struct bfa_tskim_s * tskim )
{
tskim - > notify = BFA_FALSE ;
bfa_stats ( tskim - > itnim , tm_iocdowns ) ;
bfa_sm_send_event ( tskim , BFA_TSKIM_SM_HWFAIL ) ;
}
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Cleanup TM command and associated IOs as part of ITNIM offline .
*/
void
bfa_tskim_cleanup ( struct bfa_tskim_s * tskim )
{
tskim - > notify = BFA_TRUE ;
bfa_stats ( tskim - > itnim , tm_cleanups ) ;
bfa_sm_send_event ( tskim , BFA_TSKIM_SM_CLEANUP ) ;
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* Memory allocation and initialization .
2010-09-15 22:50:55 +04:00
*/
void
2011-06-25 07:24:29 +04:00
bfa_tskim_attach ( struct bfa_fcpim_s * fcpim )
2010-09-15 22:50:55 +04:00
{
struct bfa_tskim_s * tskim ;
2011-06-25 07:24:29 +04:00
struct bfa_fcp_mod_s * fcp = fcpim - > fcp ;
2010-09-15 22:50:55 +04:00
u16 i ;
INIT_LIST_HEAD ( & fcpim - > tskim_free_q ) ;
2011-06-25 07:24:08 +04:00
INIT_LIST_HEAD ( & fcpim - > tskim_unused_q ) ;
2010-09-15 22:50:55 +04:00
2011-06-25 07:24:29 +04:00
tskim = ( struct bfa_tskim_s * ) bfa_mem_kva_curp ( fcp ) ;
2010-09-15 22:50:55 +04:00
fcpim - > tskim_arr = tskim ;
for ( i = 0 ; i < fcpim - > num_tskim_reqs ; i + + , tskim + + ) {
/*
* initialize TSKIM
*/
2010-10-19 04:08:54 +04:00
memset ( tskim , 0 , sizeof ( struct bfa_tskim_s ) ) ;
2010-09-15 22:50:55 +04:00
tskim - > tsk_tag = i ;
tskim - > bfa = fcpim - > bfa ;
tskim - > fcpim = fcpim ;
tskim - > notify = BFA_FALSE ;
bfa_reqq_winit ( & tskim - > reqq_wait , bfa_tskim_qresume ,
tskim ) ;
bfa_sm_set_state ( tskim , bfa_tskim_sm_uninit ) ;
list_add_tail ( & tskim - > qe , & fcpim - > tskim_free_q ) ;
}
2011-06-25 07:24:29 +04:00
bfa_mem_kva_curp ( fcp ) = ( u8 * ) tskim ;
2010-09-15 22:50:55 +04:00
}
void
bfa_tskim_isr ( struct bfa_s * bfa , struct bfi_msg_s * m )
{
2011-06-14 02:53:58 +04:00
struct bfa_fcpim_s * fcpim = BFA_FCPIM ( bfa ) ;
2010-09-15 22:50:55 +04:00
struct bfi_tskim_rsp_s * rsp = ( struct bfi_tskim_rsp_s * ) m ;
struct bfa_tskim_s * tskim ;
2010-10-19 04:10:50 +04:00
u16 tsk_tag = be16_to_cpu ( rsp - > tsk_tag ) ;
2010-09-15 22:50:55 +04:00
tskim = BFA_TSKIM_FROM_TAG ( fcpim , tsk_tag ) ;
2010-12-27 08:46:35 +03:00
WARN_ON ( tskim - > tsk_tag ! = tsk_tag ) ;
2010-09-15 22:50:55 +04:00
tskim - > tsk_status = rsp - > tsk_status ;
2010-10-19 04:17:23 +04:00
/*
2010-09-15 22:50:55 +04:00
* Firmware sends BFI_TSKIM_STS_ABORTED status for abort
* requests . All other statuses are for normal completions .
*/
if ( rsp - > tsk_status = = BFI_TSKIM_STS_ABORTED ) {
bfa_stats ( tskim - > itnim , tm_cleanup_comps ) ;
bfa_sm_send_event ( tskim , BFA_TSKIM_SM_CLEANUP_DONE ) ;
} else {
bfa_stats ( tskim - > itnim , tm_fw_rsps ) ;
bfa_sm_send_event ( tskim , BFA_TSKIM_SM_DONE ) ;
}
}
struct bfa_tskim_s *
bfa_tskim_alloc ( struct bfa_s * bfa , struct bfad_tskim_s * dtsk )
{
2011-06-14 02:53:58 +04:00
struct bfa_fcpim_s * fcpim = BFA_FCPIM ( bfa ) ;
2010-09-15 22:50:55 +04:00
struct bfa_tskim_s * tskim ;
bfa_q_deq ( & fcpim - > tskim_free_q , & tskim ) ;
if ( tskim )
tskim - > dtsk = dtsk ;
return tskim ;
}
void
bfa_tskim_free ( struct bfa_tskim_s * tskim )
{
2010-12-27 08:46:35 +03:00
WARN_ON ( ! bfa_q_is_on_q_func ( & tskim - > itnim - > tsk_q , & tskim - > qe ) ) ;
2010-09-15 22:50:55 +04:00
list_del ( & tskim - > qe ) ;
list_add_tail ( & tskim - > qe , & tskim - > fcpim - > tskim_free_q ) ;
}
2010-10-19 04:17:23 +04:00
/*
2010-12-10 06:13:20 +03:00
* Start a task management command .
2010-09-15 22:50:55 +04:00
*
* @ param [ in ] tskim BFA task management command instance
* @ param [ in ] itnim i - t nexus for the task management command
* @ param [ in ] lun lun , if applicable
* @ param [ in ] tm_cmnd Task management command code .
* @ param [ in ] t_secs Timeout in seconds
*
* @ return None .
*/
void
2010-12-10 06:11:39 +03:00
bfa_tskim_start ( struct bfa_tskim_s * tskim , struct bfa_itnim_s * itnim ,
struct scsi_lun lun ,
2010-09-15 22:50:55 +04:00
enum fcp_tm_cmnd tm_cmnd , u8 tsecs )
{
tskim - > itnim = itnim ;
tskim - > lun = lun ;
tskim - > tm_cmnd = tm_cmnd ;
tskim - > tsecs = tsecs ;
tskim - > notify = BFA_FALSE ;
bfa_stats ( itnim , tm_cmnds ) ;
list_add_tail ( & tskim - > qe , & itnim - > tsk_q ) ;
bfa_sm_send_event ( tskim , BFA_TSKIM_SM_START ) ;
}
2011-06-14 02:53:58 +04:00
2011-06-25 07:24:08 +04:00
void
bfa_tskim_res_recfg ( struct bfa_s * bfa , u16 num_tskim_fw )
{
struct bfa_fcpim_s * fcpim = BFA_FCPIM ( bfa ) ;
struct list_head * qe ;
int i ;
for ( i = 0 ; i < ( fcpim - > num_tskim_reqs - num_tskim_fw ) ; i + + ) {
bfa_q_deq_tail ( & fcpim - > tskim_free_q , & qe ) ;
list_add_tail ( qe , & fcpim - > tskim_unused_q ) ;
}
}
2011-06-14 02:53:58 +04:00
/* BFA FCP module - parent module for fcpim */
BFA_MODULE ( fcp ) ;
static void
2011-06-25 07:24:29 +04:00
bfa_fcp_meminfo ( struct bfa_iocfc_cfg_s * cfg , struct bfa_meminfo_s * minfo ,
struct bfa_s * bfa )
2011-06-14 02:53:58 +04:00
{
2011-06-25 07:24:29 +04:00
struct bfa_fcp_mod_s * fcp = BFA_FCP_MOD ( bfa ) ;
struct bfa_mem_kva_s * fcp_kva = BFA_MEM_FCP_KVA ( bfa ) ;
struct bfa_mem_dma_s * seg_ptr ;
u16 nsegs , idx , per_seg_ios , num_io_req ;
u32 km_len = 0 ;
2011-06-14 02:53:58 +04:00
/*
* ZERO for num_ioim_reqs and num_fwtio_reqs is allowed config value .
* So if the values are non zero , adjust them appropriately .
*/
if ( cfg - > fwcfg . num_ioim_reqs & &
cfg - > fwcfg . num_ioim_reqs < BFA_IOIM_MIN )
cfg - > fwcfg . num_ioim_reqs = BFA_IOIM_MIN ;
else if ( cfg - > fwcfg . num_ioim_reqs > BFA_IOIM_MAX )
cfg - > fwcfg . num_ioim_reqs = BFA_IOIM_MAX ;
if ( cfg - > fwcfg . num_fwtio_reqs > BFA_FWTIO_MAX )
cfg - > fwcfg . num_fwtio_reqs = BFA_FWTIO_MAX ;
num_io_req = ( cfg - > fwcfg . num_ioim_reqs + cfg - > fwcfg . num_fwtio_reqs ) ;
if ( num_io_req > BFA_IO_MAX ) {
if ( cfg - > fwcfg . num_ioim_reqs & & cfg - > fwcfg . num_fwtio_reqs ) {
cfg - > fwcfg . num_ioim_reqs = BFA_IO_MAX / 2 ;
cfg - > fwcfg . num_fwtio_reqs = BFA_IO_MAX / 2 ;
} else if ( cfg - > fwcfg . num_fwtio_reqs )
cfg - > fwcfg . num_fwtio_reqs = BFA_FWTIO_MAX ;
else
cfg - > fwcfg . num_ioim_reqs = BFA_IOIM_MAX ;
}
2011-06-25 07:24:29 +04:00
bfa_fcpim_meminfo ( cfg , & km_len ) ;
2011-06-14 02:53:58 +04:00
num_io_req = ( cfg - > fwcfg . num_ioim_reqs + cfg - > fwcfg . num_fwtio_reqs ) ;
2011-06-25 07:24:29 +04:00
km_len + = num_io_req * sizeof ( struct bfa_iotag_s ) ;
km_len + = cfg - > fwcfg . num_rports * sizeof ( struct bfa_itn_s ) ;
/* dma memory */
nsegs = BFI_MEM_DMA_NSEGS ( num_io_req , BFI_IOIM_SNSLEN ) ;
per_seg_ios = BFI_MEM_NREQS_SEG ( BFI_IOIM_SNSLEN ) ;
bfa_mem_dma_seg_iter ( fcp , seg_ptr , nsegs , idx ) {
if ( num_io_req > = per_seg_ios ) {
num_io_req - = per_seg_ios ;
bfa_mem_dma_setup ( minfo , seg_ptr ,
per_seg_ios * BFI_IOIM_SNSLEN ) ;
} else
bfa_mem_dma_setup ( minfo , seg_ptr ,
num_io_req * BFI_IOIM_SNSLEN ) ;
}
/* kva memory */
bfa_mem_kva_setup ( minfo , fcp_kva , km_len ) ;
2011-06-14 02:53:58 +04:00
}
static void
bfa_fcp_attach ( struct bfa_s * bfa , void * bfad , struct bfa_iocfc_cfg_s * cfg ,
2011-06-25 07:24:29 +04:00
struct bfa_pcidev_s * pcidev )
2011-06-14 02:53:58 +04:00
{
struct bfa_fcp_mod_s * fcp = BFA_FCP_MOD ( bfa ) ;
2011-06-25 07:24:29 +04:00
struct bfa_mem_dma_s * seg_ptr ;
u16 idx , nsegs , num_io_req ;
2011-06-14 02:53:58 +04:00
fcp - > num_ioim_reqs = cfg - > fwcfg . num_ioim_reqs ;
fcp - > num_fwtio_reqs = cfg - > fwcfg . num_fwtio_reqs ;
2011-06-25 07:24:29 +04:00
fcp - > num_itns = cfg - > fwcfg . num_rports ;
2011-06-14 02:53:58 +04:00
fcp - > bfa = bfa ;
2011-06-25 07:24:29 +04:00
/*
* Setup the pool of snsbase addr ' s , that is passed to fw as
* part of bfi_iocfc_cfg_s .
*/
num_io_req = ( cfg - > fwcfg . num_ioim_reqs + cfg - > fwcfg . num_fwtio_reqs ) ;
nsegs = BFI_MEM_DMA_NSEGS ( num_io_req , BFI_IOIM_SNSLEN ) ;
bfa_mem_dma_seg_iter ( fcp , seg_ptr , nsegs , idx ) {
if ( ! bfa_mem_dma_virt ( seg_ptr ) )
break ;
fcp - > snsbase [ idx ] . pa = bfa_mem_dma_phys ( seg_ptr ) ;
fcp - > snsbase [ idx ] . kva = bfa_mem_dma_virt ( seg_ptr ) ;
bfa_iocfc_set_snsbase ( bfa , idx , fcp - > snsbase [ idx ] . pa ) ;
}
2011-06-14 02:53:58 +04:00
2011-06-25 07:24:29 +04:00
bfa_fcpim_attach ( fcp , bfad , cfg , pcidev ) ;
2011-06-14 02:53:58 +04:00
2011-06-25 07:24:29 +04:00
bfa_iotag_attach ( fcp ) ;
2011-06-14 02:53:58 +04:00
2011-06-25 07:24:29 +04:00
fcp - > itn_arr = ( struct bfa_itn_s * ) bfa_mem_kva_curp ( fcp ) ;
bfa_mem_kva_curp ( fcp ) = ( u8 * ) fcp - > itn_arr +
2011-06-14 02:53:58 +04:00
( fcp - > num_itns * sizeof ( struct bfa_itn_s ) ) ;
memset ( fcp - > itn_arr , 0 ,
( fcp - > num_itns * sizeof ( struct bfa_itn_s ) ) ) ;
}
static void
bfa_fcp_detach ( struct bfa_s * bfa )
{
}
static void
bfa_fcp_start ( struct bfa_s * bfa )
{
2011-07-21 04:04:03 +04:00
struct bfa_fcp_mod_s * fcp = BFA_FCP_MOD ( bfa ) ;
/*
* bfa_init ( ) with flash read is complete . now invalidate the stale
* content of lun mask like unit attention , rp tag and lp tag .
*/
bfa_ioim_lm_init ( fcp - > bfa ) ;
2011-06-14 02:53:58 +04:00
}
static void
bfa_fcp_stop ( struct bfa_s * bfa )
{
}
static void
bfa_fcp_iocdisable ( struct bfa_s * bfa )
{
struct bfa_fcp_mod_s * fcp = BFA_FCP_MOD ( bfa ) ;
2011-06-25 07:24:08 +04:00
/* Enqueue unused ioim resources to free_q */
list_splice_tail_init ( & fcp - > iotag_unused_q , & fcp - > iotag_ioim_free_q ) ;
2011-06-14 02:53:58 +04:00
bfa_fcpim_iocdisable ( fcp ) ;
}
2011-06-25 07:24:08 +04:00
void
bfa_fcp_res_recfg ( struct bfa_s * bfa , u16 num_ioim_fw )
{
struct bfa_fcp_mod_s * mod = BFA_FCP_MOD ( bfa ) ;
struct list_head * qe ;
int i ;
for ( i = 0 ; i < ( mod - > num_ioim_reqs - num_ioim_fw ) ; i + + ) {
bfa_q_deq_tail ( & mod - > iotag_ioim_free_q , & qe ) ;
list_add_tail ( qe , & mod - > iotag_unused_q ) ;
}
}
2011-06-14 02:53:58 +04:00
void
bfa_itn_create ( struct bfa_s * bfa , struct bfa_rport_s * rport ,
void ( * isr ) ( struct bfa_s * bfa , struct bfi_msg_s * m ) )
{
struct bfa_fcp_mod_s * fcp = BFA_FCP_MOD ( bfa ) ;
struct bfa_itn_s * itn ;
itn = BFA_ITN_FROM_TAG ( fcp , rport - > rport_tag ) ;
itn - > isr = isr ;
}
/*
* Itn interrupt processing .
*/
void
bfa_itn_isr ( struct bfa_s * bfa , struct bfi_msg_s * m )
{
struct bfa_fcp_mod_s * fcp = BFA_FCP_MOD ( bfa ) ;
union bfi_itn_i2h_msg_u msg ;
struct bfa_itn_s * itn ;
msg . msg = m ;
itn = BFA_ITN_FROM_TAG ( fcp , msg . create_rsp - > bfa_handle ) ;
if ( itn - > isr )
itn - > isr ( bfa , m ) ;
else
WARN_ON ( 1 ) ;
}
void
2011-06-25 07:24:29 +04:00
bfa_iotag_attach ( struct bfa_fcp_mod_s * fcp )
2011-06-14 02:53:58 +04:00
{
struct bfa_iotag_s * iotag ;
u16 num_io_req , i ;
2011-06-25 07:24:29 +04:00
iotag = ( struct bfa_iotag_s * ) bfa_mem_kva_curp ( fcp ) ;
2011-06-14 02:53:58 +04:00
fcp - > iotag_arr = iotag ;
INIT_LIST_HEAD ( & fcp - > iotag_ioim_free_q ) ;
INIT_LIST_HEAD ( & fcp - > iotag_tio_free_q ) ;
2011-06-25 07:24:08 +04:00
INIT_LIST_HEAD ( & fcp - > iotag_unused_q ) ;
2011-06-14 02:53:58 +04:00
num_io_req = fcp - > num_ioim_reqs + fcp - > num_fwtio_reqs ;
for ( i = 0 ; i < num_io_req ; i + + , iotag + + ) {
memset ( iotag , 0 , sizeof ( struct bfa_iotag_s ) ) ;
iotag - > tag = i ;
if ( i < fcp - > num_ioim_reqs )
list_add_tail ( & iotag - > qe , & fcp - > iotag_ioim_free_q ) ;
else
list_add_tail ( & iotag - > qe , & fcp - > iotag_tio_free_q ) ;
}
2011-06-25 07:24:29 +04:00
bfa_mem_kva_curp ( fcp ) = ( u8 * ) iotag ;
2011-06-14 02:53:58 +04:00
}