2008-12-10 02:10:17 +03:00
/*
* Copyright ( c ) 2007 Intel Corporation . All rights reserved .
*
* This program is free software ; you can redistribute it and / or modify it
* under the terms and conditions of the GNU General Public License ,
* version 2 , as published by the Free Software Foundation .
*
* This program is distributed in the hope 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 .
*
* You should have received a copy of the GNU General Public License along with
* this program ; if not , write to the Free Software Foundation , Inc . ,
* 51 Franklin St - Fifth Floor , Boston , MA 02110 - 1301 USA .
*
* Maintained at www . Open - FCoE . org
*/
/*
* PORT LOCKING NOTES
*
* These comments only apply to the ' port code ' which consists of the lport ,
* disc and rport blocks .
*
* MOTIVATION
*
* The lport , disc and rport blocks all have mutexes that are used to protect
* those objects . The main motivation for these locks is to prevent from
* having an lport reset just before we send a frame . In that scenario the
* lport ' s FID would get set to zero and then we ' d send a frame with an
* invalid SID . We also need to ensure that states don ' t change unexpectedly
* while processing another state .
*
* HEIRARCHY
*
* The following heirarchy defines the locking rules . A greater lock
* may be held before acquiring a lesser lock , but a lesser lock should never
* be held while attempting to acquire a greater lock . Here is the heirarchy -
*
* lport > disc , lport > rport , disc > rport
*
* CALLBACKS
*
* The callbacks cause complications with this scheme . There is a callback
* from the rport ( to either lport or disc ) and a callback from disc
* ( to the lport ) .
*
* As rports exit the rport state machine a callback is made to the owner of
* the rport to notify success or failure . Since the callback is likely to
* cause the lport or disc to grab its lock we cannot hold the rport lock
* while making the callback . To ensure that the rport is not free ' d while
* processing the callback the rport callbacks are serialized through a
* single - threaded workqueue . An rport would never be free ' d while in a
* callback handler becuase no other rport work in this queue can be executed
* at the same time .
*
* When discovery succeeds or fails a callback is made to the lport as
* notification . Currently , succesful discovery causes the lport to take no
* action . A failure will cause the lport to reset . There is likely a circular
* locking problem with this implementation .
*/
/*
* LPORT LOCKING
*
* The critical sections protected by the lport ' s mutex are quite broad and
* may be improved upon in the future . The lport code and its locking doesn ' t
* influence the I / O path , so excessive locking doesn ' t penalize I / O
* performance .
*
* The strategy is to lock whenever processing a request or response . Note
* that every _enter_ * function corresponds to a state change . They generally
* change the lports state and then send a request out on the wire . We lock
* before calling any of these functions to protect that state change . This
* means that the entry points into the lport block manage the locks while
* the state machine can transition between states ( i . e . _enter_ * functions )
* while always staying protected .
*
* When handling responses we also hold the lport mutex broadly . When the
* lport receives the response frame it locks the mutex and then calls the
* appropriate handler for the particuar response . Generally a response will
* trigger a state change and so the lock must already be held .
*
* Retries also have to consider the locking . The retries occur from a work
* context and the work function will lock the lport and then retry the state
* ( i . e . _enter_ * function ) .
*/
# include <linux/timer.h>
# include <asm/unaligned.h>
# include <scsi/fc/fc_gs.h>
# include <scsi/libfc.h>
# include <scsi/fc_encode.h>
2009-11-03 22:47:34 +03:00
# include <linux/scatterlist.h>
2008-12-10 02:10:17 +03:00
2009-11-03 22:45:58 +03:00
# include "fc_libfc.h"
2008-12-10 02:10:17 +03:00
/* Fabric IDs to use for point-to-point mode, chosen on whims. */
# define FC_LOCAL_PTP_FID_LO 0x010101
# define FC_LOCAL_PTP_FID_HI 0x010102
# define DNS_DELAY 3 /* Discovery delay after RSCN (in seconds)*/
static void fc_lport_error ( struct fc_lport * , struct fc_frame * ) ;
static void fc_lport_enter_reset ( struct fc_lport * ) ;
static void fc_lport_enter_flogi ( struct fc_lport * ) ;
static void fc_lport_enter_dns ( struct fc_lport * ) ;
2009-11-03 22:47:12 +03:00
static void fc_lport_enter_ns ( struct fc_lport * , enum fc_lport_state ) ;
2008-12-10 02:10:17 +03:00
static void fc_lport_enter_scr ( struct fc_lport * ) ;
static void fc_lport_enter_ready ( struct fc_lport * ) ;
static void fc_lport_enter_logo ( struct fc_lport * ) ;
static const char * fc_lport_state_names [ ] = {
2009-07-30 04:04:22 +04:00
[ LPORT_ST_DISABLED ] = " disabled " ,
2008-12-10 02:10:17 +03:00
[ LPORT_ST_FLOGI ] = " FLOGI " ,
[ LPORT_ST_DNS ] = " dNS " ,
2009-11-03 22:46:51 +03:00
[ LPORT_ST_RNN_ID ] = " RNN_ID " ,
2009-11-03 22:46:56 +03:00
[ LPORT_ST_RSNN_NN ] = " RSNN_NN " ,
2009-11-03 22:47:01 +03:00
[ LPORT_ST_RSPN_ID ] = " RSPN_ID " ,
2008-12-10 02:10:17 +03:00
[ LPORT_ST_RFT_ID ] = " RFT_ID " ,
[ LPORT_ST_SCR ] = " SCR " ,
[ LPORT_ST_READY ] = " Ready " ,
[ LPORT_ST_LOGO ] = " LOGO " ,
[ LPORT_ST_RESET ] = " reset " ,
} ;
2009-11-03 22:47:34 +03:00
/**
* struct fc_bsg_info - FC Passthrough managemet structure
* @ job : The passthrough job
* @ lport : The local port to pass through a command
* @ rsp_code : The expected response code
2009-11-03 22:47:39 +03:00
* @ sg : job - > reply_payload . sg_list
2009-11-03 22:47:34 +03:00
* @ nents : job - > reply_payload . sg_cnt
* @ offset : The offset into the response data
*/
struct fc_bsg_info {
struct fc_bsg_job * job ;
struct fc_lport * lport ;
u16 rsp_code ;
struct scatterlist * sg ;
u32 nents ;
size_t offset ;
} ;
2009-11-03 22:47:39 +03:00
/**
* fc_frame_drop ( ) - Dummy frame handler
* @ lport : The local port the frame was received on
* @ fp : The received frame
*/
2008-12-10 02:10:17 +03:00
static int fc_frame_drop ( struct fc_lport * lport , struct fc_frame * fp )
{
fc_frame_free ( fp ) ;
return 0 ;
}
/**
2009-02-27 21:55:45 +03:00
* fc_lport_rport_callback ( ) - Event handler for rport events
2008-12-10 02:10:17 +03:00
* @ lport : The lport which is receiving the event
2009-08-26 01:00:50 +04:00
* @ rdata : private remote port data
2008-12-10 02:10:17 +03:00
* @ event : The event that occured
*
* Locking Note : The rport lock should not be held when calling
* this function .
*/
static void fc_lport_rport_callback ( struct fc_lport * lport ,
2009-08-26 01:00:50 +04:00
struct fc_rport_priv * rdata ,
2008-12-10 02:10:17 +03:00
enum fc_rport_event event )
{
2009-06-11 02:31:10 +04:00
FC_LPORT_DBG ( lport , " Received a %d event for port (%6x) \n " , event ,
2009-08-26 01:01:01 +04:00
rdata - > ids . port_id ) ;
2008-12-10 02:10:17 +03:00
2009-08-26 01:01:44 +04:00
mutex_lock ( & lport - > lp_mutex ) ;
2008-12-10 02:10:17 +03:00
switch ( event ) {
2009-08-26 01:01:12 +04:00
case RPORT_EV_READY :
2009-08-26 01:01:44 +04:00
if ( lport - > state = = LPORT_ST_DNS ) {
2009-11-03 22:47:39 +03:00
lport - > dns_rdata = rdata ;
2009-11-03 22:47:12 +03:00
fc_lport_enter_ns ( lport , LPORT_ST_RNN_ID ) ;
2009-08-26 01:01:44 +04:00
} else {
FC_LPORT_DBG ( lport , " Received an READY event "
" on port (%6x) for the directory "
" server, but the lport is not "
" in the DNS state, it's in the "
" %d state " , rdata - > ids . port_id ,
lport - > state ) ;
lport - > tt . rport_logoff ( rdata ) ;
}
2008-12-10 02:10:17 +03:00
break ;
case RPORT_EV_LOGO :
case RPORT_EV_FAILED :
case RPORT_EV_STOP :
2009-11-03 22:47:39 +03:00
lport - > dns_rdata = NULL ;
2008-12-10 02:10:17 +03:00
break ;
case RPORT_EV_NONE :
break ;
}
2009-08-26 01:01:44 +04:00
mutex_unlock ( & lport - > lp_mutex ) ;
2008-12-10 02:10:17 +03:00
}
/**
2009-02-27 21:55:45 +03:00
* fc_lport_state ( ) - Return a string which represents the lport ' s state
2008-12-10 02:10:17 +03:00
* @ lport : The lport whose state is to converted to a string
*/
static const char * fc_lport_state ( struct fc_lport * lport )
{
const char * cp ;
cp = fc_lport_state_names [ lport - > state ] ;
if ( ! cp )
cp = " unknown " ;
return cp ;
}
/**
2009-02-27 21:55:45 +03:00
* fc_lport_ptp_setup ( ) - Create an rport for point - to - point mode
2009-11-03 22:47:39 +03:00
* @ lport : The lport to attach the ptp rport to
* @ remote_fid : The FID of the ptp rport
2008-12-10 02:10:17 +03:00
* @ remote_wwpn : The WWPN of the ptp rport
* @ remote_wwnn : The WWNN of the ptp rport
*/
static void fc_lport_ptp_setup ( struct fc_lport * lport ,
u32 remote_fid , u64 remote_wwpn ,
u64 remote_wwnn )
{
2009-08-26 01:01:50 +04:00
mutex_lock ( & lport - > disc . disc_mutex ) ;
2009-11-03 22:47:39 +03:00
if ( lport - > ptp_rdata )
lport - > tt . rport_logoff ( lport - > ptp_rdata ) ;
lport - > ptp_rdata = lport - > tt . rport_create ( lport , remote_fid ) ;
lport - > ptp_rdata - > ids . port_name = remote_wwpn ;
lport - > ptp_rdata - > ids . node_name = remote_wwnn ;
2009-08-26 01:01:50 +04:00
mutex_unlock ( & lport - > disc . disc_mutex ) ;
2008-12-10 02:10:17 +03:00
2009-11-03 22:47:39 +03:00
lport - > tt . rport_login ( lport - > ptp_rdata ) ;
2008-12-10 02:10:17 +03:00
fc_lport_enter_ready ( lport ) ;
}
2009-11-03 22:47:39 +03:00
/**
* fc_get_host_port_type ( ) - Return the port type of the given Scsi_Host
* @ shost : The SCSI host whose port type is to be determined
*/
2008-12-10 02:10:17 +03:00
void fc_get_host_port_type ( struct Scsi_Host * shost )
{
/* TODO - currently just NPORT */
fc_host_port_type ( shost ) = FC_PORTTYPE_NPORT ;
}
EXPORT_SYMBOL ( fc_get_host_port_type ) ;
2009-11-03 22:47:39 +03:00
/**
* fc_get_host_port_state ( ) - Return the port state of the given Scsi_Host
* @ shost : The SCSI host whose port state is to be determined
*/
2008-12-10 02:10:17 +03:00
void fc_get_host_port_state ( struct Scsi_Host * shost )
{
2009-11-03 22:47:39 +03:00
struct fc_lport * lport = shost_priv ( shost ) ;
2008-12-10 02:10:17 +03:00
2009-11-03 22:47:39 +03:00
mutex_lock ( & lport - > lp_mutex ) ;
if ( ! lport - > link_up )
2009-11-03 22:46:19 +03:00
fc_host_port_state ( shost ) = FC_PORTSTATE_LINKDOWN ;
2008-12-10 02:10:17 +03:00
else
2009-11-03 22:47:39 +03:00
switch ( lport - > state ) {
2009-11-03 22:46:19 +03:00
case LPORT_ST_READY :
fc_host_port_state ( shost ) = FC_PORTSTATE_ONLINE ;
break ;
default :
fc_host_port_state ( shost ) = FC_PORTSTATE_OFFLINE ;
}
2009-11-03 22:47:39 +03:00
mutex_unlock ( & lport - > lp_mutex ) ;
2008-12-10 02:10:17 +03:00
}
EXPORT_SYMBOL ( fc_get_host_port_state ) ;
2009-11-03 22:47:39 +03:00
/**
* fc_get_host_speed ( ) - Return the speed of the given Scsi_Host
* @ shost : The SCSI host whose port speed is to be determined
*/
2008-12-10 02:10:17 +03:00
void fc_get_host_speed ( struct Scsi_Host * shost )
{
struct fc_lport * lport = shost_priv ( shost ) ;
fc_host_speed ( shost ) = lport - > link_speed ;
}
EXPORT_SYMBOL ( fc_get_host_speed ) ;
2009-11-03 22:47:39 +03:00
/**
* fc_get_host_stats ( ) - Return the Scsi_Host ' s statistics
* @ shost : The SCSI host whose statistics are to be returned
*/
2008-12-10 02:10:17 +03:00
struct fc_host_statistics * fc_get_host_stats ( struct Scsi_Host * shost )
{
struct fc_host_statistics * fcoe_stats ;
2009-11-03 22:47:39 +03:00
struct fc_lport * lport = shost_priv ( shost ) ;
2008-12-10 02:10:17 +03:00
struct timespec v0 , v1 ;
2009-04-01 02:51:50 +04:00
unsigned int cpu ;
2008-12-10 02:10:17 +03:00
2009-11-03 22:47:39 +03:00
fcoe_stats = & lport - > host_stats ;
2008-12-10 02:10:17 +03:00
memset ( fcoe_stats , 0 , sizeof ( struct fc_host_statistics ) ) ;
jiffies_to_timespec ( jiffies , & v0 ) ;
2009-11-03 22:47:39 +03:00
jiffies_to_timespec ( lport - > boot_time , & v1 ) ;
2008-12-10 02:10:17 +03:00
fcoe_stats - > seconds_since_last_reset = ( v0 . tv_sec - v1 . tv_sec ) ;
2009-04-01 02:51:50 +04:00
for_each_possible_cpu ( cpu ) {
struct fcoe_dev_stats * stats ;
2009-11-03 22:47:39 +03:00
stats = per_cpu_ptr ( lport - > dev_stats , cpu ) ;
2009-04-01 02:51:50 +04:00
2008-12-10 02:10:17 +03:00
fcoe_stats - > tx_frames + = stats - > TxFrames ;
fcoe_stats - > tx_words + = stats - > TxWords ;
fcoe_stats - > rx_frames + = stats - > RxFrames ;
fcoe_stats - > rx_words + = stats - > RxWords ;
fcoe_stats - > error_frames + = stats - > ErrorFrames ;
fcoe_stats - > invalid_crc_count + = stats - > InvalidCRCCount ;
fcoe_stats - > fcp_input_requests + = stats - > InputRequests ;
fcoe_stats - > fcp_output_requests + = stats - > OutputRequests ;
fcoe_stats - > fcp_control_requests + = stats - > ControlRequests ;
fcoe_stats - > fcp_input_megabytes + = stats - > InputMegabytes ;
fcoe_stats - > fcp_output_megabytes + = stats - > OutputMegabytes ;
fcoe_stats - > link_failure_count + = stats - > LinkFailureCount ;
}
fcoe_stats - > lip_count = - 1 ;
fcoe_stats - > nos_count = - 1 ;
fcoe_stats - > loss_of_sync_count = - 1 ;
fcoe_stats - > loss_of_signal_count = - 1 ;
fcoe_stats - > prim_seq_protocol_err_count = - 1 ;
fcoe_stats - > dumped_frames = - 1 ;
return fcoe_stats ;
}
EXPORT_SYMBOL ( fc_get_host_stats ) ;
2009-11-03 22:47:39 +03:00
/**
* fc_lport_flogi_fill ( ) - Fill in FLOGI command for request
* @ lport : The local port the FLOGI is for
* @ flogi : The FLOGI command
* @ op : The opcode
2008-12-10 02:10:17 +03:00
*/
2009-11-03 22:47:39 +03:00
static void fc_lport_flogi_fill ( struct fc_lport * lport ,
struct fc_els_flogi * flogi ,
unsigned int op )
2008-12-10 02:10:17 +03:00
{
struct fc_els_csp * sp ;
struct fc_els_cssp * cp ;
memset ( flogi , 0 , sizeof ( * flogi ) ) ;
flogi - > fl_cmd = ( u8 ) op ;
put_unaligned_be64 ( lport - > wwpn , & flogi - > fl_wwpn ) ;
put_unaligned_be64 ( lport - > wwnn , & flogi - > fl_wwnn ) ;
sp = & flogi - > fl_csp ;
sp - > sp_hi_ver = 0x20 ;
sp - > sp_lo_ver = 0x20 ;
sp - > sp_bb_cred = htons ( 10 ) ; /* this gets set by gateway */
sp - > sp_bb_data = htons ( ( u16 ) lport - > mfs ) ;
cp = & flogi - > fl_cssp [ 3 - 1 ] ; /* class 3 parameters */
cp - > cp_class = htons ( FC_CPC_VALID | FC_CPC_SEQ ) ;
if ( op ! = ELS_FLOGI ) {
sp - > sp_features = htons ( FC_SP_FT_CIRO ) ;
sp - > sp_tot_seq = htons ( 255 ) ; /* seq. we accept */
sp - > sp_rel_off = htons ( 0x1f ) ;
sp - > sp_e_d_tov = htonl ( lport - > e_d_tov ) ;
cp - > cp_rdfs = htons ( ( u16 ) lport - > mfs ) ;
cp - > cp_con_seq = htons ( 255 ) ;
cp - > cp_open_seq = 1 ;
}
}
2009-11-03 22:47:39 +03:00
/**
* fc_lport_add_fc4_type ( ) - Add a supported FC - 4 type to a local port
* @ lport : The local port to add a new FC - 4 type to
* @ type : The new FC - 4 type
2008-12-10 02:10:17 +03:00
*/
static void fc_lport_add_fc4_type ( struct fc_lport * lport , enum fc_fh_type type )
{
__be32 * mp ;
mp = & lport - > fcts . ff_type_map [ type / FC_NS_BPW ] ;
* mp = htonl ( ntohl ( * mp ) | 1UL < < ( type % FC_NS_BPW ) ) ;
}
/**
2009-02-27 21:55:45 +03:00
* fc_lport_recv_rlir_req ( ) - Handle received Registered Link Incident Report .
2009-11-03 22:47:39 +03:00
* @ sp : The sequence in the RLIR exchange
* @ fp : The RLIR request frame
2008-12-10 02:10:17 +03:00
* @ lport : Fibre Channel local port recieving the RLIR
*
2009-10-22 03:27:17 +04:00
* Locking Note : The lport lock is expected to be held before calling
2008-12-10 02:10:17 +03:00
* this function .
*/
static void fc_lport_recv_rlir_req ( struct fc_seq * sp , struct fc_frame * fp ,
struct fc_lport * lport )
{
2009-06-11 02:31:10 +04:00
FC_LPORT_DBG ( lport , " Received RLIR request while in state %s \n " ,
fc_lport_state ( lport ) ) ;
2008-12-10 02:10:17 +03:00
lport - > tt . seq_els_rsp_send ( sp , ELS_LS_ACC , NULL ) ;
fc_frame_free ( fp ) ;
}
/**
2009-02-27 21:55:45 +03:00
* fc_lport_recv_echo_req ( ) - Handle received ECHO request
2009-11-03 22:47:39 +03:00
* @ sp : The sequence in the ECHO exchange
* @ fp : ECHO request frame
* @ lport : The local port recieving the ECHO
2008-12-10 02:10:17 +03:00
*
2009-10-22 03:27:17 +04:00
* Locking Note : The lport lock is expected to be held before calling
2008-12-10 02:10:17 +03:00
* this function .
*/
static void fc_lport_recv_echo_req ( struct fc_seq * sp , struct fc_frame * in_fp ,
struct fc_lport * lport )
{
struct fc_frame * fp ;
struct fc_exch * ep = fc_seq_exch ( sp ) ;
unsigned int len ;
void * pp ;
void * dp ;
u32 f_ctl ;
2009-10-22 03:27:17 +04:00
FC_LPORT_DBG ( lport , " Received ECHO request while in state %s \n " ,
2009-06-11 02:31:10 +04:00
fc_lport_state ( lport ) ) ;
2008-12-10 02:10:17 +03:00
len = fr_len ( in_fp ) - sizeof ( struct fc_frame_header ) ;
pp = fc_frame_payload_get ( in_fp , len ) ;
if ( len < sizeof ( __be32 ) )
len = sizeof ( __be32 ) ;
fp = fc_frame_alloc ( lport , len ) ;
if ( fp ) {
dp = fc_frame_payload_get ( fp , len ) ;
memcpy ( dp , pp , len ) ;
2009-10-22 03:27:17 +04:00
* ( ( __be32 * ) dp ) = htonl ( ELS_LS_ACC < < 24 ) ;
2008-12-10 02:10:17 +03:00
sp = lport - > tt . seq_start_next ( sp ) ;
f_ctl = FC_FC_EX_CTX | FC_FC_LAST_SEQ | FC_FC_END_SEQ ;
fc_fill_fc_hdr ( fp , FC_RCTL_ELS_REP , ep - > did , ep - > sid ,
FC_TYPE_ELS , f_ctl , 0 ) ;
lport - > tt . seq_send ( lport , sp , fp ) ;
}
fc_frame_free ( in_fp ) ;
}
/**
2009-10-22 03:27:17 +04:00
* fc_lport_recv_rnid_req ( ) - Handle received Request Node ID data request
* @ sp : The sequence in the RNID exchange
* @ fp : The RNID request frame
* @ lport : The local port recieving the RNID
2008-12-10 02:10:17 +03:00
*
2009-10-22 03:27:17 +04:00
* Locking Note : The lport lock is expected to be held before calling
2008-12-10 02:10:17 +03:00
* this function .
*/
static void fc_lport_recv_rnid_req ( struct fc_seq * sp , struct fc_frame * in_fp ,
struct fc_lport * lport )
{
struct fc_frame * fp ;
struct fc_exch * ep = fc_seq_exch ( sp ) ;
struct fc_els_rnid * req ;
struct {
struct fc_els_rnid_resp rnid ;
struct fc_els_rnid_cid cid ;
struct fc_els_rnid_gen gen ;
} * rp ;
struct fc_seq_els_data rjt_data ;
u8 fmt ;
size_t len ;
u32 f_ctl ;
2009-06-11 02:31:10 +04:00
FC_LPORT_DBG ( lport , " Received RNID request while in state %s \n " ,
fc_lport_state ( lport ) ) ;
2008-12-10 02:10:17 +03:00
req = fc_frame_payload_get ( in_fp , sizeof ( * req ) ) ;
if ( ! req ) {
rjt_data . fp = NULL ;
rjt_data . reason = ELS_RJT_LOGIC ;
rjt_data . explan = ELS_EXPL_NONE ;
lport - > tt . seq_els_rsp_send ( sp , ELS_LS_RJT , & rjt_data ) ;
} else {
fmt = req - > rnid_fmt ;
len = sizeof ( * rp ) ;
if ( fmt ! = ELS_RNIDF_GEN | |
ntohl ( lport - > rnid_gen . rnid_atype ) = = 0 ) {
fmt = ELS_RNIDF_NONE ; /* nothing to provide */
len - = sizeof ( rp - > gen ) ;
}
fp = fc_frame_alloc ( lport , len ) ;
if ( fp ) {
rp = fc_frame_payload_get ( fp , len ) ;
memset ( rp , 0 , len ) ;
rp - > rnid . rnid_cmd = ELS_LS_ACC ;
rp - > rnid . rnid_fmt = fmt ;
rp - > rnid . rnid_cid_len = sizeof ( rp - > cid ) ;
rp - > cid . rnid_wwpn = htonll ( lport - > wwpn ) ;
rp - > cid . rnid_wwnn = htonll ( lport - > wwnn ) ;
if ( fmt = = ELS_RNIDF_GEN ) {
rp - > rnid . rnid_sid_len = sizeof ( rp - > gen ) ;
memcpy ( & rp - > gen , & lport - > rnid_gen ,
sizeof ( rp - > gen ) ) ;
}
sp = lport - > tt . seq_start_next ( sp ) ;
f_ctl = FC_FC_EX_CTX | FC_FC_LAST_SEQ ;
f_ctl | = FC_FC_END_SEQ | FC_FC_SEQ_INIT ;
fc_fill_fc_hdr ( fp , FC_RCTL_ELS_REP , ep - > did , ep - > sid ,
FC_TYPE_ELS , f_ctl , 0 ) ;
lport - > tt . seq_send ( lport , sp , fp ) ;
}
}
fc_frame_free ( in_fp ) ;
}
/**
2009-02-27 21:55:45 +03:00
* fc_lport_recv_logo_req ( ) - Handle received fabric LOGO request
2009-11-03 22:47:39 +03:00
* @ sp : The sequence in the LOGO exchange
* @ fp : The LOGO request frame
* @ lport : The local port recieving the LOGO
2008-12-10 02:10:17 +03:00
*
* Locking Note : The lport lock is exected to be held before calling
* this function .
*/
static void fc_lport_recv_logo_req ( struct fc_seq * sp , struct fc_frame * fp ,
struct fc_lport * lport )
{
lport - > tt . seq_els_rsp_send ( sp , ELS_LS_ACC , NULL ) ;
fc_lport_enter_reset ( lport ) ;
fc_frame_free ( fp ) ;
}
/**
2009-02-27 21:55:45 +03:00
* fc_fabric_login ( ) - Start the lport state machine
2009-11-03 22:47:39 +03:00
* @ lport : The local port that should log into the fabric
2008-12-10 02:10:17 +03:00
*
* Locking Note : This function should not be called
* with the lport lock held .
*/
int fc_fabric_login ( struct fc_lport * lport )
{
int rc = - 1 ;
mutex_lock ( & lport - > lp_mutex ) ;
2009-07-30 04:04:22 +04:00
if ( lport - > state = = LPORT_ST_DISABLED ) {
2008-12-10 02:10:17 +03:00
fc_lport_enter_reset ( lport ) ;
rc = 0 ;
}
mutex_unlock ( & lport - > lp_mutex ) ;
return rc ;
}
EXPORT_SYMBOL ( fc_fabric_login ) ;
/**
2009-11-03 22:46:19 +03:00
* __fc_linkup ( ) - Handler for transport linkup events
2008-12-10 02:10:17 +03:00
* @ lport : The lport whose link is up
2009-11-03 22:46:19 +03:00
*
* Locking : must be called with the lp_mutex held
2008-12-10 02:10:17 +03:00
*/
2009-11-03 22:46:19 +03:00
void __fc_linkup ( struct fc_lport * lport )
2008-12-10 02:10:17 +03:00
{
2009-02-27 21:54:57 +03:00
if ( ! lport - > link_up ) {
lport - > link_up = 1 ;
2008-12-10 02:10:17 +03:00
if ( lport - > state = = LPORT_ST_RESET )
fc_lport_enter_flogi ( lport ) ;
}
2009-11-03 22:46:19 +03:00
}
/**
* fc_linkup ( ) - Handler for transport linkup events
2009-11-03 22:47:39 +03:00
* @ lport : The local port whose link is up
2009-11-03 22:46:19 +03:00
*/
void fc_linkup ( struct fc_lport * lport )
{
printk ( KERN_INFO " libfc: Link up on port (%6x) \n " ,
fc_host_port_id ( lport - > host ) ) ;
mutex_lock ( & lport - > lp_mutex ) ;
__fc_linkup ( lport ) ;
2008-12-10 02:10:17 +03:00
mutex_unlock ( & lport - > lp_mutex ) ;
}
EXPORT_SYMBOL ( fc_linkup ) ;
/**
2009-11-03 22:46:19 +03:00
* __fc_linkdown ( ) - Handler for transport linkdown events
2008-12-10 02:10:17 +03:00
* @ lport : The lport whose link is down
2009-11-03 22:46:19 +03:00
*
* Locking : must be called with the lp_mutex held
2008-12-10 02:10:17 +03:00
*/
2009-11-03 22:46:19 +03:00
void __fc_linkdown ( struct fc_lport * lport )
2008-12-10 02:10:17 +03:00
{
2009-02-27 21:54:57 +03:00
if ( lport - > link_up ) {
lport - > link_up = 0 ;
2008-12-10 02:10:17 +03:00
fc_lport_enter_reset ( lport ) ;
lport - > tt . fcp_cleanup ( lport ) ;
}
2009-11-03 22:46:19 +03:00
}
/**
* fc_linkdown ( ) - Handler for transport linkdown events
2009-11-03 22:47:39 +03:00
* @ lport : The local port whose link is down
2009-11-03 22:46:19 +03:00
*/
void fc_linkdown ( struct fc_lport * lport )
{
printk ( KERN_INFO " libfc: Link down on port (%6x) \n " ,
fc_host_port_id ( lport - > host ) ) ;
mutex_lock ( & lport - > lp_mutex ) ;
__fc_linkdown ( lport ) ;
2008-12-10 02:10:17 +03:00
mutex_unlock ( & lport - > lp_mutex ) ;
}
EXPORT_SYMBOL ( fc_linkdown ) ;
/**
2009-02-27 21:55:45 +03:00
* fc_fabric_logoff ( ) - Logout of the fabric
2009-11-03 22:47:39 +03:00
* @ lport : The local port to logoff the fabric
2008-12-10 02:10:17 +03:00
*
* Return value :
* 0 for success , - 1 for failure
2009-02-27 21:55:45 +03:00
*/
2008-12-10 02:10:17 +03:00
int fc_fabric_logoff ( struct fc_lport * lport )
{
lport - > tt . disc_stop_final ( lport ) ;
mutex_lock ( & lport - > lp_mutex ) ;
2009-11-03 22:47:39 +03:00
if ( lport - > dns_rdata )
lport - > tt . rport_logoff ( lport - > dns_rdata ) ;
2009-04-22 03:27:09 +04:00
mutex_unlock ( & lport - > lp_mutex ) ;
lport - > tt . rport_flush_queue ( ) ;
mutex_lock ( & lport - > lp_mutex ) ;
2008-12-10 02:10:17 +03:00
fc_lport_enter_logo ( lport ) ;
mutex_unlock ( & lport - > lp_mutex ) ;
2009-02-27 21:55:13 +03:00
cancel_delayed_work_sync ( & lport - > retry_work ) ;
2008-12-10 02:10:17 +03:00
return 0 ;
}
EXPORT_SYMBOL ( fc_fabric_logoff ) ;
/**
2009-11-03 22:47:39 +03:00
* fc_lport_destroy ( ) - Unregister a fc_lport
* @ lport : The local port to unregister
2008-12-10 02:10:17 +03:00
*
* Note :
* exit routine for fc_lport instance
* clean - up all the allocated memory
* and free up other system resources .
*
2009-02-27 21:55:45 +03:00
*/
2008-12-10 02:10:17 +03:00
int fc_lport_destroy ( struct fc_lport * lport )
{
2009-04-22 03:27:14 +04:00
mutex_lock ( & lport - > lp_mutex ) ;
2009-07-30 04:04:22 +04:00
lport - > state = LPORT_ST_DISABLED ;
2009-04-22 03:27:14 +04:00
lport - > link_up = 0 ;
2008-12-10 02:10:17 +03:00
lport - > tt . frame_send = fc_frame_drop ;
2009-04-22 03:27:14 +04:00
mutex_unlock ( & lport - > lp_mutex ) ;
2008-12-10 02:10:17 +03:00
lport - > tt . fcp_abort_io ( lport ) ;
2009-07-30 04:04:33 +04:00
lport - > tt . disc_stop_final ( lport ) ;
2009-02-27 21:54:35 +03:00
lport - > tt . exch_mgr_reset ( lport , 0 , 0 ) ;
2008-12-10 02:10:17 +03:00
return 0 ;
}
EXPORT_SYMBOL ( fc_lport_destroy ) ;
/**
2009-11-03 22:47:39 +03:00
* fc_set_mfs ( ) - Set the maximum frame size for a local port
* @ lport : The local port to set the MFS for
* @ mfs : The new MFS
2009-02-27 21:55:45 +03:00
*/
2008-12-10 02:10:17 +03:00
int fc_set_mfs ( struct fc_lport * lport , u32 mfs )
{
unsigned int old_mfs ;
int rc = - EINVAL ;
mutex_lock ( & lport - > lp_mutex ) ;
old_mfs = lport - > mfs ;
if ( mfs > = FC_MIN_MAX_FRAME ) {
mfs & = ~ 3 ;
if ( mfs > FC_MAX_FRAME )
mfs = FC_MAX_FRAME ;
mfs - = sizeof ( struct fc_frame_header ) ;
lport - > mfs = mfs ;
rc = 0 ;
}
if ( ! rc & & mfs < old_mfs )
fc_lport_enter_reset ( lport ) ;
mutex_unlock ( & lport - > lp_mutex ) ;
return rc ;
}
EXPORT_SYMBOL ( fc_set_mfs ) ;
/**
2009-02-27 21:55:45 +03:00
* fc_lport_disc_callback ( ) - Callback for discovery events
2009-11-03 22:47:39 +03:00
* @ lport : The local port receiving the event
2008-12-10 02:10:17 +03:00
* @ event : The discovery event
*/
void fc_lport_disc_callback ( struct fc_lport * lport , enum fc_disc_event event )
{
switch ( event ) {
case DISC_EV_SUCCESS :
2009-06-11 02:31:10 +04:00
FC_LPORT_DBG ( lport , " Discovery succeeded \n " ) ;
2008-12-10 02:10:17 +03:00
break ;
case DISC_EV_FAILED :
2009-06-11 02:31:10 +04:00
printk ( KERN_ERR " libfc: Discovery failed for port (%6x) \n " ,
fc_host_port_id ( lport - > host ) ) ;
2008-12-10 02:10:17 +03:00
mutex_lock ( & lport - > lp_mutex ) ;
fc_lport_enter_reset ( lport ) ;
mutex_unlock ( & lport - > lp_mutex ) ;
break ;
case DISC_EV_NONE :
WARN_ON ( 1 ) ;
break ;
}
}
/**
2009-02-27 21:55:45 +03:00
* fc_rport_enter_ready ( ) - Enter the ready state and start discovery
2009-11-03 22:47:39 +03:00
* @ lport : The local port that is ready
2008-12-10 02:10:17 +03:00
*
* Locking Note : The lport lock is expected to be held before calling
* this routine .
*/
static void fc_lport_enter_ready ( struct fc_lport * lport )
{
2009-06-11 02:31:10 +04:00
FC_LPORT_DBG ( lport , " Entered READY from state %s \n " ,
fc_lport_state ( lport ) ) ;
2008-12-10 02:10:17 +03:00
fc_lport_state_enter ( lport , LPORT_ST_READY ) ;
2009-11-03 22:46:19 +03:00
if ( lport - > vport )
fc_vport_set_state ( lport - > vport , FC_VPORT_ACTIVE ) ;
fc_vports_linkchange ( lport ) ;
2008-12-10 02:10:17 +03:00
2009-11-03 22:47:39 +03:00
if ( ! lport - > ptp_rdata )
2009-08-26 01:02:49 +04:00
lport - > tt . disc_start ( fc_lport_disc_callback , lport ) ;
2008-12-10 02:10:17 +03:00
}
2009-11-03 22:49:05 +03:00
/**
* fc_lport_set_port_id ( ) - set the local port Port ID
* @ lport : The local port which will have its Port ID set .
* @ port_id : The new port ID .
* @ fp : The frame containing the incoming request , or NULL .
*
* Locking Note : The lport lock is expected to be held before calling
* this function .
*/
static void fc_lport_set_port_id ( struct fc_lport * lport , u32 port_id ,
struct fc_frame * fp )
{
if ( port_id )
printk ( KERN_INFO " host%d: Assigned Port ID %6x \n " ,
lport - > host - > host_no , port_id ) ;
fc_host_port_id ( lport - > host ) = port_id ;
if ( lport - > tt . lport_set_port_id )
lport - > tt . lport_set_port_id ( lport , port_id , fp ) ;
}
2008-12-10 02:10:17 +03:00
/**
2009-02-27 21:55:45 +03:00
* fc_lport_recv_flogi_req ( ) - Receive a FLOGI request
2008-12-10 02:10:17 +03:00
* @ sp_in : The sequence the FLOGI is on
2009-11-03 22:47:39 +03:00
* @ rx_fp : The FLOGI frame
* @ lport : The local port that recieved the request
2008-12-10 02:10:17 +03:00
*
* A received FLOGI request indicates a point - to - point connection .
* Accept it with the common service parameters indicating our N port .
* Set up to do a PLOGI if we have the higher - number WWPN .
*
2009-10-22 03:27:17 +04:00
* Locking Note : The lport lock is expected to be held before calling
2008-12-10 02:10:17 +03:00
* this function .
*/
static void fc_lport_recv_flogi_req ( struct fc_seq * sp_in ,
struct fc_frame * rx_fp ,
struct fc_lport * lport )
{
struct fc_frame * fp ;
struct fc_frame_header * fh ;
struct fc_seq * sp ;
struct fc_exch * ep ;
struct fc_els_flogi * flp ;
struct fc_els_flogi * new_flp ;
u64 remote_wwpn ;
u32 remote_fid ;
u32 local_fid ;
u32 f_ctl ;
2009-06-11 02:31:10 +04:00
FC_LPORT_DBG ( lport , " Received FLOGI request while in state %s \n " ,
fc_lport_state ( lport ) ) ;
2008-12-10 02:10:17 +03:00
fh = fc_frame_header_get ( rx_fp ) ;
remote_fid = ntoh24 ( fh - > fh_s_id ) ;
flp = fc_frame_payload_get ( rx_fp , sizeof ( * flp ) ) ;
if ( ! flp )
goto out ;
remote_wwpn = get_unaligned_be64 ( & flp - > fl_wwpn ) ;
if ( remote_wwpn = = lport - > wwpn ) {
2009-06-11 02:31:10 +04:00
printk ( KERN_WARNING " libfc: Received FLOGI from port "
" with same WWPN %llx \n " , remote_wwpn ) ;
2008-12-10 02:10:17 +03:00
goto out ;
}
2009-06-11 02:31:10 +04:00
FC_LPORT_DBG ( lport , " FLOGI from port WWPN %llx \n " , remote_wwpn ) ;
2008-12-10 02:10:17 +03:00
/*
* XXX what is the right thing to do for FIDs ?
* The originator might expect our S_ID to be 0xfffffe .
* But if so , both of us could end up with the same FID .
*/
local_fid = FC_LOCAL_PTP_FID_LO ;
if ( remote_wwpn < lport - > wwpn ) {
local_fid = FC_LOCAL_PTP_FID_HI ;
if ( ! remote_fid | | remote_fid = = local_fid )
remote_fid = FC_LOCAL_PTP_FID_LO ;
} else if ( ! remote_fid ) {
remote_fid = FC_LOCAL_PTP_FID_HI ;
}
2009-11-03 22:49:05 +03:00
fc_lport_set_port_id ( lport , local_fid , rx_fp ) ;
2008-12-10 02:10:17 +03:00
fp = fc_frame_alloc ( lport , sizeof ( * flp ) ) ;
if ( fp ) {
sp = lport - > tt . seq_start_next ( fr_seq ( rx_fp ) ) ;
new_flp = fc_frame_payload_get ( fp , sizeof ( * flp ) ) ;
fc_lport_flogi_fill ( lport , new_flp , ELS_FLOGI ) ;
new_flp - > fl_cmd = ( u8 ) ELS_LS_ACC ;
/*
* Send the response . If this fails , the originator should
* repeat the sequence .
*/
f_ctl = FC_FC_EX_CTX | FC_FC_LAST_SEQ | FC_FC_END_SEQ ;
ep = fc_seq_exch ( sp ) ;
fc_fill_fc_hdr ( fp , FC_RCTL_ELS_REP , ep - > did , ep - > sid ,
FC_TYPE_ELS , f_ctl , 0 ) ;
lport - > tt . seq_send ( lport , sp , fp ) ;
} else {
fc_lport_error ( lport , fp ) ;
}
fc_lport_ptp_setup ( lport , remote_fid , remote_wwpn ,
get_unaligned_be64 ( & flp - > fl_wwnn ) ) ;
out :
sp = fr_seq ( rx_fp ) ;
fc_frame_free ( rx_fp ) ;
}
/**
2009-02-27 21:55:45 +03:00
* fc_lport_recv_req ( ) - The generic lport request handler
2009-11-03 22:47:39 +03:00
* @ lport : The local port that received the request
* @ sp : The sequence the request is on
* @ fp : The request frame
2008-12-10 02:10:17 +03:00
*
* This function will see if the lport handles the request or
* if an rport should handle the request .
*
* Locking Note : This function should not be called with the lport
* lock held becuase it will grab the lock .
*/
static void fc_lport_recv_req ( struct fc_lport * lport , struct fc_seq * sp ,
struct fc_frame * fp )
{
struct fc_frame_header * fh = fc_frame_header_get ( fp ) ;
void ( * recv ) ( struct fc_seq * , struct fc_frame * , struct fc_lport * ) ;
mutex_lock ( & lport - > lp_mutex ) ;
/*
* Handle special ELS cases like FLOGI , LOGO , and
* RSCN here . These don ' t require a session .
* Even if we had a session , it might not be ready .
*/
2009-07-30 04:04:33 +04:00
if ( ! lport - > link_up )
fc_frame_free ( fp ) ;
else if ( fh - > fh_type = = FC_TYPE_ELS & &
fh - > fh_r_ctl = = FC_RCTL_ELS_REQ ) {
2008-12-10 02:10:17 +03:00
/*
* Check opcode .
*/
2009-08-26 01:03:10 +04:00
recv = lport - > tt . rport_recv_req ;
2008-12-10 02:10:17 +03:00
switch ( fc_frame_payload_op ( fp ) ) {
case ELS_FLOGI :
recv = fc_lport_recv_flogi_req ;
break ;
case ELS_LOGO :
fh = fc_frame_header_get ( fp ) ;
if ( ntoh24 ( fh - > fh_s_id ) = = FC_FID_FLOGI )
recv = fc_lport_recv_logo_req ;
break ;
case ELS_RSCN :
recv = lport - > tt . disc_recv_req ;
break ;
case ELS_ECHO :
recv = fc_lport_recv_echo_req ;
break ;
case ELS_RLIR :
recv = fc_lport_recv_rlir_req ;
break ;
case ELS_RNID :
recv = fc_lport_recv_rnid_req ;
break ;
}
2009-08-26 01:03:10 +04:00
recv ( sp , fp , lport ) ;
2008-12-10 02:10:17 +03:00
} else {
2009-06-11 02:31:10 +04:00
FC_LPORT_DBG ( lport , " dropping invalid frame (eof %x) \n " ,
fr_eof ( fp ) ) ;
2008-12-10 02:10:17 +03:00
fc_frame_free ( fp ) ;
}
mutex_unlock ( & lport - > lp_mutex ) ;
/*
* The common exch_done for all request may not be good
* if any request requires longer hold on exhange . XXX
*/
lport - > tt . exch_done ( sp ) ;
}
/**
2009-11-03 22:47:39 +03:00
* fc_lport_reset ( ) - Reset a local port
* @ lport : The local port which should be reset
2008-12-10 02:10:17 +03:00
*
* Locking Note : This functions should not be called with the
* lport lock held .
*/
int fc_lport_reset ( struct fc_lport * lport )
{
2009-02-27 21:55:13 +03:00
cancel_delayed_work_sync ( & lport - > retry_work ) ;
2008-12-10 02:10:17 +03:00
mutex_lock ( & lport - > lp_mutex ) ;
fc_lport_enter_reset ( lport ) ;
mutex_unlock ( & lport - > lp_mutex ) ;
return 0 ;
}
EXPORT_SYMBOL ( fc_lport_reset ) ;
/**
2009-11-03 22:47:39 +03:00
* fc_lport_reset_locked ( ) - Reset the local port w / the lport lock held
* @ lport : The local port to be reset
2008-12-10 02:10:17 +03:00
*
* Locking Note : The lport lock is expected to be held before calling
* this routine .
*/
2009-07-30 04:04:27 +04:00
static void fc_lport_reset_locked ( struct fc_lport * lport )
2008-12-10 02:10:17 +03:00
{
2009-11-03 22:47:39 +03:00
if ( lport - > dns_rdata )
lport - > tt . rport_logoff ( lport - > dns_rdata ) ;
2008-12-10 02:10:17 +03:00
2009-11-03 22:47:39 +03:00
lport - > ptp_rdata = NULL ;
2008-12-10 02:10:17 +03:00
lport - > tt . disc_stop ( lport ) ;
2009-02-27 21:54:35 +03:00
lport - > tt . exch_mgr_reset ( lport , 0 , 0 ) ;
2008-12-10 02:10:17 +03:00
fc_host_fabric_name ( lport - > host ) = 0 ;
2009-11-03 22:49:05 +03:00
if ( fc_host_port_id ( lport - > host ) )
fc_lport_set_port_id ( lport , 0 , NULL ) ;
2009-07-30 04:04:27 +04:00
}
2008-12-10 02:10:17 +03:00
2009-07-30 04:04:27 +04:00
/**
* fc_lport_enter_reset ( ) - Reset the local port
2009-11-03 22:47:39 +03:00
* @ lport : The local port to be reset
2009-07-30 04:04:27 +04:00
*
* Locking Note : The lport lock is expected to be held before calling
* this routine .
*/
static void fc_lport_enter_reset ( struct fc_lport * lport )
{
FC_LPORT_DBG ( lport , " Entered RESET state from %s state \n " ,
fc_lport_state ( lport ) ) ;
2009-11-03 22:46:19 +03:00
if ( lport - > vport ) {
if ( lport - > link_up )
fc_vport_set_state ( lport - > vport , FC_VPORT_INITIALIZING ) ;
else
fc_vport_set_state ( lport - > vport , FC_VPORT_LINKDOWN ) ;
}
2009-07-30 04:04:27 +04:00
fc_lport_state_enter ( lport , LPORT_ST_RESET ) ;
2009-11-03 22:46:19 +03:00
fc_vports_linkchange ( lport ) ;
2009-07-30 04:04:27 +04:00
fc_lport_reset_locked ( lport ) ;
2009-02-27 21:54:57 +03:00
if ( lport - > link_up )
2008-12-10 02:10:17 +03:00
fc_lport_enter_flogi ( lport ) ;
}
2009-07-30 04:04:27 +04:00
/**
2009-11-03 22:47:39 +03:00
* fc_lport_enter_disabled ( ) - Disable the local port
* @ lport : The local port to be reset
2009-07-30 04:04:27 +04:00
*
* Locking Note : The lport lock is expected to be held before calling
* this routine .
*/
static void fc_lport_enter_disabled ( struct fc_lport * lport )
{
FC_LPORT_DBG ( lport , " Entered disabled state from %s state \n " ,
fc_lport_state ( lport ) ) ;
fc_lport_state_enter ( lport , LPORT_ST_DISABLED ) ;
2009-11-03 22:46:19 +03:00
fc_vports_linkchange ( lport ) ;
2009-07-30 04:04:27 +04:00
fc_lport_reset_locked ( lport ) ;
}
2008-12-10 02:10:17 +03:00
/**
2009-02-27 21:55:45 +03:00
* fc_lport_error ( ) - Handler for any errors
2009-11-03 22:47:39 +03:00
* @ lport : The local port that the error was on
* @ fp : The error code encoded in a frame pointer
2008-12-10 02:10:17 +03:00
*
* If the error was caused by a resource allocation failure
* then wait for half a second and retry , otherwise retry
* after the e_d_tov time .
*/
static void fc_lport_error ( struct fc_lport * lport , struct fc_frame * fp )
{
unsigned long delay = 0 ;
2009-06-11 02:31:10 +04:00
FC_LPORT_DBG ( lport , " Error %ld in state %s, retries %d \n " ,
PTR_ERR ( fp ) , fc_lport_state ( lport ) ,
lport - > retry_count ) ;
2008-12-10 02:10:17 +03:00
if ( ! fp | | PTR_ERR ( fp ) = = - FC_EX_TIMEOUT ) {
/*
* Memory allocation failure , or the exchange timed out .
* Retry after delay
*/
if ( lport - > retry_count < lport - > max_retry_count ) {
lport - > retry_count + + ;
if ( ! fp )
delay = msecs_to_jiffies ( 500 ) ;
else
delay = msecs_to_jiffies ( lport - > e_d_tov ) ;
schedule_delayed_work ( & lport - > retry_work , delay ) ;
} else {
switch ( lport - > state ) {
2009-07-30 04:04:22 +04:00
case LPORT_ST_DISABLED :
2008-12-10 02:10:17 +03:00
case LPORT_ST_READY :
case LPORT_ST_RESET :
2009-11-03 22:46:51 +03:00
case LPORT_ST_RNN_ID :
2009-11-03 22:46:56 +03:00
case LPORT_ST_RSNN_NN :
2009-11-03 22:47:01 +03:00
case LPORT_ST_RSPN_ID :
2008-12-10 02:10:17 +03:00
case LPORT_ST_RFT_ID :
case LPORT_ST_SCR :
case LPORT_ST_DNS :
case LPORT_ST_FLOGI :
case LPORT_ST_LOGO :
fc_lport_enter_reset ( lport ) ;
break ;
}
}
}
}
/**
2009-11-03 22:47:07 +03:00
* fc_lport_ns_resp ( ) - Handle response to a name server
2009-11-03 22:47:39 +03:00
* registration exchange
* @ sp : current sequence in exchange
* @ fp : response frame
2008-12-10 02:10:17 +03:00
* @ lp_arg : Fibre Channel host port instance
*
* Locking Note : This function will be called without the lport lock
2009-11-03 22:47:39 +03:00
* held , but it will lock , call an _enter_ * function or fc_lport_error ( )
2008-12-10 02:10:17 +03:00
* and then unlock the lport .
*/
2009-11-03 22:47:07 +03:00
static void fc_lport_ns_resp ( struct fc_seq * sp , struct fc_frame * fp ,
void * lp_arg )
2008-12-10 02:10:17 +03:00
{
struct fc_lport * lport = lp_arg ;
struct fc_frame_header * fh ;
struct fc_ct_hdr * ct ;
2009-11-03 22:47:07 +03:00
FC_LPORT_DBG ( lport , " Received a ns %s \n " , fc_els_resp_type ( fp ) ) ;
2009-08-26 01:03:21 +04:00
2008-12-10 02:10:17 +03:00
if ( fp = = ERR_PTR ( - FC_EX_CLOSED ) )
return ;
mutex_lock ( & lport - > lp_mutex ) ;
2009-11-03 22:47:07 +03:00
if ( lport - > state < LPORT_ST_RNN_ID | | lport - > state > LPORT_ST_RFT_ID ) {
FC_LPORT_DBG ( lport , " Received a name server response, "
2009-11-03 22:47:39 +03:00
" but in state %s \n " , fc_lport_state ( lport ) ) ;
2009-04-22 03:26:58 +04:00
if ( IS_ERR ( fp ) )
goto err ;
2008-12-10 02:10:17 +03:00
goto out ;
}
2009-04-22 03:26:58 +04:00
if ( IS_ERR ( fp ) ) {
fc_lport_error ( lport , fp ) ;
goto err ;
}
2008-12-10 02:10:17 +03:00
fh = fc_frame_header_get ( fp ) ;
ct = fc_frame_payload_get ( fp , sizeof ( * ct ) ) ;
if ( fh & & ct & & fh - > fh_type = = FC_TYPE_CT & &
ct - > ct_fs_type = = FC_FST_DIR & &
ct - > ct_fs_subtype = = FC_NS_SUBTYPE & &
ntohs ( ct - > ct_cmd ) = = FC_FS_ACC )
2009-11-03 22:47:07 +03:00
switch ( lport - > state ) {
case LPORT_ST_RNN_ID :
2009-11-03 22:47:12 +03:00
fc_lport_enter_ns ( lport , LPORT_ST_RSNN_NN ) ;
2009-11-03 22:47:07 +03:00
break ;
case LPORT_ST_RSNN_NN :
2009-11-03 22:47:12 +03:00
fc_lport_enter_ns ( lport , LPORT_ST_RSPN_ID ) ;
2009-11-03 22:47:07 +03:00
break ;
case LPORT_ST_RSPN_ID :
2009-11-03 22:47:12 +03:00
fc_lport_enter_ns ( lport , LPORT_ST_RFT_ID ) ;
2009-11-03 22:47:07 +03:00
break ;
case LPORT_ST_RFT_ID :
fc_lport_enter_scr ( lport ) ;
break ;
default :
/* should have already been caught by state checks */
break ;
}
2009-11-03 22:46:51 +03:00
else
fc_lport_error ( lport , fp ) ;
out :
fc_frame_free ( fp ) ;
err :
mutex_unlock ( & lport - > lp_mutex ) ;
}
2008-12-10 02:10:17 +03:00
/**
2009-02-27 21:55:45 +03:00
* fc_lport_scr_resp ( ) - Handle response to State Change Register ( SCR ) request
2009-11-03 22:47:39 +03:00
* @ sp : current sequence in SCR exchange
* @ fp : response frame
2008-12-10 02:10:17 +03:00
* @ lp_arg : Fibre Channel lport port instance that sent the registration request
*
* Locking Note : This function will be called without the lport lock
* held , but it will lock , call an _enter_ * function or fc_lport_error
* and then unlock the lport .
*/
static void fc_lport_scr_resp ( struct fc_seq * sp , struct fc_frame * fp ,
void * lp_arg )
{
struct fc_lport * lport = lp_arg ;
u8 op ;
2009-08-26 01:03:21 +04:00
FC_LPORT_DBG ( lport , " Received a SCR %s \n " , fc_els_resp_type ( fp ) ) ;
2008-12-10 02:10:17 +03:00
if ( fp = = ERR_PTR ( - FC_EX_CLOSED ) )
return ;
mutex_lock ( & lport - > lp_mutex ) ;
if ( lport - > state ! = LPORT_ST_SCR ) {
2009-06-11 02:31:10 +04:00
FC_LPORT_DBG ( lport , " Received a SCR response, but in state "
" %s \n " , fc_lport_state ( lport ) ) ;
2009-04-22 03:26:58 +04:00
if ( IS_ERR ( fp ) )
goto err ;
2008-12-10 02:10:17 +03:00
goto out ;
}
2009-04-22 03:26:58 +04:00
if ( IS_ERR ( fp ) ) {
fc_lport_error ( lport , fp ) ;
goto err ;
}
2008-12-10 02:10:17 +03:00
op = fc_frame_payload_op ( fp ) ;
if ( op = = ELS_LS_ACC )
fc_lport_enter_ready ( lport ) ;
else
fc_lport_error ( lport , fp ) ;
out :
fc_frame_free ( fp ) ;
err :
mutex_unlock ( & lport - > lp_mutex ) ;
}
/**
2009-11-03 22:47:39 +03:00
* fc_lport_enter_scr ( ) - Send a SCR ( State Change Register ) request
* @ lport : The local port to register for state changes
2008-12-10 02:10:17 +03:00
*
* Locking Note : The lport lock is expected to be held before calling
* this routine .
*/
static void fc_lport_enter_scr ( struct fc_lport * lport )
{
struct fc_frame * fp ;
2009-06-11 02:31:10 +04:00
FC_LPORT_DBG ( lport , " Entered SCR state from %s state \n " ,
fc_lport_state ( lport ) ) ;
2008-12-10 02:10:17 +03:00
fc_lport_state_enter ( lport , LPORT_ST_SCR ) ;
fp = fc_frame_alloc ( lport , sizeof ( struct fc_els_scr ) ) ;
if ( ! fp ) {
fc_lport_error ( lport , fp ) ;
return ;
}
2009-08-26 01:00:55 +04:00
if ( ! lport - > tt . elsct_send ( lport , FC_FID_FCTRL , fp , ELS_SCR ,
2008-12-10 02:10:17 +03:00
fc_lport_scr_resp , lport , lport - > e_d_tov ) )
2009-10-22 03:28:09 +04:00
fc_lport_error ( lport , NULL ) ;
2008-12-10 02:10:17 +03:00
}
/**
2009-11-03 22:47:12 +03:00
* fc_lport_enter_ns ( ) - register some object with the name server
2008-12-10 02:10:17 +03:00
* @ lport : Fibre Channel local port to register
*
* Locking Note : The lport lock is expected to be held before calling
* this routine .
*/
2009-11-03 22:47:12 +03:00
static void fc_lport_enter_ns ( struct fc_lport * lport , enum fc_lport_state state )
2009-11-03 22:47:01 +03:00
{
struct fc_frame * fp ;
2009-11-03 22:47:12 +03:00
enum fc_ns_req cmd ;
int size = sizeof ( struct fc_ct_hdr ) ;
2009-11-03 22:47:01 +03:00
size_t len ;
2009-11-03 22:47:12 +03:00
FC_LPORT_DBG ( lport , " Entered %s state from %s state \n " ,
fc_lport_state_names [ state ] ,
2009-11-03 22:47:01 +03:00
fc_lport_state ( lport ) ) ;
2009-11-03 22:47:12 +03:00
fc_lport_state_enter ( lport , state ) ;
2009-11-03 22:47:01 +03:00
2009-11-03 22:47:12 +03:00
switch ( state ) {
case LPORT_ST_RNN_ID :
cmd = FC_NS_RNN_ID ;
size + = sizeof ( struct fc_ns_rn_id ) ;
break ;
case LPORT_ST_RSNN_NN :
len = strnlen ( fc_host_symbolic_name ( lport - > host ) , 255 ) ;
/* if there is no symbolic name, skip to RFT_ID */
if ( ! len )
return fc_lport_enter_ns ( lport , LPORT_ST_RFT_ID ) ;
cmd = FC_NS_RSNN_NN ;
size + = sizeof ( struct fc_ns_rsnn ) + len ;
break ;
case LPORT_ST_RSPN_ID :
len = strnlen ( fc_host_symbolic_name ( lport - > host ) , 255 ) ;
/* if there is no symbolic name, skip to RFT_ID */
if ( ! len )
return fc_lport_enter_ns ( lport , LPORT_ST_RFT_ID ) ;
cmd = FC_NS_RSPN_ID ;
size + = sizeof ( struct fc_ns_rspn ) + len ;
break ;
case LPORT_ST_RFT_ID :
cmd = FC_NS_RFT_ID ;
size + = sizeof ( struct fc_ns_rft ) ;
break ;
default :
fc_lport_error ( lport , NULL ) ;
2009-11-03 22:46:56 +03:00
return ;
}
2009-11-03 22:47:12 +03:00
fp = fc_frame_alloc ( lport , size ) ;
2009-11-03 22:46:51 +03:00
if ( ! fp ) {
fc_lport_error ( lport , fp ) ;
return ;
}
2009-11-03 22:47:12 +03:00
if ( ! lport - > tt . elsct_send ( lport , FC_FID_DIR_SERV , fp , cmd ,
2009-11-03 22:47:07 +03:00
fc_lport_ns_resp ,
2009-11-03 22:46:51 +03:00
lport , lport - > e_d_tov ) )
fc_lport_error ( lport , fp ) ;
}
2008-12-10 02:10:17 +03:00
static struct fc_rport_operations fc_lport_rport_ops = {
. event_callback = fc_lport_rport_callback ,
} ;
/**
2009-11-03 22:47:39 +03:00
* fc_rport_enter_dns ( ) - Create a fc_rport for the name server
* @ lport : The local port requesting a remote port for the name server
2008-12-10 02:10:17 +03:00
*
* Locking Note : The lport lock is expected to be held before calling
* this routine .
*/
static void fc_lport_enter_dns ( struct fc_lport * lport )
{
2009-08-26 01:00:34 +04:00
struct fc_rport_priv * rdata ;
2008-12-10 02:10:17 +03:00
2009-06-11 02:31:10 +04:00
FC_LPORT_DBG ( lport , " Entered DNS state from %s state \n " ,
fc_lport_state ( lport ) ) ;
2008-12-10 02:10:17 +03:00
fc_lport_state_enter ( lport , LPORT_ST_DNS ) ;
2009-08-26 01:01:50 +04:00
mutex_lock ( & lport - > disc . disc_mutex ) ;
2009-08-26 01:02:59 +04:00
rdata = lport - > tt . rport_create ( lport , FC_FID_DIR_SERV ) ;
2009-08-26 01:01:50 +04:00
mutex_unlock ( & lport - > disc . disc_mutex ) ;
2009-08-26 01:00:50 +04:00
if ( ! rdata )
2008-12-10 02:10:17 +03:00
goto err ;
rdata - > ops = & fc_lport_rport_ops ;
2009-08-26 01:00:50 +04:00
lport - > tt . rport_login ( rdata ) ;
2008-12-10 02:10:17 +03:00
return ;
err :
fc_lport_error ( lport , NULL ) ;
}
/**
2009-11-03 22:47:39 +03:00
* fc_lport_timeout ( ) - Handler for the retry_work timer
* @ work : The work struct of the local port
2008-12-10 02:10:17 +03:00
*/
static void fc_lport_timeout ( struct work_struct * work )
{
struct fc_lport * lport =
container_of ( work , struct fc_lport ,
retry_work . work ) ;
mutex_lock ( & lport - > lp_mutex ) ;
switch ( lport - > state ) {
2009-07-30 04:04:22 +04:00
case LPORT_ST_DISABLED :
2009-10-22 03:27:22 +04:00
WARN_ON ( 1 ) ;
break ;
2008-12-10 02:10:17 +03:00
case LPORT_ST_READY :
WARN_ON ( 1 ) ;
break ;
2009-10-22 03:27:22 +04:00
case LPORT_ST_RESET :
break ;
2008-12-10 02:10:17 +03:00
case LPORT_ST_FLOGI :
fc_lport_enter_flogi ( lport ) ;
break ;
case LPORT_ST_DNS :
fc_lport_enter_dns ( lport ) ;
break ;
2009-11-03 22:46:51 +03:00
case LPORT_ST_RNN_ID :
2009-11-03 22:46:56 +03:00
case LPORT_ST_RSNN_NN :
2009-11-03 22:47:01 +03:00
case LPORT_ST_RSPN_ID :
2008-12-10 02:10:17 +03:00
case LPORT_ST_RFT_ID :
2009-11-03 22:47:12 +03:00
fc_lport_enter_ns ( lport , lport - > state ) ;
2008-12-10 02:10:17 +03:00
break ;
case LPORT_ST_SCR :
fc_lport_enter_scr ( lport ) ;
break ;
case LPORT_ST_LOGO :
fc_lport_enter_logo ( lport ) ;
break ;
}
mutex_unlock ( & lport - > lp_mutex ) ;
}
/**
2009-02-27 21:55:45 +03:00
* fc_lport_logo_resp ( ) - Handle response to LOGO request
2009-11-03 22:47:39 +03:00
* @ sp : The sequence that the LOGO was on
* @ fp : The LOGO frame
* @ lp_arg : The lport port that received the LOGO request
2008-12-10 02:10:17 +03:00
*
* Locking Note : This function will be called without the lport lock
2009-11-03 22:47:39 +03:00
* held , but it will lock , call an _enter_ * function or fc_lport_error ( )
2008-12-10 02:10:17 +03:00
* and then unlock the lport .
*/
2009-11-03 22:46:29 +03:00
void fc_lport_logo_resp ( struct fc_seq * sp , struct fc_frame * fp ,
2009-11-03 22:47:39 +03:00
void * lp_arg )
2008-12-10 02:10:17 +03:00
{
struct fc_lport * lport = lp_arg ;
u8 op ;
2009-08-26 01:03:21 +04:00
FC_LPORT_DBG ( lport , " Received a LOGO %s \n " , fc_els_resp_type ( fp ) ) ;
2008-12-10 02:10:17 +03:00
if ( fp = = ERR_PTR ( - FC_EX_CLOSED ) )
return ;
mutex_lock ( & lport - > lp_mutex ) ;
if ( lport - > state ! = LPORT_ST_LOGO ) {
2009-06-11 02:31:10 +04:00
FC_LPORT_DBG ( lport , " Received a LOGO response, but in state "
" %s \n " , fc_lport_state ( lport ) ) ;
2009-04-22 03:26:58 +04:00
if ( IS_ERR ( fp ) )
goto err ;
2008-12-10 02:10:17 +03:00
goto out ;
}
2009-04-22 03:26:58 +04:00
if ( IS_ERR ( fp ) ) {
fc_lport_error ( lport , fp ) ;
goto err ;
}
2008-12-10 02:10:17 +03:00
op = fc_frame_payload_op ( fp ) ;
if ( op = = ELS_LS_ACC )
2009-07-30 04:04:27 +04:00
fc_lport_enter_disabled ( lport ) ;
2008-12-10 02:10:17 +03:00
else
fc_lport_error ( lport , fp ) ;
out :
fc_frame_free ( fp ) ;
err :
mutex_unlock ( & lport - > lp_mutex ) ;
}
2009-11-03 22:46:29 +03:00
EXPORT_SYMBOL ( fc_lport_logo_resp ) ;
2008-12-10 02:10:17 +03:00
/**
2009-02-27 21:55:45 +03:00
* fc_rport_enter_logo ( ) - Logout of the fabric
2009-11-03 22:47:39 +03:00
* @ lport : The local port to be logged out
2008-12-10 02:10:17 +03:00
*
* Locking Note : The lport lock is expected to be held before calling
* this routine .
*/
static void fc_lport_enter_logo ( struct fc_lport * lport )
{
struct fc_frame * fp ;
struct fc_els_logo * logo ;
2009-06-11 02:31:10 +04:00
FC_LPORT_DBG ( lport , " Entered LOGO state from %s state \n " ,
fc_lport_state ( lport ) ) ;
2008-12-10 02:10:17 +03:00
fc_lport_state_enter ( lport , LPORT_ST_LOGO ) ;
2009-11-03 22:46:19 +03:00
fc_vports_linkchange ( lport ) ;
2008-12-10 02:10:17 +03:00
fp = fc_frame_alloc ( lport , sizeof ( * logo ) ) ;
if ( ! fp ) {
fc_lport_error ( lport , fp ) ;
return ;
}
2009-08-26 01:00:55 +04:00
if ( ! lport - > tt . elsct_send ( lport , FC_FID_FLOGI , fp , ELS_LOGO ,
fc_lport_logo_resp , lport , lport - > e_d_tov ) )
2009-10-22 03:28:09 +04:00
fc_lport_error ( lport , NULL ) ;
2008-12-10 02:10:17 +03:00
}
/**
2009-02-27 21:55:45 +03:00
* fc_lport_flogi_resp ( ) - Handle response to FLOGI request
2009-11-03 22:47:39 +03:00
* @ sp : The sequence that the FLOGI was on
* @ fp : The FLOGI response frame
* @ lp_arg : The lport port that received the FLOGI response
2008-12-10 02:10:17 +03:00
*
* Locking Note : This function will be called without the lport lock
2009-11-03 22:47:39 +03:00
* held , but it will lock , call an _enter_ * function or fc_lport_error ( )
2008-12-10 02:10:17 +03:00
* and then unlock the lport .
*/
2009-11-03 22:46:29 +03:00
void fc_lport_flogi_resp ( struct fc_seq * sp , struct fc_frame * fp ,
2009-11-03 22:47:39 +03:00
void * lp_arg )
2008-12-10 02:10:17 +03:00
{
struct fc_lport * lport = lp_arg ;
struct fc_frame_header * fh ;
struct fc_els_flogi * flp ;
u32 did ;
u16 csp_flags ;
unsigned int r_a_tov ;
unsigned int e_d_tov ;
u16 mfs ;
2009-08-26 01:03:21 +04:00
FC_LPORT_DBG ( lport , " Received a FLOGI %s \n " , fc_els_resp_type ( fp ) ) ;
2008-12-10 02:10:17 +03:00
if ( fp = = ERR_PTR ( - FC_EX_CLOSED ) )
return ;
mutex_lock ( & lport - > lp_mutex ) ;
if ( lport - > state ! = LPORT_ST_FLOGI ) {
2009-06-11 02:31:10 +04:00
FC_LPORT_DBG ( lport , " Received a FLOGI response, but in state "
" %s \n " , fc_lport_state ( lport ) ) ;
2009-04-22 03:26:58 +04:00
if ( IS_ERR ( fp ) )
goto err ;
2008-12-10 02:10:17 +03:00
goto out ;
}
2009-04-22 03:26:58 +04:00
if ( IS_ERR ( fp ) ) {
fc_lport_error ( lport , fp ) ;
goto err ;
}
2008-12-10 02:10:17 +03:00
fh = fc_frame_header_get ( fp ) ;
did = ntoh24 ( fh - > fh_d_id ) ;
if ( fc_frame_payload_op ( fp ) = = ELS_LS_ACC & & did ! = 0 ) {
flp = fc_frame_payload_get ( fp , sizeof ( * flp ) ) ;
if ( flp ) {
mfs = ntohs ( flp - > fl_csp . sp_bb_data ) &
FC_SP_BB_DATA_MASK ;
if ( mfs > = FC_SP_MIN_MAX_PAYLOAD & &
mfs < lport - > mfs )
lport - > mfs = mfs ;
csp_flags = ntohs ( flp - > fl_csp . sp_features ) ;
r_a_tov = ntohl ( flp - > fl_csp . sp_r_a_tov ) ;
e_d_tov = ntohl ( flp - > fl_csp . sp_e_d_tov ) ;
if ( csp_flags & FC_SP_FT_EDTR )
e_d_tov / = 1000000 ;
2009-11-03 22:46:24 +03:00
lport - > npiv_enabled = ! ! ( csp_flags & FC_SP_FT_NPIV_ACC ) ;
2008-12-10 02:10:17 +03:00
if ( ( csp_flags & FC_SP_FT_FPORT ) = = 0 ) {
if ( e_d_tov > lport - > e_d_tov )
lport - > e_d_tov = e_d_tov ;
lport - > r_a_tov = 2 * e_d_tov ;
2009-11-03 22:49:05 +03:00
fc_lport_set_port_id ( lport , did , fp ) ;
2009-06-11 02:31:10 +04:00
printk ( KERN_INFO " libfc: Port (%6x) entered "
" point to point mode \n " , did ) ;
2008-12-10 02:10:17 +03:00
fc_lport_ptp_setup ( lport , ntoh24 ( fh - > fh_s_id ) ,
get_unaligned_be64 (
& flp - > fl_wwpn ) ,
get_unaligned_be64 (
& flp - > fl_wwnn ) ) ;
} else {
lport - > e_d_tov = e_d_tov ;
lport - > r_a_tov = r_a_tov ;
fc_host_fabric_name ( lport - > host ) =
get_unaligned_be64 ( & flp - > fl_wwnn ) ;
2009-11-03 22:49:05 +03:00
fc_lport_set_port_id ( lport , did , fp ) ;
2008-12-10 02:10:17 +03:00
fc_lport_enter_dns ( lport ) ;
}
}
} else {
2009-06-11 02:31:10 +04:00
FC_LPORT_DBG ( lport , " Bad FLOGI response \n " ) ;
2008-12-10 02:10:17 +03:00
}
out :
fc_frame_free ( fp ) ;
err :
mutex_unlock ( & lport - > lp_mutex ) ;
}
2009-11-03 22:46:29 +03:00
EXPORT_SYMBOL ( fc_lport_flogi_resp ) ;
2008-12-10 02:10:17 +03:00
/**
2009-02-27 21:55:45 +03:00
* fc_rport_enter_flogi ( ) - Send a FLOGI request to the fabric manager
2008-12-10 02:10:17 +03:00
* @ lport : Fibre Channel local port to be logged in to the fabric
*
* Locking Note : The lport lock is expected to be held before calling
* this routine .
*/
void fc_lport_enter_flogi ( struct fc_lport * lport )
{
struct fc_frame * fp ;
2009-06-11 02:31:10 +04:00
FC_LPORT_DBG ( lport , " Entered FLOGI state from %s state \n " ,
fc_lport_state ( lport ) ) ;
2008-12-10 02:10:17 +03:00
fc_lport_state_enter ( lport , LPORT_ST_FLOGI ) ;
fp = fc_frame_alloc ( lport , sizeof ( struct fc_els_flogi ) ) ;
if ( ! fp )
return fc_lport_error ( lport , fp ) ;
2009-11-03 22:46:24 +03:00
if ( ! lport - > tt . elsct_send ( lport , FC_FID_FLOGI , fp ,
lport - > vport ? ELS_FDISC : ELS_FLOGI ,
2008-12-10 02:10:17 +03:00
fc_lport_flogi_resp , lport , lport - > e_d_tov ) )
2009-10-22 03:28:09 +04:00
fc_lport_error ( lport , NULL ) ;
2008-12-10 02:10:17 +03:00
}
2009-11-03 22:47:39 +03:00
/**
* fc_lport_config ( ) - Configure a fc_lport
* @ lport : The local port to be configured
*/
2008-12-10 02:10:17 +03:00
int fc_lport_config ( struct fc_lport * lport )
{
INIT_DELAYED_WORK ( & lport - > retry_work , fc_lport_timeout ) ;
mutex_init ( & lport - > lp_mutex ) ;
2009-07-30 04:04:22 +04:00
fc_lport_state_enter ( lport , LPORT_ST_DISABLED ) ;
2008-12-10 02:10:17 +03:00
fc_lport_add_fc4_type ( lport , FC_TYPE_FCP ) ;
fc_lport_add_fc4_type ( lport , FC_TYPE_CT ) ;
return 0 ;
}
EXPORT_SYMBOL ( fc_lport_config ) ;
2009-11-03 22:47:39 +03:00
/**
* fc_lport_init ( ) - Initialize the lport layer for a local port
* @ lport : The local port to initialize the exchange layer for
*/
2008-12-10 02:10:17 +03:00
int fc_lport_init ( struct fc_lport * lport )
{
if ( ! lport - > tt . lport_recv )
lport - > tt . lport_recv = fc_lport_recv_req ;
if ( ! lport - > tt . lport_reset )
lport - > tt . lport_reset = fc_lport_reset ;
fc_host_port_type ( lport - > host ) = FC_PORTTYPE_NPORT ;
fc_host_node_name ( lport - > host ) = lport - > wwnn ;
fc_host_port_name ( lport - > host ) = lport - > wwpn ;
fc_host_supported_classes ( lport - > host ) = FC_COS_CLASS3 ;
memset ( fc_host_supported_fc4s ( lport - > host ) , 0 ,
sizeof ( fc_host_supported_fc4s ( lport - > host ) ) ) ;
fc_host_supported_fc4s ( lport - > host ) [ 2 ] = 1 ;
fc_host_supported_fc4s ( lport - > host ) [ 7 ] = 1 ;
/* This value is also unchanging */
memset ( fc_host_active_fc4s ( lport - > host ) , 0 ,
sizeof ( fc_host_active_fc4s ( lport - > host ) ) ) ;
fc_host_active_fc4s ( lport - > host ) [ 2 ] = 1 ;
fc_host_active_fc4s ( lport - > host ) [ 7 ] = 1 ;
fc_host_maxframe_size ( lport - > host ) = lport - > mfs ;
fc_host_supported_speeds ( lport - > host ) = 0 ;
if ( lport - > link_supported_speeds & FC_PORTSPEED_1GBIT )
fc_host_supported_speeds ( lport - > host ) | = FC_PORTSPEED_1GBIT ;
if ( lport - > link_supported_speeds & FC_PORTSPEED_10GBIT )
fc_host_supported_speeds ( lport - > host ) | = FC_PORTSPEED_10GBIT ;
return 0 ;
}
EXPORT_SYMBOL ( fc_lport_init ) ;
2009-11-03 22:47:34 +03:00
/**
2009-11-03 22:47:39 +03:00
* fc_lport_bsg_resp ( ) - The common response handler for FC Passthrough requests
* @ sp : The sequence for the FC Passthrough response
* @ fp : The response frame
* @ info_arg : The BSG info that the response is for
2009-11-03 22:47:34 +03:00
*/
static void fc_lport_bsg_resp ( struct fc_seq * sp , struct fc_frame * fp ,
void * info_arg )
{
struct fc_bsg_info * info = info_arg ;
struct fc_bsg_job * job = info - > job ;
struct fc_lport * lport = info - > lport ;
struct fc_frame_header * fh ;
size_t len ;
void * buf ;
if ( IS_ERR ( fp ) ) {
job - > reply - > result = ( PTR_ERR ( fp ) = = - FC_EX_CLOSED ) ?
- ECONNABORTED : - ETIMEDOUT ;
job - > reply_len = sizeof ( uint32_t ) ;
job - > state_flags | = FC_RQST_STATE_DONE ;
job - > job_done ( job ) ;
kfree ( info ) ;
return ;
}
mutex_lock ( & lport - > lp_mutex ) ;
fh = fc_frame_header_get ( fp ) ;
len = fr_len ( fp ) - sizeof ( * fh ) ;
buf = fc_frame_payload_get ( fp , 0 ) ;
if ( fr_sof ( fp ) = = FC_SOF_I3 & & ! ntohs ( fh - > fh_seq_cnt ) ) {
/* Get the response code from the first frame payload */
unsigned short cmd = ( info - > rsp_code = = FC_FS_ACC ) ?
ntohs ( ( ( struct fc_ct_hdr * ) buf ) - > ct_cmd ) :
( unsigned short ) fc_frame_payload_op ( fp ) ;
/* Save the reply status of the job */
job - > reply - > reply_data . ctels_reply . status =
( cmd = = info - > rsp_code ) ?
FC_CTELS_STATUS_OK : FC_CTELS_STATUS_REJECT ;
}
job - > reply - > reply_payload_rcv_len + =
fc_copy_buffer_to_sglist ( buf , len , info - > sg , & info - > nents ,
& info - > offset , KM_BIO_SRC_IRQ , NULL ) ;
if ( fr_eof ( fp ) = = FC_EOF_T & &
( ntoh24 ( fh - > fh_f_ctl ) & ( FC_FC_LAST_SEQ | FC_FC_END_SEQ ) ) = =
( FC_FC_LAST_SEQ | FC_FC_END_SEQ ) ) {
if ( job - > reply - > reply_payload_rcv_len >
job - > reply_payload . payload_len )
job - > reply - > reply_payload_rcv_len =
job - > reply_payload . payload_len ;
job - > reply - > result = 0 ;
job - > state_flags | = FC_RQST_STATE_DONE ;
job - > job_done ( job ) ;
kfree ( info ) ;
}
fc_frame_free ( fp ) ;
mutex_unlock ( & lport - > lp_mutex ) ;
}
/**
2009-11-03 22:47:39 +03:00
* fc_lport_els_request ( ) - Send ELS passthrough request
* @ job : The BSG Passthrough job
2009-11-03 22:47:34 +03:00
* @ lport : The local port sending the request
2009-11-03 22:47:39 +03:00
* @ did : The destination port id
2009-11-03 22:47:34 +03:00
*
* Locking Note : The lport lock is expected to be held before calling
* this routine .
*/
static int fc_lport_els_request ( struct fc_bsg_job * job ,
struct fc_lport * lport ,
u32 did , u32 tov )
{
struct fc_bsg_info * info ;
struct fc_frame * fp ;
struct fc_frame_header * fh ;
char * pp ;
int len ;
fp = fc_frame_alloc ( lport , sizeof ( struct fc_frame_header ) +
job - > request_payload . payload_len ) ;
if ( ! fp )
return - ENOMEM ;
len = job - > request_payload . payload_len ;
pp = fc_frame_payload_get ( fp , len ) ;
sg_copy_to_buffer ( job - > request_payload . sg_list ,
job - > request_payload . sg_cnt ,
pp , len ) ;
fh = fc_frame_header_get ( fp ) ;
fh - > fh_r_ctl = FC_RCTL_ELS_REQ ;
hton24 ( fh - > fh_d_id , did ) ;
hton24 ( fh - > fh_s_id , fc_host_port_id ( lport - > host ) ) ;
fh - > fh_type = FC_TYPE_ELS ;
hton24 ( fh - > fh_f_ctl , FC_FC_FIRST_SEQ |
FC_FC_END_SEQ | FC_FC_SEQ_INIT ) ;
fh - > fh_cs_ctl = 0 ;
fh - > fh_df_ctl = 0 ;
fh - > fh_parm_offset = 0 ;
info = kzalloc ( sizeof ( struct fc_bsg_info ) , GFP_KERNEL ) ;
if ( ! info ) {
fc_frame_free ( fp ) ;
return - ENOMEM ;
}
info - > job = job ;
info - > lport = lport ;
info - > rsp_code = ELS_LS_ACC ;
info - > nents = job - > reply_payload . sg_cnt ;
info - > sg = job - > reply_payload . sg_list ;
if ( ! lport - > tt . exch_seq_send ( lport , fp , fc_lport_bsg_resp ,
NULL , info , tov ) )
return - ECOMM ;
return 0 ;
}
/**
2009-11-03 22:47:39 +03:00
* fc_lport_ct_request ( ) - Send CT Passthrough request
* @ job : The BSG Passthrough job
2009-11-03 22:47:34 +03:00
* @ lport : The local port sending the request
* @ did : The destination FC - ID
2009-11-03 22:47:39 +03:00
* @ tov : The timeout period to wait for the response
2009-11-03 22:47:34 +03:00
*
* Locking Note : The lport lock is expected to be held before calling
* this routine .
*/
static int fc_lport_ct_request ( struct fc_bsg_job * job ,
struct fc_lport * lport , u32 did , u32 tov )
{
struct fc_bsg_info * info ;
struct fc_frame * fp ;
struct fc_frame_header * fh ;
struct fc_ct_req * ct ;
size_t len ;
fp = fc_frame_alloc ( lport , sizeof ( struct fc_ct_hdr ) +
job - > request_payload . payload_len ) ;
if ( ! fp )
return - ENOMEM ;
len = job - > request_payload . payload_len ;
ct = fc_frame_payload_get ( fp , len ) ;
sg_copy_to_buffer ( job - > request_payload . sg_list ,
job - > request_payload . sg_cnt ,
ct , len ) ;
fh = fc_frame_header_get ( fp ) ;
fh - > fh_r_ctl = FC_RCTL_DD_UNSOL_CTL ;
hton24 ( fh - > fh_d_id , did ) ;
hton24 ( fh - > fh_s_id , fc_host_port_id ( lport - > host ) ) ;
fh - > fh_type = FC_TYPE_CT ;
hton24 ( fh - > fh_f_ctl , FC_FC_FIRST_SEQ |
FC_FC_END_SEQ | FC_FC_SEQ_INIT ) ;
fh - > fh_cs_ctl = 0 ;
fh - > fh_df_ctl = 0 ;
fh - > fh_parm_offset = 0 ;
info = kzalloc ( sizeof ( struct fc_bsg_info ) , GFP_KERNEL ) ;
if ( ! info ) {
fc_frame_free ( fp ) ;
return - ENOMEM ;
}
info - > job = job ;
info - > lport = lport ;
info - > rsp_code = FC_FS_ACC ;
info - > nents = job - > reply_payload . sg_cnt ;
info - > sg = job - > reply_payload . sg_list ;
if ( ! lport - > tt . exch_seq_send ( lport , fp , fc_lport_bsg_resp ,
NULL , info , tov ) )
return - ECOMM ;
return 0 ;
}
/**
* fc_lport_bsg_request ( ) - The common entry point for sending
2009-11-03 22:47:39 +03:00
* FC Passthrough requests
* @ job : The BSG passthrough job
2009-11-03 22:47:34 +03:00
*/
int fc_lport_bsg_request ( struct fc_bsg_job * job )
{
struct request * rsp = job - > req - > next_rq ;
struct Scsi_Host * shost = job - > shost ;
struct fc_lport * lport = shost_priv ( shost ) ;
struct fc_rport * rport ;
struct fc_rport_priv * rdata ;
int rc = - EINVAL ;
u32 did ;
job - > reply - > reply_payload_rcv_len = 0 ;
rsp - > resid_len = job - > reply_payload . payload_len ;
mutex_lock ( & lport - > lp_mutex ) ;
switch ( job - > request - > msgcode ) {
case FC_BSG_RPT_ELS :
rport = job - > rport ;
if ( ! rport )
break ;
rdata = rport - > dd_data ;
rc = fc_lport_els_request ( job , lport , rport - > port_id ,
rdata - > e_d_tov ) ;
break ;
case FC_BSG_RPT_CT :
rport = job - > rport ;
if ( ! rport )
break ;
rdata = rport - > dd_data ;
rc = fc_lport_ct_request ( job , lport , rport - > port_id ,
rdata - > e_d_tov ) ;
break ;
case FC_BSG_HST_CT :
did = ntoh24 ( job - > request - > rqst_data . h_ct . port_id ) ;
if ( did = = FC_FID_DIR_SERV )
2009-11-03 22:47:39 +03:00
rdata = lport - > dns_rdata ;
2009-11-03 22:47:34 +03:00
else
rdata = lport - > tt . rport_lookup ( lport , did ) ;
if ( ! rdata )
break ;
rc = fc_lport_ct_request ( job , lport , did , rdata - > e_d_tov ) ;
break ;
case FC_BSG_HST_ELS_NOLOGIN :
did = ntoh24 ( job - > request - > rqst_data . h_els . port_id ) ;
rc = fc_lport_els_request ( job , lport , did , lport - > e_d_tov ) ;
break ;
}
mutex_unlock ( & lport - > lp_mutex ) ;
return rc ;
}
EXPORT_SYMBOL ( fc_lport_bsg_request ) ;