Re: [PATCH] [Target_Core_Mod/pSCSI]: Add optional legacy scsi_execute_async() usage for Linux/SCSI passthrough

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On 04/17/2009 04:09 AM, Nicholas A. Bellinger wrote:
> Greetings all,
> 
> This patch modifies the Target_Core_Mod/ConfigFS v3.0 PSCSI subsystem plugin for target_core_mod v3.0 to add
> a configurable method of choosing legacy scsi_execute_async() usage for underlying SCSI HTCL struct scsi_device.
> This method is still the only method of issuing I/O to the Linux SCSI subsystem is used for non TYPE_DISK hardware
> SCSI and SATA devices that appear under Linux-SCSI, but have not yet been updated to accept raw struct request
> operations directly.  Some of these include TYPE_ROM, TYPE_TYPE and TYPE_MEDIUM_CHANGER hardware.
> 
> This method can be used be used via ConfigFS as follows:
> 
> 	export TARGET=/sys/kernel/config/target/core/
>         # load target_core_mod
> 	modprobe target_core_mod
> 	# The Linux/SCSI scsi host id is part of pscsi_$HOST_ID HBA name..
> 	mkdir -p $TARGET/pscsi_0/sr0
> 	# echo CTL to the control configfs attribute
> 	echo scsi_channel_id=0,scsi_target_id=0,scsi_lun_id=0 > $TARGET/pscsi_0/sr0/control
>         # Activate the target_core_mod/ConfigFS PSCSI storage object
> 	echo 1 > $TARGET/pscsi_0/sr0/enable
> 
> At this point, creating symlinks for SCSI target ports to configfs enabled fabric modules
> against PSCSI hardware passthrough using te HCTL reference will work.
> 
> This patch are made against lio-core-2.6.git/master and tested on
> v2.6.29 x86 32-bit HVM with TYPE_DISK and v2.6.29 ppc64 with TYPE_ROM.
> The lio-core-2.6.git tree can be found at: 
> 
> http://git.kernel.org/?p=linux/kernel/git/nab/lio-core-2.6.git;a=summary
> 
> :-)
> 

Are you aware that scsi_execute_async() has gone in 2.6.30-rc1?

I'm not sure what would be the best alternative for you. I would say
a bio, but it is still being debated. Your current options are:

1. bio_alloc then loop () bio_add_pc_page, and finally blk_rq_append_bio
   (Which block people don't like)
2. sglist => page-pointers-array translation and blk_rq_map_user with 
   struct rq_map_data mode. (not possible with all kind of sglists)
2. sglist => iovec translation and blk_rq_map_user_iov()
   (Very very ugly mapping of pages to virtual pointers)

I have a similar situation with my OSD code.

Boaz

> Signed-off-by: Nicholas A. Bellinger <nab@xxxxxxxxxxxxxxx>
> ---
>  drivers/target/target_core_pscsi.c |   88 +++++++++++++++++++++++++++++++-----
>  include/target/target_core_pscsi.h |    2 +
>  2 files changed, 79 insertions(+), 11 deletions(-)
> 
> diff --git a/drivers/target/target_core_pscsi.c b/drivers/target/target_core_pscsi.c
> index 0962563..66510f4 100644
> --- a/drivers/target/target_core_pscsi.c
> +++ b/drivers/target/target_core_pscsi.c
> @@ -399,6 +399,8 @@ void *pscsi_allocate_virtdevice(se_hba_t *hba, const char *name)
>  		return NULL;
>  	}
>  	pdv->pdv_se_hba = hba;
> +	/* Use legacy path unless using ConfigFS FD method */
> +	pdv->pdv_legacy = 1;
>  
>  	printk(KERN_INFO "PSCSI: Allocated pdv: %p for %s\n", pdv, name);
>  	return (void *)pdv;
> @@ -555,9 +557,10 @@ int pscsi_activate_device(se_device_t *dev)
>  	struct scsi_device *sd = (struct scsi_device *) pdv->pdv_sd;
>  	struct Scsi_Host *sh = sd->host;
>  
> -	printk(KERN_INFO "CORE_PSCSI[%d] - Activating Device with TCQ: %d at"
> +	printk(KERN_INFO "CORE_PSCSI[%d] - Activating %s Device with TCQ: %d at"
>  		" SCSI Location (Channel/Target/LUN) %d/%d/%d\n", sh->host_no,
> -		 sd->queue_depth, sd->channel, sd->id, sd->lun);
> +		(pdv->pdv_legacy) ? "Legacy" : "REQ",  sd->queue_depth,
> +		sd->channel, sd->id, sd->lun);
>  
>  	return 0;
>  }
> @@ -572,8 +575,9 @@ void pscsi_deactivate_device(se_device_t *dev)
>  	struct scsi_device *sd = (struct scsi_device *) pdv->pdv_sd;
>  	struct Scsi_Host *sh = sd->host;
>  
> -	printk(KERN_INFO "CORE_PSCSI[%d] - Deactivating Device with TCQ: %d at"
> -		" SCSI Location (Channel/Target/LUN) %d/%d/%d\n", sh->host_no,
> +	printk(KERN_INFO "CORE_PSCSI[%d] - Deactivating %s Device with TCQ: %d"
> +		" at SCSI Location (Channel/Target/LUN) %d/%d/%d\n",
> +		sh->host_no, (pdv->pdv_legacy) ? "Legacy" : "REQ",
>  		sd->queue_depth, sd->channel, sd->id, sd->lun);
>  }
>  
> @@ -843,6 +847,30 @@ static int pscsi_blk_get_request(se_task_t *task)
>  	return 0;
>  }
>  
> +static int pscsi_do_task_legacy(
> +	se_task_t *task,
> +	pscsi_plugin_task_t *pt,
> +	pscsi_dev_virt_t *pdv)
> +{
> +	se_cmd_t *cmd = TASK_CMD(task);
> +	void *pscsi_buf = (task->task_sg_num != 0) ? task->task_sg :
> +			T_TASK(cmd)->t_task_buf;
> +	int ret;
> +
> +	ret = scsi_execute_async(pdv->pdv_sd, pt->pscsi_cdb,
> +			COMMAND_SIZE(pt->pscsi_cdb[0]), pt->pscsi_direction,
> +			pscsi_buf, task->task_size, task->task_sg_num,
> +			(pdv->pdv_sd->type == TYPE_DISK) ? PS_TIMEOUT_DISK :
> +			PS_TIMEOUT_OTHER, PS_RETRY, (void *)task,
> +			pscsi_req_done_legacy, GFP_KERNEL);
> +	if (ret != 0) {
> +		printk(KERN_ERR "PSCSI Execute(): returned: %d\n", ret);
> +		return PYX_TRANSPORT_LOGICAL_UNIT_COMMUNICATION_FAILURE;
> +	}
> +
> +	return 0;
> +}
> +
>  /*      pscsi_do_task(): (Part of se_subsystem_api_t template)
>   *
>   *
> @@ -852,6 +880,9 @@ int pscsi_do_task(se_task_t *task)
>  	pscsi_plugin_task_t *pt = (pscsi_plugin_task_t *) task->transport_req;
>  	pscsi_dev_virt_t *pdv = (pscsi_dev_virt_t *) task->se_dev->dev_ptr;
>  	struct gendisk *gd = NULL;
> +
> +	if (pdv->pdv_legacy)
> +		return pscsi_do_task_legacy(task, pt, pdv);
>  	/*
>  	 * Grab pointer to struct gendisk for TYPE_DISK and TYPE_ROM
>  	 * cases (eg: cases where struct scsi_device has a backing
> @@ -1104,6 +1135,7 @@ se_device_t *pscsi_create_virtdevice_from_fd(
>  		 * Keep track of the struct block_device for now..
>  		 */
>  		pdv->pdv_bd = bd;
> +		pdv->pdv_legacy = 0;
>  		/*
>  		 * pscsi_create_type_[disk,rom]() will release host_lock..
>  		 */
> @@ -1286,6 +1318,9 @@ int pscsi_map_task_SG(se_task_t *task)
>  			PAGE_SIZE - 1) >> PAGE_SHIFT;
>  	int nr_vecs = 0, ret = 0;
>  
> +	if (pdv->pdv_legacy)
> +		return 0;
> +
>  	if (!task->task_size)
>  		return 0;
>  	/*
> @@ -1387,6 +1422,9 @@ int pscsi_map_task_non_SG(se_task_t *task)
>  	pscsi_dev_virt_t *pdv = (pscsi_dev_virt_t *) task->se_dev->dev_ptr;
>  	int ret = 0;
>  
> +	if (pdv->pdv_legacy)
> +		return 0;
> +
>  	if (!task->task_size)
>  		return 0;
>  
> @@ -1407,8 +1445,12 @@ int pscsi_map_task_non_SG(se_task_t *task)
>  int pscsi_CDB_inquiry(se_task_t *task, u32 size)
>  {
>  	pscsi_plugin_task_t *pt = (pscsi_plugin_task_t *) task->transport_req;
> +	pscsi_dev_virt_t *pdv = (pscsi_dev_virt_t *) task->se_dev->dev_ptr;
>  
>  	pt->pscsi_direction = DMA_FROM_DEVICE;
> +	if (pdv->pdv_legacy)
> +		return pscsi_map_task_non_SG(task);
> +
>  	if (pscsi_blk_get_request(task) < 0)
>  		return -1;
>  
> @@ -1418,10 +1460,11 @@ int pscsi_CDB_inquiry(se_task_t *task, u32 size)
>  int pscsi_CDB_none(se_task_t *task, u32 size)
>  {
>  	pscsi_plugin_task_t *pt = (pscsi_plugin_task_t *) task->transport_req;
> +	pscsi_dev_virt_t *pdv = (pscsi_dev_virt_t *) task->se_dev->dev_ptr;
>  
>  	pt->pscsi_direction = DMA_NONE;
>  
> -	return pscsi_blk_get_request(task);
> +	return (pdv->pdv_legacy) ? 0 : pscsi_blk_get_request(task);
>  }
>  
>  /*	pscsi_CDB_read_non_SG():
> @@ -1431,8 +1474,11 @@ int pscsi_CDB_none(se_task_t *task, u32 size)
>  int pscsi_CDB_read_non_SG(se_task_t *task, u32 size)
>  {
>  	pscsi_plugin_task_t *pt = (pscsi_plugin_task_t *) task->transport_req;
> +	pscsi_dev_virt_t *pdv = (pscsi_dev_virt_t *) task->se_dev->dev_ptr;
>  
>  	pt->pscsi_direction = DMA_FROM_DEVICE;
> +	if (pdv->pdv_legacy)
> +		return pscsi_map_task_non_SG(task);
>  
>  	if (pscsi_blk_get_request(task) < 0)
>  		return PYX_TRANSPORT_LOGICAL_UNIT_COMMUNICATION_FAILURE;
> @@ -1447,11 +1493,13 @@ int pscsi_CDB_read_non_SG(se_task_t *task, u32 size)
>  int pscsi_CDB_read_SG(se_task_t *task, u32 size)
>  {
>  	pscsi_plugin_task_t *pt = (pscsi_plugin_task_t *) task->transport_req;
> +	pscsi_dev_virt_t *pdv = (pscsi_dev_virt_t *) task->se_dev->dev_ptr;
>  
>  	pt->pscsi_direction = DMA_FROM_DEVICE;
>  
> -	if (pscsi_blk_get_request(task) < 0)
> -		return PYX_TRANSPORT_LOGICAL_UNIT_COMMUNICATION_FAILURE;
> +	if (!(pdv->pdv_legacy))
> +		if (pscsi_blk_get_request(task) < 0)
> +			return PYX_TRANSPORT_LOGICAL_UNIT_COMMUNICATION_FAILURE;
>  
>  	if (pscsi_map_task_SG(task) < 0)
>  		return PYX_TRANSPORT_LOGICAL_UNIT_COMMUNICATION_FAILURE;
> @@ -1466,11 +1514,13 @@ int pscsi_CDB_read_SG(se_task_t *task, u32 size)
>  int pscsi_CDB_write_non_SG(se_task_t *task, u32 size)
>  {
>  	pscsi_plugin_task_t *pt = (pscsi_plugin_task_t *) task->transport_req;
> +	pscsi_dev_virt_t *pdv = (pscsi_dev_virt_t *) task->se_dev->dev_ptr;
>  
>  	pt->pscsi_direction = DMA_TO_DEVICE;
>  
> -	if (pscsi_blk_get_request(task) < 0)
> -		return PYX_TRANSPORT_LOGICAL_UNIT_COMMUNICATION_FAILURE;
> +	if (!(pdv->pdv_legacy))
> +		if (pscsi_blk_get_request(task) < 0)
> +			return PYX_TRANSPORT_LOGICAL_UNIT_COMMUNICATION_FAILURE;
>  
>  	return pscsi_map_task_non_SG(task);
>  }
> @@ -1482,11 +1532,13 @@ int pscsi_CDB_write_non_SG(se_task_t *task, u32 size)
>  int pscsi_CDB_write_SG(se_task_t *task, u32 size)
>  {
>  	pscsi_plugin_task_t *pt = (pscsi_plugin_task_t *) task->transport_req;
> +	pscsi_dev_virt_t *pdv = (pscsi_dev_virt_t *) task->se_dev->dev_ptr;
>  
>  	pt->pscsi_direction = DMA_TO_DEVICE;
>  
> -	if (pscsi_blk_get_request(task) < 0)
> -		return PYX_TRANSPORT_LOGICAL_UNIT_COMMUNICATION_FAILURE;
> +	if (!(pdv->pdv_legacy))
> +		if (pscsi_blk_get_request(task) < 0)
> +			return PYX_TRANSPORT_LOGICAL_UNIT_COMMUNICATION_FAILURE;
>  
>  	if (pscsi_map_task_SG(task) < 0)
>  		return PYX_TRANSPORT_LOGICAL_UNIT_COMMUNICATION_FAILURE;
> @@ -1643,6 +1695,20 @@ static inline void pscsi_process_SAM_status(
>  	return;
>  }
>  
> +void pscsi_req_done_legacy(void *data, char *sense, int result, int data_len)
> +{
> +	se_task_t *task = (se_task_t *)data;
> +	pscsi_plugin_task_t *pt = (pscsi_plugin_task_t *)task->transport_req;
> +
> +	pt->pscsi_result = result;
> +	pt->pscsi_resid = data_len;
> +
> +	if (result != 0)
> +		memcpy(pt->pscsi_sense, sense, SCSI_SENSE_BUFFERSIZE);	
> +
> +	pscsi_process_SAM_status(task, pt);
> +}
> +
>  void pscsi_req_done(struct request *req, int uptodate)
>  {
>  	se_task_t *task = (se_task_t *)req->end_io_data;
> diff --git a/include/target/target_core_pscsi.h b/include/target/target_core_pscsi.h
> index b05b793..0e1d6cf 100644
> --- a/include/target/target_core_pscsi.h
> +++ b/include/target/target_core_pscsi.h
> @@ -94,6 +94,7 @@ extern u32 pscsi_get_dma_length(u32, se_device_t *);
>  extern u32 pscsi_get_max_sectors(se_device_t *);
>  extern u32 pscsi_get_queue_depth(se_device_t *);
>  extern void pscsi_shutdown_hba(struct se_hba_s *);
> +extern void pscsi_req_done_legacy(void *, char *, int, int);
>  extern void pscsi_req_done(struct request *, int);
>  #endif
>  
> @@ -120,6 +121,7 @@ typedef struct pscsi_plugin_task_s {
>  
>  typedef struct pscsi_dev_virt_s {
>  	int	pdv_flags;
> +	int	pdv_legacy; /* Use scsi_execute_async() from HTCL */
>  	int	pdv_channel_id;
>  	int	pdv_target_id;
>  	int	pdv_lun_id;

--
To unsubscribe from this list: send the line "unsubscribe linux-scsi" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]
[Index of Archives]     [SCSI Target Devel]     [Linux SCSI Target Infrastructure]     [Kernel Newbies]     [IDE]     [Security]     [Git]     [Netfilter]     [Bugtraq]     [Yosemite News]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux RAID]     [Linux ATA RAID]     [Linux IIO]     [Samba]     [Device Mapper]
  Powered by Linux