diff options
Diffstat (limited to 'drivers/s390/scsi')
| -rw-r--r-- | drivers/s390/scsi/zfcp_aux.c | 204 | ||||
| -rw-r--r-- | drivers/s390/scsi/zfcp_ccw.c | 18 | ||||
| -rw-r--r-- | drivers/s390/scsi/zfcp_dbf.c | 13 | ||||
| -rw-r--r-- | drivers/s390/scsi/zfcp_def.h | 45 | ||||
| -rw-r--r-- | drivers/s390/scsi/zfcp_erp.c | 397 | ||||
| -rw-r--r-- | drivers/s390/scsi/zfcp_ext.h | 25 | ||||
| -rw-r--r-- | drivers/s390/scsi/zfcp_fsf.c | 415 | ||||
| -rw-r--r-- | drivers/s390/scsi/zfcp_qdio.c | 78 | ||||
| -rw-r--r-- | drivers/s390/scsi/zfcp_scsi.c | 174 | 
9 files changed, 638 insertions, 731 deletions
| diff --git a/drivers/s390/scsi/zfcp_aux.c b/drivers/s390/scsi/zfcp_aux.c index 9cd789b8acd4..5d39b2df0cc4 100644 --- a/drivers/s390/scsi/zfcp_aux.c +++ b/drivers/s390/scsi/zfcp_aux.c @@ -112,6 +112,109 @@ _zfcp_hex_dump(char *addr, int count)  		printk("\n");  } + +/****************************************************************/ +/****** Functions to handle the request ID hash table    ********/ +/****************************************************************/ + +#define ZFCP_LOG_AREA			ZFCP_LOG_AREA_FSF + +static int zfcp_reqlist_init(struct zfcp_adapter *adapter) +{ +	int i; + +	adapter->req_list = kcalloc(REQUEST_LIST_SIZE, sizeof(struct list_head), +				    GFP_KERNEL); + +	if (!adapter->req_list) +		return -ENOMEM; + +	for (i=0; i<REQUEST_LIST_SIZE; i++) +		INIT_LIST_HEAD(&adapter->req_list[i]); + +	return 0; +} + +static void zfcp_reqlist_free(struct zfcp_adapter *adapter) +{ +	struct zfcp_fsf_req *request, *tmp; +	unsigned int i; + +	for (i=0; i<REQUEST_LIST_SIZE; i++) { +		if (list_empty(&adapter->req_list[i])) +			continue; + +		list_for_each_entry_safe(request, tmp, +					 &adapter->req_list[i], list) +			list_del(&request->list); +	} + +	kfree(adapter->req_list); +} + +void zfcp_reqlist_add(struct zfcp_adapter *adapter, +		      struct zfcp_fsf_req *fsf_req) +{ +	unsigned int i; + +	i = fsf_req->req_id % REQUEST_LIST_SIZE; +	list_add_tail(&fsf_req->list, &adapter->req_list[i]); +} + +void zfcp_reqlist_remove(struct zfcp_adapter *adapter, unsigned long req_id) +{ +	struct zfcp_fsf_req *request, *tmp; +	unsigned int i, counter; +	u64 dbg_tmp[2]; + +	i = req_id % REQUEST_LIST_SIZE; +	BUG_ON(list_empty(&adapter->req_list[i])); + +	counter = 0; +	list_for_each_entry_safe(request, tmp, &adapter->req_list[i], list) { +		if (request->req_id == req_id) { +			dbg_tmp[0] = (u64) atomic_read(&adapter->reqs_active); +			dbg_tmp[1] = (u64) counter; +			debug_event(adapter->erp_dbf, 4, (void *) dbg_tmp, 16); +			list_del(&request->list); +			break; +		} +		counter++; +	} +} + +struct zfcp_fsf_req *zfcp_reqlist_ismember(struct zfcp_adapter *adapter, +					   unsigned long req_id) +{ +	struct zfcp_fsf_req *request, *tmp; +	unsigned int i; + +	/* 0 is reserved as an invalid req_id */ +	if (req_id == 0) +		return NULL; + +	i = req_id % REQUEST_LIST_SIZE; + +	list_for_each_entry_safe(request, tmp, &adapter->req_list[i], list) +		if (request->req_id == req_id) +			return request; + +	return NULL; +} + +int zfcp_reqlist_isempty(struct zfcp_adapter *adapter) +{ +	unsigned int i; + +	for (i=0; i<REQUEST_LIST_SIZE; i++) +		if (!list_empty(&adapter->req_list[i])) +			return 0; + +	return 1; +} + +#undef ZFCP_LOG_AREA +  /****************************************************************/  /************** Uncategorised Functions *************************/  /****************************************************************/ @@ -200,11 +303,45 @@ zfcp_init_device_configure(void)  	return;  } +static int calc_alignment(int size) +{ +	int align = 1; + +	if (!size) +		return 0; + +	while ((size - align) > 0) +		align <<= 1; + +	return align; +} +  static int __init  zfcp_module_init(void)  { +	int retval = -ENOMEM; +	int size, align; + +	size = sizeof(struct zfcp_fsf_req_qtcb); +	align = calc_alignment(size); +	zfcp_data.fsf_req_qtcb_cache = +		kmem_cache_create("zfcp_fsf", size, align, 0, NULL, NULL); +	if (!zfcp_data.fsf_req_qtcb_cache) +		goto out; -	int retval = 0; +	size = sizeof(struct fsf_status_read_buffer); +	align = calc_alignment(size); +	zfcp_data.sr_buffer_cache = +		kmem_cache_create("zfcp_sr", size, align, 0, NULL, NULL); +	if (!zfcp_data.sr_buffer_cache) +		goto out_sr_cache; + +	size = sizeof(struct zfcp_gid_pn_data); +	align = calc_alignment(size); +	zfcp_data.gid_pn_cache = +		kmem_cache_create("zfcp_gid", size, align, 0, NULL, NULL); +	if (!zfcp_data.gid_pn_cache) +		goto out_gid_cache;  	atomic_set(&zfcp_data.loglevel, loglevel); @@ -214,15 +351,16 @@ zfcp_module_init(void)  	/* initialize adapters to be removed list head */  	INIT_LIST_HEAD(&zfcp_data.adapter_remove_lh); -	zfcp_transport_template = fc_attach_transport(&zfcp_transport_functions); -	if (!zfcp_transport_template) -		return -ENODEV; +	zfcp_data.scsi_transport_template = +		fc_attach_transport(&zfcp_transport_functions); +	if (!zfcp_data.scsi_transport_template) +		goto out_transport;  	retval = misc_register(&zfcp_cfdc_misc);  	if (retval != 0) {  		ZFCP_LOG_INFO("registration of misc device "  			      "zfcp_cfdc failed\n"); -		goto out; +		goto out_misc;  	}  	ZFCP_LOG_TRACE("major/minor for zfcp_cfdc: %d/%d\n", @@ -234,9 +372,6 @@ zfcp_module_init(void)  	/* initialise configuration rw lock */  	rwlock_init(&zfcp_data.config_lock); -	/* save address of data structure managing the driver module */ -	zfcp_data.scsi_host_template.module = THIS_MODULE; -  	/* setup dynamic I/O */  	retval = zfcp_ccw_register();  	if (retval) { @@ -251,6 +386,14 @@ zfcp_module_init(void)   out_ccw_register:  	misc_deregister(&zfcp_cfdc_misc); + out_misc: +	fc_release_transport(zfcp_data.scsi_transport_template); + out_transport: +	kmem_cache_destroy(zfcp_data.gid_pn_cache); + out_gid_cache: +	kmem_cache_destroy(zfcp_data.sr_buffer_cache); + out_sr_cache: +	kmem_cache_destroy(zfcp_data.fsf_req_qtcb_cache);   out:  	return retval;  } @@ -836,20 +979,20 @@ static int  zfcp_allocate_low_mem_buffers(struct zfcp_adapter *adapter)  {  	adapter->pool.fsf_req_erp = -		mempool_create_kmalloc_pool(ZFCP_POOL_FSF_REQ_ERP_NR, -				sizeof(struct zfcp_fsf_req_pool_element)); +		mempool_create_slab_pool(ZFCP_POOL_FSF_REQ_ERP_NR, +					 zfcp_data.fsf_req_qtcb_cache);  	if (!adapter->pool.fsf_req_erp)  		return -ENOMEM;  	adapter->pool.fsf_req_scsi = -		mempool_create_kmalloc_pool(ZFCP_POOL_FSF_REQ_SCSI_NR, -				sizeof(struct zfcp_fsf_req_pool_element)); +		mempool_create_slab_pool(ZFCP_POOL_FSF_REQ_SCSI_NR, +					 zfcp_data.fsf_req_qtcb_cache);  	if (!adapter->pool.fsf_req_scsi)  		return -ENOMEM;  	adapter->pool.fsf_req_abort = -		mempool_create_kmalloc_pool(ZFCP_POOL_FSF_REQ_ABORT_NR, -				sizeof(struct zfcp_fsf_req_pool_element)); +		mempool_create_slab_pool(ZFCP_POOL_FSF_REQ_ABORT_NR, +					 zfcp_data.fsf_req_qtcb_cache);  	if (!adapter->pool.fsf_req_abort)  		return -ENOMEM; @@ -860,14 +1003,14 @@ zfcp_allocate_low_mem_buffers(struct zfcp_adapter *adapter)  		return -ENOMEM;  	adapter->pool.data_status_read = -		mempool_create_kmalloc_pool(ZFCP_POOL_STATUS_READ_NR, -					sizeof(struct fsf_status_read_buffer)); +		mempool_create_slab_pool(ZFCP_POOL_STATUS_READ_NR, +					 zfcp_data.sr_buffer_cache);  	if (!adapter->pool.data_status_read)  		return -ENOMEM;  	adapter->pool.data_gid_pn = -		mempool_create_kmalloc_pool(ZFCP_POOL_DATA_GID_PN_NR, -					    sizeof(struct zfcp_gid_pn_data)); +		mempool_create_slab_pool(ZFCP_POOL_DATA_GID_PN_NR, +					 zfcp_data.gid_pn_cache);  	if (!adapter->pool.data_gid_pn)  		return -ENOMEM; @@ -961,8 +1104,12 @@ zfcp_adapter_enqueue(struct ccw_device *ccw_device)  	INIT_LIST_HEAD(&adapter->port_remove_lh);  	/* initialize list of fsf requests */ -	spin_lock_init(&adapter->fsf_req_list_lock); -	INIT_LIST_HEAD(&adapter->fsf_req_list_head); +	spin_lock_init(&adapter->req_list_lock); +	retval = zfcp_reqlist_init(adapter); +	if (retval) { +		ZFCP_LOG_INFO("request list initialization failed\n"); +		goto failed_low_mem_buffers; +	}  	/* initialize debug locks */ @@ -988,9 +1135,6 @@ zfcp_adapter_enqueue(struct ccw_device *ccw_device)  	/* initialize lock of associated request queue */  	rwlock_init(&adapter->request_queue.queue_lock); -	/* intitialise SCSI ER timer */ -	init_timer(&adapter->scsi_er_timer); -  	/* mark adapter unusable as long as sysfs registration is not complete */  	atomic_set_mask(ZFCP_STATUS_COMMON_REMOVE, &adapter->status); @@ -1041,8 +1185,6 @@ zfcp_adapter_enqueue(struct ccw_device *ccw_device)   *		!0 - struct zfcp_adapter  data structure could not be removed   *			(e.g. still used)   * locks:	adapter list write lock is assumed to be held by caller - *              adapter->fsf_req_list_lock is taken and released within this  - *              function and must not be held on entry   */  void  zfcp_adapter_dequeue(struct zfcp_adapter *adapter) @@ -1054,14 +1196,14 @@ zfcp_adapter_dequeue(struct zfcp_adapter *adapter)  	zfcp_sysfs_adapter_remove_files(&adapter->ccw_device->dev);  	dev_set_drvdata(&adapter->ccw_device->dev, NULL);  	/* sanity check: no pending FSF requests */ -	spin_lock_irqsave(&adapter->fsf_req_list_lock, flags); -	retval = !list_empty(&adapter->fsf_req_list_head); -	spin_unlock_irqrestore(&adapter->fsf_req_list_lock, flags); -	if (retval) { +	spin_lock_irqsave(&adapter->req_list_lock, flags); +	retval = zfcp_reqlist_isempty(adapter); +	spin_unlock_irqrestore(&adapter->req_list_lock, flags); +	if (!retval) {  		ZFCP_LOG_NORMAL("bug: adapter %s (%p) still in use, "  				"%i requests outstanding\n",  				zfcp_get_busid_by_adapter(adapter), adapter, -				atomic_read(&adapter->fsf_reqs_active)); +				atomic_read(&adapter->reqs_active));  		retval = -EBUSY;  		goto out;  	} @@ -1087,6 +1229,7 @@ zfcp_adapter_dequeue(struct zfcp_adapter *adapter)  	zfcp_free_low_mem_buffers(adapter);  	/* free memory of adapter data structure and queues */  	zfcp_qdio_free_queues(adapter); +	zfcp_reqlist_free(adapter);  	kfree(adapter->fc_stats);  	kfree(adapter->stats_reset_data);  	ZFCP_LOG_TRACE("freeing adapter structure\n"); @@ -1507,7 +1650,6 @@ zfcp_ns_gid_pn_request(struct zfcp_erp_action *erp_action)  	gid_pn->ct.handler = zfcp_ns_gid_pn_handler;  	gid_pn->ct.handler_data = (unsigned long) gid_pn;          gid_pn->ct.timeout = ZFCP_NS_GID_PN_TIMEOUT; -        gid_pn->ct.timer = &erp_action->timer;  	gid_pn->port = erp_action->port;  	ret = zfcp_fsf_send_ct(&gid_pn->ct, adapter->pool.fsf_req_erp, diff --git a/drivers/s390/scsi/zfcp_ccw.c b/drivers/s390/scsi/zfcp_ccw.c index 57d8e4bfb8d9..81680efa1721 100644 --- a/drivers/s390/scsi/zfcp_ccw.c +++ b/drivers/s390/scsi/zfcp_ccw.c @@ -164,6 +164,11 @@ zfcp_ccw_set_online(struct ccw_device *ccw_device)  	retval = zfcp_adapter_scsi_register(adapter);  	if (retval)  		goto out_scsi_register; + +	/* initialize request counter */ +	BUG_ON(!zfcp_reqlist_isempty(adapter)); +	adapter->req_no = 0; +  	zfcp_erp_modify_adapter_status(adapter, ZFCP_STATUS_COMMON_RUNNING,  				       ZFCP_SET);  	zfcp_erp_adapter_reopen(adapter, ZFCP_STATUS_COMMON_ERP_FAILED); @@ -270,19 +275,6 @@ zfcp_ccw_register(void)  }  /** - * zfcp_ccw_unregister - ccw unregister function - * - * Unregisters the driver from common i/o layer. Function will be called at - * module unload/system shutdown. - */ -void __exit -zfcp_ccw_unregister(void) -{ -	zfcp_sysfs_driver_remove_files(&zfcp_ccw_driver.driver); -	ccw_driver_unregister(&zfcp_ccw_driver); -} - -/**   * zfcp_ccw_shutdown - gets called on reboot/shutdown   *   * Makes sure that QDIO queues are down when the system gets stopped. diff --git a/drivers/s390/scsi/zfcp_dbf.c b/drivers/s390/scsi/zfcp_dbf.c index c033145d0f19..0aa3b1ac76af 100644 --- a/drivers/s390/scsi/zfcp_dbf.c +++ b/drivers/s390/scsi/zfcp_dbf.c @@ -707,7 +707,7 @@ _zfcp_scsi_dbf_event_common(const char *tag, const char *tag2, int level,  			    struct zfcp_adapter *adapter,  			    struct scsi_cmnd *scsi_cmnd,  			    struct zfcp_fsf_req *fsf_req, -			    struct zfcp_fsf_req *old_fsf_req) +			    unsigned long old_req_id)  {  	struct zfcp_scsi_dbf_record *rec = &adapter->scsi_dbf_buf;  	struct zfcp_dbf_dump *dump = (struct zfcp_dbf_dump *)rec; @@ -768,8 +768,7 @@ _zfcp_scsi_dbf_event_common(const char *tag, const char *tag2, int level,  				rec->fsf_seqno = fsf_req->seq_no;  				rec->fsf_issued = fsf_req->issued;  			} -			rec->type.old_fsf_reqid = -				    (unsigned long) old_fsf_req; +			rec->type.old_fsf_reqid = old_req_id;  		} else {  			strncpy(dump->tag, "dump", ZFCP_DBF_TAG_SIZE);  			dump->total_size = buflen; @@ -794,17 +793,17 @@ zfcp_scsi_dbf_event_result(const char *tag, int level,  			   struct zfcp_fsf_req *fsf_req)  {  	_zfcp_scsi_dbf_event_common("rslt", tag, level, -			adapter, scsi_cmnd, fsf_req, NULL); +			adapter, scsi_cmnd, fsf_req, 0);  }  inline void  zfcp_scsi_dbf_event_abort(const char *tag, struct zfcp_adapter *adapter,  			  struct scsi_cmnd *scsi_cmnd,  			  struct zfcp_fsf_req *new_fsf_req, -			  struct zfcp_fsf_req *old_fsf_req) +			  unsigned long old_req_id)  {  	_zfcp_scsi_dbf_event_common("abrt", tag, 1, -			adapter, scsi_cmnd, new_fsf_req, old_fsf_req); +			adapter, scsi_cmnd, new_fsf_req, old_req_id);  }  inline void @@ -814,7 +813,7 @@ zfcp_scsi_dbf_event_devreset(const char *tag, u8 flag, struct zfcp_unit *unit,  	struct zfcp_adapter *adapter = unit->port->adapter;  	_zfcp_scsi_dbf_event_common(flag == FCP_TARGET_RESET ? "trst" : "lrst", -			tag, 1, adapter, scsi_cmnd, NULL, NULL); +			tag, 1, adapter, scsi_cmnd, NULL, 0);  }  static int diff --git a/drivers/s390/scsi/zfcp_def.h b/drivers/s390/scsi/zfcp_def.h index 2df512a18e2c..8f882690994d 100644 --- a/drivers/s390/scsi/zfcp_def.h +++ b/drivers/s390/scsi/zfcp_def.h @@ -19,7 +19,6 @@   * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.    */  -  #ifndef ZFCP_DEF_H  #define ZFCP_DEF_H @@ -32,6 +31,10 @@  #include <linux/blkdev.h>  #include <linux/delay.h>  #include <linux/timer.h> +#include <linux/slab.h> +#include <linux/mempool.h> +#include <linux/syscalls.h> +#include <linux/ioctl.h>  #include <scsi/scsi.h>  #include <scsi/scsi_tcq.h>  #include <scsi/scsi_cmnd.h> @@ -39,20 +42,17 @@  #include <scsi/scsi_host.h>  #include <scsi/scsi_transport.h>  #include <scsi/scsi_transport_fc.h> -#include "zfcp_fsf.h"  #include <asm/ccwdev.h>  #include <asm/qdio.h>  #include <asm/debug.h>  #include <asm/ebcdic.h> -#include <linux/mempool.h> -#include <linux/syscalls.h> -#include <linux/ioctl.h> +#include "zfcp_fsf.h"  /********************* GENERAL DEFINES *********************************/  /* zfcp version number, it consists of major, minor, and patch-level number */ -#define ZFCP_VERSION		"4.7.0" +#define ZFCP_VERSION		"4.8.0"  /**   * zfcp_sg_to_address - determine kernel address from struct scatterlist @@ -80,7 +80,7 @@ zfcp_address_to_sg(void *address, struct scatterlist *list)  #define REQUEST_LIST_SIZE 128  /********************* SCSI SPECIFIC DEFINES *********************************/ -#define ZFCP_SCSI_ER_TIMEOUT                    (100*HZ) +#define ZFCP_SCSI_ER_TIMEOUT                    (10*HZ)  /********************* CIO/QDIO SPECIFIC DEFINES *****************************/ @@ -137,7 +137,7 @@ zfcp_address_to_sg(void *address, struct scatterlist *list)  #define ZFCP_EXCHANGE_CONFIG_DATA_RETRIES	7  /* timeout value for "default timer" for fsf requests */ -#define ZFCP_FSF_REQUEST_TIMEOUT (60*HZ); +#define ZFCP_FSF_REQUEST_TIMEOUT (60*HZ)  /*************** FIBRE CHANNEL PROTOCOL SPECIFIC DEFINES ********************/ @@ -543,7 +543,7 @@ do { \  } while (0)  #if ZFCP_LOG_LEVEL_LIMIT < ZFCP_LOG_LEVEL_NORMAL -# define ZFCP_LOG_NORMAL(fmt, args...) +# define ZFCP_LOG_NORMAL(fmt, args...)	do { } while (0)  #else  # define ZFCP_LOG_NORMAL(fmt, args...) \  do { \ @@ -553,7 +553,7 @@ do { \  #endif  #if ZFCP_LOG_LEVEL_LIMIT < ZFCP_LOG_LEVEL_INFO -# define ZFCP_LOG_INFO(fmt, args...) +# define ZFCP_LOG_INFO(fmt, args...)	do { } while (0)  #else  # define ZFCP_LOG_INFO(fmt, args...) \  do { \ @@ -563,14 +563,14 @@ do { \  #endif  #if ZFCP_LOG_LEVEL_LIMIT < ZFCP_LOG_LEVEL_DEBUG -# define ZFCP_LOG_DEBUG(fmt, args...) +# define ZFCP_LOG_DEBUG(fmt, args...)	do { } while (0)  #else  # define ZFCP_LOG_DEBUG(fmt, args...) \  	ZFCP_LOG(ZFCP_LOG_LEVEL_DEBUG, fmt , ##args)  #endif  #if ZFCP_LOG_LEVEL_LIMIT < ZFCP_LOG_LEVEL_TRACE -# define ZFCP_LOG_TRACE(fmt, args...) +# define ZFCP_LOG_TRACE(fmt, args...)	do { } while (0)  #else  # define ZFCP_LOG_TRACE(fmt, args...) \  	ZFCP_LOG(ZFCP_LOG_LEVEL_TRACE, fmt , ##args) @@ -779,7 +779,6 @@ typedef void (*zfcp_send_ct_handler_t)(unsigned long);   * @handler_data: data passed to handler function   * @pool: pointer to memory pool for ct request structure   * @timeout: FSF timeout for this request - * @timer: timer (e.g. for request initiated by erp)   * @completion: completion for synchronization purposes   * @status: used to pass error status to calling function   */ @@ -793,7 +792,6 @@ struct zfcp_send_ct {  	unsigned long handler_data;  	mempool_t *pool;  	int timeout; -	struct timer_list *timer;  	struct completion *completion;  	int status;  }; @@ -821,7 +819,6 @@ typedef void (*zfcp_send_els_handler_t)(unsigned long);   * @resp_count: number of elements in response scatter-gather list   * @handler: handler function (called for response to the request)   * @handler_data: data passed to handler function - * @timer: timer (e.g. for request initiated by erp)   * @completion: completion for synchronization purposes   * @ls_code: hex code of ELS command   * @status: used to pass error status to calling function @@ -836,7 +833,6 @@ struct zfcp_send_els {  	unsigned int resp_count;  	zfcp_send_els_handler_t handler;  	unsigned long handler_data; -	struct timer_list *timer;  	struct completion *completion;  	int ls_code;  	int status; @@ -886,11 +882,10 @@ struct zfcp_adapter {  	struct list_head        port_remove_lh;    /* head of ports to be  						      removed */  	u32			ports;	           /* number of remote ports */ -        struct timer_list       scsi_er_timer;     /* SCSI err recovery watch */ -	struct list_head	fsf_req_list_head; /* head of FSF req list */ -	spinlock_t		fsf_req_list_lock; /* lock for ops on list of -						      FSF requests */ -        atomic_t       		fsf_reqs_active;   /* # active FSF reqs */ +	atomic_t		reqs_active;	   /* # active FSF reqs */ +	unsigned long		req_no;		   /* unique FSF req number */ +	struct list_head	*req_list;	   /* list of pending reqs */ +	spinlock_t		req_list_lock;	   /* request list lock */  	struct zfcp_qdio_queue	request_queue;	   /* request queue */  	u32			fsf_req_seq_no;	   /* FSF cmnd seq number */  	wait_queue_head_t	request_wq;	   /* can be used to wait for @@ -986,6 +981,7 @@ struct zfcp_unit {  /* FSF request */  struct zfcp_fsf_req {  	struct list_head       list;	       /* list of FSF requests */ +	unsigned long	       req_id;	       /* unique request ID */  	struct zfcp_adapter    *adapter;       /* adapter request belongs to */  	u8		       sbal_number;    /* nr of SBALs free for use */  	u8		       sbal_first;     /* first SBAL for this request */ @@ -1002,6 +998,7 @@ struct zfcp_fsf_req {  	struct fsf_qtcb	       *qtcb;	       /* address of associated QTCB */  	u32		       seq_no;         /* Sequence number of request */          unsigned long          data;           /* private data of request */  +	struct timer_list      timer;	       /* used for erp or scsi er */  	struct zfcp_erp_action *erp_action;    /* used if this request is  						  issued on behalf of erp */  	mempool_t	       *pool;	       /* used if request was alloacted @@ -1015,6 +1012,7 @@ typedef void zfcp_fsf_req_handler_t(struct zfcp_fsf_req*);  /* driver data */  struct zfcp_data {  	struct scsi_host_template scsi_host_template; +	struct scsi_transport_template *scsi_transport_template;          atomic_t                status;             /* Module status flags */  	struct list_head	adapter_list_head;  /* head of adapter list */  	struct list_head	adapter_remove_lh;  /* head of adapters to be @@ -1030,6 +1028,9 @@ struct zfcp_data {  	wwn_t                   init_wwpn;  	fcp_lun_t               init_fcp_lun;  	char 			*driver_version; +	kmem_cache_t		*fsf_req_qtcb_cache; +	kmem_cache_t		*sr_buffer_cache; +	kmem_cache_t		*gid_pn_cache;  };  /** @@ -1050,7 +1051,7 @@ struct zfcp_sg_list {  #define ZFCP_POOL_DATA_GID_PN_NR	1  /* struct used by memory pools for fsf_requests */ -struct zfcp_fsf_req_pool_element { +struct zfcp_fsf_req_qtcb {  	struct zfcp_fsf_req fsf_req;  	struct fsf_qtcb qtcb;  }; diff --git a/drivers/s390/scsi/zfcp_erp.c b/drivers/s390/scsi/zfcp_erp.c index 4682c8b8bd24..862a411a4aa0 100644 --- a/drivers/s390/scsi/zfcp_erp.c +++ b/drivers/s390/scsi/zfcp_erp.c @@ -64,8 +64,6 @@ static int zfcp_erp_strategy_check_action(struct zfcp_erp_action *, int);  static int zfcp_erp_adapter_strategy(struct zfcp_erp_action *);  static int zfcp_erp_adapter_strategy_generic(struct zfcp_erp_action *, int);  static int zfcp_erp_adapter_strategy_close(struct zfcp_erp_action *); -static int zfcp_erp_adapter_strategy_close_qdio(struct zfcp_erp_action *); -static int zfcp_erp_adapter_strategy_close_fsf(struct zfcp_erp_action *);  static int zfcp_erp_adapter_strategy_open(struct zfcp_erp_action *);  static int zfcp_erp_adapter_strategy_open_qdio(struct zfcp_erp_action *);  static int zfcp_erp_adapter_strategy_open_fsf(struct zfcp_erp_action *); @@ -93,10 +91,10 @@ static int zfcp_erp_unit_strategy_clearstati(struct zfcp_unit *);  static int zfcp_erp_unit_strategy_close(struct zfcp_erp_action *);  static int zfcp_erp_unit_strategy_open(struct zfcp_erp_action *); -static int zfcp_erp_action_dismiss_adapter(struct zfcp_adapter *); -static int zfcp_erp_action_dismiss_port(struct zfcp_port *); -static int zfcp_erp_action_dismiss_unit(struct zfcp_unit *); -static int zfcp_erp_action_dismiss(struct zfcp_erp_action *); +static void zfcp_erp_action_dismiss_adapter(struct zfcp_adapter *); +static void zfcp_erp_action_dismiss_port(struct zfcp_port *); +static void zfcp_erp_action_dismiss_unit(struct zfcp_unit *); +static void zfcp_erp_action_dismiss(struct zfcp_erp_action *);  static int zfcp_erp_action_enqueue(int, struct zfcp_adapter *,  				   struct zfcp_port *, struct zfcp_unit *); @@ -112,8 +110,62 @@ static inline void zfcp_erp_action_to_ready(struct zfcp_erp_action *);  static inline void zfcp_erp_action_to_running(struct zfcp_erp_action *);  static void zfcp_erp_memwait_handler(unsigned long); -static void zfcp_erp_timeout_handler(unsigned long); -static inline void zfcp_erp_timeout_init(struct zfcp_erp_action *); + +/** + * zfcp_close_qdio - close qdio queues for an adapter + */ +static void zfcp_close_qdio(struct zfcp_adapter *adapter) +{ +	struct zfcp_qdio_queue *req_queue; +	int first, count; + +	if (!atomic_test_mask(ZFCP_STATUS_ADAPTER_QDIOUP, &adapter->status)) +		return; + +	/* clear QDIOUP flag, thus do_QDIO is not called during qdio_shutdown */ +	req_queue = &adapter->request_queue; +	write_lock_irq(&req_queue->queue_lock); +	atomic_clear_mask(ZFCP_STATUS_ADAPTER_QDIOUP, &adapter->status); +	write_unlock_irq(&req_queue->queue_lock); + +	debug_text_event(adapter->erp_dbf, 3, "qdio_down2a"); +	while (qdio_shutdown(adapter->ccw_device, +			     QDIO_FLAG_CLEANUP_USING_CLEAR) == -EINPROGRESS) +		msleep(1000); +	debug_text_event(adapter->erp_dbf, 3, "qdio_down2b"); + +	/* cleanup used outbound sbals */ +	count = atomic_read(&req_queue->free_count); +	if (count < QDIO_MAX_BUFFERS_PER_Q) { +		first = (req_queue->free_index+count) % QDIO_MAX_BUFFERS_PER_Q; +		count = QDIO_MAX_BUFFERS_PER_Q - count; +		zfcp_qdio_zero_sbals(req_queue->buffer, first, count); +	} +	req_queue->free_index = 0; +	atomic_set(&req_queue->free_count, 0); +	req_queue->distance_from_int = 0; +	adapter->response_queue.free_index = 0; +	atomic_set(&adapter->response_queue.free_count, 0); +} + +/** + * zfcp_close_fsf - stop FSF operations for an adapter + * + * Dismiss and cleanup all pending fsf_reqs (this wakes up all initiators of + * requests waiting for completion; especially this returns SCSI commands + * with error state). + */ +static void zfcp_close_fsf(struct zfcp_adapter *adapter) +{ +	/* close queues to ensure that buffers are not accessed by adapter */ +	zfcp_close_qdio(adapter); +	zfcp_fsf_req_dismiss_all(adapter); +	/* reset FSF request sequence number */ +	adapter->fsf_req_seq_no = 0; +	/* all ports and units are closed */ +	zfcp_erp_modify_adapter_status(adapter, +				       ZFCP_STATUS_COMMON_OPEN, ZFCP_CLEAR); +}  /**   * zfcp_fsf_request_timeout_handler - called if a request timed out @@ -122,42 +174,20 @@ static inline void zfcp_erp_timeout_init(struct zfcp_erp_action *);   * This function needs to be called if requests (ELS, Generic Service,   * or SCSI commands) exceed a certain time limit. The assumption is   * that after the time limit the adapter get stuck. So we trigger a reopen of - * the adapter. This should not be used for error recovery, SCSI abort - * commands and SCSI requests from SCSI mid-layer. + * the adapter.   */ -void -zfcp_fsf_request_timeout_handler(unsigned long data) +static void zfcp_fsf_request_timeout_handler(unsigned long data)  { -	struct zfcp_adapter *adapter; - -	adapter = (struct zfcp_adapter *) data; - +	struct zfcp_adapter *adapter = (struct zfcp_adapter *) data;  	zfcp_erp_adapter_reopen(adapter, 0);  } -/* - * function:	zfcp_fsf_scsi_er_timeout_handler - * - * purpose:     This function needs to be called whenever a SCSI error recovery - *              action (abort/reset) does not return. - *              Re-opening the adapter means that the command can be returned - *              by zfcp (it is guarranteed that it does not return via the - *              adapter anymore). The buffer can then be used again. - *     - * returns:     sod all - */ -void -zfcp_fsf_scsi_er_timeout_handler(unsigned long data) +void zfcp_fsf_start_timer(struct zfcp_fsf_req *fsf_req, unsigned long timeout)  { -	struct zfcp_adapter *adapter = (struct zfcp_adapter *) data; - -	ZFCP_LOG_NORMAL("warning: SCSI error recovery timed out. " -			"Restarting all operations on the adapter %s\n", -			zfcp_get_busid_by_adapter(adapter)); -	debug_text_event(adapter->erp_dbf, 1, "eh_lmem_tout"); -	zfcp_erp_adapter_reopen(adapter, 0); - -	return; +	fsf_req->timer.function = zfcp_fsf_request_timeout_handler; +	fsf_req->timer.data = (unsigned long) fsf_req->adapter; +	fsf_req->timer.expires = timeout; +	add_timer(&fsf_req->timer);  }  /* @@ -167,7 +197,7 @@ zfcp_fsf_scsi_er_timeout_handler(unsigned long data)   *		initiates adapter recovery which is done   *		asynchronously   * - * returns:	0	- initiated action succesfully + * returns:	0	- initiated action successfully   *		<0	- failed to initiate action   */  int @@ -203,7 +233,7 @@ zfcp_erp_adapter_reopen_internal(struct zfcp_adapter *adapter, int clear_mask)   * purpose:	Wrappper for zfcp_erp_adapter_reopen_internal   *              used to ensure the correct locking   * - * returns:	0	- initiated action succesfully + * returns:	0	- initiated action successfully   *		<0	- failed to initiate action   */  int @@ -273,7 +303,6 @@ zfcp_erp_adisc(struct zfcp_port *port)  	struct zfcp_ls_adisc *adisc;  	void *address = NULL;  	int retval = 0; -	struct timer_list *timer;  	send_els = kzalloc(sizeof(struct zfcp_send_els), GFP_ATOMIC);  	if (send_els == NULL) @@ -320,22 +349,11 @@ zfcp_erp_adisc(struct zfcp_port *port)  		      (wwn_t) adisc->wwnn, adisc->hard_nport_id,  		      adisc->nport_id); -	timer = kmalloc(sizeof(struct timer_list), GFP_ATOMIC); -	if (!timer) -		goto nomem; - -	init_timer(timer); -	timer->function = zfcp_fsf_request_timeout_handler; -	timer->data = (unsigned long) adapter; -	timer->expires = ZFCP_FSF_REQUEST_TIMEOUT; -	send_els->timer = timer; -  	retval = zfcp_fsf_send_els(send_els);  	if (retval != 0) {  		ZFCP_LOG_NORMAL("error: initiation of Send ELS failed for port "  				"0x%08x on adapter %s\n", send_els->d_id,  				zfcp_get_busid_by_adapter(adapter)); -		del_timer(send_els->timer);  		goto freemem;  	} @@ -347,7 +365,6 @@ zfcp_erp_adisc(struct zfcp_port *port)  	if (address != NULL)  		__free_pages(send_els->req->page, 0);  	if (send_els != NULL) { -		kfree(send_els->timer);  		kfree(send_els->req);  		kfree(send_els->resp);  		kfree(send_els); @@ -373,9 +390,6 @@ zfcp_erp_adisc_handler(unsigned long data)  	struct zfcp_ls_adisc_acc *adisc;  	send_els = (struct zfcp_send_els *) data; - -	del_timer(send_els->timer); -  	adapter = send_els->adapter;  	port = send_els->port;  	d_id = send_els->d_id; @@ -424,7 +438,6 @@ zfcp_erp_adisc_handler(unsigned long data)   out:  	zfcp_port_put(port);  	__free_pages(send_els->req->page, 0); -	kfree(send_els->timer);  	kfree(send_els->req);  	kfree(send_els->resp);  	kfree(send_els); @@ -469,7 +482,7 @@ zfcp_test_link(struct zfcp_port *port)   *		initiates Forced Reopen recovery which is done   *		asynchronously   * - * returns:	0	- initiated action succesfully + * returns:	0	- initiated action successfully   *		<0	- failed to initiate action   */  static int @@ -509,7 +522,7 @@ zfcp_erp_port_forced_reopen_internal(struct zfcp_port *port, int clear_mask)   * purpose:	Wrappper for zfcp_erp_port_forced_reopen_internal   *              used to ensure the correct locking   * - * returns:	0	- initiated action succesfully + * returns:	0	- initiated action successfully   *		<0	- failed to initiate action   */  int @@ -536,7 +549,7 @@ zfcp_erp_port_forced_reopen(struct zfcp_port *port, int clear_mask)   *		initiates Reopen recovery which is done   *		asynchronously   * - * returns:	0	- initiated action succesfully + * returns:	0	- initiated action successfully   *		<0	- failed to initiate action   */  static int @@ -605,7 +618,7 @@ zfcp_erp_port_reopen(struct zfcp_port *port, int clear_mask)   *		initiates Reopen recovery which is done   *		asynchronously   * - * returns:	0	- initiated action succesfully + * returns:	0	- initiated action successfully   *		<0	- failed to initiate action   */  static int @@ -670,17 +683,10 @@ zfcp_erp_unit_reopen(struct zfcp_unit *unit, int clear_mask)  	return retval;  } -/* - * function:	 - * - * purpose:	disable I/O, - *		return any open requests and clean them up, - *		aim: no pending and incoming I/O - * - * returns: +/** + * zfcp_erp_adapter_block - mark adapter as blocked, block scsi requests   */ -static void -zfcp_erp_adapter_block(struct zfcp_adapter *adapter, int clear_mask) +static void zfcp_erp_adapter_block(struct zfcp_adapter *adapter, int clear_mask)  {  	debug_text_event(adapter->erp_dbf, 6, "a_bl");  	zfcp_erp_modify_adapter_status(adapter, @@ -688,15 +694,10 @@ zfcp_erp_adapter_block(struct zfcp_adapter *adapter, int clear_mask)  				       clear_mask, ZFCP_CLEAR);  } -/* - * function:	 - * - * purpose:	enable I/O - * - * returns: +/** + * zfcp_erp_adapter_unblock - mark adapter as unblocked, allow scsi requests   */ -static void -zfcp_erp_adapter_unblock(struct zfcp_adapter *adapter) +static void zfcp_erp_adapter_unblock(struct zfcp_adapter *adapter)  {  	debug_text_event(adapter->erp_dbf, 6, "a_ubl");  	atomic_set_mask(ZFCP_STATUS_COMMON_UNBLOCKED, &adapter->status); @@ -848,18 +849,16 @@ zfcp_erp_strategy_check_fsfreq(struct zfcp_erp_action *erp_action)  	struct zfcp_adapter *adapter = erp_action->adapter;  	if (erp_action->fsf_req) { -		/* take lock to ensure that request is not being deleted meanwhile */ -		spin_lock(&adapter->fsf_req_list_lock); -		/* check whether fsf req does still exist */ -		list_for_each_entry(fsf_req, &adapter->fsf_req_list_head, list) -		    if (fsf_req == erp_action->fsf_req) -			break; -		if (fsf_req && (fsf_req->erp_action == erp_action)) { +		/* take lock to ensure that request is not deleted meanwhile */ +		spin_lock(&adapter->req_list_lock); +		if ((!zfcp_reqlist_ismember(adapter, +					    erp_action->fsf_req->req_id)) && +		    (fsf_req->erp_action == erp_action)) {  			/* fsf_req still exists */  			debug_text_event(adapter->erp_dbf, 3, "a_ca_req");  			debug_event(adapter->erp_dbf, 3, &fsf_req,  				    sizeof (unsigned long)); -			/* dismiss fsf_req of timed out or dismissed erp_action */ +			/* dismiss fsf_req of timed out/dismissed erp_action */  			if (erp_action->status & (ZFCP_STATUS_ERP_DISMISSED |  						  ZFCP_STATUS_ERP_TIMEDOUT)) {  				debug_text_event(adapter->erp_dbf, 3, @@ -892,77 +891,50 @@ zfcp_erp_strategy_check_fsfreq(struct zfcp_erp_action *erp_action)  			 */  			erp_action->fsf_req = NULL;  		} -		spin_unlock(&adapter->fsf_req_list_lock); +		spin_unlock(&adapter->req_list_lock);  	} else  		debug_text_event(adapter->erp_dbf, 3, "a_ca_noreq");  	return retval;  } -/* - * purpose:	generic handler for asynchronous events related to erp_action events - *		(normal completion, time-out, dismissing, retry after - *		low memory condition) - * - * note:	deletion of timer is not required (e.g. in case of a time-out), - *		but a second try does no harm, - *		we leave it in here to allow for greater simplification +/** + * zfcp_erp_async_handler_nolock - complete erp_action   * - * returns:	0 - there was an action to handle - *		!0 - otherwise + * Used for normal completion, time-out, dismissal and failure after + * low memory condition.   */ -static int -zfcp_erp_async_handler_nolock(struct zfcp_erp_action *erp_action, -			      unsigned long set_mask) +static void zfcp_erp_async_handler_nolock(struct zfcp_erp_action *erp_action, +					  unsigned long set_mask)  { -	int retval;  	struct zfcp_adapter *adapter = erp_action->adapter;  	if (zfcp_erp_action_exists(erp_action) == ZFCP_ERP_ACTION_RUNNING) {  		debug_text_event(adapter->erp_dbf, 2, "a_asyh_ex");  		debug_event(adapter->erp_dbf, 2, &erp_action->action,  			    sizeof (int)); -		if (!(set_mask & ZFCP_STATUS_ERP_TIMEDOUT)) -			del_timer(&erp_action->timer);  		erp_action->status |= set_mask;  		zfcp_erp_action_ready(erp_action); -		retval = 0;  	} else {  		/* action is ready or gone - nothing to do */  		debug_text_event(adapter->erp_dbf, 3, "a_asyh_gone");  		debug_event(adapter->erp_dbf, 3, &erp_action->action,  			    sizeof (int)); -		retval = 1;  	} - -	return retval;  } -/* - * purpose:	generic handler for asynchronous events related to erp_action - *               events	(normal completion, time-out, dismissing, retry after - *		low memory condition) - * - * note:	deletion of timer is not required (e.g. in case of a time-out), - *		but a second try does no harm, - *		we leave it in here to allow for greater simplification - * - * returns:	0 - there was an action to handle - *		!0 - otherwise +/** + * zfcp_erp_async_handler - wrapper for erp_async_handler_nolock w/ locking   */ -int -zfcp_erp_async_handler(struct zfcp_erp_action *erp_action, -		       unsigned long set_mask) +void zfcp_erp_async_handler(struct zfcp_erp_action *erp_action, +			    unsigned long set_mask)  {  	struct zfcp_adapter *adapter = erp_action->adapter;  	unsigned long flags; -	int retval;  	write_lock_irqsave(&adapter->erp_lock, flags); -	retval = zfcp_erp_async_handler_nolock(erp_action, set_mask); +	zfcp_erp_async_handler_nolock(erp_action, set_mask);  	write_unlock_irqrestore(&adapter->erp_lock, flags); - -	return retval;  }  /* @@ -987,8 +959,7 @@ zfcp_erp_memwait_handler(unsigned long data)   *		action gets an appropriate flag and will be processed   *		accordingly   */ -static void -zfcp_erp_timeout_handler(unsigned long data) +void zfcp_erp_timeout_handler(unsigned long data)  {  	struct zfcp_erp_action *erp_action = (struct zfcp_erp_action *) data;  	struct zfcp_adapter *adapter = erp_action->adapter; @@ -999,17 +970,15 @@ zfcp_erp_timeout_handler(unsigned long data)  	zfcp_erp_async_handler(erp_action, ZFCP_STATUS_ERP_TIMEDOUT);  } -/* - * purpose:	is called for an erp_action which needs to be ended - *		though not being done, - *		this is usually required if an higher is generated, - *		action gets an appropriate flag and will be processed - *		accordingly +/** + * zfcp_erp_action_dismiss - dismiss an erp_action   * - * locks:	erp_lock held (thus we need to call another handler variant) + * adapter->erp_lock must be held + *  + * Dismissal of an erp_action is usually required if an erp_action of + * higher priority is generated.   */ -static int -zfcp_erp_action_dismiss(struct zfcp_erp_action *erp_action) +static void zfcp_erp_action_dismiss(struct zfcp_erp_action *erp_action)  {  	struct zfcp_adapter *adapter = erp_action->adapter; @@ -1017,8 +986,6 @@ zfcp_erp_action_dismiss(struct zfcp_erp_action *erp_action)  	debug_event(adapter->erp_dbf, 2, &erp_action->action, sizeof (int));  	zfcp_erp_async_handler_nolock(erp_action, ZFCP_STATUS_ERP_DISMISSED); - -	return 0;  }  int @@ -1805,7 +1772,7 @@ zfcp_erp_modify_unit_status(struct zfcp_unit *unit, u32 mask, int set_or_clear)   * purpose:	Wrappper for zfcp_erp_port_reopen_all_internal   *              used to ensure the correct locking   * - * returns:	0	- initiated action succesfully + * returns:	0	- initiated action successfully   *		<0	- failed to initiate action   */  int @@ -1968,8 +1935,7 @@ zfcp_erp_adapter_strategy_generic(struct zfcp_erp_action *erp_action, int close)  			  &erp_action->adapter->status);   failed_openfcp: -	zfcp_erp_adapter_strategy_close_qdio(erp_action); -	zfcp_erp_adapter_strategy_close_fsf(erp_action); +	zfcp_close_fsf(erp_action->adapter);   failed_qdio:   out:  	return retval; @@ -2074,69 +2040,6 @@ zfcp_erp_adapter_strategy_open_qdio(struct zfcp_erp_action *erp_action)  	return retval;  } -/* - * function:    zfcp_qdio_cleanup - * - * purpose:	cleans up QDIO operation for the specified adapter - * - * returns:	0 - successful cleanup - *		!0 - failed cleanup - */ -int -zfcp_erp_adapter_strategy_close_qdio(struct zfcp_erp_action *erp_action) -{ -	int retval = ZFCP_ERP_SUCCEEDED; -	int first_used; -	int used_count; -	struct zfcp_adapter *adapter = erp_action->adapter; - -	if (!atomic_test_mask(ZFCP_STATUS_ADAPTER_QDIOUP, &adapter->status)) { -		ZFCP_LOG_DEBUG("error: attempt to shut down inactive QDIO " -			       "queues on adapter %s\n", -			       zfcp_get_busid_by_adapter(adapter)); -		retval = ZFCP_ERP_FAILED; -		goto out; -	} - -	/* -	 * Get queue_lock and clear QDIOUP flag. Thus it's guaranteed that -	 * do_QDIO won't be called while qdio_shutdown is in progress. -	 */ - -	write_lock_irq(&adapter->request_queue.queue_lock); -	atomic_clear_mask(ZFCP_STATUS_ADAPTER_QDIOUP, &adapter->status); -	write_unlock_irq(&adapter->request_queue.queue_lock); - -	debug_text_event(adapter->erp_dbf, 3, "qdio_down2a"); -	while (qdio_shutdown(adapter->ccw_device, -			     QDIO_FLAG_CLEANUP_USING_CLEAR) == -EINPROGRESS) -		msleep(1000); -	debug_text_event(adapter->erp_dbf, 3, "qdio_down2b"); - -	/* -	 * First we had to stop QDIO operation. -	 * Now it is safe to take the following actions. -	 */ - -	/* Cleanup only necessary when there are unacknowledged buffers */ -	if (atomic_read(&adapter->request_queue.free_count) -	    < QDIO_MAX_BUFFERS_PER_Q) { -		first_used = (adapter->request_queue.free_index + -			      atomic_read(&adapter->request_queue.free_count)) -			% QDIO_MAX_BUFFERS_PER_Q; -		used_count = QDIO_MAX_BUFFERS_PER_Q - -			atomic_read(&adapter->request_queue.free_count); -		zfcp_qdio_zero_sbals(adapter->request_queue.buffer, -				     first_used, used_count); -	} -	adapter->response_queue.free_index = 0; -	atomic_set(&adapter->response_queue.free_count, 0); -	adapter->request_queue.free_index = 0; -	atomic_set(&adapter->request_queue.free_count, 0); -	adapter->request_queue.distance_from_int = 0; - out: -	return retval; -}  static int  zfcp_erp_adapter_strategy_open_fsf(struct zfcp_erp_action *erp_action) @@ -2168,10 +2071,9 @@ zfcp_erp_adapter_strategy_open_fsf_xconfig(struct zfcp_erp_action *erp_action)  		atomic_clear_mask(ZFCP_STATUS_ADAPTER_HOST_CON_INIT,  				  &adapter->status);  		ZFCP_LOG_DEBUG("Doing exchange config data\n"); -		write_lock(&adapter->erp_lock); +		write_lock_irq(&adapter->erp_lock);  		zfcp_erp_action_to_running(erp_action); -		write_unlock(&adapter->erp_lock); -		zfcp_erp_timeout_init(erp_action); +		write_unlock_irq(&adapter->erp_lock);  		if (zfcp_fsf_exchange_config_data(erp_action)) {  			retval = ZFCP_ERP_FAILED;  			debug_text_event(adapter->erp_dbf, 5, "a_fstx_xf"); @@ -2236,11 +2138,10 @@ zfcp_erp_adapter_strategy_open_fsf_xport(struct zfcp_erp_action *erp_action)  	adapter = erp_action->adapter;  	atomic_clear_mask(ZFCP_STATUS_ADAPTER_XPORT_OK, &adapter->status); -	write_lock(&adapter->erp_lock); +	write_lock_irq(&adapter->erp_lock);  	zfcp_erp_action_to_running(erp_action); -	write_unlock(&adapter->erp_lock); +	write_unlock_irq(&adapter->erp_lock); -	zfcp_erp_timeout_init(erp_action);  	ret = zfcp_fsf_exchange_port_data(erp_action, adapter, NULL);  	if (ret == -EOPNOTSUPP) {  		debug_text_event(adapter->erp_dbf, 3, "a_xport_notsupp"); @@ -2258,11 +2159,11 @@ zfcp_erp_adapter_strategy_open_fsf_xport(struct zfcp_erp_action *erp_action)  			      "%s)\n", zfcp_get_busid_by_adapter(adapter));  		ret = ZFCP_ERP_FAILED;  	} -	if (!atomic_test_mask(ZFCP_STATUS_ADAPTER_XPORT_OK, &adapter->status)) { -		ZFCP_LOG_INFO("error: exchange port data failed (adapter " + +	/* don't treat as error for the sake of compatibility */ +	if (!atomic_test_mask(ZFCP_STATUS_ADAPTER_XPORT_OK, &adapter->status)) +		ZFCP_LOG_INFO("warning: exchange port data failed (adapter "  			      "%s\n", zfcp_get_busid_by_adapter(adapter)); -		ret = ZFCP_ERP_FAILED; -	}  	return ret;  } @@ -2293,35 +2194,6 @@ zfcp_erp_adapter_strategy_open_fsf_statusread(struct zfcp_erp_action  }  /* - * function:    zfcp_fsf_cleanup - * - * purpose:	cleanup FSF operation for specified adapter - * - * returns:	0 - FSF operation successfully cleaned up - *		!0 - failed to cleanup FSF operation for this adapter - */ -static int -zfcp_erp_adapter_strategy_close_fsf(struct zfcp_erp_action *erp_action) -{ -	int retval = ZFCP_ERP_SUCCEEDED; -	struct zfcp_adapter *adapter = erp_action->adapter; - -	/* -	 * wake waiting initiators of requests, -	 * return SCSI commands (with error status), -	 * clean up all requests (synchronously) -	 */ -	zfcp_fsf_req_dismiss_all(adapter); -	/* reset FSF request sequence number */ -	adapter->fsf_req_seq_no = 0; -	/* all ports and units are closed */ -	zfcp_erp_modify_adapter_status(adapter, -				       ZFCP_STATUS_COMMON_OPEN, ZFCP_CLEAR); - -	return retval; -} - -/*   * function:	   *   * purpose:	this routine executes the 'Reopen Physical Port' action @@ -2657,7 +2529,6 @@ zfcp_erp_port_forced_strategy_close(struct zfcp_erp_action *erp_action)  	struct zfcp_adapter *adapter = erp_action->adapter;  	struct zfcp_port *port = erp_action->port; -	zfcp_erp_timeout_init(erp_action);  	retval = zfcp_fsf_close_physical_port(erp_action);  	if (retval == -ENOMEM) {  		debug_text_event(adapter->erp_dbf, 5, "o_pfstc_nomem"); @@ -2714,7 +2585,6 @@ zfcp_erp_port_strategy_close(struct zfcp_erp_action *erp_action)  	struct zfcp_adapter *adapter = erp_action->adapter;  	struct zfcp_port *port = erp_action->port; -	zfcp_erp_timeout_init(erp_action);  	retval = zfcp_fsf_close_port(erp_action);  	if (retval == -ENOMEM) {  		debug_text_event(adapter->erp_dbf, 5, "p_pstc_nomem"); @@ -2752,7 +2622,6 @@ zfcp_erp_port_strategy_open_port(struct zfcp_erp_action *erp_action)  	struct zfcp_adapter *adapter = erp_action->adapter;  	struct zfcp_port *port = erp_action->port; -	zfcp_erp_timeout_init(erp_action);  	retval = zfcp_fsf_open_port(erp_action);  	if (retval == -ENOMEM) {  		debug_text_event(adapter->erp_dbf, 5, "p_psto_nomem"); @@ -2790,7 +2659,6 @@ zfcp_erp_port_strategy_open_common_lookup(struct zfcp_erp_action *erp_action)  	struct zfcp_adapter *adapter = erp_action->adapter;  	struct zfcp_port *port = erp_action->port; -	zfcp_erp_timeout_init(erp_action);  	retval = zfcp_ns_gid_pn_request(erp_action);  	if (retval == -ENOMEM) {  		debug_text_event(adapter->erp_dbf, 5, "p_pstn_nomem"); @@ -2916,7 +2784,6 @@ zfcp_erp_unit_strategy_close(struct zfcp_erp_action *erp_action)  	struct zfcp_adapter *adapter = erp_action->adapter;  	struct zfcp_unit *unit = erp_action->unit; -	zfcp_erp_timeout_init(erp_action);  	retval = zfcp_fsf_close_unit(erp_action);  	if (retval == -ENOMEM) {  		debug_text_event(adapter->erp_dbf, 5, "u_ustc_nomem"); @@ -2957,7 +2824,6 @@ zfcp_erp_unit_strategy_open(struct zfcp_erp_action *erp_action)  	struct zfcp_adapter *adapter = erp_action->adapter;  	struct zfcp_unit *unit = erp_action->unit; -	zfcp_erp_timeout_init(erp_action);  	retval = zfcp_fsf_open_unit(erp_action);  	if (retval == -ENOMEM) {  		debug_text_event(adapter->erp_dbf, 5, "u_usto_nomem"); @@ -2982,14 +2848,13 @@ zfcp_erp_unit_strategy_open(struct zfcp_erp_action *erp_action)  	return retval;  } -static inline void -zfcp_erp_timeout_init(struct zfcp_erp_action *erp_action) +void zfcp_erp_start_timer(struct zfcp_fsf_req *fsf_req)  { -	init_timer(&erp_action->timer); -	erp_action->timer.function = zfcp_erp_timeout_handler; -	erp_action->timer.data = (unsigned long) erp_action; -	/* jiffies will be added in zfcp_fsf_req_send */ -	erp_action->timer.expires = ZFCP_ERP_FSFREQ_TIMEOUT; +	BUG_ON(!fsf_req->erp_action); +	fsf_req->timer.function = zfcp_erp_timeout_handler; +	fsf_req->timer.data = (unsigned long) fsf_req->erp_action; +	fsf_req->timer.expires = jiffies + ZFCP_ERP_FSFREQ_TIMEOUT; +	add_timer(&fsf_req->timer);  }  /* @@ -3293,10 +3158,8 @@ zfcp_erp_action_cleanup(int action, struct zfcp_adapter *adapter,  } -static int -zfcp_erp_action_dismiss_adapter(struct zfcp_adapter *adapter) +static void zfcp_erp_action_dismiss_adapter(struct zfcp_adapter *adapter)  { -	int retval = 0;  	struct zfcp_port *port;  	debug_text_event(adapter->erp_dbf, 5, "a_actab"); @@ -3305,14 +3168,10 @@ zfcp_erp_action_dismiss_adapter(struct zfcp_adapter *adapter)  	else  		list_for_each_entry(port, &adapter->port_list_head, list)  		    zfcp_erp_action_dismiss_port(port); - -	return retval;  } -static int -zfcp_erp_action_dismiss_port(struct zfcp_port *port) +static void zfcp_erp_action_dismiss_port(struct zfcp_port *port)  { -	int retval = 0;  	struct zfcp_unit *unit;  	struct zfcp_adapter *adapter = port->adapter; @@ -3323,22 +3182,16 @@ zfcp_erp_action_dismiss_port(struct zfcp_port *port)  	else  		list_for_each_entry(unit, &port->unit_list_head, list)  		    zfcp_erp_action_dismiss_unit(unit); - -	return retval;  } -static int -zfcp_erp_action_dismiss_unit(struct zfcp_unit *unit) +static void zfcp_erp_action_dismiss_unit(struct zfcp_unit *unit)  { -	int retval = 0;  	struct zfcp_adapter *adapter = unit->port->adapter;  	debug_text_event(adapter->erp_dbf, 5, "u_actab");  	debug_event(adapter->erp_dbf, 5, &unit->fcp_lun, sizeof (fcp_lun_t));  	if (atomic_test_mask(ZFCP_STATUS_COMMON_ERP_INUSE, &unit->status))  		zfcp_erp_action_dismiss(&unit->erp_action); - -	return retval;  }  static inline void diff --git a/drivers/s390/scsi/zfcp_ext.h b/drivers/s390/scsi/zfcp_ext.h index d02366004cdd..b8794d77285d 100644 --- a/drivers/s390/scsi/zfcp_ext.h +++ b/drivers/s390/scsi/zfcp_ext.h @@ -55,7 +55,6 @@ extern void   zfcp_unit_dequeue(struct zfcp_unit *);  /******************************* S/390 IO ************************************/  extern int  zfcp_ccw_register(void); -extern void zfcp_ccw_unregister(void);  extern void zfcp_qdio_zero_sbals(struct qdio_buffer **, int, int);  extern int  zfcp_qdio_allocate(struct zfcp_adapter *); @@ -63,7 +62,6 @@ extern int  zfcp_qdio_allocate_queues(struct zfcp_adapter *);  extern void zfcp_qdio_free_queues(struct zfcp_adapter *);  extern int  zfcp_qdio_determine_pci(struct zfcp_qdio_queue *,  				    struct zfcp_fsf_req *); -extern int  zfcp_qdio_reqid_check(struct zfcp_adapter *, void *);  extern volatile struct qdio_buffer_element *zfcp_qdio_sbale_req  	(struct zfcp_fsf_req *, int, int); @@ -89,8 +87,8 @@ extern int  zfcp_fsf_exchange_port_data(struct zfcp_erp_action *,  					struct fsf_qtcb_bottom_port *);  extern int  zfcp_fsf_control_file(struct zfcp_adapter *, struct zfcp_fsf_req **,  				  u32, u32, struct zfcp_sg_list *); -extern void zfcp_fsf_request_timeout_handler(unsigned long); -extern void zfcp_fsf_scsi_er_timeout_handler(unsigned long); +extern void zfcp_fsf_start_timer(struct zfcp_fsf_req *, unsigned long); +extern void zfcp_erp_start_timer(struct zfcp_fsf_req *);  extern int  zfcp_fsf_req_dismiss_all(struct zfcp_adapter *);  extern int  zfcp_fsf_status_read(struct zfcp_adapter *, int);  extern int zfcp_fsf_req_create(struct zfcp_adapter *, u32, int, mempool_t *, @@ -100,8 +98,7 @@ extern int zfcp_fsf_send_ct(struct zfcp_send_ct *, mempool_t *,  extern int zfcp_fsf_send_els(struct zfcp_send_els *);  extern int  zfcp_fsf_send_fcp_command_task(struct zfcp_adapter *,  					   struct zfcp_unit *, -					   struct scsi_cmnd *, -					   struct timer_list*, int); +					   struct scsi_cmnd *, int, int);  extern int  zfcp_fsf_req_complete(struct zfcp_fsf_req *);  extern void zfcp_fsf_incoming_els(struct zfcp_fsf_req *);  extern void zfcp_fsf_req_free(struct zfcp_fsf_req *); @@ -125,14 +122,11 @@ extern char *zfcp_get_fcp_rsp_info_ptr(struct fcp_rsp_iu *);  extern void set_host_byte(u32 *, char);  extern void set_driver_byte(u32 *, char);  extern char *zfcp_get_fcp_sns_info_ptr(struct fcp_rsp_iu *); -extern void zfcp_fsf_start_scsi_er_timer(struct zfcp_adapter *);  extern fcp_dl_t zfcp_get_fcp_dl(struct fcp_cmnd_iu *);  extern int zfcp_scsi_command_async(struct zfcp_adapter *,struct zfcp_unit *, -				   struct scsi_cmnd *, struct timer_list *); -extern int zfcp_scsi_command_sync(struct zfcp_unit *, struct scsi_cmnd *, -				  struct timer_list *); -extern struct scsi_transport_template *zfcp_transport_template; +				   struct scsi_cmnd *, int); +extern int zfcp_scsi_command_sync(struct zfcp_unit *, struct scsi_cmnd *, int);  extern struct fc_function_template zfcp_transport_functions;  /******************************** ERP ****************************************/ @@ -156,7 +150,7 @@ extern void zfcp_erp_unit_failed(struct zfcp_unit *);  extern int  zfcp_erp_thread_setup(struct zfcp_adapter *);  extern int  zfcp_erp_thread_kill(struct zfcp_adapter *);  extern int  zfcp_erp_wait(struct zfcp_adapter *); -extern int  zfcp_erp_async_handler(struct zfcp_erp_action *, unsigned long); +extern void zfcp_erp_async_handler(struct zfcp_erp_action *, unsigned long);  extern int  zfcp_test_link(struct zfcp_port *); @@ -187,8 +181,13 @@ extern void zfcp_scsi_dbf_event_result(const char *, int, struct zfcp_adapter *,  				       struct zfcp_fsf_req *);  extern void zfcp_scsi_dbf_event_abort(const char *, struct zfcp_adapter *,  				      struct scsi_cmnd *, struct zfcp_fsf_req *, -				      struct zfcp_fsf_req *); +				      unsigned long);  extern void zfcp_scsi_dbf_event_devreset(const char *, u8, struct zfcp_unit *,  					 struct scsi_cmnd *); +extern void zfcp_reqlist_add(struct zfcp_adapter *, struct zfcp_fsf_req *); +extern void zfcp_reqlist_remove(struct zfcp_adapter *, unsigned long); +extern struct zfcp_fsf_req *zfcp_reqlist_ismember(struct zfcp_adapter *, +						  unsigned long); +extern int zfcp_reqlist_isempty(struct zfcp_adapter *);  #endif	/* ZFCP_EXT_H */ diff --git a/drivers/s390/scsi/zfcp_fsf.c b/drivers/s390/scsi/zfcp_fsf.c index 6335f9229184..277826cdd0c8 100644 --- a/drivers/s390/scsi/zfcp_fsf.c +++ b/drivers/s390/scsi/zfcp_fsf.c @@ -42,14 +42,13 @@ static inline int zfcp_fsf_req_sbal_check(  static inline int zfcp_use_one_sbal(  	struct scatterlist *, int, struct scatterlist *, int);  static struct zfcp_fsf_req *zfcp_fsf_req_alloc(mempool_t *, int); -static int zfcp_fsf_req_send(struct zfcp_fsf_req *, struct timer_list *); +static int zfcp_fsf_req_send(struct zfcp_fsf_req *);  static int zfcp_fsf_protstatus_eval(struct zfcp_fsf_req *);  static int zfcp_fsf_fsfstatus_eval(struct zfcp_fsf_req *);  static int zfcp_fsf_fsfstatus_qual_eval(struct zfcp_fsf_req *);  static void zfcp_fsf_link_down_info_eval(struct zfcp_adapter *,  	struct fsf_link_down_info *);  static int zfcp_fsf_req_dispatch(struct zfcp_fsf_req *); -static void zfcp_fsf_req_dismiss(struct zfcp_fsf_req *);  /* association between FSF command and FSF QTCB type */  static u32 fsf_qtcb_type[] = { @@ -101,14 +100,19 @@ zfcp_fsf_req_alloc(mempool_t *pool, int req_flags)  	if (req_flags & ZFCP_REQ_NO_QTCB)  		size = sizeof(struct zfcp_fsf_req);  	else -		size = sizeof(struct zfcp_fsf_req_pool_element); +		size = sizeof(struct zfcp_fsf_req_qtcb); -	if (likely(pool != NULL)) +	if (likely(pool))  		ptr = mempool_alloc(pool, GFP_ATOMIC); -	else -		ptr = kmalloc(size, GFP_ATOMIC); +	else { +		if (req_flags & ZFCP_REQ_NO_QTCB) +			ptr = kmalloc(size, GFP_ATOMIC); +		else +			ptr = kmem_cache_alloc(zfcp_data.fsf_req_qtcb_cache, +					       SLAB_ATOMIC); +	} -	if (unlikely(NULL == ptr)) +	if (unlikely(!ptr))  		goto out;  	memset(ptr, 0, size); @@ -116,9 +120,8 @@ zfcp_fsf_req_alloc(mempool_t *pool, int req_flags)  	if (req_flags & ZFCP_REQ_NO_QTCB) {  		fsf_req = (struct zfcp_fsf_req *) ptr;  	} else { -		fsf_req = &((struct zfcp_fsf_req_pool_element *) ptr)->fsf_req; -		fsf_req->qtcb = -			&((struct zfcp_fsf_req_pool_element *) ptr)->qtcb; +		fsf_req = &((struct zfcp_fsf_req_qtcb *) ptr)->fsf_req; +		fsf_req->qtcb =	&((struct zfcp_fsf_req_qtcb *) ptr)->qtcb;  	}  	fsf_req->pool = pool; @@ -140,55 +143,63 @@ zfcp_fsf_req_alloc(mempool_t *pool, int req_flags)  void  zfcp_fsf_req_free(struct zfcp_fsf_req *fsf_req)  { -	if (likely(fsf_req->pool != NULL)) +	if (likely(fsf_req->pool)) {  		mempool_free(fsf_req, fsf_req->pool); -	else -		kfree(fsf_req); -} - -/* - * function:	 - * - * purpose:	 - * - * returns: - * - * note: qdio queues shall be down (no ongoing inbound processing) - */ -int -zfcp_fsf_req_dismiss_all(struct zfcp_adapter *adapter) -{ -	struct zfcp_fsf_req *fsf_req, *tmp; -	unsigned long flags; -	LIST_HEAD(remove_queue); - -	spin_lock_irqsave(&adapter->fsf_req_list_lock, flags); -	list_splice_init(&adapter->fsf_req_list_head, &remove_queue); -	atomic_set(&adapter->fsf_reqs_active, 0); -	spin_unlock_irqrestore(&adapter->fsf_req_list_lock, flags); +		return; +	} -	list_for_each_entry_safe(fsf_req, tmp, &remove_queue, list) { -		list_del(&fsf_req->list); -		zfcp_fsf_req_dismiss(fsf_req); +	if (fsf_req->qtcb) { +		kmem_cache_free(zfcp_data.fsf_req_qtcb_cache, fsf_req); +		return;  	} -	return 0; +	kfree(fsf_req);  } -/* - * function:	 - * - * purpose:	 - * - * returns: +/** + * zfcp_fsf_req_dismiss - dismiss a single fsf request   */ -static void -zfcp_fsf_req_dismiss(struct zfcp_fsf_req *fsf_req) +static void zfcp_fsf_req_dismiss(struct zfcp_adapter *adapter, +				 struct zfcp_fsf_req *fsf_req, +				 unsigned int counter)  { +	u64 dbg_tmp[2]; + +	dbg_tmp[0] = (u64) atomic_read(&adapter->reqs_active); +	dbg_tmp[1] = (u64) counter; +	debug_event(adapter->erp_dbf, 4, (void *) dbg_tmp, 16); +	list_del(&fsf_req->list);  	fsf_req->status |= ZFCP_STATUS_FSFREQ_DISMISSED;  	zfcp_fsf_req_complete(fsf_req);  } +/** + * zfcp_fsf_req_dismiss_all - dismiss all remaining fsf requests + */ +int zfcp_fsf_req_dismiss_all(struct zfcp_adapter *adapter) +{ +	struct zfcp_fsf_req *request, *tmp; +	unsigned long flags; +	unsigned int i, counter; + +	spin_lock_irqsave(&adapter->req_list_lock, flags); +	atomic_set(&adapter->reqs_active, 0); +	for (i=0; i<REQUEST_LIST_SIZE; i++) { +		if (list_empty(&adapter->req_list[i])) +			continue; + +		counter = 0; +		list_for_each_entry_safe(request, tmp, +					 &adapter->req_list[i], list) { +			zfcp_fsf_req_dismiss(adapter, request, counter); +			counter++; +		} +	} +	spin_unlock_irqrestore(&adapter->req_list_lock, flags); + +	return 0; +} +  /*   * function:    zfcp_fsf_req_complete   * @@ -214,8 +225,10 @@ zfcp_fsf_req_complete(struct zfcp_fsf_req *fsf_req)  		 */  		zfcp_fsf_status_read_handler(fsf_req);  		goto out; -	} else +	} else { +		del_timer(&fsf_req->timer);  		zfcp_fsf_protstatus_eval(fsf_req); +	}  	/*  	 * fsf_req may be deleted due to waking up functions, so  @@ -774,8 +787,7 @@ zfcp_fsf_status_read(struct zfcp_adapter *adapter, int req_flags)  	sbale->addr = (void *) status_buffer;  	sbale->length = sizeof(struct fsf_status_read_buffer); -	/* start QDIO request for this FSF request */ -	retval = zfcp_fsf_req_send(fsf_req, NULL); +	retval = zfcp_fsf_req_send(fsf_req);  	if (retval) {  		ZFCP_LOG_DEBUG("error: Could not set-up unsolicited status "  			       "environment.\n"); @@ -1101,8 +1113,8 @@ zfcp_fsf_abort_fcp_command(unsigned long old_req_id,  			   struct zfcp_unit *unit, int req_flags)  {  	volatile struct qdio_buffer_element *sbale; -	unsigned long lock_flags;  	struct zfcp_fsf_req *fsf_req = NULL; +	unsigned long lock_flags;  	int retval = 0;  	/* setup new FSF request */ @@ -1132,12 +1144,9 @@ zfcp_fsf_abort_fcp_command(unsigned long old_req_id,  	/* set handle of request which should be aborted */  	fsf_req->qtcb->bottom.support.req_handle = (u64) old_req_id; -	/* start QDIO request for this FSF request */ - -	zfcp_fsf_start_scsi_er_timer(adapter); -	retval = zfcp_fsf_req_send(fsf_req, NULL); +	zfcp_fsf_start_timer(fsf_req, ZFCP_SCSI_ER_TIMEOUT); +	retval = zfcp_fsf_req_send(fsf_req);  	if (retval) { -		del_timer(&adapter->scsi_er_timer);  		ZFCP_LOG_INFO("error: Failed to send abort command request "  			      "on adapter %s, port 0x%016Lx, unit 0x%016Lx\n",  			      zfcp_get_busid_by_adapter(adapter), @@ -1173,8 +1182,6 @@ zfcp_fsf_abort_fcp_command_handler(struct zfcp_fsf_req *new_fsf_req)  	unsigned char status_qual =  	    new_fsf_req->qtcb->header.fsf_status_qual.word[0]; -	del_timer(&new_fsf_req->adapter->scsi_er_timer); -  	if (new_fsf_req->status & ZFCP_STATUS_FSFREQ_ERROR) {  		/* do not set ZFCP_STATUS_FSFREQ_ABORTSUCCEEDED */  		goto skip_fsfstatus; @@ -1380,11 +1387,6 @@ zfcp_fsf_send_ct(struct zfcp_send_ct *ct, mempool_t *pool,  		goto failed_req;  	} -        if (erp_action != NULL) { -                erp_action->fsf_req = fsf_req; -                fsf_req->erp_action = erp_action; -        } -  	sbale = zfcp_qdio_sbale_req(fsf_req, fsf_req->sbal_curr, 0);          if (zfcp_use_one_sbal(ct->req, ct->req_count,                                ct->resp, ct->resp_count)){ @@ -1451,8 +1453,14 @@ zfcp_fsf_send_ct(struct zfcp_send_ct *ct, mempool_t *pool,  	zfcp_san_dbf_event_ct_request(fsf_req); -	/* start QDIO request for this FSF request */ -	ret = zfcp_fsf_req_send(fsf_req, ct->timer); +	if (erp_action) { +		erp_action->fsf_req = fsf_req; +		fsf_req->erp_action = erp_action; +		zfcp_erp_start_timer(fsf_req); +	} else +		zfcp_fsf_start_timer(fsf_req, ZFCP_FSF_REQUEST_TIMEOUT); + +	ret = zfcp_fsf_req_send(fsf_req);  	if (ret) {  		ZFCP_LOG_DEBUG("error: initiation of CT request failed "  			       "(adapter %s, port 0x%016Lx)\n", @@ -1749,8 +1757,8 @@ zfcp_fsf_send_els(struct zfcp_send_els *els)  	zfcp_san_dbf_event_els_request(fsf_req); -	/* start QDIO request for this FSF request */ -	ret = zfcp_fsf_req_send(fsf_req, els->timer); +	zfcp_fsf_start_timer(fsf_req, ZFCP_FSF_REQUEST_TIMEOUT); +	ret = zfcp_fsf_req_send(fsf_req);  	if (ret) {  		ZFCP_LOG_DEBUG("error: initiation of ELS request failed "  			       "(adapter %s, port d_id: 0x%08x)\n", @@ -1947,6 +1955,7 @@ int  zfcp_fsf_exchange_config_data(struct zfcp_erp_action *erp_action)  {  	volatile struct qdio_buffer_element *sbale; +	struct zfcp_fsf_req *fsf_req;  	unsigned long lock_flags;  	int retval = 0; @@ -1955,7 +1964,7 @@ zfcp_fsf_exchange_config_data(struct zfcp_erp_action *erp_action)  				     FSF_QTCB_EXCHANGE_CONFIG_DATA,  				     ZFCP_REQ_AUTO_CLEANUP,  				     erp_action->adapter->pool.fsf_req_erp, -				     &lock_flags, &(erp_action->fsf_req)); +				     &lock_flags, &fsf_req);  	if (retval < 0) {  		ZFCP_LOG_INFO("error: Could not create exchange configuration "  			      "data request for adapter %s.\n", @@ -1963,26 +1972,26 @@ zfcp_fsf_exchange_config_data(struct zfcp_erp_action *erp_action)  		goto out;  	} -	sbale = zfcp_qdio_sbale_req(erp_action->fsf_req, -                                    erp_action->fsf_req->sbal_curr, 0); +	sbale = zfcp_qdio_sbale_req(fsf_req, fsf_req->sbal_curr, 0);          sbale[0].flags |= SBAL_FLAGS0_TYPE_READ;          sbale[1].flags |= SBAL_FLAGS_LAST_ENTRY; -	erp_action->fsf_req->erp_action = erp_action; -	erp_action->fsf_req->qtcb->bottom.config.feature_selection = +	fsf_req->qtcb->bottom.config.feature_selection =  			FSF_FEATURE_CFDC |  			FSF_FEATURE_LUN_SHARING |  			FSF_FEATURE_NOTIFICATION_LOST |  			FSF_FEATURE_UPDATE_ALERT; +	fsf_req->erp_action = erp_action; +	erp_action->fsf_req = fsf_req; -	/* start QDIO request for this FSF request */ -	retval = zfcp_fsf_req_send(erp_action->fsf_req, &erp_action->timer); +	zfcp_erp_start_timer(fsf_req); +	retval = zfcp_fsf_req_send(fsf_req);  	if (retval) {  		ZFCP_LOG_INFO  		    ("error: Could not send exchange configuration data "  		     "command on the adapter %s\n",  		     zfcp_get_busid_by_adapter(erp_action->adapter)); -		zfcp_fsf_req_free(erp_action->fsf_req); +		zfcp_fsf_req_free(fsf_req);  		erp_action->fsf_req = NULL;  		goto out;  	} @@ -2212,10 +2221,9 @@ zfcp_fsf_exchange_port_data(struct zfcp_erp_action *erp_action,  			    struct fsf_qtcb_bottom_port *data)  {  	volatile struct qdio_buffer_element *sbale; -	int retval = 0; -	unsigned long lock_flags;          struct zfcp_fsf_req *fsf_req; -	struct timer_list *timer; +	unsigned long lock_flags; +	int retval = 0;  	if (!(adapter->adapter_features & FSF_FEATURE_HBAAPI_MANAGEMENT)) {  		ZFCP_LOG_INFO("error: exchange port data " @@ -2227,7 +2235,7 @@ zfcp_fsf_exchange_port_data(struct zfcp_erp_action *erp_action,  	/* setup new FSF request */  	retval = zfcp_fsf_req_create(adapter, FSF_QTCB_EXCHANGE_PORT_DATA,  				     erp_action ? ZFCP_REQ_AUTO_CLEANUP : 0, -				     0, &lock_flags, &fsf_req); +				     NULL, &lock_flags, &fsf_req);  	if (retval < 0) {  		ZFCP_LOG_INFO("error: Out of resources. Could not create an "                                "exchange port data request for" @@ -2248,22 +2256,11 @@ zfcp_fsf_exchange_port_data(struct zfcp_erp_action *erp_action,  	if (erp_action) {  		erp_action->fsf_req = fsf_req;  		fsf_req->erp_action = erp_action; -		timer = &erp_action->timer; -	} else { -		timer = kmalloc(sizeof(struct timer_list), GFP_ATOMIC); -		if (!timer) { -			write_unlock_irqrestore(&adapter->request_queue.queue_lock, -						lock_flags); -			zfcp_fsf_req_free(fsf_req); -			return -ENOMEM; -		} -		init_timer(timer); -		timer->function = zfcp_fsf_request_timeout_handler; -		timer->data = (unsigned long) adapter; -		timer->expires = ZFCP_FSF_REQUEST_TIMEOUT; -	} +		zfcp_erp_start_timer(fsf_req); +	} else +		zfcp_fsf_start_timer(fsf_req, ZFCP_FSF_REQUEST_TIMEOUT); -	retval = zfcp_fsf_req_send(fsf_req, timer); +	retval = zfcp_fsf_req_send(fsf_req);  	if (retval) {  		ZFCP_LOG_INFO("error: Could not send an exchange port data "                                "command on the adapter %s\n", @@ -2271,8 +2268,6 @@ zfcp_fsf_exchange_port_data(struct zfcp_erp_action *erp_action,  		zfcp_fsf_req_free(fsf_req);  		if (erp_action)  			erp_action->fsf_req = NULL; -		else -			kfree(timer);  		write_unlock_irqrestore(&adapter->request_queue.queue_lock,  					lock_flags);  		return retval; @@ -2283,9 +2278,7 @@ zfcp_fsf_exchange_port_data(struct zfcp_erp_action *erp_action,  	if (!erp_action) {  		wait_event(fsf_req->completion_wq,  			   fsf_req->status & ZFCP_STATUS_FSFREQ_COMPLETED); -		del_timer_sync(timer);  		zfcp_fsf_req_free(fsf_req); -		kfree(timer);  	}  	return retval;  } @@ -2367,6 +2360,7 @@ int  zfcp_fsf_open_port(struct zfcp_erp_action *erp_action)  {  	volatile struct qdio_buffer_element *sbale; +	struct zfcp_fsf_req *fsf_req;  	unsigned long lock_flags;  	int retval = 0; @@ -2375,7 +2369,7 @@ zfcp_fsf_open_port(struct zfcp_erp_action *erp_action)  				     FSF_QTCB_OPEN_PORT_WITH_DID,  				     ZFCP_WAIT_FOR_SBAL | ZFCP_REQ_AUTO_CLEANUP,  				     erp_action->adapter->pool.fsf_req_erp, -				     &lock_flags, &(erp_action->fsf_req)); +				     &lock_flags, &fsf_req);  	if (retval < 0) {  		ZFCP_LOG_INFO("error: Could not create open port request "  			      "for port 0x%016Lx on adapter %s.\n", @@ -2384,24 +2378,24 @@ zfcp_fsf_open_port(struct zfcp_erp_action *erp_action)  		goto out;  	} -	sbale = zfcp_qdio_sbale_req(erp_action->fsf_req, -                                    erp_action->fsf_req->sbal_curr, 0); +	sbale = zfcp_qdio_sbale_req(fsf_req, fsf_req->sbal_curr, 0);          sbale[0].flags |= SBAL_FLAGS0_TYPE_READ;          sbale[1].flags |= SBAL_FLAGS_LAST_ENTRY; -	erp_action->fsf_req->qtcb->bottom.support.d_id = erp_action->port->d_id; +	fsf_req->qtcb->bottom.support.d_id = erp_action->port->d_id;  	atomic_set_mask(ZFCP_STATUS_COMMON_OPENING, &erp_action->port->status); -	erp_action->fsf_req->data = (unsigned long) erp_action->port; -	erp_action->fsf_req->erp_action = erp_action; +	fsf_req->data = (unsigned long) erp_action->port; +	fsf_req->erp_action = erp_action; +	erp_action->fsf_req = fsf_req; -	/* start QDIO request for this FSF request */ -	retval = zfcp_fsf_req_send(erp_action->fsf_req, &erp_action->timer); +	zfcp_erp_start_timer(fsf_req); +	retval = zfcp_fsf_req_send(fsf_req);  	if (retval) {  		ZFCP_LOG_INFO("error: Could not send open port request for "  			      "port 0x%016Lx on adapter %s.\n",  			      erp_action->port->wwpn,  			      zfcp_get_busid_by_adapter(erp_action->adapter)); -		zfcp_fsf_req_free(erp_action->fsf_req); +		zfcp_fsf_req_free(fsf_req);  		erp_action->fsf_req = NULL;  		goto out;  	} @@ -2623,6 +2617,7 @@ int  zfcp_fsf_close_port(struct zfcp_erp_action *erp_action)  {  	volatile struct qdio_buffer_element *sbale; +	struct zfcp_fsf_req *fsf_req;  	unsigned long lock_flags;  	int retval = 0; @@ -2631,7 +2626,7 @@ zfcp_fsf_close_port(struct zfcp_erp_action *erp_action)  				     FSF_QTCB_CLOSE_PORT,  				     ZFCP_WAIT_FOR_SBAL | ZFCP_REQ_AUTO_CLEANUP,  				     erp_action->adapter->pool.fsf_req_erp, -				     &lock_flags, &(erp_action->fsf_req)); +				     &lock_flags, &fsf_req);  	if (retval < 0) {  		ZFCP_LOG_INFO("error: Could not create a close port request "  			      "for port 0x%016Lx on adapter %s.\n", @@ -2640,25 +2635,25 @@ zfcp_fsf_close_port(struct zfcp_erp_action *erp_action)  		goto out;  	} -	sbale = zfcp_qdio_sbale_req(erp_action->fsf_req, -                                    erp_action->fsf_req->sbal_curr, 0); +	sbale = zfcp_qdio_sbale_req(fsf_req, fsf_req->sbal_curr, 0);          sbale[0].flags |= SBAL_FLAGS0_TYPE_READ;          sbale[1].flags |= SBAL_FLAGS_LAST_ENTRY;  	atomic_set_mask(ZFCP_STATUS_COMMON_CLOSING, &erp_action->port->status); -	erp_action->fsf_req->data = (unsigned long) erp_action->port; -	erp_action->fsf_req->erp_action = erp_action; -	erp_action->fsf_req->qtcb->header.port_handle = -	    erp_action->port->handle; - -	/* start QDIO request for this FSF request */ -	retval = zfcp_fsf_req_send(erp_action->fsf_req, &erp_action->timer); +	fsf_req->data = (unsigned long) erp_action->port; +	fsf_req->erp_action = erp_action; +	fsf_req->qtcb->header.port_handle = erp_action->port->handle; +	fsf_req->erp_action = erp_action; +	erp_action->fsf_req = fsf_req; + +	zfcp_erp_start_timer(fsf_req); +	retval = zfcp_fsf_req_send(fsf_req);  	if (retval) {  		ZFCP_LOG_INFO("error: Could not send a close port request for "  			      "port 0x%016Lx on adapter %s.\n",  			      erp_action->port->wwpn,  			      zfcp_get_busid_by_adapter(erp_action->adapter)); -		zfcp_fsf_req_free(erp_action->fsf_req); +		zfcp_fsf_req_free(fsf_req);  		erp_action->fsf_req = NULL;  		goto out;  	} @@ -2755,16 +2750,17 @@ zfcp_fsf_close_port_handler(struct zfcp_fsf_req *fsf_req)  int  zfcp_fsf_close_physical_port(struct zfcp_erp_action *erp_action)  { -	int retval = 0; -	unsigned long lock_flags;  	volatile struct qdio_buffer_element *sbale; +	struct zfcp_fsf_req *fsf_req; +	unsigned long lock_flags; +	int retval = 0;  	/* setup new FSF request */  	retval = zfcp_fsf_req_create(erp_action->adapter,  				     FSF_QTCB_CLOSE_PHYSICAL_PORT,  				     ZFCP_WAIT_FOR_SBAL | ZFCP_REQ_AUTO_CLEANUP,  				     erp_action->adapter->pool.fsf_req_erp, -				     &lock_flags, &erp_action->fsf_req); +				     &lock_flags, &fsf_req);  	if (retval < 0) {  		ZFCP_LOG_INFO("error: Could not create close physical port "  			      "request (adapter %s, port 0x%016Lx)\n", @@ -2774,8 +2770,7 @@ zfcp_fsf_close_physical_port(struct zfcp_erp_action *erp_action)  		goto out;  	} -	sbale = zfcp_qdio_sbale_req(erp_action->fsf_req, -				    erp_action->fsf_req->sbal_curr, 0); +	sbale = zfcp_qdio_sbale_req(fsf_req, fsf_req->sbal_curr, 0);  	sbale[0].flags |= SBAL_FLAGS0_TYPE_READ;  	sbale[1].flags |= SBAL_FLAGS_LAST_ENTRY; @@ -2783,20 +2778,19 @@ zfcp_fsf_close_physical_port(struct zfcp_erp_action *erp_action)  	atomic_set_mask(ZFCP_STATUS_PORT_PHYS_CLOSING,  			&erp_action->port->status);  	/* save a pointer to this port */ -	erp_action->fsf_req->data = (unsigned long) erp_action->port; -	/* port to be closed */ -	erp_action->fsf_req->qtcb->header.port_handle = -	    erp_action->port->handle; -	erp_action->fsf_req->erp_action = erp_action; - -	/* start QDIO request for this FSF request */ -	retval = zfcp_fsf_req_send(erp_action->fsf_req, &erp_action->timer); +	fsf_req->data = (unsigned long) erp_action->port; +	fsf_req->qtcb->header.port_handle = erp_action->port->handle; +	fsf_req->erp_action = erp_action; +	erp_action->fsf_req = fsf_req; + +	zfcp_erp_start_timer(fsf_req); +	retval = zfcp_fsf_req_send(fsf_req);  	if (retval) {  		ZFCP_LOG_INFO("error: Could not send close physical port "  			      "request (adapter %s, port 0x%016Lx)\n",  			      zfcp_get_busid_by_adapter(erp_action->adapter),  			      erp_action->port->wwpn); -		zfcp_fsf_req_free(erp_action->fsf_req); +		zfcp_fsf_req_free(fsf_req);  		erp_action->fsf_req = NULL;  		goto out;  	} @@ -2961,6 +2955,7 @@ int  zfcp_fsf_open_unit(struct zfcp_erp_action *erp_action)  {  	volatile struct qdio_buffer_element *sbale; +	struct zfcp_fsf_req *fsf_req;  	unsigned long lock_flags;  	int retval = 0; @@ -2969,7 +2964,7 @@ zfcp_fsf_open_unit(struct zfcp_erp_action *erp_action)  				     FSF_QTCB_OPEN_LUN,  				     ZFCP_WAIT_FOR_SBAL | ZFCP_REQ_AUTO_CLEANUP,  				     erp_action->adapter->pool.fsf_req_erp, -				     &lock_flags, &(erp_action->fsf_req)); +				     &lock_flags, &fsf_req);  	if (retval < 0) {  		ZFCP_LOG_INFO("error: Could not create open unit request for "  			      "unit 0x%016Lx on port 0x%016Lx on adapter %s.\n", @@ -2979,24 +2974,22 @@ zfcp_fsf_open_unit(struct zfcp_erp_action *erp_action)  		goto out;  	} -	sbale = zfcp_qdio_sbale_req(erp_action->fsf_req, -                                    erp_action->fsf_req->sbal_curr, 0); +	sbale = zfcp_qdio_sbale_req(fsf_req, fsf_req->sbal_curr, 0);          sbale[0].flags |= SBAL_FLAGS0_TYPE_READ;          sbale[1].flags |= SBAL_FLAGS_LAST_ENTRY; -	erp_action->fsf_req->qtcb->header.port_handle = -		erp_action->port->handle; -	erp_action->fsf_req->qtcb->bottom.support.fcp_lun = -		erp_action->unit->fcp_lun; +	fsf_req->qtcb->header.port_handle = erp_action->port->handle; +	fsf_req->qtcb->bottom.support.fcp_lun =	erp_action->unit->fcp_lun;  	if (!(erp_action->adapter->connection_features & FSF_FEATURE_NPIV_MODE)) -		erp_action->fsf_req->qtcb->bottom.support.option = +		fsf_req->qtcb->bottom.support.option =  			FSF_OPEN_LUN_SUPPRESS_BOXING;  	atomic_set_mask(ZFCP_STATUS_COMMON_OPENING, &erp_action->unit->status); -	erp_action->fsf_req->data = (unsigned long) erp_action->unit; -	erp_action->fsf_req->erp_action = erp_action; +	fsf_req->data = (unsigned long) erp_action->unit; +	fsf_req->erp_action = erp_action; +	erp_action->fsf_req = fsf_req; -	/* start QDIO request for this FSF request */ -	retval = zfcp_fsf_req_send(erp_action->fsf_req, &erp_action->timer); +	zfcp_erp_start_timer(fsf_req); +	retval = zfcp_fsf_req_send(erp_action->fsf_req);  	if (retval) {  		ZFCP_LOG_INFO("error: Could not send an open unit request "  			      "on the adapter %s, port 0x%016Lx for " @@ -3004,7 +2997,7 @@ zfcp_fsf_open_unit(struct zfcp_erp_action *erp_action)  			      zfcp_get_busid_by_adapter(erp_action->adapter),  			      erp_action->port->wwpn,  			      erp_action->unit->fcp_lun); -		zfcp_fsf_req_free(erp_action->fsf_req); +		zfcp_fsf_req_free(fsf_req);  		erp_action->fsf_req = NULL;  		goto out;  	} @@ -3297,6 +3290,7 @@ int  zfcp_fsf_close_unit(struct zfcp_erp_action *erp_action)  {  	volatile struct qdio_buffer_element *sbale; +	struct zfcp_fsf_req *fsf_req;  	unsigned long lock_flags;  	int retval = 0; @@ -3305,7 +3299,7 @@ zfcp_fsf_close_unit(struct zfcp_erp_action *erp_action)  				     FSF_QTCB_CLOSE_LUN,  				     ZFCP_WAIT_FOR_SBAL | ZFCP_REQ_AUTO_CLEANUP,  				     erp_action->adapter->pool.fsf_req_erp, -				     &lock_flags, &(erp_action->fsf_req)); +				     &lock_flags, &fsf_req);  	if (retval < 0) {  		ZFCP_LOG_INFO("error: Could not create close unit request for "  			      "unit 0x%016Lx on port 0x%016Lx on adapter %s.\n", @@ -3315,27 +3309,26 @@ zfcp_fsf_close_unit(struct zfcp_erp_action *erp_action)  		goto out;  	} -	sbale = zfcp_qdio_sbale_req(erp_action->fsf_req, -                                    erp_action->fsf_req->sbal_curr, 0); +	sbale = zfcp_qdio_sbale_req(fsf_req, fsf_req->sbal_curr, 0);          sbale[0].flags |= SBAL_FLAGS0_TYPE_READ;          sbale[1].flags |= SBAL_FLAGS_LAST_ENTRY; -	erp_action->fsf_req->qtcb->header.port_handle = -	    erp_action->port->handle; -	erp_action->fsf_req->qtcb->header.lun_handle = erp_action->unit->handle; +	fsf_req->qtcb->header.port_handle = erp_action->port->handle; +	fsf_req->qtcb->header.lun_handle = erp_action->unit->handle;  	atomic_set_mask(ZFCP_STATUS_COMMON_CLOSING, &erp_action->unit->status); -	erp_action->fsf_req->data = (unsigned long) erp_action->unit; -	erp_action->fsf_req->erp_action = erp_action; +	fsf_req->data = (unsigned long) erp_action->unit; +	fsf_req->erp_action = erp_action; +	erp_action->fsf_req = fsf_req; -	/* start QDIO request for this FSF request */ -	retval = zfcp_fsf_req_send(erp_action->fsf_req, &erp_action->timer); +	zfcp_erp_start_timer(fsf_req); +	retval = zfcp_fsf_req_send(erp_action->fsf_req);  	if (retval) {  		ZFCP_LOG_INFO("error: Could not send a close unit request for "  			      "unit 0x%016Lx on port 0x%016Lx onadapter %s.\n",  			      erp_action->unit->fcp_lun,  			      erp_action->port->wwpn,  			      zfcp_get_busid_by_adapter(erp_action->adapter)); -		zfcp_fsf_req_free(erp_action->fsf_req); +		zfcp_fsf_req_free(fsf_req);  		erp_action->fsf_req = NULL;  		goto out;  	} @@ -3488,7 +3481,7 @@ int  zfcp_fsf_send_fcp_command_task(struct zfcp_adapter *adapter,  			       struct zfcp_unit *unit,  			       struct scsi_cmnd * scsi_cmnd, -			       struct timer_list *timer, int req_flags) +			       int use_timer, int req_flags)  {  	struct zfcp_fsf_req *fsf_req = NULL;  	struct fcp_cmnd_iu *fcp_cmnd_iu; @@ -3516,7 +3509,7 @@ zfcp_fsf_send_fcp_command_task(struct zfcp_adapter *adapter,  	fsf_req->unit = unit;  	/* associate FSF request with SCSI request (for look up on abort) */ -	scsi_cmnd->host_scribble = (char *) fsf_req; +	scsi_cmnd->host_scribble = (unsigned char *) fsf_req->req_id;  	/* associate SCSI command with FSF request */  	fsf_req->data = (unsigned long) scsi_cmnd; @@ -3629,11 +3622,10 @@ zfcp_fsf_send_fcp_command_task(struct zfcp_adapter *adapter,  	ZFCP_HEX_DUMP(ZFCP_LOG_LEVEL_DEBUG,  		      (char *) scsi_cmnd->cmnd, scsi_cmnd->cmd_len); -	/* -	 * start QDIO request for this FSF request -	 *  covered by an SBALE) -	 */ -	retval = zfcp_fsf_req_send(fsf_req, timer); +	if (use_timer) +		zfcp_fsf_start_timer(fsf_req, ZFCP_FSF_REQUEST_TIMEOUT); + +	retval = zfcp_fsf_req_send(fsf_req);  	if (unlikely(retval < 0)) {  		ZFCP_LOG_INFO("error: Could not send FCP command request "  			      "on adapter %s, port 0x%016Lx, unit 0x%016Lx\n", @@ -3718,11 +3710,9 @@ zfcp_fsf_send_fcp_command_task_management(struct zfcp_adapter *adapter,  	fcp_cmnd_iu->fcp_lun = unit->fcp_lun;  	fcp_cmnd_iu->task_management_flags = tm_flags; -	/* start QDIO request for this FSF request */ -	zfcp_fsf_start_scsi_er_timer(adapter); -	retval = zfcp_fsf_req_send(fsf_req, NULL); +	zfcp_fsf_start_timer(fsf_req, ZFCP_SCSI_ER_TIMEOUT); +	retval = zfcp_fsf_req_send(fsf_req);  	if (retval) { -		del_timer(&adapter->scsi_er_timer);  		ZFCP_LOG_INFO("error: Could not send an FCP-command (task "  			      "management) on adapter %s, port 0x%016Lx for "  			      "unit LUN 0x%016Lx\n", @@ -4226,7 +4216,6 @@ zfcp_fsf_send_fcp_command_task_management_handler(struct zfcp_fsf_req *fsf_req)  	char *fcp_rsp_info = zfcp_get_fcp_rsp_info_ptr(fcp_rsp_iu);  	struct zfcp_unit *unit = (struct zfcp_unit *) fsf_req->data; -	del_timer(&fsf_req->adapter->scsi_er_timer);  	if (fsf_req->status & ZFCP_STATUS_FSFREQ_ERROR) {  		fsf_req->status |= ZFCP_STATUS_FSFREQ_TMFUNCFAILED;  		goto skip_fsfstatus; @@ -4295,7 +4284,6 @@ zfcp_fsf_control_file(struct zfcp_adapter *adapter,  	struct zfcp_fsf_req *fsf_req;  	struct fsf_qtcb_bottom_support *bottom;  	volatile struct qdio_buffer_element *sbale; -	struct timer_list *timer;  	unsigned long lock_flags;  	int req_flags = 0;  	int direction; @@ -4327,12 +4315,6 @@ zfcp_fsf_control_file(struct zfcp_adapter *adapter,  		goto out;  	} -	timer = kmalloc(sizeof(struct timer_list), GFP_KERNEL); -	if (!timer) { -		retval = -ENOMEM; -		goto out; - 	} -  	retval = zfcp_fsf_req_create(adapter, fsf_command, req_flags,  				     NULL, &lock_flags, &fsf_req);  	if (retval < 0) { @@ -4367,12 +4349,8 @@ zfcp_fsf_control_file(struct zfcp_adapter *adapter,  	} else  		sbale[1].flags |= SBAL_FLAGS_LAST_ENTRY; -	init_timer(timer); -	timer->function = zfcp_fsf_request_timeout_handler; -	timer->data = (unsigned long) adapter; -	timer->expires = ZFCP_FSF_REQUEST_TIMEOUT; - -	retval = zfcp_fsf_req_send(fsf_req, timer); +	zfcp_fsf_start_timer(fsf_req, ZFCP_FSF_REQUEST_TIMEOUT); +	retval = zfcp_fsf_req_send(fsf_req);  	if (retval < 0) {  		ZFCP_LOG_INFO("initiation of cfdc up/download failed"  			      "(adapter %s)\n", @@ -4392,15 +4370,12 @@ zfcp_fsf_control_file(struct zfcp_adapter *adapter,  	           fsf_req->status & ZFCP_STATUS_FSFREQ_COMPLETED);  	*fsf_req_ptr = fsf_req; -	del_timer_sync(timer); -	goto free_timer; +	goto out;   free_fsf_req:  	zfcp_fsf_req_free(fsf_req);   unlock_queue_lock:  	write_unlock_irqrestore(&adapter->request_queue.queue_lock, lock_flags); - free_timer: -	kfree(timer);   out:  	return retval;  } @@ -4592,12 +4567,14 @@ static inline void  zfcp_fsf_req_qtcb_init(struct zfcp_fsf_req *fsf_req)  {  	if (likely(fsf_req->qtcb != NULL)) { -		fsf_req->qtcb->prefix.req_seq_no = fsf_req->adapter->fsf_req_seq_no; -		fsf_req->qtcb->prefix.req_id = (unsigned long)fsf_req; +		fsf_req->qtcb->prefix.req_seq_no = +			fsf_req->adapter->fsf_req_seq_no; +		fsf_req->qtcb->prefix.req_id = fsf_req->req_id;  		fsf_req->qtcb->prefix.ulp_info = ZFCP_ULP_INFO_VERSION; -		fsf_req->qtcb->prefix.qtcb_type = fsf_qtcb_type[fsf_req->fsf_command]; +		fsf_req->qtcb->prefix.qtcb_type = +			fsf_qtcb_type[fsf_req->fsf_command];  		fsf_req->qtcb->prefix.qtcb_version = ZFCP_QTCB_VERSION; -		fsf_req->qtcb->header.req_handle = (unsigned long)fsf_req; +		fsf_req->qtcb->header.req_handle = fsf_req->req_id;  		fsf_req->qtcb->header.fsf_command = fsf_req->fsf_command;  	}  } @@ -4668,8 +4645,15 @@ zfcp_fsf_req_create(struct zfcp_adapter *adapter, u32 fsf_cmd, int req_flags,  	fsf_req->adapter = adapter;  	fsf_req->fsf_command = fsf_cmd; +	INIT_LIST_HEAD(&fsf_req->list); +	 +	/* this is serialized (we are holding req_queue-lock of adapter */ +	if (adapter->req_no == 0) +		adapter->req_no++; +	fsf_req->req_id = adapter->req_no++; -        zfcp_fsf_req_qtcb_init(fsf_req); +	init_timer(&fsf_req->timer); +	zfcp_fsf_req_qtcb_init(fsf_req);  	/* initialize waitqueue which may be used to wait on   	   this request completion */ @@ -4707,7 +4691,7 @@ zfcp_fsf_req_create(struct zfcp_adapter *adapter, u32 fsf_cmd, int req_flags,  	sbale = zfcp_qdio_sbale_req(fsf_req, fsf_req->sbal_curr, 0);  	/* setup common SBALE fields */ -	sbale[0].addr = fsf_req; +	sbale[0].addr = (void *) fsf_req->req_id;  	sbale[0].flags |= SBAL_FLAGS0_COMMAND;  	if (likely(fsf_req->qtcb != NULL)) {  		sbale[1].addr = (void *) fsf_req->qtcb; @@ -4739,15 +4723,14 @@ zfcp_fsf_req_create(struct zfcp_adapter *adapter, u32 fsf_cmd, int req_flags,   * returns:	0 - request transfer succesfully started   *		!0 - start of request transfer failed   */ -static int -zfcp_fsf_req_send(struct zfcp_fsf_req *fsf_req, struct timer_list *timer) +static int zfcp_fsf_req_send(struct zfcp_fsf_req *fsf_req)  {  	struct zfcp_adapter *adapter;  	struct zfcp_qdio_queue *req_queue;  	volatile struct qdio_buffer_element *sbale;  	int inc_seq_no;  	int new_distance_from_int; -	unsigned long flags; +	u64 dbg_tmp[2];  	int retval = 0;  	adapter = fsf_req->adapter; @@ -4761,19 +4744,13 @@ zfcp_fsf_req_send(struct zfcp_fsf_req *fsf_req, struct timer_list *timer)  	ZFCP_HEX_DUMP(ZFCP_LOG_LEVEL_TRACE, (char *) sbale[1].addr,  		      sbale[1].length); -	/* put allocated FSF request at list tail */ -	spin_lock_irqsave(&adapter->fsf_req_list_lock, flags); -	list_add_tail(&fsf_req->list, &adapter->fsf_req_list_head); -	spin_unlock_irqrestore(&adapter->fsf_req_list_lock, flags); +	/* put allocated FSF request into hash table */ +	spin_lock(&adapter->req_list_lock); +	zfcp_reqlist_add(adapter, fsf_req); +	spin_unlock(&adapter->req_list_lock);  	inc_seq_no = (fsf_req->qtcb != NULL); -	/* figure out expiration time of timeout and start timeout */ -	if (unlikely(timer)) { -		timer->expires += jiffies; -		add_timer(timer); -	} -  	ZFCP_LOG_TRACE("request queue of adapter %s: "  		       "next free SBAL is %i, %i free SBALs\n",  		       zfcp_get_busid_by_adapter(adapter), @@ -4803,31 +4780,25 @@ zfcp_fsf_req_send(struct zfcp_fsf_req *fsf_req, struct timer_list *timer)  			 QDIO_FLAG_SYNC_OUTPUT,  			 0, fsf_req->sbal_first, fsf_req->sbal_number, NULL); +	dbg_tmp[0] = (unsigned long) sbale[0].addr; +	dbg_tmp[1] = (u64) retval; +	debug_event(adapter->erp_dbf, 4, (void *) dbg_tmp, 16); +  	if (unlikely(retval)) {  		/* Queues are down..... */  		retval = -EIO; -		/* -		 * FIXME(potential race): -		 * timer might be expired (absolutely unlikely) -		 */ -		if (timer) -			del_timer(timer); -		spin_lock_irqsave(&adapter->fsf_req_list_lock, flags); -		list_del(&fsf_req->list); -		spin_unlock_irqrestore(&adapter->fsf_req_list_lock, flags); -		/* -		 * adjust the number of free SBALs in request queue as well as -		 * position of first one -		 */ +		del_timer(&fsf_req->timer); +		spin_lock(&adapter->req_list_lock); +		zfcp_reqlist_remove(adapter, fsf_req->req_id); +		spin_unlock(&adapter->req_list_lock); +		/* undo changes in request queue made for this request */  		zfcp_qdio_zero_sbals(req_queue->buffer,  				     fsf_req->sbal_first, fsf_req->sbal_number);  		atomic_add(fsf_req->sbal_number, &req_queue->free_count); -		req_queue->free_index -= fsf_req->sbal_number;	 /* increase */ +		req_queue->free_index -= fsf_req->sbal_number;  		req_queue->free_index += QDIO_MAX_BUFFERS_PER_Q;  		req_queue->free_index %= QDIO_MAX_BUFFERS_PER_Q; /* wrap */ -		ZFCP_LOG_DEBUG -			("error: do_QDIO failed. Buffers could not be enqueued " -			 "to request queue.\n"); +		zfcp_erp_adapter_reopen(adapter, 0);  	} else {  		req_queue->distance_from_int = new_distance_from_int;  		/* @@ -4843,7 +4814,7 @@ zfcp_fsf_req_send(struct zfcp_fsf_req *fsf_req, struct timer_list *timer)  			adapter->fsf_req_seq_no++;  		/* count FSF requests pending */ -		atomic_inc(&adapter->fsf_reqs_active); +		atomic_inc(&adapter->reqs_active);  	}  	return retval;  } diff --git a/drivers/s390/scsi/zfcp_qdio.c b/drivers/s390/scsi/zfcp_qdio.c index 345a191926a4..dbd9f48e863e 100644 --- a/drivers/s390/scsi/zfcp_qdio.c +++ b/drivers/s390/scsi/zfcp_qdio.c @@ -282,6 +282,37 @@ zfcp_qdio_request_handler(struct ccw_device *ccw_device,  	return;  } +/** + * zfcp_qdio_reqid_check - checks for valid reqids or unsolicited status + */ +static int zfcp_qdio_reqid_check(struct zfcp_adapter *adapter,  +				 unsigned long req_id) +{ +	struct zfcp_fsf_req *fsf_req; +	unsigned long flags; + +	debug_long_event(adapter->erp_dbf, 4, req_id); + +	spin_lock_irqsave(&adapter->req_list_lock, flags); +	fsf_req = zfcp_reqlist_ismember(adapter, req_id); + +	if (!fsf_req) { +		spin_unlock_irqrestore(&adapter->req_list_lock, flags); +		ZFCP_LOG_NORMAL("error: unknown request id (%ld).\n", req_id); +		zfcp_erp_adapter_reopen(adapter, 0); +		return -EINVAL; +	} + +	zfcp_reqlist_remove(adapter, req_id); +	atomic_dec(&adapter->reqs_active); +	spin_unlock_irqrestore(&adapter->req_list_lock, flags); + +	/* finish the FSF request */ +	zfcp_fsf_req_complete(fsf_req); + +	return 0; +} +  /*   * function:   	zfcp_qdio_response_handler   * @@ -344,7 +375,7 @@ zfcp_qdio_response_handler(struct ccw_device *ccw_device,  			/* look for QDIO request identifiers in SB */  			buffere = &buffer->element[buffere_index];  			retval = zfcp_qdio_reqid_check(adapter, -						       (void *) buffere->addr); +					(unsigned long) buffere->addr);  			if (retval) {  				ZFCP_LOG_NORMAL("bug: unexpected inbound " @@ -415,51 +446,6 @@ zfcp_qdio_response_handler(struct ccw_device *ccw_device,  	return;  } -/* - * function:	zfcp_qdio_reqid_check - * - * purpose:	checks for valid reqids or unsolicited status - * - * returns:	0 - valid request id or unsolicited status - *		!0 - otherwise - */ -int -zfcp_qdio_reqid_check(struct zfcp_adapter *adapter, void *sbale_addr) -{ -	struct zfcp_fsf_req *fsf_req; - -	/* invalid (per convention used in this driver) */ -	if (unlikely(!sbale_addr)) { -		ZFCP_LOG_NORMAL("bug: invalid reqid\n"); -		return -EINVAL; -	} - -	/* valid request id and thus (hopefully :) valid fsf_req address */ -	fsf_req = (struct zfcp_fsf_req *) sbale_addr; - -	/* serialize with zfcp_fsf_req_dismiss_all */ -	spin_lock(&adapter->fsf_req_list_lock); -	if (list_empty(&adapter->fsf_req_list_head)) { -		spin_unlock(&adapter->fsf_req_list_lock); -		return 0; -	} -	list_del(&fsf_req->list); -	atomic_dec(&adapter->fsf_reqs_active); -	spin_unlock(&adapter->fsf_req_list_lock); -	 -	if (unlikely(adapter != fsf_req->adapter)) { -		ZFCP_LOG_NORMAL("bug: invalid reqid (fsf_req=%p, " -				"fsf_req->adapter=%p, adapter=%p)\n", -				fsf_req, fsf_req->adapter, adapter); -		return -EINVAL; -	} - -	/* finish the FSF request */ -	zfcp_fsf_req_complete(fsf_req); - -	return 0; -} -  /**   * zfcp_qdio_sbale_get - return pointer to SBALE of qdio_queue   * @queue: queue from which SBALE should be returned diff --git a/drivers/s390/scsi/zfcp_scsi.c b/drivers/s390/scsi/zfcp_scsi.c index 46e14f22ec18..7cafa34e4c7f 100644 --- a/drivers/s390/scsi/zfcp_scsi.c +++ b/drivers/s390/scsi/zfcp_scsi.c @@ -30,7 +30,6 @@ static int zfcp_scsi_queuecommand(struct scsi_cmnd *,  				  void (*done) (struct scsi_cmnd *));  static int zfcp_scsi_eh_abort_handler(struct scsi_cmnd *);  static int zfcp_scsi_eh_device_reset_handler(struct scsi_cmnd *); -static int zfcp_scsi_eh_bus_reset_handler(struct scsi_cmnd *);  static int zfcp_scsi_eh_host_reset_handler(struct scsi_cmnd *);  static int zfcp_task_management_function(struct zfcp_unit *, u8,  					 struct scsi_cmnd *); @@ -40,37 +39,27 @@ static struct zfcp_unit *zfcp_unit_lookup(struct zfcp_adapter *, int,  static struct device_attribute *zfcp_sysfs_sdev_attrs[]; -struct scsi_transport_template *zfcp_transport_template; -  struct zfcp_data zfcp_data = {  	.scsi_host_template = { -	      name:	               ZFCP_NAME, -	      proc_name:               "zfcp", -	      proc_info:               NULL, -	      detect:	               NULL, -	      slave_alloc:             zfcp_scsi_slave_alloc, -	      slave_configure:         zfcp_scsi_slave_configure, -	      slave_destroy:           zfcp_scsi_slave_destroy, -	      queuecommand:            zfcp_scsi_queuecommand, -	      eh_abort_handler:        zfcp_scsi_eh_abort_handler, -	      eh_device_reset_handler: zfcp_scsi_eh_device_reset_handler, -	      eh_bus_reset_handler:    zfcp_scsi_eh_bus_reset_handler, -	      eh_host_reset_handler:   zfcp_scsi_eh_host_reset_handler, -			               /* FIXME(openfcp): Tune */ -	      can_queue:               4096, -	      this_id:	               -1, -	      /* -	       * FIXME: -	       * one less? can zfcp_create_sbale cope with it? -	       */ -	      sg_tablesize:            ZFCP_MAX_SBALES_PER_REQ, -	      cmd_per_lun:             1, -	      unchecked_isa_dma:       0, -	      use_clustering:          1, -	      sdev_attrs:              zfcp_sysfs_sdev_attrs, +		.name			= ZFCP_NAME, +		.module			= THIS_MODULE, +		.proc_name		= "zfcp", +		.slave_alloc		= zfcp_scsi_slave_alloc, +		.slave_configure	= zfcp_scsi_slave_configure, +		.slave_destroy		= zfcp_scsi_slave_destroy, +		.queuecommand		= zfcp_scsi_queuecommand, +		.eh_abort_handler	= zfcp_scsi_eh_abort_handler, +		.eh_device_reset_handler = zfcp_scsi_eh_device_reset_handler, +		.eh_bus_reset_handler	= zfcp_scsi_eh_host_reset_handler, +		.eh_host_reset_handler	= zfcp_scsi_eh_host_reset_handler, +		.can_queue		= 4096, +		.this_id		= -1, +		.sg_tablesize		= ZFCP_MAX_SBALES_PER_REQ, +		.cmd_per_lun		= 1, +		.use_clustering		= 1, +		.sdev_attrs		= zfcp_sysfs_sdev_attrs,  	},  	.driver_version = ZFCP_VERSION, -	/* rest initialised with zeros */  };  /* Find start of Response Information in FCP response unit*/ @@ -177,8 +166,14 @@ zfcp_scsi_slave_alloc(struct scsi_device *sdp)  	return retval;  } -static void -zfcp_scsi_slave_destroy(struct scsi_device *sdpnt) +/** + * zfcp_scsi_slave_destroy - called when scsi device is removed + * + * Remove reference to associated scsi device for an zfcp_unit. + * Mark zfcp_unit as failed. The scsi device might be deleted via sysfs + * or a scan for this device might have failed. + */ +static void zfcp_scsi_slave_destroy(struct scsi_device *sdpnt)  {  	struct zfcp_unit *unit = (struct zfcp_unit *) sdpnt->hostdata; @@ -186,6 +181,7 @@ zfcp_scsi_slave_destroy(struct scsi_device *sdpnt)  		atomic_clear_mask(ZFCP_STATUS_UNIT_REGISTERED, &unit->status);  		sdpnt->hostdata = NULL;  		unit->device = NULL; +		zfcp_erp_unit_failed(unit);  		zfcp_unit_put(unit);  	} else {  		ZFCP_LOG_NORMAL("bug: no unit associated with SCSI device at " @@ -235,7 +231,7 @@ zfcp_scsi_command_fail(struct scsi_cmnd *scpnt, int result)   */  int  zfcp_scsi_command_async(struct zfcp_adapter *adapter, struct zfcp_unit *unit, -			struct scsi_cmnd *scpnt, struct timer_list *timer) +			struct scsi_cmnd *scpnt, int use_timer)  {  	int tmp;  	int retval; @@ -271,7 +267,7 @@ zfcp_scsi_command_async(struct zfcp_adapter *adapter, struct zfcp_unit *unit,  		goto out;  	} -	tmp = zfcp_fsf_send_fcp_command_task(adapter, unit, scpnt, timer, +	tmp = zfcp_fsf_send_fcp_command_task(adapter, unit, scpnt, use_timer,  					     ZFCP_REQ_AUTO_CLEANUP);  	if (unlikely(tmp < 0)) { @@ -295,21 +291,22 @@ zfcp_scsi_command_sync_handler(struct scsi_cmnd *scpnt)   * zfcp_scsi_command_sync - send a SCSI command and wait for completion   * @unit: unit where command is sent to   * @scpnt: scsi command to be sent - * @timer: timer to be started if request is successfully initiated + * @use_timer: indicates whether timer should be setup or not   * Return: 0   *   * Errors are indicated in scpnt->result   */  int  zfcp_scsi_command_sync(struct zfcp_unit *unit, struct scsi_cmnd *scpnt, -		       struct timer_list *timer) +		       int use_timer)  {  	int ret;  	DECLARE_COMPLETION(wait);  	scpnt->SCp.ptr = (void *) &wait;  /* silent re-use */  	scpnt->scsi_done = zfcp_scsi_command_sync_handler; -	ret = zfcp_scsi_command_async(unit->port->adapter, unit, scpnt, timer); +	ret = zfcp_scsi_command_async(unit->port->adapter, unit, scpnt, +				      use_timer);  	if (ret == 0)  		wait_for_completion(&wait); @@ -345,7 +342,7 @@ zfcp_scsi_queuecommand(struct scsi_cmnd *scpnt,  	adapter = (struct zfcp_adapter *) scpnt->device->host->hostdata[0];  	unit = (struct zfcp_unit *) scpnt->device->hostdata; -	return zfcp_scsi_command_async(adapter, unit, scpnt, NULL); +	return zfcp_scsi_command_async(adapter, unit, scpnt, 0);  }  static struct zfcp_unit * @@ -382,16 +379,15 @@ zfcp_unit_lookup(struct zfcp_adapter *adapter, int channel, unsigned int id,   * will handle late commands.  (Usually, the normal completion of late   * commands is ignored with respect to the running abort operation.)   */ -int -zfcp_scsi_eh_abort_handler(struct scsi_cmnd *scpnt) +int zfcp_scsi_eh_abort_handler(struct scsi_cmnd *scpnt)  {   	struct Scsi_Host *scsi_host;   	struct zfcp_adapter *adapter;  	struct zfcp_unit *unit; -	int retval = SUCCESS; -	struct zfcp_fsf_req *new_fsf_req = NULL; -	struct zfcp_fsf_req *old_fsf_req; +	struct zfcp_fsf_req *fsf_req;  	unsigned long flags; +	unsigned long old_req_id; +	int retval = SUCCESS;  	scsi_host = scpnt->device->host;  	adapter = (struct zfcp_adapter *) scsi_host->hostdata[0]; @@ -403,55 +399,47 @@ zfcp_scsi_eh_abort_handler(struct scsi_cmnd *scpnt)  	/* avoid race condition between late normal completion and abort */  	write_lock_irqsave(&adapter->abort_lock, flags); -	/* -	 * Check whether command has just completed and can not be aborted. -	 * Even if the command has just been completed late, we can access -	 * scpnt since the SCSI stack does not release it at least until -	 * this routine returns. (scpnt is parameter passed to this routine -	 * and must not disappear during abort even on late completion.) -	 */ -	old_fsf_req = (struct zfcp_fsf_req *) scpnt->host_scribble; -	if (!old_fsf_req) { +	/* Check whether corresponding fsf_req is still pending */ +	spin_lock(&adapter->req_list_lock); +	fsf_req = zfcp_reqlist_ismember(adapter, (unsigned long) +					scpnt->host_scribble); +	spin_unlock(&adapter->req_list_lock); +	if (!fsf_req) {  		write_unlock_irqrestore(&adapter->abort_lock, flags); -		zfcp_scsi_dbf_event_abort("lte1", adapter, scpnt, NULL, NULL); +		zfcp_scsi_dbf_event_abort("lte1", adapter, scpnt, NULL, 0);  		retval = SUCCESS;  		goto out;  	} -	old_fsf_req->data = 0; -	old_fsf_req->status |= ZFCP_STATUS_FSFREQ_ABORTING; +	fsf_req->data = 0; +	fsf_req->status |= ZFCP_STATUS_FSFREQ_ABORTING; +	old_req_id = fsf_req->req_id; -	/* don't access old_fsf_req after releasing the abort_lock */ +	/* don't access old fsf_req after releasing the abort_lock */  	write_unlock_irqrestore(&adapter->abort_lock, flags); -	/* call FSF routine which does the abort */ -	new_fsf_req = zfcp_fsf_abort_fcp_command((unsigned long) old_fsf_req, -						 adapter, unit, 0); -	if (!new_fsf_req) { + +	fsf_req = zfcp_fsf_abort_fcp_command(old_req_id, adapter, unit, 0); +	if (!fsf_req) {  		ZFCP_LOG_INFO("error: initiation of Abort FCP Cmnd failed\n");  		zfcp_scsi_dbf_event_abort("nres", adapter, scpnt, NULL, -					  old_fsf_req); +					  old_req_id);  		retval = FAILED;  		goto out;  	} -	/* wait for completion of abort */ -	__wait_event(new_fsf_req->completion_wq, -		     new_fsf_req->status & ZFCP_STATUS_FSFREQ_COMPLETED); +	__wait_event(fsf_req->completion_wq, +		     fsf_req->status & ZFCP_STATUS_FSFREQ_COMPLETED); -	/* status should be valid since signals were not permitted */ -	if (new_fsf_req->status & ZFCP_STATUS_FSFREQ_ABORTSUCCEEDED) { -		zfcp_scsi_dbf_event_abort("okay", adapter, scpnt, new_fsf_req, -					  NULL); +	if (fsf_req->status & ZFCP_STATUS_FSFREQ_ABORTSUCCEEDED) { +		zfcp_scsi_dbf_event_abort("okay", adapter, scpnt, fsf_req, 0);  		retval = SUCCESS; -	} else if (new_fsf_req->status & ZFCP_STATUS_FSFREQ_ABORTNOTNEEDED) { -		zfcp_scsi_dbf_event_abort("lte2", adapter, scpnt, new_fsf_req, -					  NULL); +	} else if (fsf_req->status & ZFCP_STATUS_FSFREQ_ABORTNOTNEEDED) { +		zfcp_scsi_dbf_event_abort("lte2", adapter, scpnt, fsf_req, 0);  		retval = SUCCESS;  	} else { -		zfcp_scsi_dbf_event_abort("fail", adapter, scpnt, new_fsf_req, -					  NULL); +		zfcp_scsi_dbf_event_abort("fail", adapter, scpnt, fsf_req, 0);  		retval = FAILED;  	} -	zfcp_fsf_req_free(new_fsf_req); +	zfcp_fsf_req_free(fsf_req);   out:  	return retval;  } @@ -550,33 +538,19 @@ zfcp_task_management_function(struct zfcp_unit *unit, u8 tm_flags,  }  /** - * zfcp_scsi_eh_bus_reset_handler - reset bus (reopen adapter) + * zfcp_scsi_eh_host_reset_handler - handler for host and bus reset   */ -int -zfcp_scsi_eh_bus_reset_handler(struct scsi_cmnd *scpnt) +int zfcp_scsi_eh_host_reset_handler(struct scsi_cmnd *scpnt)  { -	struct zfcp_unit *unit = (struct zfcp_unit*) scpnt->device->hostdata; -	struct zfcp_adapter *adapter = unit->port->adapter; - -	ZFCP_LOG_NORMAL("bus reset because of problems with " -			"unit 0x%016Lx\n", unit->fcp_lun); -	zfcp_erp_adapter_reopen(adapter, 0); -	zfcp_erp_wait(adapter); - -	return SUCCESS; -} +	struct zfcp_unit *unit; +	struct zfcp_adapter *adapter; -/** - * zfcp_scsi_eh_host_reset_handler - reset host (reopen adapter) - */ -int -zfcp_scsi_eh_host_reset_handler(struct scsi_cmnd *scpnt) -{ -	struct zfcp_unit *unit = (struct zfcp_unit*) scpnt->device->hostdata; -	struct zfcp_adapter *adapter = unit->port->adapter; +	unit = (struct zfcp_unit*) scpnt->device->hostdata; +	adapter = unit->port->adapter; -	ZFCP_LOG_NORMAL("host reset because of problems with " +	ZFCP_LOG_NORMAL("host/bus reset because of problems with "  			"unit 0x%016Lx\n", unit->fcp_lun); +  	zfcp_erp_adapter_reopen(adapter, 0);  	zfcp_erp_wait(adapter); @@ -607,7 +581,7 @@ zfcp_adapter_scsi_register(struct zfcp_adapter *adapter)  	adapter->scsi_host->max_channel = 0;  	adapter->scsi_host->unique_id = unique_id++;	/* FIXME */  	adapter->scsi_host->max_cmd_len = ZFCP_MAX_SCSI_CMND_LENGTH; -	adapter->scsi_host->transportt = zfcp_transport_template; +	adapter->scsi_host->transportt = zfcp_data.scsi_transport_template;  	/*  	 * save a pointer to our own adapter data structure within @@ -648,16 +622,6 @@ zfcp_adapter_scsi_unregister(struct zfcp_adapter *adapter)  	return;  } - -void -zfcp_fsf_start_scsi_er_timer(struct zfcp_adapter *adapter) -{ -	adapter->scsi_er_timer.function = zfcp_fsf_scsi_er_timeout_handler; -	adapter->scsi_er_timer.data = (unsigned long) adapter; -	adapter->scsi_er_timer.expires = jiffies + ZFCP_SCSI_ER_TIMEOUT; -	add_timer(&adapter->scsi_er_timer); -} -  /*   * Support functions for FC transport class   */ | 

