2017-03-24 14:11:48 +05:30
/*
2017-10-10 16:18:19 +05:30
* Copyright 2017 Broadcom . All Rights Reserved .
2017-03-24 14:11:48 +05:30
* The term " Broadcom " refers to Broadcom Limited and / or its subsidiaries .
2009-09-05 07:36:35 +05:30
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License version 2
2017-03-24 14:11:48 +05:30
* as published by the Free Software Foundation . The full GNU General
2009-09-05 07:36:35 +05:30
* Public License is included in this distribution in the file called COPYING .
*
* Contact Information :
2016-08-19 15:20:24 +05:30
* linux - drivers @ broadcom . com
2009-09-05 07:36:35 +05:30
*
*/
2012-04-03 23:41:50 -05:00
# include <linux/bsg-lib.h>
# include <scsi/scsi_transport_iscsi.h>
# include <scsi/scsi_bsg_iscsi.h>
2009-09-05 07:36:35 +05:30
# include "be_mgmt.h"
# include "be_iscsi.h"
2012-10-20 04:45:51 +05:30
# include "be_main.h"
2012-04-03 23:41:50 -05:00
unsigned int mgmt_vendor_specific_fw_cmd ( struct be_ctrl_info * ctrl ,
struct beiscsi_hba * phba ,
struct bsg_job * job ,
struct be_dma_mem * nonemb_cmd )
{
2014-05-05 21:41:24 -04:00
struct be_mcc_wrb * wrb ;
struct be_sge * mcc_sge ;
2012-04-03 23:41:50 -05:00
unsigned int tag = 0 ;
struct iscsi_bsg_request * bsg_req = job - > request ;
struct be_bsg_vendor_cmd * req = nonemb_cmd - > va ;
unsigned short region , sector_size , sector , offset ;
nonemb_cmd - > size = job - > request_payload . payload_len ;
memset ( nonemb_cmd - > va , 0 , nonemb_cmd - > size ) ;
region = bsg_req - > rqst_data . h_vendor . vendor_cmd [ 1 ] ;
sector_size = bsg_req - > rqst_data . h_vendor . vendor_cmd [ 2 ] ;
sector = bsg_req - > rqst_data . h_vendor . vendor_cmd [ 3 ] ;
offset = bsg_req - > rqst_data . h_vendor . vendor_cmd [ 4 ] ;
req - > region = region ;
req - > sector = sector ;
req - > offset = offset ;
2016-01-20 14:10:46 +05:30
if ( mutex_lock_interruptible ( & ctrl - > mbox_lock ) )
return 0 ;
2012-04-03 23:41:50 -05:00
switch ( bsg_req - > rqst_data . h_vendor . vendor_cmd [ 0 ] ) {
case BEISCSI_WRITE_FLASH :
offset = sector * sector_size + offset ;
be_cmd_hdr_prepare ( & req - > hdr , CMD_SUBSYSTEM_ISCSI ,
OPCODE_COMMON_WRITE_FLASH , sizeof ( * req ) ) ;
sg_copy_to_buffer ( job - > request_payload . sg_list ,
job - > request_payload . sg_cnt ,
nonemb_cmd - > va + offset , job - > request_len ) ;
break ;
case BEISCSI_READ_FLASH :
be_cmd_hdr_prepare ( & req - > hdr , CMD_SUBSYSTEM_ISCSI ,
OPCODE_COMMON_READ_FLASH , sizeof ( * req ) ) ;
break ;
default :
2012-08-20 23:00:18 +05:30
beiscsi_log ( phba , KERN_WARNING , BEISCSI_LOG_CONFIG ,
" BG_%d : Unsupported cmd = 0x%x \n \n " ,
bsg_req - > rqst_data . h_vendor . vendor_cmd [ 0 ] ) ;
2016-01-20 14:10:46 +05:30
mutex_unlock ( & ctrl - > mbox_lock ) ;
2016-08-19 15:20:05 +05:30
return - EPERM ;
2012-04-03 23:41:50 -05:00
}
2016-02-04 15:49:17 +05:30
wrb = alloc_mcc_wrb ( phba , & tag ) ;
if ( ! wrb ) {
2016-01-20 14:10:46 +05:30
mutex_unlock ( & ctrl - > mbox_lock ) ;
2016-02-04 15:49:17 +05:30
return 0 ;
2012-04-03 23:41:50 -05:00
}
2014-05-05 21:41:24 -04:00
mcc_sge = nonembedded_sgl ( wrb ) ;
2012-04-03 23:41:50 -05:00
be_wrb_hdr_prepare ( wrb , nonemb_cmd - > size , false ,
job - > request_payload . sg_cnt ) ;
mcc_sge - > pa_hi = cpu_to_le32 ( upper_32_bits ( nonemb_cmd - > dma ) ) ;
mcc_sge - > pa_lo = cpu_to_le32 ( nonemb_cmd - > dma & 0xFFFFFFFF ) ;
mcc_sge - > len = cpu_to_le32 ( nonemb_cmd - > size ) ;
2016-01-20 14:10:47 +05:30
be_mcc_notify ( phba , tag ) ;
2012-04-03 23:41:50 -05:00
2016-01-20 14:10:46 +05:30
mutex_unlock ( & ctrl - > mbox_lock ) ;
2012-04-03 23:41:50 -05:00
return tag ;
}
2013-09-28 15:35:50 -07:00
/**
* mgmt_open_connection ( ) - Establish a TCP CXN
* @ dst_addr : Destination Address
* @ beiscsi_ep : ptr to device endpoint struct
* @ nonemb_cmd : ptr to memory allocated for command
*
* return
* Success : Tag number of the MBX Command issued
* Failure : Error code
* */
2009-09-05 07:36:35 +05:30
int mgmt_open_connection ( struct beiscsi_hba * phba ,
struct sockaddr * dst_addr ,
2010-07-22 04:27:47 +05:30
struct beiscsi_endpoint * beiscsi_ep ,
struct be_dma_mem * nonemb_cmd )
2009-09-05 07:36:35 +05:30
{
struct hwi_controller * phwi_ctrlr ;
struct hwi_context_memory * phwi_context ;
struct sockaddr_in * daddr_in = ( struct sockaddr_in * ) dst_addr ;
struct sockaddr_in6 * daddr_in6 = ( struct sockaddr_in6 * ) dst_addr ;
struct be_ctrl_info * ctrl = & phba - > ctrl ;
2010-01-05 05:10:46 +05:30
struct be_mcc_wrb * wrb ;
2014-05-05 21:41:27 -04:00
struct tcp_connect_and_offload_in_v1 * req ;
2009-09-05 07:36:35 +05:30
unsigned short def_hdr_id ;
unsigned short def_data_id ;
struct phys_addr template_address = { 0 , 0 } ;
struct phys_addr * ptemplate_address ;
2010-01-05 05:10:46 +05:30
unsigned int tag = 0 ;
2013-09-28 15:35:50 -07:00
unsigned int i , ulp_num ;
2009-09-05 07:36:35 +05:30
unsigned short cid = beiscsi_ep - > ep_cid ;
2010-07-22 04:27:47 +05:30
struct be_sge * sge ;
2009-09-05 07:36:35 +05:30
2016-02-04 15:49:16 +05:30
if ( dst_addr - > sa_family ! = PF_INET & & dst_addr - > sa_family ! = PF_INET6 ) {
beiscsi_log ( phba , KERN_ERR , BEISCSI_LOG_CONFIG ,
" BG_%d : unknown addr family %d \n " ,
dst_addr - > sa_family ) ;
2017-10-10 16:18:12 +05:30
return 0 ;
2016-02-04 15:49:16 +05:30
}
2009-09-05 07:36:35 +05:30
phwi_ctrlr = phba - > phwi_ctrlr ;
phwi_context = phwi_ctrlr - > phwi_ctxt ;
2013-09-28 15:35:50 -07:00
ulp_num = phwi_ctrlr - > wrb_context [ BE_GET_CRI_FROM_CID ( cid ) ] . ulp_num ;
def_hdr_id = ( unsigned short ) HWI_GET_DEF_HDRQ_ID ( phba , ulp_num ) ;
def_data_id = ( unsigned short ) HWI_GET_DEF_BUFQ_ID ( phba , ulp_num ) ;
2009-09-05 07:36:35 +05:30
ptemplate_address = & template_address ;
ISCSI_GET_PDU_TEMPLATE_ADDRESS ( phba , ptemplate_address ) ;
2016-01-20 14:10:46 +05:30
if ( mutex_lock_interruptible ( & ctrl - > mbox_lock ) )
return 0 ;
2016-02-04 15:49:17 +05:30
wrb = alloc_mcc_wrb ( phba , & tag ) ;
if ( ! wrb ) {
2016-01-20 14:10:46 +05:30
mutex_unlock ( & ctrl - > mbox_lock ) ;
2016-02-04 15:49:17 +05:30
return 0 ;
2010-01-05 05:10:46 +05:30
}
2010-07-22 04:27:47 +05:30
2016-02-04 15:49:17 +05:30
sge = nonembedded_sgl ( wrb ) ;
2010-07-22 04:27:47 +05:30
req = nonemb_cmd - > va ;
memset ( req , 0 , sizeof ( * req ) ) ;
2009-09-05 07:36:35 +05:30
2014-05-05 21:41:27 -04:00
be_wrb_hdr_prepare ( wrb , nonemb_cmd - > size , false , 1 ) ;
2009-09-05 07:36:35 +05:30
be_cmd_hdr_prepare ( & req - > hdr , CMD_SUBSYSTEM_ISCSI ,
OPCODE_COMMON_ISCSI_TCP_CONNECT_AND_OFFLOAD ,
2014-05-05 21:41:27 -04:00
nonemb_cmd - > size ) ;
2009-09-05 07:36:35 +05:30
if ( dst_addr - > sa_family = = PF_INET ) {
__be32 s_addr = daddr_in - > sin_addr . s_addr ;
2016-08-19 15:20:08 +05:30
req - > ip_address . ip_type = BEISCSI_IP_TYPE_V4 ;
2012-04-03 23:41:51 -05:00
req - > ip_address . addr [ 0 ] = s_addr & 0x000000ff ;
req - > ip_address . addr [ 1 ] = ( s_addr & 0x0000ff00 ) > > 8 ;
req - > ip_address . addr [ 2 ] = ( s_addr & 0x00ff0000 ) > > 16 ;
req - > ip_address . addr [ 3 ] = ( s_addr & 0xff000000 ) > > 24 ;
2009-09-05 07:36:35 +05:30
req - > tcp_port = ntohs ( daddr_in - > sin_port ) ;
beiscsi_ep - > dst_addr = daddr_in - > sin_addr . s_addr ;
beiscsi_ep - > dst_tcpport = ntohs ( daddr_in - > sin_port ) ;
2016-08-19 15:20:08 +05:30
beiscsi_ep - > ip_type = BEISCSI_IP_TYPE_V4 ;
2016-02-04 15:49:16 +05:30
} else {
/* else its PF_INET6 family */
2016-08-19 15:20:08 +05:30
req - > ip_address . ip_type = BEISCSI_IP_TYPE_V6 ;
2012-04-03 23:41:51 -05:00
memcpy ( & req - > ip_address . addr ,
2009-09-05 07:36:35 +05:30
& daddr_in6 - > sin6_addr . in6_u . u6_addr8 , 16 ) ;
req - > tcp_port = ntohs ( daddr_in6 - > sin6_port ) ;
beiscsi_ep - > dst_tcpport = ntohs ( daddr_in6 - > sin6_port ) ;
memcpy ( & beiscsi_ep - > dst6_addr ,
& daddr_in6 - > sin6_addr . in6_u . u6_addr8 , 16 ) ;
2016-08-19 15:20:08 +05:30
beiscsi_ep - > ip_type = BEISCSI_IP_TYPE_V6 ;
2009-09-05 07:36:35 +05:30
}
req - > cid = cid ;
2009-10-23 11:52:33 +05:30
i = phba - > nxt_cqid + + ;
if ( phba - > nxt_cqid = = phba - > num_cpus )
phba - > nxt_cqid = 0 ;
req - > cq_id = phwi_context - > be_cq [ i ] . id ;
2012-08-20 23:00:18 +05:30
beiscsi_log ( phba , KERN_INFO , BEISCSI_LOG_CONFIG ,
" BG_%d : i=%d cq_id=%d \n " , i , req - > cq_id ) ;
2009-09-05 07:36:35 +05:30
req - > defq_id = def_hdr_id ;
req - > hdr_ring_id = def_hdr_id ;
req - > data_ring_id = def_data_id ;
req - > do_offload = 1 ;
req - > dataout_template_pa . lo = ptemplate_address - > lo ;
req - > dataout_template_pa . hi = ptemplate_address - > hi ;
2010-07-22 04:27:47 +05:30
sge - > pa_hi = cpu_to_le32 ( upper_32_bits ( nonemb_cmd - > dma ) ) ;
sge - > pa_lo = cpu_to_le32 ( nonemb_cmd - > dma & 0xFFFFFFFF ) ;
sge - > len = cpu_to_le32 ( nonemb_cmd - > size ) ;
2014-05-05 21:41:27 -04:00
if ( ! is_chip_be2_be3r ( phba ) ) {
req - > hdr . version = MBX_CMD_VER1 ;
2016-08-19 15:20:23 +05:30
req - > tcp_window_size = 0x8000 ;
2014-05-05 21:41:27 -04:00
req - > tcp_window_scale_count = 2 ;
}
2016-01-20 14:10:47 +05:30
be_mcc_notify ( phba , tag ) ;
2016-01-20 14:10:46 +05:30
mutex_unlock ( & ctrl - > mbox_lock ) ;
2010-01-05 05:10:46 +05:30
return tag ;
2009-09-05 07:36:35 +05:30
}
2009-10-23 11:52:33 +05:30
2012-10-20 04:45:40 +05:30
/*
2017-10-10 16:18:14 +05:30
* beiscsi_exec_nemb_cmd ( ) - execute non - embedded MBX cmd
* @ phba : driver priv structure
* @ nonemb_cmd : DMA address of the MBX command to be issued
* @ cbfn : callback func on MCC completion
* @ resp_buf : buffer to copy the MBX cmd response
* @ resp_buf_len : response length to be copied
2012-10-20 04:45:40 +05:30
*
* */
2017-10-10 16:18:14 +05:30
static int beiscsi_exec_nemb_cmd ( struct beiscsi_hba * phba ,
struct be_dma_mem * nonemb_cmd ,
void ( * cbfn ) ( struct beiscsi_hba * ,
unsigned int ) ,
void * resp_buf , u32 resp_buf_len )
2012-04-03 23:41:51 -05:00
{
struct be_ctrl_info * ctrl = & phba - > ctrl ;
2014-05-05 21:41:24 -04:00
struct be_mcc_wrb * wrb ;
2012-04-03 23:41:51 -05:00
struct be_sge * sge ;
unsigned int tag ;
int rc = 0 ;
2009-10-23 11:52:33 +05:30
2016-01-20 14:10:46 +05:30
mutex_lock ( & ctrl - > mbox_lock ) ;
2016-02-04 15:49:17 +05:30
wrb = alloc_mcc_wrb ( phba , & tag ) ;
if ( ! wrb ) {
2016-01-20 14:10:46 +05:30
mutex_unlock ( & ctrl - > mbox_lock ) ;
2012-04-03 23:41:51 -05:00
rc = - ENOMEM ;
goto free_cmd ;
2010-01-05 05:10:46 +05:30
}
2014-05-05 21:41:24 -04:00
2012-04-03 23:41:51 -05:00
sge = nonembedded_sgl ( wrb ) ;
be_wrb_hdr_prepare ( wrb , nonemb_cmd - > size , false , 1 ) ;
sge - > pa_hi = cpu_to_le32 ( upper_32_bits ( nonemb_cmd - > dma ) ) ;
2012-10-20 04:45:40 +05:30
sge - > pa_lo = cpu_to_le32 ( lower_32_bits ( nonemb_cmd - > dma ) ) ;
2012-04-03 23:41:51 -05:00
sge - > len = cpu_to_le32 ( nonemb_cmd - > size ) ;
2009-10-23 11:52:33 +05:30
2017-10-10 16:18:14 +05:30
if ( cbfn ) {
struct be_dma_mem * tag_mem ;
set_bit ( MCC_TAG_STATE_ASYNC , & ctrl - > ptag_state [ tag ] . tag_state ) ;
ctrl - > ptag_state [ tag ] . cbfn = cbfn ;
tag_mem = & phba - > ctrl . ptag_state [ tag ] . tag_mem_state ;
/* store DMA mem to be freed in callback */
tag_mem - > size = nonemb_cmd - > size ;
tag_mem - > va = nonemb_cmd - > va ;
tag_mem - > dma = nonemb_cmd - > dma ;
}
2016-01-20 14:10:47 +05:30
be_mcc_notify ( phba , tag ) ;
2016-01-20 14:10:46 +05:30
mutex_unlock ( & ctrl - > mbox_lock ) ;
2012-04-03 23:41:51 -05:00
2017-10-10 16:18:14 +05:30
/* with cbfn set, its async cmd, don't wait */
if ( cbfn )
return 0 ;
2016-02-04 15:49:12 +05:30
rc = beiscsi_mccq_compl_wait ( phba , tag , NULL , nonemb_cmd ) ;
2014-01-29 02:16:39 -05:00
2017-10-10 16:18:14 +05:30
/* copy the response, if any */
2014-01-29 02:16:39 -05:00
if ( resp_buf )
memcpy ( resp_buf , nonemb_cmd - > va , resp_buf_len ) ;
2017-10-10 16:18:14 +05:30
/**
* This is special case of NTWK_GET_IF_INFO where the size of
* response is not known . beiscsi_if_get_info checks the return
* value to free DMA buffer .
*/
if ( rc = = - EAGAIN )
return rc ;
2014-01-29 02:16:39 -05:00
2017-10-10 16:18:14 +05:30
/**
* If FW is busy that is driver timed out , DMA buffer is saved with
* the tag , only when the cmd completes this buffer is freed .
*/
if ( rc = = - EBUSY )
return rc ;
2012-10-20 04:45:40 +05:30
2012-04-03 23:41:51 -05:00
free_cmd :
pci_free_consistent ( ctrl - > pdev , nonemb_cmd - > size ,
nonemb_cmd - > va , nonemb_cmd - > dma ) ;
return rc ;
}
2017-10-10 16:18:14 +05:30
static int beiscsi_prep_nemb_cmd ( struct beiscsi_hba * phba ,
struct be_dma_mem * cmd ,
u8 subsystem , u8 opcode , u32 size )
2012-04-03 23:41:51 -05:00
{
2014-08-08 14:24:46 -07:00
cmd - > va = pci_zalloc_consistent ( phba - > ctrl . pdev , size , & cmd - > dma ) ;
2012-04-03 23:41:51 -05:00
if ( ! cmd - > va ) {
2012-08-20 23:00:18 +05:30
beiscsi_log ( phba , KERN_ERR , BEISCSI_LOG_CONFIG ,
" BG_%d : Failed to allocate memory for if info \n " ) ;
2012-04-03 23:41:51 -05:00
return - ENOMEM ;
}
cmd - > size = size ;
2017-10-10 16:18:14 +05:30
be_cmd_hdr_prepare ( cmd - > va , subsystem , opcode , size ) ;
2016-08-19 15:20:04 +05:30
beiscsi_log ( phba , KERN_INFO , BEISCSI_LOG_CONFIG ,
2017-10-10 16:18:14 +05:30
" BG_%d : subsystem %u cmd %u size %u \n " ,
subsystem , opcode , size ) ;
2012-04-03 23:41:51 -05:00
return 0 ;
2009-10-23 11:52:33 +05:30
}
2017-10-10 16:18:14 +05:30
static void __beiscsi_eq_delay_compl ( struct beiscsi_hba * phba , unsigned int tag )
{
struct be_dma_mem * tag_mem ;
/* status is ignored */
__beiscsi_mcc_compl_status ( phba , tag , NULL , NULL ) ;
tag_mem = & phba - > ctrl . ptag_state [ tag ] . tag_mem_state ;
if ( tag_mem - > size ) {
pci_free_consistent ( phba - > pcidev , tag_mem - > size ,
tag_mem - > va , tag_mem - > dma ) ;
tag_mem - > size = 0 ;
}
}
int beiscsi_modify_eq_delay ( struct beiscsi_hba * phba ,
struct be_set_eqd * set_eqd , int num )
{
struct be_cmd_req_modify_eq_delay * req ;
struct be_dma_mem nonemb_cmd ;
int i , rc ;
rc = beiscsi_prep_nemb_cmd ( phba , & nonemb_cmd , CMD_SUBSYSTEM_COMMON ,
OPCODE_COMMON_MODIFY_EQ_DELAY , sizeof ( * req ) ) ;
if ( rc )
return rc ;
req = nonemb_cmd . va ;
req - > num_eq = cpu_to_le32 ( num ) ;
for ( i = 0 ; i < num ; i + + ) {
req - > delay [ i ] . eq_id = cpu_to_le32 ( set_eqd [ i ] . eq_id ) ;
req - > delay [ i ] . phase = 0 ;
req - > delay [ i ] . delay_multiplier =
cpu_to_le32 ( set_eqd [ i ] . delay_multiplier ) ;
}
return beiscsi_exec_nemb_cmd ( phba , & nonemb_cmd ,
__beiscsi_eq_delay_compl , NULL , 0 ) ;
}
2017-10-10 16:18:15 +05:30
/**
* beiscsi_get_initiator_name - read initiator name from flash
* @ phba : device priv structure
* @ name : buffer pointer
2017-10-10 16:18:16 +05:30
* @ cfg : fetch user configured
2017-10-10 16:18:15 +05:30
*
*/
2017-10-10 16:18:16 +05:30
int beiscsi_get_initiator_name ( struct beiscsi_hba * phba , char * name , bool cfg )
2017-10-10 16:18:15 +05:30
{
struct be_dma_mem nonemb_cmd ;
struct be_cmd_hba_name resp ;
2017-10-10 16:18:16 +05:30
struct be_cmd_hba_name * req ;
2017-10-10 16:18:15 +05:30
int rc ;
rc = beiscsi_prep_nemb_cmd ( phba , & nonemb_cmd , CMD_SUBSYSTEM_ISCSI_INI ,
OPCODE_ISCSI_INI_CFG_GET_HBA_NAME , sizeof ( resp ) ) ;
if ( rc )
return rc ;
2017-10-10 16:18:16 +05:30
req = nonemb_cmd . va ;
if ( cfg )
req - > hdr . version = 1 ;
2017-10-10 16:18:15 +05:30
rc = beiscsi_exec_nemb_cmd ( phba , & nonemb_cmd , NULL ,
& resp , sizeof ( resp ) ) ;
if ( rc ) {
beiscsi_log ( phba , KERN_ERR ,
BEISCSI_LOG_CONFIG | BEISCSI_LOG_MBOX ,
" BS_%d : Initiator Name MBX Failed \n " ) ;
return rc ;
}
rc = sprintf ( name , " %s \n " , resp . initiator_name ) ;
return rc ;
}
2016-08-19 15:20:05 +05:30
unsigned int beiscsi_if_get_handle ( struct beiscsi_hba * phba )
{
struct be_ctrl_info * ctrl = & phba - > ctrl ;
struct be_mcc_wrb * wrb ;
struct be_cmd_get_all_if_id_req * req ;
struct be_cmd_get_all_if_id_req * pbe_allid ;
unsigned int tag ;
int status = 0 ;
if ( mutex_lock_interruptible ( & ctrl - > mbox_lock ) )
return - EINTR ;
wrb = alloc_mcc_wrb ( phba , & tag ) ;
if ( ! wrb ) {
mutex_unlock ( & ctrl - > mbox_lock ) ;
return - ENOMEM ;
}
req = embedded_payload ( wrb ) ;
be_wrb_hdr_prepare ( wrb , sizeof ( * req ) , true , 0 ) ;
be_cmd_hdr_prepare ( & req - > hdr , CMD_SUBSYSTEM_ISCSI ,
OPCODE_COMMON_ISCSI_NTWK_GET_ALL_IF_ID ,
sizeof ( * req ) ) ;
be_mcc_notify ( phba , tag ) ;
mutex_unlock ( & ctrl - > mbox_lock ) ;
status = beiscsi_mccq_compl_wait ( phba , tag , & wrb , NULL ) ;
if ( status ) {
beiscsi_log ( phba , KERN_WARNING , BEISCSI_LOG_CONFIG ,
" BG_%d : %s failed: %d \n " , __func__ , status ) ;
return - EBUSY ;
}
pbe_allid = embedded_payload ( wrb ) ;
/* we now support only one interface per function */
phba - > interface_handle = pbe_allid - > if_hndl_list [ 0 ] ;
return status ;
}
2016-08-19 15:20:08 +05:30
static inline bool beiscsi_if_zero_ip ( u8 * ip , u32 ip_type )
{
u32 len ;
len = ( ip_type < BEISCSI_IP_TYPE_V6 ) ? IP_V4_LEN : IP_V6_LEN ;
while ( len & & ! ip [ len - 1 ] )
len - - ;
return ( len = = 0 ) ;
}
2016-08-19 15:20:02 +05:30
static int beiscsi_if_mod_gw ( struct beiscsi_hba * phba ,
u32 action , u32 ip_type , u8 * gw )
2012-04-03 23:41:51 -05:00
{
struct be_cmd_set_def_gateway_req * req ;
struct be_dma_mem nonemb_cmd ;
int rt_val ;
2017-10-10 16:18:14 +05:30
rt_val = beiscsi_prep_nemb_cmd ( phba , & nonemb_cmd , CMD_SUBSYSTEM_ISCSI ,
OPCODE_COMMON_ISCSI_NTWK_MODIFY_DEFAULT_GATEWAY ,
sizeof ( * req ) ) ;
2012-04-03 23:41:51 -05:00
if ( rt_val )
return rt_val ;
req = nonemb_cmd . va ;
2016-08-19 15:20:02 +05:30
req - > action = action ;
req - > ip_addr . ip_type = ip_type ;
memcpy ( req - > ip_addr . addr , gw ,
2016-08-19 15:20:08 +05:30
( ip_type < BEISCSI_IP_TYPE_V6 ) ? IP_V4_LEN : IP_V6_LEN ) ;
2017-10-10 16:18:14 +05:30
return beiscsi_exec_nemb_cmd ( phba , & nonemb_cmd , NULL , NULL , 0 ) ;
2016-08-19 15:20:02 +05:30
}
2012-04-03 23:41:51 -05:00
2016-08-19 15:20:02 +05:30
int beiscsi_if_set_gw ( struct beiscsi_hba * phba , u32 ip_type , u8 * gw )
{
struct be_cmd_get_def_gateway_resp gw_resp ;
int rt_val ;
2012-04-03 23:41:51 -05:00
2016-08-19 15:20:02 +05:30
memset ( & gw_resp , 0 , sizeof ( gw_resp ) ) ;
rt_val = beiscsi_if_get_gw ( phba , ip_type , & gw_resp ) ;
if ( rt_val ) {
beiscsi_log ( phba , KERN_WARNING , BEISCSI_LOG_CONFIG ,
" BG_%d : Failed to Get Gateway Addr \n " ) ;
return rt_val ;
}
2016-08-19 15:20:08 +05:30
if ( ! beiscsi_if_zero_ip ( gw_resp . ip_addr . addr , ip_type ) ) {
rt_val = beiscsi_if_mod_gw ( phba , IP_ACTION_DEL , ip_type ,
gw_resp . ip_addr . addr ) ;
if ( rt_val ) {
beiscsi_log ( phba , KERN_WARNING , BEISCSI_LOG_CONFIG ,
" BG_%d : Failed to clear Gateway Addr Set \n " ) ;
return rt_val ;
}
2016-08-19 15:20:02 +05:30
}
rt_val = beiscsi_if_mod_gw ( phba , IP_ACTION_ADD , ip_type , gw ) ;
if ( rt_val )
beiscsi_log ( phba , KERN_WARNING , BEISCSI_LOG_CONFIG ,
" BG_%d : Failed to Set Gateway Addr \n " ) ;
return rt_val ;
}
int beiscsi_if_get_gw ( struct beiscsi_hba * phba , u32 ip_type ,
struct be_cmd_get_def_gateway_resp * resp )
{
struct be_cmd_get_def_gateway_req * req ;
struct be_dma_mem nonemb_cmd ;
int rc ;
2017-10-10 16:18:14 +05:30
rc = beiscsi_prep_nemb_cmd ( phba , & nonemb_cmd , CMD_SUBSYSTEM_ISCSI ,
OPCODE_COMMON_ISCSI_NTWK_GET_DEFAULT_GATEWAY ,
sizeof ( * resp ) ) ;
2016-08-19 15:20:02 +05:30
if ( rc )
return rc ;
req = nonemb_cmd . va ;
req - > ip_type = ip_type ;
2017-10-10 16:18:14 +05:30
return beiscsi_exec_nemb_cmd ( phba , & nonemb_cmd , NULL ,
resp , sizeof ( * resp ) ) ;
2012-04-03 23:41:51 -05:00
}
2016-08-19 15:20:03 +05:30
static int
beiscsi_if_clr_ip ( struct beiscsi_hba * phba ,
struct be_cmd_get_if_info_resp * if_info )
2012-04-03 23:41:51 -05:00
{
2016-08-19 15:20:03 +05:30
struct be_cmd_set_ip_addr_req * req ;
2012-04-03 23:41:51 -05:00
struct be_dma_mem nonemb_cmd ;
int rc ;
2017-10-10 16:18:14 +05:30
rc = beiscsi_prep_nemb_cmd ( phba , & nonemb_cmd , CMD_SUBSYSTEM_ISCSI ,
OPCODE_COMMON_ISCSI_NTWK_MODIFY_IP_ADDR ,
sizeof ( * req ) ) ;
2016-01-20 14:10:50 +05:30
if ( rc )
return rc ;
2012-04-03 23:41:51 -05:00
2016-08-19 15:20:03 +05:30
req = nonemb_cmd . va ;
req - > ip_params . record_entry_count = 1 ;
req - > ip_params . ip_record . action = IP_ACTION_DEL ;
req - > ip_params . ip_record . interface_hndl =
phba - > interface_handle ;
req - > ip_params . ip_record . ip_addr . size_of_structure =
sizeof ( struct be_ip_addr_subnet_format ) ;
req - > ip_params . ip_record . ip_addr . ip_type = if_info - > ip_addr . ip_type ;
memcpy ( req - > ip_params . ip_record . ip_addr . addr ,
if_info - > ip_addr . addr ,
sizeof ( if_info - > ip_addr . addr ) ) ;
memcpy ( req - > ip_params . ip_record . ip_addr . subnet_mask ,
if_info - > ip_addr . subnet_mask ,
sizeof ( if_info - > ip_addr . subnet_mask ) ) ;
2017-10-10 16:18:14 +05:30
rc = beiscsi_exec_nemb_cmd ( phba , & nonemb_cmd , NULL , NULL , 0 ) ;
2016-08-19 15:20:03 +05:30
if ( rc < 0 | | req - > ip_params . ip_record . status ) {
beiscsi_log ( phba , KERN_INFO , BEISCSI_LOG_CONFIG ,
" BG_%d : failed to clear IP: rc %d status %d \n " ,
rc , req - > ip_params . ip_record . status ) ;
}
return rc ;
}
2012-04-03 23:41:51 -05:00
2016-08-19 15:20:03 +05:30
static int
beiscsi_if_set_ip ( struct beiscsi_hba * phba , u8 * ip ,
u8 * subnet , u32 ip_type )
{
struct be_cmd_set_ip_addr_req * req ;
struct be_dma_mem nonemb_cmd ;
uint32_t ip_len ;
int rc ;
2017-10-10 16:18:14 +05:30
rc = beiscsi_prep_nemb_cmd ( phba , & nonemb_cmd , CMD_SUBSYSTEM_ISCSI ,
OPCODE_COMMON_ISCSI_NTWK_MODIFY_IP_ADDR ,
sizeof ( * req ) ) ;
2014-06-06 14:22:44 +02:00
if ( rc )
2012-04-03 23:41:51 -05:00
return rc ;
2016-08-19 15:20:03 +05:30
req = nonemb_cmd . va ;
req - > ip_params . record_entry_count = 1 ;
req - > ip_params . ip_record . action = IP_ACTION_ADD ;
req - > ip_params . ip_record . interface_hndl =
phba - > interface_handle ;
req - > ip_params . ip_record . ip_addr . size_of_structure =
sizeof ( struct be_ip_addr_subnet_format ) ;
req - > ip_params . ip_record . ip_addr . ip_type = ip_type ;
2016-08-19 15:20:08 +05:30
ip_len = ( ip_type < BEISCSI_IP_TYPE_V6 ) ? IP_V4_LEN : IP_V6_LEN ;
2016-08-19 15:20:03 +05:30
memcpy ( req - > ip_params . ip_record . ip_addr . addr , ip , ip_len ) ;
if ( subnet )
memcpy ( req - > ip_params . ip_record . ip_addr . subnet_mask ,
subnet , ip_len ) ;
2012-04-03 23:41:51 -05:00
2017-10-10 16:18:14 +05:30
rc = beiscsi_exec_nemb_cmd ( phba , & nonemb_cmd , NULL , NULL , 0 ) ;
2016-08-19 15:20:03 +05:30
/**
* In some cases , host needs to look into individual record status
* even though FW reported success for that IOCTL .
*/
if ( rc < 0 | | req - > ip_params . ip_record . status ) {
__beiscsi_log ( phba , KERN_ERR ,
" BG_%d : failed to set IP: rc %d status %d \n " ,
rc , req - > ip_params . ip_record . status ) ;
if ( req - > ip_params . ip_record . status )
rc = - EINVAL ;
}
return rc ;
}
2012-04-03 23:41:51 -05:00
2016-08-19 15:20:03 +05:30
int beiscsi_if_en_static ( struct beiscsi_hba * phba , u32 ip_type ,
u8 * ip , u8 * subnet )
{
struct be_cmd_get_if_info_resp * if_info ;
struct be_cmd_rel_dhcp_req * reldhcp ;
struct be_dma_mem nonemb_cmd ;
int rc ;
2012-04-03 23:41:51 -05:00
2016-08-19 15:20:06 +05:30
rc = beiscsi_if_get_info ( phba , ip_type , & if_info ) ;
2016-08-19 15:20:03 +05:30
if ( rc )
return rc ;
2012-04-03 23:41:51 -05:00
2016-08-19 15:20:03 +05:30
if ( if_info - > dhcp_state ) {
2017-10-10 16:18:14 +05:30
rc = beiscsi_prep_nemb_cmd ( phba , & nonemb_cmd ,
CMD_SUBSYSTEM_ISCSI ,
2016-08-19 15:20:03 +05:30
OPCODE_COMMON_ISCSI_NTWK_REL_STATELESS_IP_ADDR ,
sizeof ( * reldhcp ) ) ;
2012-04-03 23:41:51 -05:00
if ( rc )
2014-06-27 14:55:20 +02:00
goto exit ;
2012-04-03 23:41:51 -05:00
2016-08-19 15:20:03 +05:30
reldhcp = nonemb_cmd . va ;
reldhcp - > interface_hndl = phba - > interface_handle ;
reldhcp - > ip_type = ip_type ;
2017-10-10 16:18:14 +05:30
rc = beiscsi_exec_nemb_cmd ( phba , & nonemb_cmd , NULL , NULL , 0 ) ;
2016-08-19 15:20:03 +05:30
if ( rc < 0 ) {
2012-08-20 23:00:18 +05:30
beiscsi_log ( phba , KERN_WARNING , BEISCSI_LOG_CONFIG ,
2016-08-19 15:20:03 +05:30
" BG_%d : failed to release existing DHCP: %d \n " ,
rc ) ;
2014-06-27 14:55:20 +02:00
goto exit ;
2012-04-03 23:41:51 -05:00
}
2016-08-19 15:20:03 +05:30
}
2012-04-03 23:41:51 -05:00
2016-08-19 15:20:08 +05:30
/* first delete any IP set */
if ( ! beiscsi_if_zero_ip ( if_info - > ip_addr . addr , ip_type ) ) {
rc = beiscsi_if_clr_ip ( phba , if_info ) ;
if ( rc )
goto exit ;
}
2012-04-03 23:41:51 -05:00
2016-08-19 15:20:03 +05:30
/* if ip == NULL then this is called just to release DHCP IP */
if ( ip )
rc = beiscsi_if_set_ip ( phba , ip , subnet , ip_type ) ;
exit :
kfree ( if_info ) ;
return rc ;
}
int beiscsi_if_en_dhcp ( struct beiscsi_hba * phba , u32 ip_type )
{
struct be_cmd_get_def_gateway_resp gw_resp ;
struct be_cmd_get_if_info_resp * if_info ;
struct be_cmd_set_dhcp_req * dhcpreq ;
struct be_dma_mem nonemb_cmd ;
u8 * gw ;
int rc ;
2016-08-19 15:20:06 +05:30
rc = beiscsi_if_get_info ( phba , ip_type , & if_info ) ;
2016-08-19 15:20:03 +05:30
if ( rc )
return rc ;
if ( if_info - > dhcp_state ) {
beiscsi_log ( phba , KERN_WARNING , BEISCSI_LOG_CONFIG ,
" BG_%d : DHCP Already Enabled \n " ) ;
goto exit ;
2012-04-03 23:41:51 -05:00
}
2016-08-19 15:20:08 +05:30
/* first delete any IP set */
if ( ! beiscsi_if_zero_ip ( if_info - > ip_addr . addr , ip_type ) ) {
rc = beiscsi_if_clr_ip ( phba , if_info ) ;
if ( rc )
goto exit ;
}
2016-08-19 15:20:03 +05:30
/* delete gateway settings if mode change is to DHCP */
memset ( & gw_resp , 0 , sizeof ( gw_resp ) ) ;
/* use ip_type provided in if_info */
rc = beiscsi_if_get_gw ( phba , if_info - > ip_addr . ip_type , & gw_resp ) ;
if ( rc ) {
beiscsi_log ( phba , KERN_WARNING , BEISCSI_LOG_CONFIG ,
" BG_%d : Failed to Get Gateway Addr \n " ) ;
goto exit ;
}
gw = ( u8 * ) & gw_resp . ip_addr . addr ;
2016-08-19 15:20:08 +05:30
if ( ! beiscsi_if_zero_ip ( gw , if_info - > ip_addr . ip_type ) ) {
rc = beiscsi_if_mod_gw ( phba , IP_ACTION_DEL ,
if_info - > ip_addr . ip_type , gw ) ;
if ( rc ) {
beiscsi_log ( phba , KERN_WARNING , BEISCSI_LOG_CONFIG ,
" BG_%d : Failed to clear Gateway Addr Set \n " ) ;
goto exit ;
}
2016-08-19 15:20:03 +05:30
}
2017-10-10 16:18:14 +05:30
rc = beiscsi_prep_nemb_cmd ( phba , & nonemb_cmd , CMD_SUBSYSTEM_ISCSI ,
2012-04-03 23:41:51 -05:00
OPCODE_COMMON_ISCSI_NTWK_CONFIG_STATELESS_IP_ADDR ,
sizeof ( * dhcpreq ) ) ;
2016-08-19 15:20:03 +05:30
if ( rc )
goto exit ;
2012-04-03 23:41:51 -05:00
2016-08-19 15:20:03 +05:30
dhcpreq = nonemb_cmd . va ;
2016-08-19 15:20:08 +05:30
dhcpreq - > flags = 1 ; /* 1 - blocking; 0 - non-blocking */
2016-08-19 15:20:03 +05:30
dhcpreq - > retry_count = 1 ;
dhcpreq - > interface_hndl = phba - > interface_handle ;
dhcpreq - > ip_type = ip_type ;
2017-10-10 16:18:14 +05:30
rc = beiscsi_exec_nemb_cmd ( phba , & nonemb_cmd , NULL , NULL , 0 ) ;
2012-04-03 23:41:51 -05:00
2014-06-27 14:55:20 +02:00
exit :
kfree ( if_info ) ;
2012-04-03 23:41:51 -05:00
return rc ;
}
2016-08-19 15:20:04 +05:30
/**
* beiscsi_if_set_vlan ( ) - Issue and wait for CMD completion
* @ phba : device private structure instance
* @ vlan_tag : VLAN tag
*
* Issue the MBX Cmd and wait for the completion of the
* command .
*
* returns
* Success : 0
* Failure : Non - Xero Value
* */
int beiscsi_if_set_vlan ( struct beiscsi_hba * phba , uint16_t vlan_tag )
{
int rc ;
unsigned int tag ;
tag = be_cmd_set_vlan ( phba , vlan_tag ) ;
if ( ! tag ) {
beiscsi_log ( phba , KERN_ERR ,
( BEISCSI_LOG_CONFIG | BEISCSI_LOG_MBOX ) ,
" BG_%d : VLAN Setting Failed \n " ) ;
return - EBUSY ;
}
rc = beiscsi_mccq_compl_wait ( phba , tag , NULL , NULL ) ;
if ( rc ) {
beiscsi_log ( phba , KERN_ERR ,
( BEISCSI_LOG_CONFIG | BEISCSI_LOG_MBOX ) ,
" BS_%d : VLAN MBX Cmd Failed \n " ) ;
return rc ;
}
return rc ;
}
2016-08-19 15:20:06 +05:30
int beiscsi_if_get_info ( struct beiscsi_hba * phba , int ip_type ,
struct be_cmd_get_if_info_resp * * if_info )
2012-04-03 23:41:51 -05:00
{
struct be_cmd_get_if_info_req * req ;
struct be_dma_mem nonemb_cmd ;
2013-09-28 15:35:56 -07:00
uint32_t ioctl_size = sizeof ( struct be_cmd_get_if_info_resp ) ;
2012-04-03 23:41:51 -05:00
int rc ;
2016-08-19 15:20:05 +05:30
rc = beiscsi_if_get_handle ( phba ) ;
2016-01-20 14:10:50 +05:30
if ( rc )
return rc ;
2012-04-03 23:41:51 -05:00
2013-09-28 15:35:56 -07:00
do {
2017-10-10 16:18:14 +05:30
rc = beiscsi_prep_nemb_cmd ( phba , & nonemb_cmd ,
CMD_SUBSYSTEM_ISCSI ,
OPCODE_COMMON_ISCSI_NTWK_GET_IF_INFO ,
ioctl_size ) ;
2013-09-28 15:35:56 -07:00
if ( rc )
return rc ;
2012-04-03 23:41:51 -05:00
2013-09-28 15:35:56 -07:00
req = nonemb_cmd . va ;
req - > interface_hndl = phba - > interface_handle ;
req - > ip_type = ip_type ;
/* Allocate memory for if_info */
* if_info = kzalloc ( ioctl_size , GFP_KERNEL ) ;
if ( ! * if_info ) {
beiscsi_log ( phba , KERN_ERR ,
BEISCSI_LOG_INIT | BEISCSI_LOG_CONFIG ,
" BG_%d : Memory Allocation Failure \n " ) ;
/* Free the DMA memory for the IOCTL issuing */
pci_free_consistent ( phba - > ctrl . pdev ,
nonemb_cmd . size ,
nonemb_cmd . va ,
nonemb_cmd . dma ) ;
return - ENOMEM ;
}
2017-10-10 16:18:14 +05:30
rc = beiscsi_exec_nemb_cmd ( phba , & nonemb_cmd , NULL , * if_info ,
ioctl_size ) ;
2012-04-03 23:41:51 -05:00
2013-09-28 15:35:56 -07:00
/* Check if the error is because of Insufficent_Buffer */
if ( rc = = - EAGAIN ) {
/* Get the new memory size */
ioctl_size = ( ( struct be_cmd_resp_hdr * )
nonemb_cmd . va ) - > actual_resp_len ;
ioctl_size + = sizeof ( struct be_cmd_req_hdr ) ;
/* Free the previous allocated DMA memory */
pci_free_consistent ( phba - > ctrl . pdev , nonemb_cmd . size ,
nonemb_cmd . va ,
nonemb_cmd . dma ) ;
/* Free the virtual memory */
kfree ( * if_info ) ;
} else
break ;
} while ( true ) ;
return rc ;
2012-04-03 23:41:51 -05:00
}
int mgmt_get_nic_conf ( struct beiscsi_hba * phba ,
struct be_cmd_get_nic_conf_resp * nic )
{
struct be_dma_mem nonemb_cmd ;
int rc ;
2017-10-10 16:18:14 +05:30
rc = beiscsi_prep_nemb_cmd ( phba , & nonemb_cmd , CMD_SUBSYSTEM_ISCSI ,
OPCODE_COMMON_ISCSI_NTWK_GET_NIC_CONFIG ,
sizeof ( * nic ) ) ;
2012-04-03 23:41:51 -05:00
if ( rc )
return rc ;
2017-10-10 16:18:14 +05:30
return beiscsi_exec_nemb_cmd ( phba , & nonemb_cmd , NULL ,
nic , sizeof ( * nic ) ) ;
2012-04-03 23:41:51 -05:00
}
2016-08-19 15:20:12 +05:30
static void beiscsi_boot_process_compl ( struct beiscsi_hba * phba ,
unsigned int tag )
{
struct be_cmd_get_boot_target_resp * boot_resp ;
struct be_cmd_resp_logout_fw_sess * logo_resp ;
struct be_cmd_get_session_resp * sess_resp ;
struct be_mcc_wrb * wrb ;
struct boot_struct * bs ;
int boot_work , status ;
if ( ! test_bit ( BEISCSI_HBA_BOOT_WORK , & phba - > state ) ) {
__beiscsi_log ( phba , KERN_ERR ,
" BG_%d : %s no boot work %lx \n " ,
__func__ , phba - > state ) ;
return ;
}
if ( phba - > boot_struct . tag ! = tag ) {
__beiscsi_log ( phba , KERN_ERR ,
" BG_%d : %s tag mismatch %d:%d \n " ,
__func__ , tag , phba - > boot_struct . tag ) ;
return ;
}
bs = & phba - > boot_struct ;
boot_work = 1 ;
status = 0 ;
switch ( bs - > action ) {
case BEISCSI_BOOT_REOPEN_SESS :
status = __beiscsi_mcc_compl_status ( phba , tag , NULL , NULL ) ;
if ( ! status )
bs - > action = BEISCSI_BOOT_GET_SHANDLE ;
else
bs - > retry - - ;
break ;
case BEISCSI_BOOT_GET_SHANDLE :
status = __beiscsi_mcc_compl_status ( phba , tag , & wrb , NULL ) ;
if ( ! status ) {
boot_resp = embedded_payload ( wrb ) ;
bs - > s_handle = boot_resp - > boot_session_handle ;
}
if ( bs - > s_handle = = BE_BOOT_INVALID_SHANDLE ) {
bs - > action = BEISCSI_BOOT_REOPEN_SESS ;
bs - > retry - - ;
} else {
bs - > action = BEISCSI_BOOT_GET_SINFO ;
}
break ;
case BEISCSI_BOOT_GET_SINFO :
status = __beiscsi_mcc_compl_status ( phba , tag , NULL ,
& bs - > nonemb_cmd ) ;
if ( ! status ) {
sess_resp = bs - > nonemb_cmd . va ;
memcpy ( & bs - > boot_sess , & sess_resp - > session_info ,
sizeof ( struct mgmt_session_info ) ) ;
bs - > action = BEISCSI_BOOT_LOGOUT_SESS ;
} else {
__beiscsi_log ( phba , KERN_ERR ,
" BG_%d : get boot session info error : 0x%x \n " ,
status ) ;
boot_work = 0 ;
}
pci_free_consistent ( phba - > ctrl . pdev , bs - > nonemb_cmd . size ,
bs - > nonemb_cmd . va , bs - > nonemb_cmd . dma ) ;
bs - > nonemb_cmd . va = NULL ;
break ;
case BEISCSI_BOOT_LOGOUT_SESS :
status = __beiscsi_mcc_compl_status ( phba , tag , & wrb , NULL ) ;
if ( ! status ) {
logo_resp = embedded_payload ( wrb ) ;
if ( logo_resp - > session_status ! = BE_SESS_STATUS_CLOSE ) {
__beiscsi_log ( phba , KERN_ERR ,
" BG_%d : FW boot session logout error : 0x%x \n " ,
logo_resp - > session_status ) ;
}
}
/* continue to create boot_kset even if logout failed? */
bs - > action = BEISCSI_BOOT_CREATE_KSET ;
break ;
default :
break ;
}
/* clear the tag so no other completion matches this tag */
bs - > tag = 0 ;
if ( ! bs - > retry ) {
boot_work = 0 ;
__beiscsi_log ( phba , KERN_ERR ,
" BG_%d : failed to setup boot target: status %d action %d \n " ,
status , bs - > action ) ;
}
if ( ! boot_work ) {
/* wait for next event to start boot_work */
clear_bit ( BEISCSI_HBA_BOOT_WORK , & phba - > state ) ;
return ;
}
schedule_work ( & phba - > boot_work ) ;
}
2012-08-20 23:00:08 +05:30
/**
2016-08-19 15:20:12 +05:30
* beiscsi_boot_logout_sess ( ) - Logout from boot FW session
* @ phba : Device priv structure instance
*
* return
* the TAG used for MBOX Command
2012-08-20 23:00:08 +05:30
*
2016-08-19 15:20:12 +05:30
*/
unsigned int beiscsi_boot_logout_sess ( struct beiscsi_hba * phba )
{
struct be_ctrl_info * ctrl = & phba - > ctrl ;
struct be_mcc_wrb * wrb ;
struct be_cmd_req_logout_fw_sess * req ;
unsigned int tag ;
mutex_lock ( & ctrl - > mbox_lock ) ;
wrb = alloc_mcc_wrb ( phba , & tag ) ;
if ( ! wrb ) {
mutex_unlock ( & ctrl - > mbox_lock ) ;
return 0 ;
}
req = embedded_payload ( wrb ) ;
be_wrb_hdr_prepare ( wrb , sizeof ( * req ) , true , 0 ) ;
be_cmd_hdr_prepare ( & req - > hdr , CMD_SUBSYSTEM_ISCSI_INI ,
OPCODE_ISCSI_INI_SESSION_LOGOUT_TARGET ,
sizeof ( struct be_cmd_req_logout_fw_sess ) ) ;
/* Use the session handle copied into boot_sess */
req - > session_handle = phba - > boot_struct . boot_sess . session_handle ;
phba - > boot_struct . tag = tag ;
set_bit ( MCC_TAG_STATE_ASYNC , & ctrl - > ptag_state [ tag ] . tag_state ) ;
ctrl - > ptag_state [ tag ] . cbfn = beiscsi_boot_process_compl ;
be_mcc_notify ( phba , tag ) ;
mutex_unlock ( & ctrl - > mbox_lock ) ;
return tag ;
}
/**
* beiscsi_boot_reopen_sess ( ) - Reopen boot session
* @ phba : Device priv structure instance
2012-08-20 23:00:08 +05:30
*
* return
2016-08-19 15:20:12 +05:30
* the TAG used for MBOX Command
2012-08-20 23:00:08 +05:30
*
* */
2016-08-19 15:20:12 +05:30
unsigned int beiscsi_boot_reopen_sess ( struct beiscsi_hba * phba )
2012-08-20 23:00:08 +05:30
{
2016-08-19 15:20:12 +05:30
struct be_ctrl_info * ctrl = & phba - > ctrl ;
2012-08-20 23:00:08 +05:30
struct be_mcc_wrb * wrb ;
2016-08-19 15:20:12 +05:30
struct be_cmd_reopen_session_req * req ;
2012-10-20 04:45:40 +05:30
unsigned int tag ;
2012-08-20 23:00:08 +05:30
2016-08-19 15:20:12 +05:30
mutex_lock ( & ctrl - > mbox_lock ) ;
wrb = alloc_mcc_wrb ( phba , & tag ) ;
if ( ! wrb ) {
mutex_unlock ( & ctrl - > mbox_lock ) ;
return 0 ;
}
2012-10-20 04:45:40 +05:30
2016-08-19 15:20:12 +05:30
req = embedded_payload ( wrb ) ;
be_wrb_hdr_prepare ( wrb , sizeof ( * req ) , true , 0 ) ;
be_cmd_hdr_prepare ( & req - > hdr , CMD_SUBSYSTEM_ISCSI_INI ,
OPCODE_ISCSI_INI_DRIVER_REOPEN_ALL_SESSIONS ,
sizeof ( struct be_cmd_reopen_session_resp ) ) ;
req - > reopen_type = BE_REOPEN_BOOT_SESSIONS ;
req - > session_handle = BE_BOOT_INVALID_SHANDLE ;
2012-10-20 04:45:40 +05:30
2016-08-19 15:20:12 +05:30
phba - > boot_struct . tag = tag ;
set_bit ( MCC_TAG_STATE_ASYNC , & ctrl - > ptag_state [ tag ] . tag_state ) ;
ctrl - > ptag_state [ tag ] . cbfn = beiscsi_boot_process_compl ;
2012-08-20 23:00:08 +05:30
2016-08-19 15:20:12 +05:30
be_mcc_notify ( phba , tag ) ;
mutex_unlock ( & ctrl - > mbox_lock ) ;
return tag ;
}
2012-08-20 23:00:08 +05:30
2016-08-19 15:20:12 +05:30
/**
* beiscsi_boot_get_sinfo ( ) - Get boot session info
* @ phba : device priv structure instance
*
* Fetches the boot_struct . s_handle info from FW .
* return
* the TAG used for MBOX Command
*
* */
unsigned int beiscsi_boot_get_sinfo ( struct beiscsi_hba * phba )
{
struct be_ctrl_info * ctrl = & phba - > ctrl ;
struct be_cmd_get_session_req * req ;
struct be_dma_mem * nonemb_cmd ;
struct be_mcc_wrb * wrb ;
struct be_sge * sge ;
unsigned int tag ;
2012-10-20 04:45:40 +05:30
2016-08-19 15:20:12 +05:30
mutex_lock ( & ctrl - > mbox_lock ) ;
wrb = alloc_mcc_wrb ( phba , & tag ) ;
if ( ! wrb ) {
mutex_unlock ( & ctrl - > mbox_lock ) ;
return 0 ;
}
2012-08-20 23:00:08 +05:30
2016-08-19 15:20:12 +05:30
nonemb_cmd = & phba - > boot_struct . nonemb_cmd ;
2016-12-13 15:56:01 +05:30
nonemb_cmd - > size = sizeof ( struct be_cmd_get_session_resp ) ;
2016-08-19 15:20:12 +05:30
nonemb_cmd - > va = pci_alloc_consistent ( phba - > ctrl . pdev ,
2016-11-18 14:53:39 +03:00
nonemb_cmd - > size ,
2016-08-19 15:20:12 +05:30
& nonemb_cmd - > dma ) ;
2016-08-26 15:09:08 +05:30
if ( ! nonemb_cmd - > va ) {
mutex_unlock ( & ctrl - > mbox_lock ) ;
2016-08-19 15:20:12 +05:30
return 0 ;
2016-08-26 15:09:08 +05:30
}
2016-08-19 15:20:12 +05:30
req = nonemb_cmd - > va ;
memset ( req , 0 , sizeof ( * req ) ) ;
sge = nonembedded_sgl ( wrb ) ;
be_wrb_hdr_prepare ( wrb , sizeof ( * req ) , false , 1 ) ;
be_cmd_hdr_prepare ( & req - > hdr , CMD_SUBSYSTEM_ISCSI_INI ,
OPCODE_ISCSI_INI_SESSION_GET_A_SESSION ,
2016-12-13 15:56:01 +05:30
sizeof ( struct be_cmd_get_session_resp ) ) ;
2016-08-19 15:20:12 +05:30
req - > session_handle = phba - > boot_struct . s_handle ;
sge - > pa_hi = cpu_to_le32 ( upper_32_bits ( nonemb_cmd - > dma ) ) ;
sge - > pa_lo = cpu_to_le32 ( nonemb_cmd - > dma & 0xFFFFFFFF ) ;
sge - > len = cpu_to_le32 ( nonemb_cmd - > size ) ;
phba - > boot_struct . tag = tag ;
set_bit ( MCC_TAG_STATE_ASYNC , & ctrl - > ptag_state [ tag ] . tag_state ) ;
ctrl - > ptag_state [ tag ] . cbfn = beiscsi_boot_process_compl ;
be_mcc_notify ( phba , tag ) ;
mutex_unlock ( & ctrl - > mbox_lock ) ;
return tag ;
}
unsigned int __beiscsi_boot_get_shandle ( struct beiscsi_hba * phba , int async )
{
struct be_ctrl_info * ctrl = & phba - > ctrl ;
struct be_mcc_wrb * wrb ;
struct be_cmd_get_boot_target_req * req ;
unsigned int tag ;
mutex_lock ( & ctrl - > mbox_lock ) ;
wrb = alloc_mcc_wrb ( phba , & tag ) ;
if ( ! wrb ) {
mutex_unlock ( & ctrl - > mbox_lock ) ;
return 0 ;
}
req = embedded_payload ( wrb ) ;
be_wrb_hdr_prepare ( wrb , sizeof ( * req ) , true , 0 ) ;
be_cmd_hdr_prepare ( & req - > hdr , CMD_SUBSYSTEM_ISCSI_INI ,
OPCODE_ISCSI_INI_BOOT_GET_BOOT_TARGET ,
sizeof ( struct be_cmd_get_boot_target_resp ) ) ;
if ( async ) {
phba - > boot_struct . tag = tag ;
set_bit ( MCC_TAG_STATE_ASYNC , & ctrl - > ptag_state [ tag ] . tag_state ) ;
ctrl - > ptag_state [ tag ] . cbfn = beiscsi_boot_process_compl ;
}
be_mcc_notify ( phba , tag ) ;
mutex_unlock ( & ctrl - > mbox_lock ) ;
return tag ;
}
/**
* beiscsi_boot_get_shandle ( ) - Get boot session handle
* @ phba : device priv structure instance
* @ s_handle : session handle returned for boot session .
*
* return
* Success : 1
* Failure : negative
*
* */
int beiscsi_boot_get_shandle ( struct beiscsi_hba * phba , unsigned int * s_handle )
{
struct be_cmd_get_boot_target_resp * boot_resp ;
struct be_mcc_wrb * wrb ;
unsigned int tag ;
int rc ;
* s_handle = BE_BOOT_INVALID_SHANDLE ;
/* get configured boot session count and handle */
tag = __beiscsi_boot_get_shandle ( phba , 0 ) ;
if ( ! tag ) {
beiscsi_log ( phba , KERN_ERR ,
BEISCSI_LOG_CONFIG | BEISCSI_LOG_INIT ,
" BG_%d : Getting Boot Target Info Failed \n " ) ;
return - EAGAIN ;
}
rc = beiscsi_mccq_compl_wait ( phba , tag , & wrb , NULL ) ;
if ( rc ) {
beiscsi_log ( phba , KERN_ERR ,
BEISCSI_LOG_INIT | BEISCSI_LOG_CONFIG ,
" BG_%d : MBX CMD get_boot_target Failed \n " ) ;
return - EBUSY ;
}
boot_resp = embedded_payload ( wrb ) ;
/* check if there are any boot targets configured */
if ( ! boot_resp - > boot_session_count ) {
__beiscsi_log ( phba , KERN_INFO ,
" BG_%d : No boot targets configured \n " ) ;
return - ENXIO ;
}
/* only if FW has logged in to the boot target, s_handle is valid */
* s_handle = boot_resp - > boot_session_handle ;
return 1 ;
2012-08-20 23:00:08 +05:30
}
2012-08-20 23:00:43 +05:30
2012-10-20 04:42:25 +05:30
/**
* beiscsi_drvr_ver_disp ( ) - Display the driver Name and Version
* @ dev : ptr to device not used .
* @ attr : device attribute , not used .
* @ buf : contains formatted text driver name and version
*
* return
* size of the formatted string
* */
ssize_t
beiscsi_drvr_ver_disp ( struct device * dev , struct device_attribute * attr ,
char * buf )
{
return snprintf ( buf , PAGE_SIZE , BE_NAME " \n " ) ;
}
2012-10-20 04:44:35 +05:30
2013-04-05 20:38:28 -07:00
/**
* beiscsi_fw_ver_disp ( ) - Display Firmware Version
* @ dev : ptr to device not used .
* @ attr : device attribute , not used .
* @ buf : contains formatted text Firmware version
*
* return
* size of the formatted string
* */
ssize_t
beiscsi_fw_ver_disp ( struct device * dev , struct device_attribute * attr ,
char * buf )
{
struct Scsi_Host * shost = class_to_shost ( dev ) ;
struct beiscsi_hba * phba = iscsi_host_priv ( shost ) ;
return snprintf ( buf , PAGE_SIZE , " %s \n " , phba - > fw_ver_str ) ;
}
2013-04-05 20:38:29 -07:00
/**
2013-09-28 15:35:52 -07:00
* beiscsi_active_session_disp ( ) - Display Sessions Active
2013-04-05 20:38:29 -07:00
* @ dev : ptr to device not used .
* @ attr : device attribute , not used .
* @ buf : contains formatted text Session Count
*
* return
* size of the formatted string
* */
ssize_t
2013-09-28 15:35:52 -07:00
beiscsi_active_session_disp ( struct device * dev , struct device_attribute * attr ,
2013-04-05 20:38:29 -07:00
char * buf )
{
struct Scsi_Host * shost = class_to_shost ( dev ) ;
struct beiscsi_hba * phba = iscsi_host_priv ( shost ) ;
2013-09-28 15:35:49 -07:00
uint16_t avlbl_cids = 0 , ulp_num , len = 0 , total_cids = 0 ;
for ( ulp_num = 0 ; ulp_num < BEISCSI_ULP_COUNT ; ulp_num + + ) {
if ( test_bit ( ulp_num , ( void * ) & phba - > fw_config . ulp_supported ) ) {
avlbl_cids = BEISCSI_ULP_AVLBL_CID ( phba , ulp_num ) ;
total_cids = BEISCSI_GET_CID_COUNT ( phba , ulp_num ) ;
len + = snprintf ( buf + len , PAGE_SIZE - len ,
" ULP%d : %d \n " , ulp_num ,
( total_cids - avlbl_cids ) ) ;
} else
len + = snprintf ( buf + len , PAGE_SIZE - len ,
" ULP%d : %d \n " , ulp_num , 0 ) ;
}
2013-04-05 20:38:29 -07:00
2013-09-28 15:35:49 -07:00
return len ;
2013-04-05 20:38:29 -07:00
}
2013-09-28 15:35:52 -07:00
/**
* beiscsi_free_session_disp ( ) - Display Avaliable Session
* @ dev : ptr to device not used .
* @ attr : device attribute , not used .
* @ buf : contains formatted text Session Count
*
* return
* size of the formatted string
* */
ssize_t
beiscsi_free_session_disp ( struct device * dev , struct device_attribute * attr ,
char * buf )
{
struct Scsi_Host * shost = class_to_shost ( dev ) ;
struct beiscsi_hba * phba = iscsi_host_priv ( shost ) ;
uint16_t ulp_num , len = 0 ;
for ( ulp_num = 0 ; ulp_num < BEISCSI_ULP_COUNT ; ulp_num + + ) {
if ( test_bit ( ulp_num , ( void * ) & phba - > fw_config . ulp_supported ) )
len + = snprintf ( buf + len , PAGE_SIZE - len ,
" ULP%d : %d \n " , ulp_num ,
BEISCSI_ULP_AVLBL_CID ( phba , ulp_num ) ) ;
else
len + = snprintf ( buf + len , PAGE_SIZE - len ,
" ULP%d : %d \n " , ulp_num , 0 ) ;
}
return len ;
}
2012-10-20 04:45:06 +05:30
/**
* beiscsi_adap_family_disp ( ) - Display adapter family .
* @ dev : ptr to device to get priv structure
* @ attr : device attribute , not used .
* @ buf : contains formatted text driver name and version
*
* return
* size of the formatted string
* */
ssize_t
beiscsi_adap_family_disp ( struct device * dev , struct device_attribute * attr ,
char * buf )
{
uint16_t dev_id = 0 ;
struct Scsi_Host * shost = class_to_shost ( dev ) ;
struct beiscsi_hba * phba = iscsi_host_priv ( shost ) ;
dev_id = phba - > pcidev - > device ;
switch ( dev_id ) {
case BE_DEVICE_ID1 :
case OC_DEVICE_ID1 :
case OC_DEVICE_ID2 :
2016-12-13 15:56:05 +05:30
return snprintf ( buf , PAGE_SIZE ,
" Obsolete/Unsupported BE2 Adapter Family \n " ) ;
2012-10-20 04:45:06 +05:30
break ;
case BE_DEVICE_ID2 :
case OC_DEVICE_ID3 :
return snprintf ( buf , PAGE_SIZE , " BE3-R Adapter Family \n " ) ;
break ;
case OC_SKH_ID1 :
return snprintf ( buf , PAGE_SIZE , " Skyhawk-R Adapter Family \n " ) ;
break ;
default :
return snprintf ( buf , PAGE_SIZE ,
2013-04-18 00:12:55 +09:00
" Unknown Adapter Family: 0x%x \n " , dev_id ) ;
2012-10-20 04:45:06 +05:30
break ;
}
}
2013-09-28 15:35:53 -07:00
/**
* beiscsi_phys_port ( ) - Display Physical Port Identifier
* @ dev : ptr to device not used .
* @ attr : device attribute , not used .
* @ buf : contains formatted text port identifier
*
* return
* size of the formatted string
* */
ssize_t
beiscsi_phys_port_disp ( struct device * dev , struct device_attribute * attr ,
char * buf )
{
struct Scsi_Host * shost = class_to_shost ( dev ) ;
struct beiscsi_hba * phba = iscsi_host_priv ( shost ) ;
2016-12-13 15:56:01 +05:30
return snprintf ( buf , PAGE_SIZE , " Port Identifier : %u \n " ,
2013-09-28 15:35:53 -07:00
phba - > fw_config . phys_port ) ;
}
2012-10-20 04:45:06 +05:30
2012-10-20 04:44:35 +05:30
void beiscsi_offload_cxn_v0 ( struct beiscsi_offload_params * params ,
struct wrb_handle * pwrb_handle ,
2015-08-20 04:44:30 +05:30
struct be_mem_descriptor * mem_descr ,
struct hwi_wrb_context * pwrb_context )
2012-10-20 04:44:35 +05:30
{
struct iscsi_wrb * pwrb = pwrb_handle - > pwrb ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb ,
max_send_data_segment_length , pwrb ,
params - > dw [ offsetof ( struct amap_beiscsi_offload_params ,
max_send_data_segment_length ) / 32 ] ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb , type , pwrb ,
BE_TGT_CTX_UPDT_CMD ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb ,
first_burst_length ,
pwrb ,
params - > dw [ offsetof ( struct amap_beiscsi_offload_params ,
first_burst_length ) / 32 ] ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb , erl , pwrb ,
( params - > dw [ offsetof ( struct amap_beiscsi_offload_params ,
erl ) / 32 ] & OFFLD_PARAMS_ERL ) ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb , dde , pwrb ,
( params - > dw [ offsetof ( struct amap_beiscsi_offload_params ,
dde ) / 32 ] & OFFLD_PARAMS_DDE ) > > 2 ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb , hde , pwrb ,
( params - > dw [ offsetof ( struct amap_beiscsi_offload_params ,
hde ) / 32 ] & OFFLD_PARAMS_HDE ) > > 3 ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb , ir2t , pwrb ,
( params - > dw [ offsetof ( struct amap_beiscsi_offload_params ,
ir2t ) / 32 ] & OFFLD_PARAMS_IR2T ) > > 4 ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb , imd , pwrb ,
( params - > dw [ offsetof ( struct amap_beiscsi_offload_params ,
imd ) / 32 ] & OFFLD_PARAMS_IMD ) > > 5 ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb , stat_sn ,
pwrb ,
( params - > dw [ offsetof ( struct amap_beiscsi_offload_params ,
exp_statsn ) / 32 ] + 1 ) ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb , wrb_idx ,
pwrb , pwrb_handle - > wrb_index ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb ,
max_burst_length , pwrb , params - > dw [ offsetof
( struct amap_beiscsi_offload_params ,
max_burst_length ) / 32 ] ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb , ptr2nextwrb ,
2015-08-20 04:44:30 +05:30
pwrb , pwrb_handle - > wrb_index ) ;
if ( pwrb_context - > plast_wrb )
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb ,
ptr2nextwrb ,
pwrb_context - > plast_wrb ,
pwrb_handle - > wrb_index ) ;
pwrb_context - > plast_wrb = pwrb ;
2012-10-20 04:44:35 +05:30
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb ,
session_state , pwrb , 0 ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb , compltonack ,
pwrb , 1 ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb , notpredblq ,
pwrb , 0 ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb , mode , pwrb ,
0 ) ;
mem_descr + = ISCSI_MEM_GLOBAL_HEADER ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb ,
pad_buffer_addr_hi , pwrb ,
mem_descr - > mem_array [ 0 ] . bus_address . u . a32 . address_hi ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb ,
pad_buffer_addr_lo , pwrb ,
mem_descr - > mem_array [ 0 ] . bus_address . u . a32 . address_lo ) ;
}
void beiscsi_offload_cxn_v2 ( struct beiscsi_offload_params * params ,
2015-08-20 04:44:30 +05:30
struct wrb_handle * pwrb_handle ,
struct hwi_wrb_context * pwrb_context )
2012-10-20 04:44:35 +05:30
{
struct iscsi_wrb * pwrb = pwrb_handle - > pwrb ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb_v2 ,
max_burst_length , pwrb , params - > dw [ offsetof
( struct amap_beiscsi_offload_params ,
max_burst_length ) / 32 ] ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb_v2 ,
type , pwrb ,
BE_TGT_CTX_UPDT_CMD ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb_v2 ,
ptr2nextwrb ,
2015-08-20 04:44:30 +05:30
pwrb , pwrb_handle - > wrb_index ) ;
if ( pwrb_context - > plast_wrb )
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb_v2 ,
ptr2nextwrb ,
pwrb_context - > plast_wrb ,
pwrb_handle - > wrb_index ) ;
pwrb_context - > plast_wrb = pwrb ;
2012-10-20 04:44:35 +05:30
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb_v2 , wrb_idx ,
pwrb , pwrb_handle - > wrb_index ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb_v2 ,
max_send_data_segment_length , pwrb ,
params - > dw [ offsetof ( struct amap_beiscsi_offload_params ,
max_send_data_segment_length ) / 32 ] ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb_v2 ,
first_burst_length , pwrb ,
params - > dw [ offsetof ( struct amap_beiscsi_offload_params ,
first_burst_length ) / 32 ] ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb_v2 ,
2013-09-28 15:35:41 -07:00
max_recv_dataseg_len , pwrb ,
params - > dw [ offsetof ( struct amap_beiscsi_offload_params ,
max_recv_data_segment_length ) / 32 ] ) ;
2012-10-20 04:44:35 +05:30
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb_v2 ,
max_cxns , pwrb , BEISCSI_MAX_CXNS ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb_v2 , erl , pwrb ,
( params - > dw [ offsetof ( struct amap_beiscsi_offload_params ,
erl ) / 32 ] & OFFLD_PARAMS_ERL ) ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb_v2 , dde , pwrb ,
( params - > dw [ offsetof ( struct amap_beiscsi_offload_params ,
dde ) / 32 ] & OFFLD_PARAMS_DDE ) > > 2 ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb_v2 , hde , pwrb ,
( params - > dw [ offsetof ( struct amap_beiscsi_offload_params ,
hde ) / 32 ] & OFFLD_PARAMS_HDE ) > > 3 ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb_v2 ,
ir2t , pwrb ,
( params - > dw [ offsetof ( struct amap_beiscsi_offload_params ,
ir2t ) / 32 ] & OFFLD_PARAMS_IR2T ) > > 4 ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb_v2 , imd , pwrb ,
( params - > dw [ offsetof ( struct amap_beiscsi_offload_params ,
imd ) / 32 ] & OFFLD_PARAMS_IMD ) > > 5 ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb_v2 ,
data_seq_inorder ,
pwrb ,
( params - > dw [ offsetof ( struct amap_beiscsi_offload_params ,
data_seq_inorder ) / 32 ] &
OFFLD_PARAMS_DATA_SEQ_INORDER ) > > 6 ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb_v2 ,
pdu_seq_inorder ,
pwrb ,
( params - > dw [ offsetof ( struct amap_beiscsi_offload_params ,
pdu_seq_inorder ) / 32 ] &
OFFLD_PARAMS_PDU_SEQ_INORDER ) > > 7 ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb_v2 , max_r2t ,
pwrb ,
( params - > dw [ offsetof ( struct amap_beiscsi_offload_params ,
max_r2t ) / 32 ] &
OFFLD_PARAMS_MAX_R2T ) > > 8 ) ;
AMAP_SET_BITS ( struct amap_iscsi_target_context_update_wrb_v2 , stat_sn ,
pwrb ,
( params - > dw [ offsetof ( struct amap_beiscsi_offload_params ,
exp_statsn ) / 32 ] + 1 ) ) ;
}
2016-12-13 15:55:55 +05:30
2017-03-24 14:11:41 +05:30
unsigned int beiscsi_invalidate_cxn ( struct beiscsi_hba * phba ,
struct beiscsi_endpoint * beiscsi_ep )
{
struct be_invalidate_connection_params_in * req ;
struct be_ctrl_info * ctrl = & phba - > ctrl ;
struct be_mcc_wrb * wrb ;
unsigned int tag = 0 ;
mutex_lock ( & ctrl - > mbox_lock ) ;
wrb = alloc_mcc_wrb ( phba , & tag ) ;
if ( ! wrb ) {
mutex_unlock ( & ctrl - > mbox_lock ) ;
return 0 ;
}
req = embedded_payload ( wrb ) ;
be_wrb_hdr_prepare ( wrb , sizeof ( union be_invalidate_connection_params ) ,
true , 0 ) ;
be_cmd_hdr_prepare ( & req - > hdr , CMD_SUBSYSTEM_ISCSI_INI ,
OPCODE_ISCSI_INI_DRIVER_INVALIDATE_CONNECTION ,
sizeof ( * req ) ) ;
req - > session_handle = beiscsi_ep - > fw_handle ;
req - > cid = beiscsi_ep - > ep_cid ;
if ( beiscsi_ep - > conn )
req - > cleanup_type = BE_CLEANUP_TYPE_INVALIDATE ;
else
req - > cleanup_type = BE_CLEANUP_TYPE_ISSUE_TCP_RST ;
/**
* 0 - non - persistent targets
* 1 - save session info on flash
*/
req - > save_cfg = 0 ;
be_mcc_notify ( phba , tag ) ;
mutex_unlock ( & ctrl - > mbox_lock ) ;
return tag ;
}
unsigned int beiscsi_upload_cxn ( struct beiscsi_hba * phba ,
struct beiscsi_endpoint * beiscsi_ep )
{
struct be_ctrl_info * ctrl = & phba - > ctrl ;
struct be_mcc_wrb * wrb ;
struct be_tcp_upload_params_in * req ;
unsigned int tag ;
mutex_lock ( & ctrl - > mbox_lock ) ;
wrb = alloc_mcc_wrb ( phba , & tag ) ;
if ( ! wrb ) {
mutex_unlock ( & ctrl - > mbox_lock ) ;
return 0 ;
}
req = embedded_payload ( wrb ) ;
be_wrb_hdr_prepare ( wrb , sizeof ( union be_tcp_upload_params ) , true , 0 ) ;
be_cmd_hdr_prepare ( & req - > hdr , CMD_COMMON_TCP_UPLOAD ,
OPCODE_COMMON_TCP_UPLOAD , sizeof ( * req ) ) ;
req - > id = beiscsi_ep - > ep_cid ;
if ( beiscsi_ep - > conn )
req - > upload_type = BE_UPLOAD_TYPE_GRACEFUL ;
else
req - > upload_type = BE_UPLOAD_TYPE_ABORT ;
be_mcc_notify ( phba , tag ) ;
mutex_unlock ( & ctrl - > mbox_lock ) ;
return tag ;
}
2016-12-13 15:55:55 +05:30
int beiscsi_mgmt_invalidate_icds ( struct beiscsi_hba * phba ,
struct invldt_cmd_tbl * inv_tbl ,
unsigned int nents )
{
struct be_ctrl_info * ctrl = & phba - > ctrl ;
struct invldt_cmds_params_in * req ;
struct be_dma_mem nonemb_cmd ;
struct be_mcc_wrb * wrb ;
unsigned int i , tag ;
struct be_sge * sge ;
int rc ;
if ( ! nents | | nents > BE_INVLDT_CMD_TBL_SZ )
return - EINVAL ;
nonemb_cmd . size = sizeof ( union be_invldt_cmds_params ) ;
nonemb_cmd . va = pci_zalloc_consistent ( phba - > ctrl . pdev ,
nonemb_cmd . size ,
& nonemb_cmd . dma ) ;
if ( ! nonemb_cmd . va ) {
beiscsi_log ( phba , KERN_ERR , BEISCSI_LOG_EH ,
" BM_%d : invldt_cmds_params alloc failed \n " ) ;
return - ENOMEM ;
}
mutex_lock ( & ctrl - > mbox_lock ) ;
wrb = alloc_mcc_wrb ( phba , & tag ) ;
if ( ! wrb ) {
mutex_unlock ( & ctrl - > mbox_lock ) ;
pci_free_consistent ( phba - > ctrl . pdev , nonemb_cmd . size ,
nonemb_cmd . va , nonemb_cmd . dma ) ;
return - ENOMEM ;
}
req = nonemb_cmd . va ;
be_wrb_hdr_prepare ( wrb , nonemb_cmd . size , false , 1 ) ;
be_cmd_hdr_prepare ( & req - > hdr , CMD_SUBSYSTEM_ISCSI ,
OPCODE_COMMON_ISCSI_ERROR_RECOVERY_INVALIDATE_COMMANDS ,
sizeof ( * req ) ) ;
req - > ref_handle = 0 ;
req - > cleanup_type = CMD_ISCSI_COMMAND_INVALIDATE ;
for ( i = 0 ; i < nents ; i + + ) {
req - > table [ i ] . icd = inv_tbl [ i ] . icd ;
req - > table [ i ] . cid = inv_tbl [ i ] . cid ;
req - > icd_count + + ;
}
sge = nonembedded_sgl ( wrb ) ;
sge - > pa_hi = cpu_to_le32 ( upper_32_bits ( nonemb_cmd . dma ) ) ;
sge - > pa_lo = cpu_to_le32 ( lower_32_bits ( nonemb_cmd . dma ) ) ;
sge - > len = cpu_to_le32 ( nonemb_cmd . size ) ;
be_mcc_notify ( phba , tag ) ;
mutex_unlock ( & ctrl - > mbox_lock ) ;
rc = beiscsi_mccq_compl_wait ( phba , tag , NULL , & nonemb_cmd ) ;
if ( rc ! = - EBUSY )
pci_free_consistent ( phba - > ctrl . pdev , nonemb_cmd . size ,
nonemb_cmd . va , nonemb_cmd . dma ) ;
return rc ;
}