diff options
Diffstat (limited to 'drivers/vhost/scsi.c')
| -rw-r--r-- | drivers/vhost/scsi.c | 58 | 
1 files changed, 16 insertions, 42 deletions
| diff --git a/drivers/vhost/scsi.c b/drivers/vhost/scsi.c index 5de21ad4bd05..d16c04dcc144 100644 --- a/drivers/vhost/scsi.c +++ b/drivers/vhost/scsi.c @@ -85,7 +85,7 @@ struct vhost_scsi_cmd {  	/* The number of scatterlists associated with this cmd */  	u32 tvc_sgl_count;  	u32 tvc_prot_sgl_count; -	/* Saved unpacked SCSI LUN for vhost_scsi_submission_work() */ +	/* Saved unpacked SCSI LUN for vhost_scsi_target_queue_cmd() */  	u32 tvc_lun;  	/* Pointer to the SGL formatted memory from virtio-scsi */  	struct scatterlist *tvc_sgl; @@ -101,8 +101,6 @@ struct vhost_scsi_cmd {  	struct vhost_scsi_nexus *tvc_nexus;  	/* The TCM I/O descriptor that is accessed via container_of() */  	struct se_cmd tvc_se_cmd; -	/* work item used for cmwq dispatch to vhost_scsi_submission_work() */ -	struct work_struct work;  	/* Copy of the incoming SCSI command descriptor block (CDB) */  	unsigned char tvc_cdb[VHOST_SCSI_MAX_CDB_SIZE];  	/* Sense buffer that will be mapped into outgoing status */ @@ -240,8 +238,6 @@ struct vhost_scsi_ctx {  	struct iov_iter out_iter;  }; -static struct workqueue_struct *vhost_scsi_workqueue; -  /* Global spinlock to protect vhost_scsi TPG list for vhost IOCTL access */  static DEFINE_MUTEX(vhost_scsi_mutex);  static LIST_HEAD(vhost_scsi_list); @@ -614,7 +610,7 @@ vhost_scsi_get_cmd(struct vhost_virtqueue *vq, struct vhost_scsi_tpg *tpg,  		return ERR_PTR(-EIO);  	} -	tag = sbitmap_get(&svq->scsi_tags, 0, false); +	tag = sbitmap_get(&svq->scsi_tags);  	if (tag < 0) {  		pr_err("Unable to obtain tag for vhost_scsi_cmd\n");  		return ERR_PTR(-ENOMEM); @@ -782,14 +778,11 @@ static int vhost_scsi_to_tcm_attr(int attr)  	return TCM_SIMPLE_TAG;  } -static void vhost_scsi_submission_work(struct work_struct *work) +static void vhost_scsi_target_queue_cmd(struct vhost_scsi_cmd *cmd)  { -	struct vhost_scsi_cmd *cmd = -		container_of(work, struct vhost_scsi_cmd, work); -	struct vhost_scsi_nexus *tv_nexus;  	struct se_cmd *se_cmd = &cmd->tvc_se_cmd; +	struct vhost_scsi_nexus *tv_nexus;  	struct scatterlist *sg_ptr, *sg_prot_ptr = NULL; -	int rc;  	/* FIXME: BIDI operation */  	if (cmd->tvc_sgl_count) { @@ -805,18 +798,17 @@ static void vhost_scsi_submission_work(struct work_struct *work)  	tv_nexus = cmd->tvc_nexus;  	se_cmd->tag = 0; -	rc = target_submit_cmd_map_sgls(se_cmd, tv_nexus->tvn_se_sess, -			cmd->tvc_cdb, &cmd->tvc_sense_buf[0], +	target_init_cmd(se_cmd, tv_nexus->tvn_se_sess, &cmd->tvc_sense_buf[0],  			cmd->tvc_lun, cmd->tvc_exp_data_len,  			vhost_scsi_to_tcm_attr(cmd->tvc_task_attr), -			cmd->tvc_data_direction, TARGET_SCF_ACK_KREF, -			sg_ptr, cmd->tvc_sgl_count, NULL, 0, sg_prot_ptr, -			cmd->tvc_prot_sgl_count); -	if (rc < 0) { -		transport_send_check_condition_and_sense(se_cmd, -				TCM_LOGICAL_UNIT_COMMUNICATION_FAILURE, 0); -		transport_generic_free_cmd(se_cmd, 0); -	} +			cmd->tvc_data_direction, TARGET_SCF_ACK_KREF); + +	if (target_submit_prep(se_cmd, cmd->tvc_cdb, sg_ptr, +			       cmd->tvc_sgl_count, NULL, 0, sg_prot_ptr, +			       cmd->tvc_prot_sgl_count, GFP_KERNEL)) +		return; + +	target_queue_submission(se_cmd);  }  static void @@ -1132,14 +1124,7 @@ vhost_scsi_handle_vq(struct vhost_scsi *vs, struct vhost_virtqueue *vq)  		 * vhost_scsi_queue_data_in() and vhost_scsi_queue_status()  		 */  		cmd->tvc_vq_desc = vc.head; -		/* -		 * Dispatch cmd descriptor for cmwq execution in process -		 * context provided by vhost_scsi_workqueue.  This also ensures -		 * cmd is executed on the same kworker CPU as this vhost -		 * thread to gain positive L2 cache locality effects. -		 */ -		INIT_WORK(&cmd->work, vhost_scsi_submission_work); -		queue_work(vhost_scsi_workqueue, &cmd->work); +		vhost_scsi_target_queue_cmd(cmd);  		ret = 0;  err:  		/* @@ -1512,7 +1497,7 @@ static int vhost_scsi_setup_vq_cmds(struct vhost_virtqueue *vq, int max_cmds)  		return 0;  	if (sbitmap_init_node(&svq->scsi_tags, max_cmds, -1, GFP_KERNEL, -			      NUMA_NO_NODE)) +			      NUMA_NO_NODE, false, true))  		return -ENOMEM;  	svq->max_cmds = max_cmds; @@ -2486,17 +2471,9 @@ static int __init vhost_scsi_init(void)  		" on "UTS_RELEASE"\n", VHOST_SCSI_VERSION, utsname()->sysname,  		utsname()->machine); -	/* -	 * Use our own dedicated workqueue for submitting I/O into -	 * target core to avoid contention within system_wq. -	 */ -	vhost_scsi_workqueue = alloc_workqueue("vhost_scsi", 0, 0); -	if (!vhost_scsi_workqueue) -		goto out; -  	ret = vhost_scsi_register();  	if (ret < 0) -		goto out_destroy_workqueue; +		goto out;  	ret = target_register_template(&vhost_scsi_ops);  	if (ret < 0) @@ -2506,8 +2483,6 @@ static int __init vhost_scsi_init(void)  out_vhost_scsi_deregister:  	vhost_scsi_deregister(); -out_destroy_workqueue: -	destroy_workqueue(vhost_scsi_workqueue);  out:  	return ret;  }; @@ -2516,7 +2491,6 @@ static void vhost_scsi_exit(void)  {  	target_unregister_template(&vhost_scsi_ops);  	vhost_scsi_deregister(); -	destroy_workqueue(vhost_scsi_workqueue);  };  MODULE_DESCRIPTION("VHOST_SCSI series fabric driver"); | 
