| 
									
										
										
										
											2022-06-03 19:15:11 +00:00
										 |  |  | /*   SPDX-License-Identifier: BSD-3-Clause
 | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  |  *   Copyright(c) Intel Corporation. All rights reserved. | 
					
						
							|  |  |  |  *   All rights reserved. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "spdk/stdinc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <linux/virtio_scsi.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "spdk/env.h"
 | 
					
						
							| 
									
										
										
										
											2018-06-11 20:32:15 +00:00
										 |  |  | #include "spdk/thread.h"
 | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | #include "spdk/scsi.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-02 16:00:22 +00:00
										 |  |  | #include "spdk/scsi_spec.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-06 18:35:56 +00:00
										 |  |  | #include "spdk/util.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-22 13:41:24 +00:00
										 |  |  | #include "spdk/likely.h"
 | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "spdk/vhost.h"
 | 
					
						
							|  |  |  | #include "vhost_internal.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Features supported by SPDK VHOST lib. */ | 
					
						
							| 
									
										
										
										
											2017-08-09 16:59:53 +00:00
										 |  |  | #define SPDK_VHOST_SCSI_FEATURES	(SPDK_VHOST_FEATURES | \
 | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 					(1ULL << VIRTIO_SCSI_F_INOUT) | \ | 
					
						
							|  |  |  | 					(1ULL << VIRTIO_SCSI_F_HOTPLUG) | \ | 
					
						
							|  |  |  | 					(1ULL << VIRTIO_SCSI_F_CHANGE ) | \ | 
					
						
							|  |  |  | 					(1ULL << VIRTIO_SCSI_F_T10_PI )) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Features that are specified in VIRTIO SCSI but currently not supported:
 | 
					
						
							|  |  |  |  * - Live migration not supported yet | 
					
						
							|  |  |  |  * - T10 PI | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2017-08-09 16:59:53 +00:00
										 |  |  | #define SPDK_VHOST_SCSI_DISABLED_FEATURES	(SPDK_VHOST_DISABLED_FEATURES | \
 | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 						(1ULL << VIRTIO_SCSI_F_T10_PI )) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-08 10:28:52 +00:00
										 |  |  | /* Vhost-user-scsi support protocol features */ | 
					
						
							|  |  |  | #define SPDK_VHOST_SCSI_PROTOCOL_FEATURES	(1ULL << VHOST_USER_PROTOCOL_F_INFLIGHT_SHMFD)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-29 15:01:16 +00:00
										 |  |  | #define MGMT_POLL_PERIOD_US (1000 * 5)
 | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define VIRTIO_SCSI_CONTROLQ   0
 | 
					
						
							|  |  |  | #define VIRTIO_SCSI_EVENTQ   1
 | 
					
						
							|  |  |  | #define VIRTIO_SCSI_REQUESTQ   2
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-04 15:20:21 +00:00
										 |  |  | enum spdk_scsi_dev_vhost_status { | 
					
						
							|  |  |  | 	/* Target ID is empty. */ | 
					
						
							|  |  |  | 	VHOST_SCSI_DEV_EMPTY, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-27 15:26:22 +00:00
										 |  |  | 	/* Target is still being added. */ | 
					
						
							|  |  |  | 	VHOST_SCSI_DEV_ADDING, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-04 15:20:21 +00:00
										 |  |  | 	/* Target ID occupied. */ | 
					
						
							|  |  |  | 	VHOST_SCSI_DEV_PRESENT, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Target ID is occupied but removal is in progress. */ | 
					
						
							|  |  |  | 	VHOST_SCSI_DEV_REMOVING, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* In session - device (SCSI target) seen but removed. */ | 
					
						
							|  |  |  | 	VHOST_SCSI_DEV_REMOVED, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-05 08:00:30 +00:00
										 |  |  | /** Context for a SCSI target in a vhost device */ | 
					
						
							| 
									
										
										
										
											2017-07-25 13:42:27 +00:00
										 |  |  | struct spdk_scsi_dev_vhost_state { | 
					
						
							| 
									
										
										
										
											2018-12-16 22:06:48 +00:00
										 |  |  | 	struct spdk_scsi_dev *dev; | 
					
						
							| 
									
										
										
										
											2019-03-04 15:20:21 +00:00
										 |  |  | 	enum spdk_scsi_dev_vhost_status status; | 
					
						
							| 
									
										
										
										
											2017-07-25 13:42:27 +00:00
										 |  |  | 	spdk_vhost_event_fn remove_cb; | 
					
						
							|  |  |  | 	void *remove_ctx; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | struct spdk_vhost_scsi_dev { | 
					
						
							| 
									
										
										
										
											2019-08-09 16:09:38 +00:00
										 |  |  | 	int ref; | 
					
						
							|  |  |  | 	bool registered; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	struct spdk_vhost_dev vdev; | 
					
						
							| 
									
										
										
										
											2017-07-25 13:42:27 +00:00
										 |  |  | 	struct spdk_scsi_dev_vhost_state scsi_dev_state[SPDK_VHOST_SCSI_CTRLR_MAX_DEVS]; | 
					
						
							| 
									
										
										
										
											2019-04-08 21:39:30 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2018-12-13 15:34:15 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-05 08:00:30 +00:00
										 |  |  | /** Context for a SCSI target in a vhost session */ | 
					
						
							|  |  |  | struct spdk_scsi_dev_session_state { | 
					
						
							|  |  |  | 	struct spdk_scsi_dev *dev; | 
					
						
							|  |  |  | 	enum spdk_scsi_dev_vhost_status status; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-13 15:34:15 +00:00
										 |  |  | struct spdk_vhost_scsi_session { | 
					
						
							|  |  |  | 	struct spdk_vhost_session vsession; | 
					
						
							| 
									
										
										
										
											2017-07-07 14:10:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-13 15:34:15 +00:00
										 |  |  | 	struct spdk_vhost_scsi_dev *svdev; | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | 	/** Local copy of the device state */ | 
					
						
							| 
									
										
										
										
											2019-03-05 08:00:30 +00:00
										 |  |  | 	struct spdk_scsi_dev_session_state scsi_dev_state[SPDK_VHOST_SCSI_CTRLR_MAX_DEVS]; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	struct spdk_poller *requestq_poller; | 
					
						
							| 
									
										
										
										
											2017-06-29 15:01:16 +00:00
										 |  |  | 	struct spdk_poller *mgmt_poller; | 
					
						
							| 
									
										
										
										
											2019-03-16 23:31:45 +00:00
										 |  |  | 	struct spdk_poller *stop_poller; | 
					
						
							| 
									
										
										
										
											2018-12-13 15:34:15 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-10 18:30:45 +00:00
										 |  |  | struct spdk_vhost_scsi_task { | 
					
						
							| 
									
										
										
										
											2017-06-23 10:02:34 +00:00
										 |  |  | 	struct spdk_scsi_task	scsi; | 
					
						
							| 
									
										
										
										
											2017-07-28 16:03:49 +00:00
										 |  |  | 	struct iovec iovs[SPDK_VHOST_IOVS_MAX]; | 
					
						
							| 
									
										
										
										
											2017-06-23 10:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	union { | 
					
						
							|  |  |  | 		struct virtio_scsi_cmd_resp *resp; | 
					
						
							|  |  |  | 		struct virtio_scsi_ctrl_tmf_resp *tmf_resp; | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-15 23:54:53 +00:00
										 |  |  | 	struct spdk_vhost_scsi_session *svsession; | 
					
						
							| 
									
										
										
										
											2017-06-23 10:02:34 +00:00
										 |  |  | 	struct spdk_scsi_dev *scsi_dev; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-08 19:39:03 +00:00
										 |  |  | 	/** Number of bytes that were written. */ | 
					
						
							|  |  |  | 	uint32_t used_len; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-23 10:02:34 +00:00
										 |  |  | 	int req_idx; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-10 20:19:40 +00:00
										 |  |  | 	/* If set, the task is currently used for I/O processing. */ | 
					
						
							|  |  |  | 	bool used; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-31 14:38:35 +00:00
										 |  |  | 	struct spdk_vhost_virtqueue *vq; | 
					
						
							| 
									
										
										
										
											2017-06-23 10:02:34 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | static int vhost_scsi_start(struct spdk_vhost_session *vsession); | 
					
						
							|  |  |  | static int vhost_scsi_stop(struct spdk_vhost_session *vsession); | 
					
						
							|  |  |  | static void vhost_scsi_dump_info_json(struct spdk_vhost_dev *vdev, | 
					
						
							|  |  |  | 				      struct spdk_json_write_ctx *w); | 
					
						
							|  |  |  | static void vhost_scsi_write_config_json(struct spdk_vhost_dev *vdev, | 
					
						
							| 
									
										
										
										
											2018-03-19 18:01:57 +00:00
										 |  |  | 		struct spdk_json_write_ctx *w); | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | static int vhost_scsi_dev_remove(struct spdk_vhost_dev *vdev); | 
					
						
							| 
									
										
										
										
											2020-09-25 10:17:05 +00:00
										 |  |  | static int vhost_scsi_dev_param_changed(struct spdk_vhost_dev *vdev, | 
					
						
							|  |  |  | 					unsigned scsi_tgt_num); | 
					
						
							| 
									
										
										
										
											2017-05-24 11:16:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-03 10:24:34 +00:00
										 |  |  | static const struct spdk_vhost_user_dev_backend spdk_vhost_scsi_user_device_backend = { | 
					
						
							| 
									
										
										
										
											2018-12-13 15:34:15 +00:00
										 |  |  | 	.session_ctx_size = sizeof(struct spdk_vhost_scsi_session) - sizeof(struct spdk_vhost_session), | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 	.start_session =  vhost_scsi_start, | 
					
						
							|  |  |  | 	.stop_session = vhost_scsi_stop, | 
					
						
							| 
									
										
										
										
											2022-03-03 10:24:34 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct spdk_vhost_dev_backend spdk_vhost_scsi_device_backend = { | 
					
						
							| 
									
										
										
										
											2022-05-13 08:37:06 +00:00
										 |  |  | 	.type = VHOST_BACKEND_SCSI, | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 	.dump_info_json = vhost_scsi_dump_info_json, | 
					
						
							|  |  |  | 	.write_config_json = vhost_scsi_write_config_json, | 
					
						
							|  |  |  | 	.remove_device = vhost_scsi_dev_remove, | 
					
						
							| 
									
										
										
										
											2017-05-24 11:16:40 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 13:25:54 +00:00
										 |  |  | static inline void | 
					
						
							|  |  |  | scsi_task_init(struct spdk_vhost_scsi_task *task) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	memset(&task->scsi, 0, sizeof(task->scsi)); | 
					
						
							|  |  |  | 	/* Tmf_resp pointer and resp pointer are in a union.
 | 
					
						
							|  |  |  | 	 * Here means task->tmf_resp = task->resp = NULL. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	task->resp = NULL; | 
					
						
							|  |  |  | 	task->used = true; | 
					
						
							|  |  |  | 	task->used_len = 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-23 10:02:34 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | vhost_scsi_task_put(struct spdk_vhost_scsi_task *task) | 
					
						
							| 
									
										
										
										
											2017-06-23 10:02:34 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	spdk_scsi_task_put(&task->scsi); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | vhost_scsi_task_free_cb(struct spdk_scsi_task *scsi_task) | 
					
						
							| 
									
										
										
										
											2017-06-23 10:02:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-07-10 18:30:45 +00:00
										 |  |  | 	struct spdk_vhost_scsi_task *task = SPDK_CONTAINEROF(scsi_task, struct spdk_vhost_scsi_task, scsi); | 
					
						
							| 
									
										
										
										
											2018-12-15 23:54:53 +00:00
										 |  |  | 	struct spdk_vhost_session *vsession = &task->svsession->vsession; | 
					
						
							| 
									
										
										
										
											2017-06-23 10:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-13 10:51:34 +00:00
										 |  |  | 	assert(vsession->task_cnt > 0); | 
					
						
							|  |  |  | 	vsession->task_cnt--; | 
					
						
							| 
									
										
										
										
											2017-10-10 20:19:40 +00:00
										 |  |  | 	task->used = false; | 
					
						
							| 
									
										
										
										
											2017-06-23 10:02:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-21 08:09:59 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | remove_scsi_tgt(struct spdk_vhost_scsi_dev *svdev, | 
					
						
							|  |  |  | 		unsigned scsi_tgt_num) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct spdk_scsi_dev_vhost_state *state; | 
					
						
							|  |  |  | 	struct spdk_scsi_dev *dev; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	state = &svdev->scsi_dev_state[scsi_tgt_num]; | 
					
						
							|  |  |  | 	dev = state->dev; | 
					
						
							|  |  |  | 	state->dev = NULL; | 
					
						
							| 
									
										
										
										
											2019-03-04 15:20:21 +00:00
										 |  |  | 	assert(state->status == VHOST_SCSI_DEV_REMOVING); | 
					
						
							|  |  |  | 	state->status = VHOST_SCSI_DEV_EMPTY; | 
					
						
							| 
									
										
										
										
											2019-04-11 09:10:47 +00:00
										 |  |  | 	spdk_scsi_dev_destruct(dev, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | 	if (state->remove_cb) { | 
					
						
							|  |  |  | 		state->remove_cb(&svdev->vdev, state->remove_ctx); | 
					
						
							|  |  |  | 		state->remove_cb = NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-11-08 09:44:00 +00:00
										 |  |  | 	SPDK_INFOLOG(vhost, "removed target 'Target %u'\n", scsi_tgt_num); | 
					
						
							| 
									
										
										
										
											2019-08-09 16:09:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (--svdev->ref == 0 && svdev->registered == false) { | 
					
						
							|  |  |  | 		free(svdev); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 21:30:56 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | vhost_scsi_dev_process_removed_cpl_cb(struct spdk_vhost_dev *vdev, void *ctx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unsigned scsi_tgt_num = (unsigned)(uintptr_t)ctx; | 
					
						
							|  |  |  | 	struct spdk_vhost_scsi_dev *svdev = SPDK_CONTAINEROF(vdev, | 
					
						
							|  |  |  | 					    struct spdk_vhost_scsi_dev, vdev); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* all sessions have already detached the device */ | 
					
						
							|  |  |  | 	if (svdev->scsi_dev_state[scsi_tgt_num].status != VHOST_SCSI_DEV_REMOVING) { | 
					
						
							|  |  |  | 		/* device was already removed in the meantime */ | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	remove_scsi_tgt(svdev, scsi_tgt_num); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | vhost_scsi_session_process_removed(struct spdk_vhost_dev *vdev, | 
					
						
							|  |  |  | 				   struct spdk_vhost_session *vsession, void *ctx) | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	unsigned scsi_tgt_num = (unsigned)(uintptr_t)ctx; | 
					
						
							| 
									
										
										
										
											2019-07-20 21:47:55 +00:00
										 |  |  | 	struct spdk_vhost_scsi_session *svsession = (struct spdk_vhost_scsi_session *)vsession; | 
					
						
							|  |  |  | 	struct spdk_scsi_dev_session_state *state = &svsession->scsi_dev_state[scsi_tgt_num]; | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (state->dev != NULL) { | 
					
						
							|  |  |  | 		/* there's still a session that references this device,
 | 
					
						
							|  |  |  | 		 * so abort our foreach chain here. We'll be called | 
					
						
							|  |  |  | 		 * again from this session's management poller after it | 
					
						
							|  |  |  | 		 * is removed in there | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-22 17:29:21 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | process_removed_devs(struct spdk_vhost_scsi_session *svsession) | 
					
						
							| 
									
										
										
										
											2017-06-22 17:29:21 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct spdk_scsi_dev *dev; | 
					
						
							| 
									
										
										
										
											2019-03-05 08:00:30 +00:00
										 |  |  | 	struct spdk_scsi_dev_session_state *state; | 
					
						
							| 
									
										
										
										
											2017-06-22 17:29:21 +00:00
										 |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (i = 0; i < SPDK_VHOST_SCSI_CTRLR_MAX_DEVS; ++i) { | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | 		state = &svsession->scsi_dev_state[i]; | 
					
						
							| 
									
										
										
										
											2018-12-16 22:06:48 +00:00
										 |  |  | 		dev = state->dev; | 
					
						
							| 
									
										
										
										
											2017-06-22 17:29:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-15 07:46:15 +00:00
										 |  |  | 		if (dev && state->status == VHOST_SCSI_DEV_REMOVING && | 
					
						
							|  |  |  | 		    !spdk_scsi_dev_has_pending_tasks(dev, NULL)) { | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | 			/* detach the device from this session */ | 
					
						
							| 
									
										
										
										
											2017-06-22 17:29:21 +00:00
										 |  |  | 			spdk_scsi_dev_free_io_channels(dev); | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | 			state->dev = NULL; | 
					
						
							| 
									
										
										
										
											2019-03-04 15:20:21 +00:00
										 |  |  | 			state->status = VHOST_SCSI_DEV_REMOVED; | 
					
						
							| 
									
										
										
										
											2019-01-13 17:24:52 +00:00
										 |  |  | 			/* try to detach it globally */ | 
					
						
							|  |  |  | 			spdk_vhost_lock(); | 
					
						
							| 
									
										
										
										
											2022-01-11 12:11:22 +00:00
										 |  |  | 			vhost_user_dev_foreach_session(&svsession->svdev->vdev, | 
					
						
							|  |  |  | 						       vhost_scsi_session_process_removed, | 
					
						
							|  |  |  | 						       vhost_scsi_dev_process_removed_cpl_cb, | 
					
						
							|  |  |  | 						       (void *)(uintptr_t)i); | 
					
						
							| 
									
										
										
										
											2019-01-13 17:24:52 +00:00
										 |  |  | 			spdk_vhost_unlock(); | 
					
						
							| 
									
										
										
										
											2017-06-22 17:29:21 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-06 18:35:56 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2018-12-15 23:54:53 +00:00
										 |  |  | eventq_enqueue(struct spdk_vhost_scsi_session *svsession, unsigned scsi_dev_num, | 
					
						
							|  |  |  | 	       uint32_t event, uint32_t reason) | 
					
						
							| 
									
										
										
										
											2017-06-06 18:35:56 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-12-15 23:54:53 +00:00
										 |  |  | 	struct spdk_vhost_session *vsession = &svsession->vsession; | 
					
						
							| 
									
										
										
										
											2017-08-31 14:38:35 +00:00
										 |  |  | 	struct spdk_vhost_virtqueue *vq; | 
					
						
							| 
									
										
										
										
											2017-10-09 15:28:00 +00:00
										 |  |  | 	struct vring_desc *desc, *desc_table; | 
					
						
							| 
									
										
										
										
											2017-07-25 11:55:42 +00:00
										 |  |  | 	struct virtio_scsi_event *desc_ev; | 
					
						
							| 
									
										
										
										
											2017-10-09 15:28:00 +00:00
										 |  |  | 	uint32_t desc_table_size, req_size = 0; | 
					
						
							| 
									
										
										
										
											2017-07-25 11:55:42 +00:00
										 |  |  | 	uint16_t req; | 
					
						
							| 
									
										
										
										
											2017-10-09 15:28:00 +00:00
										 |  |  | 	int rc; | 
					
						
							| 
									
										
										
										
											2017-06-06 18:35:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-25 12:38:18 +00:00
										 |  |  | 	assert(scsi_dev_num < SPDK_VHOST_SCSI_CTRLR_MAX_DEVS); | 
					
						
							| 
									
										
										
										
											2018-12-13 10:51:34 +00:00
										 |  |  | 	vq = &vsession->virtqueue[VIRTIO_SCSI_EVENTQ]; | 
					
						
							| 
									
										
										
										
											2017-06-06 18:35:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 	if (vq->vring.desc == NULL || vhost_vq_avail_ring_get(vq, &req, 1) != 1) { | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 		SPDK_ERRLOG("%s: failed to send virtio event (no avail ring entries?).\n", | 
					
						
							|  |  |  | 			    vsession->name); | 
					
						
							| 
									
										
										
										
											2017-07-25 11:55:42 +00:00
										 |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2017-06-06 18:35:56 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-07-25 11:55:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 	rc = vhost_vq_get_desc(vsession, vq, req, &desc, &desc_table, &desc_table_size); | 
					
						
							| 
									
										
										
										
											2017-10-09 15:28:00 +00:00
										 |  |  | 	if (rc != 0 || desc->len < sizeof(*desc_ev)) { | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 		SPDK_ERRLOG("%s: invalid eventq descriptor at index %"PRIu16".\n", | 
					
						
							|  |  |  | 			    vsession->name, req); | 
					
						
							| 
									
										
										
										
											2017-10-09 15:28:00 +00:00
										 |  |  | 		goto out; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-07-25 11:55:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 	desc_ev = vhost_gpa_to_vva(vsession, desc->addr, sizeof(*desc_ev)); | 
					
						
							| 
									
										
										
										
											2017-10-10 10:32:57 +00:00
										 |  |  | 	if (desc_ev == NULL) { | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 		SPDK_ERRLOG("%s: eventq descriptor at index %"PRIu16" points " | 
					
						
							|  |  |  | 			    "to unmapped guest memory address %p.\n", | 
					
						
							|  |  |  | 			    vsession->name, req, (void *)(uintptr_t)desc->addr); | 
					
						
							| 
									
										
										
										
											2017-10-09 15:28:00 +00:00
										 |  |  | 		goto out; | 
					
						
							| 
									
										
										
										
											2017-07-25 11:55:42 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-09 15:28:00 +00:00
										 |  |  | 	desc_ev->event = event; | 
					
						
							|  |  |  | 	desc_ev->lun[0] = 1; | 
					
						
							|  |  |  | 	desc_ev->lun[1] = scsi_dev_num; | 
					
						
							|  |  |  | 	/* virtio LUN id 0 can refer either to the entire device
 | 
					
						
							|  |  |  | 	 * or actual LUN 0 (the only supported by vhost for now) | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	desc_ev->lun[2] = 0 >> 8; | 
					
						
							|  |  |  | 	desc_ev->lun[3] = 0 & 0xFF; | 
					
						
							|  |  |  | 	/* virtio doesn't specify any strict format for LUN id (bytes 2 and 3)
 | 
					
						
							|  |  |  | 	 * current implementation relies on linux kernel sources | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	memset(&desc_ev->lun[4], 0, 4); | 
					
						
							|  |  |  | 	desc_ev->reason = reason; | 
					
						
							|  |  |  | 	req_size = sizeof(*desc_ev); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | out: | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 	vhost_vq_used_ring_enqueue(vsession, vq, req, req_size); | 
					
						
							| 
									
										
										
										
											2017-06-06 18:35:56 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2017-07-10 18:30:45 +00:00
										 |  |  | submit_completion(struct spdk_vhost_scsi_task *task) | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-12-15 23:54:53 +00:00
										 |  |  | 	struct spdk_vhost_session *vsession = &task->svsession->vsession; | 
					
						
							| 
									
										
										
										
											2018-12-13 10:51:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 	vhost_vq_used_ring_enqueue(vsession, task->vq, task->req_idx, | 
					
						
							|  |  |  | 				   task->used_len); | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | 	SPDK_DEBUGLOG(vhost_scsi, "Finished task (%p) req_idx=%d\n", task, task->req_idx); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 	vhost_scsi_task_put(task); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-23 10:02:34 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | vhost_scsi_task_mgmt_cpl(struct spdk_scsi_task *scsi_task) | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-07-10 18:30:45 +00:00
										 |  |  | 	struct spdk_vhost_scsi_task *task = SPDK_CONTAINEROF(scsi_task, struct spdk_vhost_scsi_task, scsi); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	submit_completion(task); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-23 10:02:34 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | vhost_scsi_task_cpl(struct spdk_scsi_task *scsi_task) | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-07-10 18:30:45 +00:00
										 |  |  | 	struct spdk_vhost_scsi_task *task = SPDK_CONTAINEROF(scsi_task, struct spdk_vhost_scsi_task, scsi); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* The SCSI task has completed.  Do final processing and then post
 | 
					
						
							|  |  |  | 	   notification to the virtqueue's "used" ring. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	task->resp->status = task->scsi.status; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (task->scsi.status != SPDK_SCSI_STATUS_GOOD) { | 
					
						
							|  |  |  | 		memcpy(task->resp->sense, task->scsi.sense_data, task->scsi.sense_data_len); | 
					
						
							|  |  |  | 		task->resp->sense_len = task->scsi.sense_data_len; | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | 		SPDK_DEBUGLOG(vhost_scsi, "Task (%p) req_idx=%d failed - status=%u\n", task, task->req_idx, | 
					
						
							| 
									
										
										
										
											2018-09-11 12:31:26 +00:00
										 |  |  | 			      task->scsi.status); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-12-28 20:35:37 +00:00
										 |  |  | 	assert(task->scsi.transfer_len == task->scsi.length); | 
					
						
							|  |  |  | 	task->resp->resid = task->scsi.length - task->scsi.data_transferred; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	submit_completion(task); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2017-07-10 18:30:45 +00:00
										 |  |  | task_submit(struct spdk_vhost_scsi_task *task) | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	task->resp->response = VIRTIO_SCSI_S_OK; | 
					
						
							|  |  |  | 	spdk_scsi_dev_queue_task(task->scsi_dev, &task->scsi); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2017-07-10 18:30:45 +00:00
										 |  |  | mgmt_task_submit(struct spdk_vhost_scsi_task *task, enum spdk_scsi_task_func func) | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	task->tmf_resp->response = VIRTIO_SCSI_S_OK; | 
					
						
							| 
									
										
										
										
											2018-12-09 22:57:30 +00:00
										 |  |  | 	task->scsi.function = func; | 
					
						
							|  |  |  | 	spdk_scsi_dev_queue_mgmt_task(task->scsi_dev, &task->scsi); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2017-07-10 18:30:45 +00:00
										 |  |  | invalid_request(struct spdk_vhost_scsi_task *task) | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-12-15 23:54:53 +00:00
										 |  |  | 	struct spdk_vhost_session *vsession = &task->svsession->vsession; | 
					
						
							| 
									
										
										
										
											2018-12-13 10:51:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 	vhost_vq_used_ring_enqueue(vsession, task->vq, task->req_idx, | 
					
						
							|  |  |  | 				   task->used_len); | 
					
						
							|  |  |  | 	vhost_scsi_task_put(task); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | 	SPDK_DEBUGLOG(vhost_scsi, "Invalid request (status=%" PRIu8")\n", | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 		      task->resp ? task->resp->response : -1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-22 12:45:27 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | vhost_scsi_task_init_target(struct spdk_vhost_scsi_task *task, const __u8 *lun) | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | 	struct spdk_vhost_scsi_session *svsession = task->svsession; | 
					
						
							| 
									
										
										
										
											2019-03-05 08:00:30 +00:00
										 |  |  | 	struct spdk_scsi_dev_session_state *state; | 
					
						
							| 
									
										
										
										
											2017-06-22 12:45:27 +00:00
										 |  |  | 	uint16_t lun_id = (((uint16_t)lun[2] << 8) | lun[3]) & 0x3FFF; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | 	SPDK_LOGDUMP(vhost_scsi_queue, "LUN", lun, 8); | 
					
						
							| 
									
										
										
										
											2017-06-22 12:45:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	/* First byte must be 1 and second is target */ | 
					
						
							| 
									
										
										
										
											2017-12-07 23:23:48 +00:00
										 |  |  | 	if (lun[0] != 1 || lun[1] >= SPDK_VHOST_SCSI_CTRLR_MAX_DEVS) { | 
					
						
							| 
									
										
										
										
											2017-06-22 12:45:27 +00:00
										 |  |  | 		return -1; | 
					
						
							| 
									
										
										
										
											2017-12-07 23:23:48 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | 	state = &svsession->scsi_dev_state[lun[1]]; | 
					
						
							| 
									
										
										
										
											2018-12-16 22:06:48 +00:00
										 |  |  | 	task->scsi_dev = state->dev; | 
					
						
							| 
									
										
										
										
											2019-03-04 15:20:21 +00:00
										 |  |  | 	if (state->dev == NULL || state->status != VHOST_SCSI_DEV_PRESENT) { | 
					
						
							| 
									
										
										
										
											2017-07-25 11:55:42 +00:00
										 |  |  | 		/* If dev has been hotdetached, return 0 to allow sending
 | 
					
						
							|  |  |  | 		 * additional hotremove event via sense codes. | 
					
						
							| 
									
										
										
										
											2017-06-22 17:29:21 +00:00
										 |  |  | 		 */ | 
					
						
							| 
									
										
										
										
											2019-03-04 15:20:21 +00:00
										 |  |  | 		return state->status != VHOST_SCSI_DEV_EMPTY ? 0 : -1; | 
					
						
							| 
									
										
										
										
											2017-06-22 12:45:27 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-22 12:45:27 +00:00
										 |  |  | 	task->scsi.target_port = spdk_scsi_dev_find_port_by_id(task->scsi_dev, 0); | 
					
						
							| 
									
										
										
										
											2018-12-16 22:06:48 +00:00
										 |  |  | 	task->scsi.lun = spdk_scsi_dev_get_lun(state->dev, lun_id); | 
					
						
							| 
									
										
										
										
											2017-06-22 12:45:27 +00:00
										 |  |  | 	return 0; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2017-07-10 18:30:45 +00:00
										 |  |  | process_ctrl_request(struct spdk_vhost_scsi_task *task) | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-12-15 23:54:53 +00:00
										 |  |  | 	struct spdk_vhost_session *vsession = &task->svsession->vsession; | 
					
						
							| 
									
										
										
										
											2017-10-09 15:28:00 +00:00
										 |  |  | 	struct vring_desc *desc, *desc_table; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	struct virtio_scsi_ctrl_tmf_req *ctrl_req; | 
					
						
							|  |  |  | 	struct virtio_scsi_ctrl_an_resp *an_resp; | 
					
						
							| 
									
										
										
										
											2018-01-08 19:39:03 +00:00
										 |  |  | 	uint32_t desc_table_size, used_len = 0; | 
					
						
							| 
									
										
										
										
											2017-10-09 15:28:00 +00:00
										 |  |  | 	int rc; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 	spdk_scsi_task_construct(&task->scsi, vhost_scsi_task_mgmt_cpl, vhost_scsi_task_free_cb); | 
					
						
							|  |  |  | 	rc = vhost_vq_get_desc(vsession, task->vq, task->req_idx, &desc, &desc_table, | 
					
						
							|  |  |  | 			       &desc_table_size); | 
					
						
							| 
									
										
										
										
											2017-10-09 15:28:00 +00:00
										 |  |  | 	if (spdk_unlikely(rc != 0)) { | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 		SPDK_ERRLOG("%s: invalid controlq descriptor at index %d.\n", | 
					
						
							|  |  |  | 			    vsession->name, task->req_idx); | 
					
						
							| 
									
										
										
										
											2017-10-09 15:28:00 +00:00
										 |  |  | 		goto out; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 	ctrl_req = vhost_gpa_to_vva(vsession, desc->addr, sizeof(*ctrl_req)); | 
					
						
							| 
									
										
										
										
											2018-04-23 22:14:07 +00:00
										 |  |  | 	if (ctrl_req == NULL) { | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 		SPDK_ERRLOG("%s: invalid task management request at index %d.\n", | 
					
						
							|  |  |  | 			    vsession->name, task->req_idx); | 
					
						
							| 
									
										
										
										
											2018-04-23 22:14:07 +00:00
										 |  |  | 		goto out; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | 	SPDK_DEBUGLOG(vhost_scsi_queue, | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 		      "Processing controlq descriptor: desc %d/%p, desc_addr %p, len %d, flags %d, last_used_idx %d; kickfd %d; size %d\n", | 
					
						
							| 
									
										
										
										
											2019-02-24 23:46:37 +00:00
										 |  |  | 		      task->req_idx, desc, (void *)desc->addr, desc->len, desc->flags, task->vq->last_used_idx, | 
					
						
							| 
									
										
										
										
											2017-08-31 14:38:35 +00:00
										 |  |  | 		      task->vq->vring.kickfd, task->vq->vring.size); | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | 	SPDK_LOGDUMP(vhost_scsi_queue, "Request descriptor", (uint8_t *)ctrl_req, desc->len); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 	vhost_scsi_task_init_target(task, ctrl_req->lun); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 	vhost_vring_desc_get_next(&desc, desc_table, desc_table_size); | 
					
						
							| 
									
										
										
										
											2017-10-09 15:28:00 +00:00
										 |  |  | 	if (spdk_unlikely(desc == NULL)) { | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 		SPDK_ERRLOG("%s: no response descriptor for controlq request %d.\n", | 
					
						
							|  |  |  | 			    vsession->name, task->req_idx); | 
					
						
							| 
									
										
										
										
											2017-10-09 15:28:00 +00:00
										 |  |  | 		goto out; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	/* Process the TMF request */ | 
					
						
							|  |  |  | 	switch (ctrl_req->type) { | 
					
						
							|  |  |  | 	case VIRTIO_SCSI_T_TMF: | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 		task->tmf_resp = vhost_gpa_to_vva(vsession, desc->addr, sizeof(*task->tmf_resp)); | 
					
						
							| 
									
										
										
										
											2017-10-13 13:30:05 +00:00
										 |  |  | 		if (spdk_unlikely(desc->len < sizeof(struct virtio_scsi_ctrl_tmf_resp) || task->tmf_resp == NULL)) { | 
					
						
							|  |  |  | 			SPDK_ERRLOG("%s: TMF response descriptor at index %d points to invalid guest memory region\n", | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 				    vsession->name, task->req_idx); | 
					
						
							| 
									
										
										
										
											2017-10-13 13:30:05 +00:00
										 |  |  | 			goto out; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/* Check if we are processing a valid request */ | 
					
						
							|  |  |  | 		if (task->scsi_dev == NULL) { | 
					
						
							|  |  |  | 			task->tmf_resp->response = VIRTIO_SCSI_S_BAD_TARGET; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		switch (ctrl_req->subtype) { | 
					
						
							|  |  |  | 		case VIRTIO_SCSI_T_TMF_LOGICAL_UNIT_RESET: | 
					
						
							|  |  |  | 			/* Handle LUN reset */ | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | 			SPDK_DEBUGLOG(vhost_scsi_queue, "%s: LUN reset\n", vsession->name); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			mgmt_task_submit(task, SPDK_SCSI_TASK_FUNC_LUN_RESET); | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 			task->tmf_resp->response = VIRTIO_SCSI_S_ABORTED; | 
					
						
							|  |  |  | 			/* Unsupported command */ | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | 			SPDK_DEBUGLOG(vhost_scsi_queue, "%s: unsupported TMF command %x\n", | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 				      vsession->name, ctrl_req->subtype); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case VIRTIO_SCSI_T_AN_QUERY: | 
					
						
							|  |  |  | 	case VIRTIO_SCSI_T_AN_SUBSCRIBE: { | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 		an_resp = vhost_gpa_to_vva(vsession, desc->addr, sizeof(*an_resp)); | 
					
						
							| 
									
										
										
										
											2017-10-13 13:30:05 +00:00
										 |  |  | 		if (spdk_unlikely(desc->len < sizeof(struct virtio_scsi_ctrl_an_resp) || an_resp == NULL)) { | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 			SPDK_WARNLOG("%s: asynchronous response descriptor points to invalid guest memory region\n", | 
					
						
							|  |  |  | 				     vsession->name); | 
					
						
							| 
									
										
										
										
											2017-10-13 13:30:05 +00:00
										 |  |  | 			goto out; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 		an_resp->response = VIRTIO_SCSI_S_ABORTED; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	default: | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | 		SPDK_DEBUGLOG(vhost_scsi_queue, "%s: Unsupported control command %x\n", | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 			      vsession->name, ctrl_req->type); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 		break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-08 19:39:03 +00:00
										 |  |  | 	used_len = sizeof(struct virtio_scsi_ctrl_tmf_resp); | 
					
						
							| 
									
										
										
										
											2017-10-09 15:28:00 +00:00
										 |  |  | out: | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 	vhost_vq_used_ring_enqueue(vsession, task->vq, task->req_idx, used_len); | 
					
						
							|  |  |  | 	vhost_scsi_task_put(task); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * Process task's descriptor chain and setup data related fields. | 
					
						
							|  |  |  |  * Return | 
					
						
							|  |  |  |  *   -1 if request is invalid and must be aborted, | 
					
						
							| 
									
										
										
										
											2017-07-25 16:51:31 +00:00
										 |  |  |  *    0 if all data are set. | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2017-07-10 18:30:45 +00:00
										 |  |  | task_data_setup(struct spdk_vhost_scsi_task *task, | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 		struct virtio_scsi_cmd_req **req) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-12-15 23:54:53 +00:00
										 |  |  | 	struct spdk_vhost_session *vsession = &task->svsession->vsession; | 
					
						
							| 
									
										
										
										
											2017-10-09 15:28:00 +00:00
										 |  |  | 	struct vring_desc *desc, *desc_table; | 
					
						
							| 
									
										
										
										
											2017-06-01 13:15:54 +00:00
										 |  |  | 	struct iovec *iovs = task->iovs; | 
					
						
							| 
									
										
										
										
											2018-01-09 17:53:17 +00:00
										 |  |  | 	uint16_t iovcnt = 0; | 
					
						
							| 
									
										
										
										
											2017-10-09 15:28:00 +00:00
										 |  |  | 	uint32_t desc_table_len, len = 0; | 
					
						
							|  |  |  | 	int rc; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 	spdk_scsi_task_construct(&task->scsi, vhost_scsi_task_cpl, vhost_scsi_task_free_cb); | 
					
						
							| 
									
										
										
										
											2017-10-09 18:21:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 	rc = vhost_vq_get_desc(vsession, task->vq, task->req_idx, &desc, &desc_table, &desc_table_len); | 
					
						
							| 
									
										
										
										
											2017-10-09 15:28:00 +00:00
										 |  |  | 	/* First descriptor must be readable */ | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 	if (spdk_unlikely(rc != 0  || vhost_vring_desc_is_wr(desc) || | 
					
						
							| 
									
										
										
										
											2017-10-13 13:30:05 +00:00
										 |  |  | 			  desc->len < sizeof(struct virtio_scsi_cmd_req))) { | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 		SPDK_WARNLOG("%s: invalid first request descriptor at index %"PRIu16".\n", | 
					
						
							|  |  |  | 			     vsession->name, task->req_idx); | 
					
						
							| 
									
										
										
										
											2017-10-10 10:32:57 +00:00
										 |  |  | 		goto invalid_task; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 	*req = vhost_gpa_to_vva(vsession, desc->addr, sizeof(**req)); | 
					
						
							| 
									
										
										
										
											2017-10-13 13:30:05 +00:00
										 |  |  | 	if (spdk_unlikely(*req == NULL)) { | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 		SPDK_WARNLOG("%s: request descriptor at index %d points to invalid guest memory region\n", | 
					
						
							|  |  |  | 			     vsession->name, task->req_idx); | 
					
						
							| 
									
										
										
										
											2017-10-13 13:30:05 +00:00
										 |  |  | 		goto invalid_task; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-09 15:28:00 +00:00
										 |  |  | 	/* Each request must have at least 2 descriptors (e.g. request and response) */ | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 	vhost_vring_desc_get_next(&desc, desc_table, desc_table_len); | 
					
						
							| 
									
										
										
										
											2017-10-09 15:28:00 +00:00
										 |  |  | 	if (desc == NULL) { | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 		SPDK_WARNLOG("%s: descriptor chain at index %d contains neither payload nor response buffer.\n", | 
					
						
							|  |  |  | 			     vsession->name, task->req_idx); | 
					
						
							| 
									
										
										
										
											2017-10-10 10:32:57 +00:00
										 |  |  | 		goto invalid_task; | 
					
						
							| 
									
										
										
										
											2017-10-09 15:28:00 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 	task->scsi.dxfer_dir = vhost_vring_desc_is_wr(desc) ? SPDK_SCSI_DIR_FROM_DEV : | 
					
						
							| 
									
										
										
										
											2017-05-24 12:52:07 +00:00
										 |  |  | 			       SPDK_SCSI_DIR_TO_DEV; | 
					
						
							| 
									
										
										
										
											2017-06-01 13:15:54 +00:00
										 |  |  | 	task->scsi.iovs = iovs; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (task->scsi.dxfer_dir == SPDK_SCSI_DIR_FROM_DEV) { | 
					
						
							|  |  |  | 		/*
 | 
					
						
							|  |  |  | 		 * FROM_DEV (READ): [RD_req][WR_resp][WR_buf0]...[WR_bufN] | 
					
						
							|  |  |  | 		 */ | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 		task->resp = vhost_gpa_to_vva(vsession, desc->addr, sizeof(*task->resp)); | 
					
						
							| 
									
										
										
										
											2017-10-13 13:30:05 +00:00
										 |  |  | 		if (spdk_unlikely(desc->len < sizeof(struct virtio_scsi_cmd_resp) || task->resp == NULL)) { | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 			SPDK_WARNLOG("%s: response descriptor at index %d points to invalid guest memory region\n", | 
					
						
							|  |  |  | 				     vsession->name, task->req_idx); | 
					
						
							| 
									
										
										
										
											2017-10-13 13:30:05 +00:00
										 |  |  | 			goto invalid_task; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 		rc = vhost_vring_desc_get_next(&desc, desc_table, desc_table_len); | 
					
						
							| 
									
										
										
										
											2017-10-09 15:28:00 +00:00
										 |  |  | 		if (spdk_unlikely(rc != 0)) { | 
					
						
							|  |  |  | 			SPDK_WARNLOG("%s: invalid descriptor chain at request index %d (descriptor id overflow?).\n", | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 				     vsession->name, task->req_idx); | 
					
						
							| 
									
										
										
										
											2017-10-10 10:32:57 +00:00
										 |  |  | 			goto invalid_task; | 
					
						
							| 
									
										
										
										
											2017-10-09 15:28:00 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (desc == NULL) { | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 			/*
 | 
					
						
							|  |  |  | 			 * TEST UNIT READY command and some others might not contain any payload and this is not an error. | 
					
						
							|  |  |  | 			 */ | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | 			SPDK_DEBUGLOG(vhost_scsi_data, | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 				      "No payload descriptors for FROM DEV command req_idx=%"PRIu16".\n", task->req_idx); | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | 			SPDK_LOGDUMP(vhost_scsi_data, "CDB=", (*req)->cdb, VIRTIO_SCSI_CDB_SIZE); | 
					
						
							| 
									
										
										
										
											2018-01-08 19:39:03 +00:00
										 |  |  | 			task->used_len = sizeof(struct virtio_scsi_cmd_resp); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 			task->scsi.iovcnt = 1; | 
					
						
							|  |  |  | 			task->scsi.iovs[0].iov_len = 0; | 
					
						
							|  |  |  | 			task->scsi.length = 0; | 
					
						
							|  |  |  | 			task->scsi.transfer_len = 0; | 
					
						
							|  |  |  | 			return 0; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* All remaining descriptors are data. */ | 
					
						
							| 
									
										
										
										
											2018-01-09 17:53:17 +00:00
										 |  |  | 		while (desc) { | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 			if (spdk_unlikely(!vhost_vring_desc_is_wr(desc))) { | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 				SPDK_WARNLOG("%s: FROM DEV cmd: descriptor nr %" PRIu16" in payload chain is read only.\n", | 
					
						
							|  |  |  | 					     vsession->name, iovcnt); | 
					
						
							| 
									
										
										
										
											2017-10-10 10:32:57 +00:00
										 |  |  | 				goto invalid_task; | 
					
						
							| 
									
										
										
										
											2017-10-09 15:28:00 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 			if (spdk_unlikely(vhost_vring_desc_to_iov(vsession, iovs, &iovcnt, desc))) { | 
					
						
							| 
									
										
										
										
											2017-10-10 10:32:57 +00:00
										 |  |  | 				goto invalid_task; | 
					
						
							| 
									
										
										
										
											2017-07-28 15:36:53 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 			len += desc->len; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 			rc = vhost_vring_desc_get_next(&desc, desc_table, desc_table_len); | 
					
						
							| 
									
										
										
										
											2017-10-09 15:28:00 +00:00
										 |  |  | 			if (spdk_unlikely(rc != 0)) { | 
					
						
							|  |  |  | 				SPDK_WARNLOG("%s: invalid payload in descriptor chain starting at index %d.\n", | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 					     vsession->name, task->req_idx); | 
					
						
							| 
									
										
										
										
											2017-10-10 10:32:57 +00:00
										 |  |  | 				goto invalid_task; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-01-08 19:39:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		task->used_len = sizeof(struct virtio_scsi_cmd_resp) + len; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | 		SPDK_DEBUGLOG(vhost_scsi_data, "TO DEV"); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 		/*
 | 
					
						
							|  |  |  | 		 * TO_DEV (WRITE):[RD_req][RD_buf0]...[RD_bufN][WR_resp] | 
					
						
							|  |  |  | 		 * No need to check descriptor WR flag as this is done while setting scsi.dxfer_dir. | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* Process descriptors up to response. */ | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 		while (!vhost_vring_desc_is_wr(desc)) { | 
					
						
							|  |  |  | 			if (spdk_unlikely(vhost_vring_desc_to_iov(vsession, iovs, &iovcnt, desc))) { | 
					
						
							| 
									
										
										
										
											2017-10-10 10:32:57 +00:00
										 |  |  | 				goto invalid_task; | 
					
						
							| 
									
										
										
										
											2017-07-28 15:36:53 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 			len += desc->len; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 			vhost_vring_desc_get_next(&desc, desc_table, desc_table_len); | 
					
						
							| 
									
										
										
										
											2017-10-09 15:28:00 +00:00
										 |  |  | 			if (spdk_unlikely(desc == NULL)) { | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 				SPDK_WARNLOG("%s: TO_DEV cmd: no response descriptor.\n", vsession->name); | 
					
						
							| 
									
										
										
										
											2017-10-10 10:32:57 +00:00
										 |  |  | 				goto invalid_task; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 		task->resp = vhost_gpa_to_vva(vsession, desc->addr, sizeof(*task->resp)); | 
					
						
							| 
									
										
										
										
											2017-10-13 13:30:05 +00:00
										 |  |  | 		if (spdk_unlikely(desc->len < sizeof(struct virtio_scsi_cmd_resp) || task->resp == NULL)) { | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 			SPDK_WARNLOG("%s: response descriptor at index %d points to invalid guest memory region\n", | 
					
						
							|  |  |  | 				     vsession->name, task->req_idx); | 
					
						
							| 
									
										
										
										
											2017-10-13 13:30:05 +00:00
										 |  |  | 			goto invalid_task; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-01-08 19:39:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		task->used_len = sizeof(struct virtio_scsi_cmd_resp); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	task->scsi.iovcnt = iovcnt; | 
					
						
							|  |  |  | 	task->scsi.length = len; | 
					
						
							|  |  |  | 	task->scsi.transfer_len = len; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-10 10:32:57 +00:00
										 |  |  | invalid_task: | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | 	SPDK_DEBUGLOG(vhost_scsi_data, "%s: Invalid task at index %"PRIu16".\n", | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 		      vsession->name, task->req_idx); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	return -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2017-07-10 18:30:45 +00:00
										 |  |  | process_request(struct spdk_vhost_scsi_task *task) | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct virtio_scsi_cmd_req *req; | 
					
						
							|  |  |  | 	int result; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	result = task_data_setup(task, &req); | 
					
						
							|  |  |  | 	if (result) { | 
					
						
							|  |  |  | 		return result; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 	result = vhost_scsi_task_init_target(task, req->lun); | 
					
						
							| 
									
										
										
										
											2017-06-22 13:41:24 +00:00
										 |  |  | 	if (spdk_unlikely(result != 0)) { | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 		task->resp->response = VIRTIO_SCSI_S_BAD_TARGET; | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	task->scsi.cdb = req->cdb; | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | 	SPDK_LOGDUMP(vhost_scsi_data, "request CDB", req->cdb, VIRTIO_SCSI_CDB_SIZE); | 
					
						
							| 
									
										
										
										
											2017-06-13 16:35:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-22 13:41:24 +00:00
										 |  |  | 	if (spdk_unlikely(task->scsi.lun == NULL)) { | 
					
						
							| 
									
										
										
										
											2017-06-13 16:35:30 +00:00
										 |  |  | 		spdk_scsi_task_process_null_lun(&task->scsi); | 
					
						
							|  |  |  | 		task->resp->response = VIRTIO_SCSI_S_OK; | 
					
						
							|  |  |  | 		return 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2020-04-01 13:25:54 +00:00
										 |  |  | process_scsi_task(struct spdk_vhost_session *vsession, | 
					
						
							|  |  |  | 		  struct spdk_vhost_virtqueue *vq, | 
					
						
							|  |  |  | 		  uint16_t req_idx) | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-07-10 18:30:45 +00:00
										 |  |  | 	struct spdk_vhost_scsi_task *task; | 
					
						
							| 
									
										
										
										
											2020-04-01 13:25:54 +00:00
										 |  |  | 	int result; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 13:25:54 +00:00
										 |  |  | 	task = &((struct spdk_vhost_scsi_task *)vq->tasks)[req_idx]; | 
					
						
							|  |  |  | 	if (spdk_unlikely(task->used)) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("%s: request with idx '%"PRIu16"' is already pending.\n", | 
					
						
							|  |  |  | 			    vsession->name, req_idx); | 
					
						
							|  |  |  | 		vhost_vq_used_ring_enqueue(vsession, vq, req_idx, 0); | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-06-22 09:44:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 13:25:54 +00:00
										 |  |  | 	vsession->task_cnt++; | 
					
						
							|  |  |  | 	scsi_task_init(task); | 
					
						
							| 
									
										
										
										
											2017-10-10 20:19:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 13:25:54 +00:00
										 |  |  | 	if (spdk_unlikely(vq->vring_idx == VIRTIO_SCSI_CONTROLQ)) { | 
					
						
							| 
									
										
										
										
											2017-06-22 09:44:41 +00:00
										 |  |  | 		process_ctrl_request(task); | 
					
						
							| 
									
										
										
										
											2020-04-01 13:25:54 +00:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 		result = process_request(task); | 
					
						
							|  |  |  | 		if (likely(result == 0)) { | 
					
						
							|  |  |  | 			task_submit(task); | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | 			SPDK_DEBUGLOG(vhost_scsi, "====== Task %p req_idx %d submitted ======\n", task, | 
					
						
							| 
									
										
										
										
											2020-04-01 13:25:54 +00:00
										 |  |  | 				      task->req_idx); | 
					
						
							|  |  |  | 		} else if (result > 0) { | 
					
						
							|  |  |  | 			vhost_scsi_task_cpl(&task->scsi); | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | 			SPDK_DEBUGLOG(vhost_scsi, "====== Task %p req_idx %d finished early ======\n", task, | 
					
						
							| 
									
										
										
										
											2020-04-01 13:25:54 +00:00
										 |  |  | 				      task->req_idx); | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			invalid_request(task); | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | 			SPDK_DEBUGLOG(vhost_scsi, "====== Task %p req_idx %d failed ======\n", task, | 
					
						
							| 
									
										
										
										
											2020-04-01 13:25:54 +00:00
										 |  |  | 				      task->req_idx); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-08 10:28:52 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | submit_inflight_desc(struct spdk_vhost_scsi_session *svsession, | 
					
						
							|  |  |  | 		     struct spdk_vhost_virtqueue *vq) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-11-24 08:06:24 +00:00
										 |  |  | 	struct spdk_vhost_session *vsession; | 
					
						
							|  |  |  | 	spdk_vhost_resubmit_info *resubmit; | 
					
						
							| 
									
										
										
										
											2020-09-08 10:28:52 +00:00
										 |  |  | 	spdk_vhost_resubmit_desc *resubmit_list; | 
					
						
							|  |  |  | 	uint16_t req_idx; | 
					
						
							| 
									
										
										
										
											2021-11-24 08:06:24 +00:00
										 |  |  | 	int i; | 
					
						
							| 
									
										
										
										
											2020-09-08 10:28:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-24 08:06:24 +00:00
										 |  |  | 	resubmit = vq->vring_inflight.resubmit_inflight; | 
					
						
							|  |  |  | 	if (spdk_likely(resubmit == NULL || resubmit->resubmit_list == NULL || | 
					
						
							|  |  |  | 			resubmit->resubmit_num == 0)) { | 
					
						
							| 
									
										
										
										
											2020-09-08 10:28:52 +00:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	resubmit_list = resubmit->resubmit_list; | 
					
						
							| 
									
										
										
										
											2021-11-24 08:06:24 +00:00
										 |  |  | 	vsession = &svsession->vsession; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (i = resubmit->resubmit_num - 1; i >= 0; --i) { | 
					
						
							| 
									
										
										
										
											2020-09-08 10:28:52 +00:00
										 |  |  | 		req_idx = resubmit_list[resubmit->resubmit_num].index; | 
					
						
							| 
									
										
										
										
											2021-11-24 08:06:24 +00:00
										 |  |  | 		SPDK_DEBUGLOG(vhost_scsi, "====== Start processing resubmit request idx %"PRIu16"======\n", | 
					
						
							| 
									
										
										
										
											2020-09-08 10:28:52 +00:00
										 |  |  | 			      req_idx); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (spdk_unlikely(req_idx >= vq->vring.size)) { | 
					
						
							|  |  |  | 			SPDK_ERRLOG("%s: request idx '%"PRIu16"' exceeds virtqueue size (%"PRIu16").\n", | 
					
						
							|  |  |  | 				    vsession->name, req_idx, vq->vring.size); | 
					
						
							|  |  |  | 			vhost_vq_used_ring_enqueue(vsession, vq, req_idx, 0); | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		process_scsi_task(vsession, vq, req_idx); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-24 08:06:24 +00:00
										 |  |  | 	resubmit->resubmit_num = 0; | 
					
						
							| 
									
										
										
										
											2020-09-08 10:28:52 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-11 10:40:11 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2020-04-01 13:25:54 +00:00
										 |  |  | process_vq(struct spdk_vhost_scsi_session *svsession, struct spdk_vhost_virtqueue *vq) | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-12-15 23:54:53 +00:00
										 |  |  | 	struct spdk_vhost_session *vsession = &svsession->vsession; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	uint16_t reqs[32]; | 
					
						
							|  |  |  | 	uint16_t reqs_cnt, i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-08 10:28:52 +00:00
										 |  |  | 	submit_inflight_desc(svsession, vq); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 	reqs_cnt = vhost_vq_avail_ring_get(vq, reqs, SPDK_COUNTOF(reqs)); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	assert(reqs_cnt <= 32); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-22 09:44:41 +00:00
										 |  |  | 	for (i = 0; i < reqs_cnt; i++) { | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | 		SPDK_DEBUGLOG(vhost_scsi, "====== Starting processing request idx %"PRIu16"======\n", | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 			      reqs[i]); | 
					
						
							| 
									
										
										
										
											2017-06-22 09:44:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-10 20:19:40 +00:00
										 |  |  | 		if (spdk_unlikely(reqs[i] >= vq->vring.size)) { | 
					
						
							|  |  |  | 			SPDK_ERRLOG("%s: request idx '%"PRIu16"' exceeds virtqueue size (%"PRIu16").\n", | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 				    vsession->name, reqs[i], vq->vring.size); | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 			vhost_vq_used_ring_enqueue(vsession, vq, reqs[i], 0); | 
					
						
							| 
									
										
										
										
											2017-10-10 20:19:40 +00:00
										 |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-08 10:28:52 +00:00
										 |  |  | 		rte_vhost_set_inflight_desc_split(vsession->vid, vq->vring_idx, reqs[i]); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 13:25:54 +00:00
										 |  |  | 		process_scsi_task(vsession, vq, reqs[i]); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-05-11 10:40:11 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return reqs_cnt; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-13 00:16:47 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2017-06-29 15:01:16 +00:00
										 |  |  | vdev_mgmt_worker(void *arg) | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-12-15 23:54:53 +00:00
										 |  |  | 	struct spdk_vhost_scsi_session *svsession = arg; | 
					
						
							|  |  |  | 	struct spdk_vhost_session *vsession = &svsession->vsession; | 
					
						
							| 
									
										
										
										
											2022-05-11 10:40:11 +00:00
										 |  |  | 	int rc = 0; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | 	process_removed_devs(svsession); | 
					
						
							| 
									
										
										
										
											2017-09-11 17:45:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-18 09:11:17 +00:00
										 |  |  | 	if (vsession->virtqueue[VIRTIO_SCSI_EVENTQ].vring.desc) { | 
					
						
							|  |  |  | 		vhost_vq_used_signal(vsession, &vsession->virtqueue[VIRTIO_SCSI_EVENTQ]); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (vsession->virtqueue[VIRTIO_SCSI_CONTROLQ].vring.desc) { | 
					
						
							| 
									
										
										
										
											2022-05-11 10:40:11 +00:00
										 |  |  | 		rc = process_vq(svsession, &vsession->virtqueue[VIRTIO_SCSI_CONTROLQ]); | 
					
						
							| 
									
										
										
										
											2020-11-18 09:11:17 +00:00
										 |  |  | 		vhost_vq_used_signal(vsession, &vsession->virtqueue[VIRTIO_SCSI_CONTROLQ]); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-03-13 00:16:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-11 10:40:11 +00:00
										 |  |  | 	return rc > 0 ? SPDK_POLLER_BUSY : SPDK_POLLER_IDLE; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-13 00:16:47 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | vdev_worker(void *arg) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-12-15 23:54:53 +00:00
										 |  |  | 	struct spdk_vhost_scsi_session *svsession = arg; | 
					
						
							|  |  |  | 	struct spdk_vhost_session *vsession = &svsession->vsession; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	uint32_t q_idx; | 
					
						
							| 
									
										
										
										
											2022-05-11 10:40:11 +00:00
										 |  |  | 	int rc = 0; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-13 10:51:34 +00:00
										 |  |  | 	for (q_idx = VIRTIO_SCSI_REQUESTQ; q_idx < vsession->max_queues; q_idx++) { | 
					
						
							| 
									
										
										
										
											2022-05-11 10:40:11 +00:00
										 |  |  | 		rc = process_vq(svsession, &vsession->virtqueue[q_idx]); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-09-11 17:45:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 	vhost_session_used_signal(vsession); | 
					
						
							| 
									
										
										
										
											2018-03-13 00:16:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-11 10:40:11 +00:00
										 |  |  | 	return rc > 0 ? SPDK_POLLER_BUSY : SPDK_POLLER_IDLE; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-02 16:00:22 +00:00
										 |  |  | static struct spdk_vhost_scsi_dev * | 
					
						
							|  |  |  | to_scsi_dev(struct spdk_vhost_dev *ctrlr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (ctrlr == NULL) { | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-13 08:37:06 +00:00
										 |  |  | 	if (ctrlr->backend->type != VHOST_BACKEND_SCSI) { | 
					
						
							| 
									
										
										
										
											2018-01-26 14:29:20 +00:00
										 |  |  | 		SPDK_ERRLOG("%s: not a vhost-scsi device.\n", ctrlr->name); | 
					
						
							| 
									
										
										
										
											2017-06-02 16:00:22 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-26 14:29:20 +00:00
										 |  |  | 	return SPDK_CONTAINEROF(ctrlr, struct spdk_vhost_scsi_dev, vdev); | 
					
						
							| 
									
										
										
										
											2017-06-02 16:00:22 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-13 15:34:15 +00:00
										 |  |  | static struct spdk_vhost_scsi_session * | 
					
						
							|  |  |  | to_scsi_session(struct spdk_vhost_session *vsession) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-05-13 08:37:06 +00:00
										 |  |  | 	assert(vsession->vdev->backend->type == VHOST_BACKEND_SCSI); | 
					
						
							| 
									
										
										
										
											2018-12-13 15:34:15 +00:00
										 |  |  | 	return (struct spdk_vhost_scsi_session *)vsession; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | int | 
					
						
							| 
									
										
										
										
											2017-08-10 18:11:14 +00:00
										 |  |  | spdk_vhost_scsi_dev_construct(const char *name, const char *cpumask) | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-08 21:39:30 +00:00
										 |  |  | 	struct spdk_vhost_scsi_dev *svdev = calloc(1, sizeof(*svdev)); | 
					
						
							| 
									
										
										
										
											2017-06-06 18:35:56 +00:00
										 |  |  | 	int rc; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (svdev == NULL) { | 
					
						
							|  |  |  | 		return -ENOMEM; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-02 13:15:51 +00:00
										 |  |  | 	svdev->vdev.virtio_features = SPDK_VHOST_SCSI_FEATURES; | 
					
						
							|  |  |  | 	svdev->vdev.disabled_features = SPDK_VHOST_SCSI_DISABLED_FEATURES; | 
					
						
							| 
									
										
										
										
											2020-09-08 10:28:52 +00:00
										 |  |  | 	svdev->vdev.protocol_features = SPDK_VHOST_SCSI_PROTOCOL_FEATURES; | 
					
						
							| 
									
										
										
										
											2019-10-02 13:15:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-08 11:06:41 +00:00
										 |  |  | 	spdk_vhost_lock(); | 
					
						
							| 
									
										
										
										
											2021-09-17 08:23:35 +00:00
										 |  |  | 	rc = vhost_dev_register(&svdev->vdev, name, cpumask, NULL, | 
					
						
							| 
									
										
										
										
											2022-03-03 10:24:34 +00:00
										 |  |  | 				&spdk_vhost_scsi_device_backend, | 
					
						
							|  |  |  | 				&spdk_vhost_scsi_user_device_backend); | 
					
						
							| 
									
										
										
										
											2017-06-06 18:35:56 +00:00
										 |  |  | 	if (rc) { | 
					
						
							| 
									
										
										
										
											2019-04-08 21:39:30 +00:00
										 |  |  | 		free(svdev); | 
					
						
							| 
									
										
										
										
											2019-08-09 16:09:38 +00:00
										 |  |  | 		spdk_vhost_unlock(); | 
					
						
							|  |  |  | 		return rc; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-09 16:09:38 +00:00
										 |  |  | 	svdev->registered = true; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-08 11:06:41 +00:00
										 |  |  | 	spdk_vhost_unlock(); | 
					
						
							|  |  |  | 	return rc; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-26 13:09:39 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | vhost_scsi_dev_remove(struct spdk_vhost_dev *vdev) | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-06-02 16:00:22 +00:00
										 |  |  | 	struct spdk_vhost_scsi_dev *svdev = to_scsi_dev(vdev); | 
					
						
							| 
									
										
										
										
											2022-03-03 09:31:02 +00:00
										 |  |  | 	struct spdk_vhost_user_dev *user_dev = vdev->ctxt; | 
					
						
							| 
									
										
										
										
											2017-08-01 16:08:32 +00:00
										 |  |  | 	int rc, i; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-03 09:31:02 +00:00
										 |  |  | 	if (user_dev->pending_async_op_num) { | 
					
						
							| 
									
										
										
										
											2022-03-03 09:58:36 +00:00
										 |  |  | 		return -EBUSY; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 15:03:04 +00:00
										 |  |  | 	assert(svdev != NULL); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	for (i = 0; i < SPDK_VHOST_SCSI_CTRLR_MAX_DEVS; ++i) { | 
					
						
							| 
									
										
										
										
											2018-12-16 22:06:48 +00:00
										 |  |  | 		if (svdev->scsi_dev_state[i].dev) { | 
					
						
							| 
									
										
										
										
											2018-01-26 13:46:26 +00:00
										 |  |  | 			if (vdev->registered) { | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 				SPDK_ERRLOG("%s: SCSI target %d is still present.\n", vdev->name, i); | 
					
						
							| 
									
										
										
										
											2018-01-26 13:46:26 +00:00
										 |  |  | 				return -EBUSY; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			rc = spdk_vhost_scsi_dev_remove_tgt(vdev, i, NULL, NULL); | 
					
						
							|  |  |  | 			if (rc != 0) { | 
					
						
							|  |  |  | 				SPDK_ERRLOG("%s: failed to force-remove target %d\n", vdev->name, i); | 
					
						
							|  |  |  | 				return rc; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 	rc = vhost_dev_unregister(vdev); | 
					
						
							| 
									
										
										
										
											2017-08-01 16:08:32 +00:00
										 |  |  | 	if (rc != 0) { | 
					
						
							|  |  |  | 		return rc; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-08-09 16:09:38 +00:00
										 |  |  | 	svdev->registered = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (svdev->ref == 0) { | 
					
						
							|  |  |  | 		free(svdev); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct spdk_scsi_dev * | 
					
						
							| 
									
										
										
										
											2017-11-22 11:20:03 +00:00
										 |  |  | spdk_vhost_scsi_dev_get_tgt(struct spdk_vhost_dev *vdev, uint8_t num) | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-06-02 16:00:22 +00:00
										 |  |  | 	struct spdk_vhost_scsi_dev *svdev; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	assert(num < SPDK_VHOST_SCSI_CTRLR_MAX_DEVS); | 
					
						
							| 
									
										
										
										
											2017-06-02 16:00:22 +00:00
										 |  |  | 	svdev = to_scsi_dev(vdev); | 
					
						
							| 
									
										
										
										
											2019-07-20 15:03:04 +00:00
										 |  |  | 	assert(svdev != NULL); | 
					
						
							|  |  |  | 	if (svdev->scsi_dev_state[num].status != VHOST_SCSI_DEV_PRESENT) { | 
					
						
							| 
									
										
										
										
											2019-03-04 16:05:29 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-06-02 16:00:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-04 16:05:29 +00:00
										 |  |  | 	assert(svdev->scsi_dev_state[num].dev != NULL); | 
					
						
							|  |  |  | 	return svdev->scsi_dev_state[num].dev; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-25 10:17:05 +00:00
										 |  |  | static unsigned | 
					
						
							|  |  |  | get_scsi_dev_num(const struct spdk_vhost_scsi_dev *svdev, | 
					
						
							|  |  |  | 		 const struct spdk_scsi_lun *lun) | 
					
						
							| 
									
										
										
										
											2017-06-06 18:35:56 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-07-25 12:38:18 +00:00
										 |  |  | 	const struct spdk_scsi_dev *scsi_dev; | 
					
						
							|  |  |  | 	unsigned scsi_dev_num; | 
					
						
							| 
									
										
										
										
											2017-06-06 18:35:56 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	assert(lun != NULL); | 
					
						
							|  |  |  | 	assert(svdev != NULL); | 
					
						
							| 
									
										
										
										
											2017-07-25 12:38:18 +00:00
										 |  |  | 	scsi_dev = spdk_scsi_lun_get_dev(lun); | 
					
						
							|  |  |  | 	for (scsi_dev_num = 0; scsi_dev_num < SPDK_VHOST_SCSI_CTRLR_MAX_DEVS; scsi_dev_num++) { | 
					
						
							| 
									
										
										
										
											2018-12-16 22:06:48 +00:00
										 |  |  | 		if (svdev->scsi_dev_state[scsi_dev_num].dev == scsi_dev) { | 
					
						
							| 
									
										
										
										
											2017-07-25 12:38:18 +00:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-25 10:17:05 +00:00
										 |  |  | 	return scsi_dev_num; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | vhost_scsi_lun_resize(const struct spdk_scsi_lun *lun, void *arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct spdk_vhost_scsi_dev *svdev = arg; | 
					
						
							|  |  |  | 	unsigned scsi_dev_num; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	scsi_dev_num = get_scsi_dev_num(svdev, lun); | 
					
						
							|  |  |  | 	if (scsi_dev_num == SPDK_VHOST_SCSI_CTRLR_MAX_DEVS) { | 
					
						
							|  |  |  | 		/* The entire device has been already removed. */ | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	vhost_scsi_dev_param_changed(&svdev->vdev, scsi_dev_num); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | vhost_scsi_lun_hotremove(const struct spdk_scsi_lun *lun, void *arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct spdk_vhost_scsi_dev *svdev = arg; | 
					
						
							|  |  |  | 	unsigned scsi_dev_num; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	scsi_dev_num = get_scsi_dev_num(svdev, lun); | 
					
						
							| 
									
										
										
										
											2017-07-25 12:38:18 +00:00
										 |  |  | 	if (scsi_dev_num == SPDK_VHOST_SCSI_CTRLR_MAX_DEVS) { | 
					
						
							| 
									
										
										
										
											2018-01-26 13:48:48 +00:00
										 |  |  | 		/* The entire device has been already removed. */ | 
					
						
							| 
									
										
										
										
											2017-07-25 12:38:18 +00:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-26 11:40:37 +00:00
										 |  |  | 	/* remove entire device */ | 
					
						
							| 
									
										
										
										
											2017-11-22 11:20:03 +00:00
										 |  |  | 	spdk_vhost_scsi_dev_remove_tgt(&svdev->vdev, scsi_dev_num, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2017-06-06 18:35:56 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 21:30:56 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | vhost_scsi_dev_add_tgt_cpl_cb(struct spdk_vhost_dev *vdev, void *ctx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unsigned scsi_tgt_num = (unsigned)(uintptr_t)ctx; | 
					
						
							|  |  |  | 	struct spdk_vhost_scsi_dev *svdev = SPDK_CONTAINEROF(vdev, | 
					
						
							|  |  |  | 					    struct spdk_vhost_scsi_dev, vdev); | 
					
						
							|  |  |  | 	struct spdk_scsi_dev_vhost_state *vhost_sdev; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	vhost_sdev = &svdev->scsi_dev_state[scsi_tgt_num]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* All sessions have added the target */ | 
					
						
							|  |  |  | 	assert(vhost_sdev->status == VHOST_SCSI_DEV_ADDING); | 
					
						
							|  |  |  | 	vhost_sdev->status = VHOST_SCSI_DEV_PRESENT; | 
					
						
							| 
									
										
										
										
											2019-08-09 16:09:38 +00:00
										 |  |  | 	svdev->ref++; | 
					
						
							| 
									
										
										
										
											2019-07-20 21:30:56 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | vhost_scsi_session_add_tgt(struct spdk_vhost_dev *vdev, | 
					
						
							|  |  |  | 			   struct spdk_vhost_session *vsession, void *ctx) | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	unsigned scsi_tgt_num = (unsigned)(uintptr_t)ctx; | 
					
						
							| 
									
										
										
										
											2019-07-20 21:47:55 +00:00
										 |  |  | 	struct spdk_vhost_scsi_session *svsession = (struct spdk_vhost_scsi_session *)vsession; | 
					
						
							|  |  |  | 	struct spdk_scsi_dev_session_state *session_sdev = &svsession->scsi_dev_state[scsi_tgt_num]; | 
					
						
							| 
									
										
										
										
											2019-03-05 08:00:30 +00:00
										 |  |  | 	struct spdk_scsi_dev_vhost_state *vhost_sdev; | 
					
						
							| 
									
										
										
										
											2019-02-15 10:29:09 +00:00
										 |  |  | 	int rc; | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-29 07:26:10 +00:00
										 |  |  | 	if (!vsession->started || session_sdev->dev != NULL) { | 
					
						
							| 
									
										
										
										
											2019-03-27 16:20:39 +00:00
										 |  |  | 		/* Nothing to do. */ | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-05 08:00:30 +00:00
										 |  |  | 	vhost_sdev = &svsession->svdev->scsi_dev_state[scsi_tgt_num]; | 
					
						
							|  |  |  | 	session_sdev->dev = vhost_sdev->dev; | 
					
						
							| 
									
										
										
										
											2019-03-27 15:26:22 +00:00
										 |  |  | 	session_sdev->status = VHOST_SCSI_DEV_PRESENT; | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-15 10:29:09 +00:00
										 |  |  | 	rc = spdk_scsi_dev_allocate_io_channels(svsession->scsi_dev_state[scsi_tgt_num].dev); | 
					
						
							|  |  |  | 	if (rc != 0) { | 
					
						
							| 
									
										
										
										
											2021-11-25 01:40:58 +00:00
										 |  |  | 		SPDK_ERRLOG("%s: Couldn't allocate io channel for SCSI target %u.\n", | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 			    vsession->name, scsi_tgt_num); | 
					
						
							| 
									
										
										
										
											2019-02-15 10:29:09 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/* unset the SCSI target so that all I/O to it will be rejected */ | 
					
						
							| 
									
										
										
										
											2019-03-05 08:00:30 +00:00
										 |  |  | 		session_sdev->dev = NULL; | 
					
						
							| 
									
										
										
										
											2019-03-04 15:20:21 +00:00
										 |  |  | 		/* Set status to EMPTY so that we won't reply with SCSI hotremove
 | 
					
						
							| 
									
										
										
										
											2019-02-15 10:29:09 +00:00
										 |  |  | 		 * sense codes - the device hasn't ever been added. | 
					
						
							|  |  |  | 		 */ | 
					
						
							| 
									
										
										
										
											2019-03-05 08:00:30 +00:00
										 |  |  | 		session_sdev->status = VHOST_SCSI_DEV_EMPTY; | 
					
						
							| 
									
										
										
										
											2019-02-15 10:29:09 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/* Return with no error. We'll continue allocating io_channels for
 | 
					
						
							|  |  |  | 		 * other sessions on this device in hopes they succeed. The sessions | 
					
						
							|  |  |  | 		 * that failed to allocate io_channels simply won't be able to | 
					
						
							|  |  |  | 		 * detect the SCSI target, nor do any I/O to it. | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 	if (vhost_dev_has_feature(vsession, VIRTIO_SCSI_F_HOTPLUG)) { | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | 		eventq_enqueue(svsession, scsi_tgt_num, | 
					
						
							|  |  |  | 			       VIRTIO_SCSI_T_TRANSPORT_RESET, VIRTIO_SCSI_EVT_RESET_RESCAN); | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 		SPDK_NOTICELOG("%s: driver does not support hotplug. " | 
					
						
							|  |  |  | 			       "Please restart it or perform a rescan.\n", | 
					
						
							|  |  |  | 			       vsession->name); | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | int | 
					
						
							| 
									
										
										
										
											2019-01-28 21:30:36 +00:00
										 |  |  | spdk_vhost_scsi_dev_add_tgt(struct spdk_vhost_dev *vdev, int scsi_tgt_num, | 
					
						
							| 
									
										
										
										
											2018-01-10 05:55:53 +00:00
										 |  |  | 			    const char *bdev_name) | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct spdk_vhost_scsi_dev *svdev; | 
					
						
							| 
									
										
										
										
											2018-12-16 22:06:48 +00:00
										 |  |  | 	struct spdk_scsi_dev_vhost_state *state; | 
					
						
							| 
									
										
										
										
											2017-11-22 11:20:03 +00:00
										 |  |  | 	char target_name[SPDK_SCSI_DEV_MAX_NAME]; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	int lun_id_list[1]; | 
					
						
							| 
									
										
										
										
											2018-01-10 05:55:53 +00:00
										 |  |  | 	const char *bdev_names_list[1]; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-25 17:26:55 +00:00
										 |  |  | 	svdev = to_scsi_dev(vdev); | 
					
						
							| 
									
										
										
										
											2020-08-27 17:22:09 +00:00
										 |  |  | 	if (!svdev) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("Before adding a SCSI target, there should be a SCSI device."); | 
					
						
							|  |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-28 21:30:36 +00:00
										 |  |  | 	if (scsi_tgt_num < 0) { | 
					
						
							|  |  |  | 		for (scsi_tgt_num = 0; scsi_tgt_num < SPDK_VHOST_SCSI_CTRLR_MAX_DEVS; scsi_tgt_num++) { | 
					
						
							|  |  |  | 			if (svdev->scsi_dev_state[scsi_tgt_num].dev == NULL) { | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (scsi_tgt_num == SPDK_VHOST_SCSI_CTRLR_MAX_DEVS) { | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 			SPDK_ERRLOG("%s: all SCSI target slots are already in use.\n", vdev->name); | 
					
						
							| 
									
										
										
										
											2019-01-28 21:30:36 +00:00
										 |  |  | 			return -ENOSPC; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		if (scsi_tgt_num >= SPDK_VHOST_SCSI_CTRLR_MAX_DEVS) { | 
					
						
							| 
									
										
										
										
											2022-03-30 07:50:33 +00:00
										 |  |  | 			SPDK_ERRLOG("%s: SCSI target number is too big (got %d, max %d), started from 0.\n", | 
					
						
							|  |  |  | 				    vdev->name, scsi_tgt_num, SPDK_VHOST_SCSI_CTRLR_MAX_DEVS - 1); | 
					
						
							| 
									
										
										
										
											2019-01-28 21:30:36 +00:00
										 |  |  | 			return -EINVAL; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-10 05:55:53 +00:00
										 |  |  | 	if (bdev_name == NULL) { | 
					
						
							| 
									
										
										
										
											2018-01-29 23:43:37 +00:00
										 |  |  | 		SPDK_ERRLOG("No lun name specified\n"); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-16 22:06:48 +00:00
										 |  |  | 	state = &svdev->scsi_dev_state[scsi_tgt_num]; | 
					
						
							|  |  |  | 	if (state->dev != NULL) { | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 		SPDK_ERRLOG("%s: SCSI target %u already occupied\n", vdev->name, scsi_tgt_num); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 		return -EEXIST; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/*
 | 
					
						
							| 
									
										
										
										
											2017-11-22 11:20:03 +00:00
										 |  |  | 	 * At this stage only one LUN per target | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2017-11-22 11:20:03 +00:00
										 |  |  | 	snprintf(target_name, sizeof(target_name), "Target %u", scsi_tgt_num); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	lun_id_list[0] = 0; | 
					
						
							| 
									
										
										
										
											2018-01-10 05:55:53 +00:00
										 |  |  | 	bdev_names_list[0] = (char *)bdev_name; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-27 15:26:22 +00:00
										 |  |  | 	state->status = VHOST_SCSI_DEV_ADDING; | 
					
						
							| 
									
										
										
										
											2020-09-25 10:17:05 +00:00
										 |  |  | 	state->dev = spdk_scsi_dev_construct_ext(target_name, bdev_names_list, lun_id_list, 1, | 
					
						
							|  |  |  | 			SPDK_SPC_PROTOCOL_IDENTIFIER_SAS, | 
					
						
							|  |  |  | 			vhost_scsi_lun_resize, svdev, | 
					
						
							|  |  |  | 			vhost_scsi_lun_hotremove, svdev); | 
					
						
							| 
									
										
										
										
											2017-07-25 11:55:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-16 22:06:48 +00:00
										 |  |  | 	if (state->dev == NULL) { | 
					
						
							| 
									
										
										
										
											2019-03-04 15:20:21 +00:00
										 |  |  | 		state->status = VHOST_SCSI_DEV_EMPTY; | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 		SPDK_ERRLOG("%s: couldn't create SCSI target %u using bdev '%s'\n", | 
					
						
							|  |  |  | 			    vdev->name, scsi_tgt_num, bdev_name); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-12-16 22:06:48 +00:00
										 |  |  | 	spdk_scsi_dev_add_port(state->dev, 0, "vhost"); | 
					
						
							| 
									
										
										
										
											2017-07-26 10:17:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | 	SPDK_INFOLOG(vhost, "%s: added SCSI target %u using bdev '%s'\n", | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 		     vdev->name, scsi_tgt_num, bdev_name); | 
					
						
							| 
									
										
										
										
											2018-06-29 11:53:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-11 12:11:22 +00:00
										 |  |  | 	vhost_user_dev_foreach_session(vdev, vhost_scsi_session_add_tgt, | 
					
						
							|  |  |  | 				       vhost_scsi_dev_add_tgt_cpl_cb, | 
					
						
							|  |  |  | 				       (void *)(uintptr_t)scsi_tgt_num); | 
					
						
							| 
									
										
										
										
											2019-01-28 21:30:36 +00:00
										 |  |  | 	return scsi_tgt_num; | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-27 17:23:49 +00:00
										 |  |  | struct scsi_tgt_hotplug_ctx { | 
					
						
							|  |  |  | 	unsigned scsi_tgt_num; | 
					
						
							|  |  |  | 	bool async_fini; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 21:30:56 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | vhost_scsi_dev_remove_tgt_cpl_cb(struct spdk_vhost_dev *vdev, void *_ctx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct scsi_tgt_hotplug_ctx *ctx = _ctx; | 
					
						
							|  |  |  | 	struct spdk_vhost_scsi_dev *svdev = SPDK_CONTAINEROF(vdev, | 
					
						
							|  |  |  | 					    struct spdk_vhost_scsi_dev, vdev); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!ctx->async_fini) { | 
					
						
							|  |  |  | 		/* there aren't any active sessions, so remove the dev and exit */ | 
					
						
							|  |  |  | 		remove_scsi_tgt(svdev, ctx->scsi_tgt_num); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	free(ctx); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | vhost_scsi_session_remove_tgt(struct spdk_vhost_dev *vdev, | 
					
						
							|  |  |  | 			      struct spdk_vhost_session *vsession, void *_ctx) | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-03-27 17:23:49 +00:00
										 |  |  | 	struct scsi_tgt_hotplug_ctx *ctx = _ctx; | 
					
						
							|  |  |  | 	unsigned scsi_tgt_num = ctx->scsi_tgt_num; | 
					
						
							| 
									
										
										
										
											2019-07-20 21:47:55 +00:00
										 |  |  | 	struct spdk_vhost_scsi_session *svsession = (struct spdk_vhost_scsi_session *)vsession; | 
					
						
							|  |  |  | 	struct spdk_scsi_dev_session_state *state = &svsession->scsi_dev_state[scsi_tgt_num]; | 
					
						
							| 
									
										
										
										
											2019-03-27 17:23:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-29 07:26:10 +00:00
										 |  |  | 	if (!vsession->started || state->dev == NULL) { | 
					
						
							| 
									
										
										
										
											2019-03-27 16:20:39 +00:00
										 |  |  | 		/* Nothing to do */ | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | 	/* Mark the target for removal */ | 
					
						
							| 
									
										
										
										
											2019-03-04 15:20:21 +00:00
										 |  |  | 	assert(state->status == VHOST_SCSI_DEV_PRESENT); | 
					
						
							|  |  |  | 	state->status = VHOST_SCSI_DEV_REMOVING; | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-25 10:17:05 +00:00
										 |  |  | 	/* Send a hotremove virtio event */ | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 	if (vhost_dev_has_feature(vsession, VIRTIO_SCSI_F_HOTPLUG)) { | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | 		eventq_enqueue(svsession, scsi_tgt_num, | 
					
						
							|  |  |  | 			       VIRTIO_SCSI_T_TRANSPORT_RESET, VIRTIO_SCSI_EVT_RESET_REMOVED); | 
					
						
							| 
									
										
										
										
											2018-06-29 11:53:04 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-27 17:23:49 +00:00
										 |  |  | 	/* Wait for the session's management poller to remove the target after
 | 
					
						
							|  |  |  | 	 * all its pending I/O has finished. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	ctx->async_fini = true; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							| 
									
										
										
										
											2017-11-22 11:20:03 +00:00
										 |  |  | spdk_vhost_scsi_dev_remove_tgt(struct spdk_vhost_dev *vdev, unsigned scsi_tgt_num, | 
					
						
							| 
									
										
										
										
											2017-07-25 13:42:27 +00:00
										 |  |  | 			       spdk_vhost_event_fn cb_fn, void *cb_arg) | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-06-02 16:00:22 +00:00
										 |  |  | 	struct spdk_vhost_scsi_dev *svdev; | 
					
						
							| 
									
										
										
										
											2017-07-25 13:42:27 +00:00
										 |  |  | 	struct spdk_scsi_dev_vhost_state *scsi_dev_state; | 
					
						
							| 
									
										
										
										
											2019-03-27 17:23:49 +00:00
										 |  |  | 	struct scsi_tgt_hotplug_ctx *ctx; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-22 11:20:03 +00:00
										 |  |  | 	if (scsi_tgt_num >= SPDK_VHOST_SCSI_CTRLR_MAX_DEVS) { | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 		SPDK_ERRLOG("%s: invalid SCSI target number %d\n", vdev->name, scsi_tgt_num); | 
					
						
							| 
									
										
										
										
											2017-06-22 17:29:21 +00:00
										 |  |  | 		return -EINVAL; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-02 16:00:22 +00:00
										 |  |  | 	svdev = to_scsi_dev(vdev); | 
					
						
							| 
									
										
										
										
											2020-08-27 17:22:09 +00:00
										 |  |  | 	if (!svdev) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("An invalid SCSI device that removing from a SCSI target."); | 
					
						
							|  |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-16 22:06:48 +00:00
										 |  |  | 	scsi_dev_state = &svdev->scsi_dev_state[scsi_tgt_num]; | 
					
						
							| 
									
										
										
										
											2020-04-28 21:20:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (scsi_dev_state->status != VHOST_SCSI_DEV_PRESENT) { | 
					
						
							|  |  |  | 		return -EBUSY; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-27 15:26:22 +00:00
										 |  |  | 	if (scsi_dev_state->dev == NULL || scsi_dev_state->status == VHOST_SCSI_DEV_ADDING) { | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 		SPDK_ERRLOG("%s: SCSI target %u is not occupied\n", vdev->name, scsi_tgt_num); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 		return -ENODEV; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-04 15:20:21 +00:00
										 |  |  | 	assert(scsi_dev_state->status != VHOST_SCSI_DEV_EMPTY); | 
					
						
							| 
									
										
										
										
											2019-03-27 17:23:49 +00:00
										 |  |  | 	ctx = calloc(1, sizeof(*ctx)); | 
					
						
							|  |  |  | 	if (ctx == NULL) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("calloc failed\n"); | 
					
						
							|  |  |  | 		return -ENOMEM; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ctx->scsi_tgt_num = scsi_tgt_num; | 
					
						
							|  |  |  | 	ctx->async_fini = false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-25 13:42:27 +00:00
										 |  |  | 	scsi_dev_state->remove_cb = cb_fn; | 
					
						
							|  |  |  | 	scsi_dev_state->remove_ctx = cb_arg; | 
					
						
							| 
									
										
										
										
											2019-03-04 15:20:21 +00:00
										 |  |  | 	scsi_dev_state->status = VHOST_SCSI_DEV_REMOVING; | 
					
						
							| 
									
										
										
										
											2018-12-16 22:53:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-11 12:11:22 +00:00
										 |  |  | 	vhost_user_dev_foreach_session(vdev, vhost_scsi_session_remove_tgt, | 
					
						
							|  |  |  | 				       vhost_scsi_dev_remove_tgt_cpl_cb, ctx); | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-25 10:17:05 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | vhost_scsi_session_param_changed(struct spdk_vhost_dev *vdev, | 
					
						
							|  |  |  | 				 struct spdk_vhost_session *vsession, void *ctx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unsigned scsi_tgt_num = (unsigned)(uintptr_t)ctx; | 
					
						
							|  |  |  | 	struct spdk_vhost_scsi_session *svsession = (struct spdk_vhost_scsi_session *)vsession; | 
					
						
							|  |  |  | 	struct spdk_scsi_dev_session_state *state = &svsession->scsi_dev_state[scsi_tgt_num]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!vsession->started || state->dev == NULL) { | 
					
						
							|  |  |  | 		/* Nothing to do */ | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Send a parameter change virtio event */ | 
					
						
							|  |  |  | 	if (vhost_dev_has_feature(vsession, VIRTIO_SCSI_F_CHANGE)) { | 
					
						
							|  |  |  | 		/*
 | 
					
						
							|  |  |  | 		 * virtio 1.0 spec says: | 
					
						
							|  |  |  | 		 * By sending this event, the device signals a change in the configuration | 
					
						
							|  |  |  | 		 * parameters of a logical unit, for example the capacity or cache mode. | 
					
						
							|  |  |  | 		 * event is set to VIRTIO_SCSI_T_PARAM_CHANGE. lun addresses a logical unit | 
					
						
							|  |  |  | 		 * in the SCSI host. The same event SHOULD also be reported as a unit | 
					
						
							|  |  |  | 		 * attention condition. reason contains the additional sense code and | 
					
						
							|  |  |  | 		 * additional sense code qualifier, respectively in bits 0…7 and 8…15. | 
					
						
							|  |  |  | 		 * Note: For example, a change in * capacity will be reported as asc | 
					
						
							|  |  |  | 		 * 0x2a, ascq 0x09 (CAPACITY DATA HAS CHANGED). | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 		eventq_enqueue(svsession, scsi_tgt_num, VIRTIO_SCSI_T_PARAM_CHANGE, 0x2a | (0x09 << 8)); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | vhost_scsi_dev_param_changed(struct spdk_vhost_dev *vdev, unsigned scsi_tgt_num) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct spdk_vhost_scsi_dev *svdev; | 
					
						
							|  |  |  | 	struct spdk_scsi_dev_vhost_state *scsi_dev_state; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (scsi_tgt_num >= SPDK_VHOST_SCSI_CTRLR_MAX_DEVS) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("%s: invalid SCSI target number %d\n", vdev->name, scsi_tgt_num); | 
					
						
							|  |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	svdev = to_scsi_dev(vdev); | 
					
						
							|  |  |  | 	if (!svdev) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("An invalid SCSI device that removing from a SCSI target."); | 
					
						
							|  |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	scsi_dev_state = &svdev->scsi_dev_state[scsi_tgt_num]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (scsi_dev_state->status != VHOST_SCSI_DEV_PRESENT) { | 
					
						
							|  |  |  | 		return -EBUSY; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (scsi_dev_state->dev == NULL || scsi_dev_state->status == VHOST_SCSI_DEV_ADDING) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("%s: SCSI target %u is not occupied\n", vdev->name, scsi_tgt_num); | 
					
						
							|  |  |  | 		return -ENODEV; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	assert(scsi_dev_state->status != VHOST_SCSI_DEV_EMPTY); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-11 12:11:22 +00:00
										 |  |  | 	vhost_user_dev_foreach_session(vdev, vhost_scsi_session_param_changed, | 
					
						
							|  |  |  | 				       NULL, (void *)(uintptr_t)scsi_tgt_num); | 
					
						
							| 
									
										
										
										
											2020-09-25 10:17:05 +00:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-07 14:10:41 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2018-12-13 15:34:15 +00:00
										 |  |  | free_task_pool(struct spdk_vhost_scsi_session *svsession) | 
					
						
							| 
									
										
										
										
											2017-07-07 14:10:41 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-12-13 15:34:15 +00:00
										 |  |  | 	struct spdk_vhost_session *vsession = &svsession->vsession; | 
					
						
							| 
									
										
										
										
											2017-10-10 20:19:40 +00:00
										 |  |  | 	struct spdk_vhost_virtqueue *vq; | 
					
						
							|  |  |  | 	uint16_t i; | 
					
						
							| 
									
										
										
										
											2017-07-07 14:10:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-13 10:51:34 +00:00
										 |  |  | 	for (i = 0; i < vsession->max_queues; i++) { | 
					
						
							|  |  |  | 		vq = &vsession->virtqueue[i]; | 
					
						
							| 
									
										
										
										
											2017-10-10 20:19:40 +00:00
										 |  |  | 		if (vq->tasks == NULL) { | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-07-07 14:10:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-08 21:39:30 +00:00
										 |  |  | 		spdk_free(vq->tasks); | 
					
						
							| 
									
										
										
										
											2017-10-10 20:19:40 +00:00
										 |  |  | 		vq->tasks = NULL; | 
					
						
							| 
									
										
										
										
											2017-07-07 14:10:41 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2018-12-13 15:34:15 +00:00
										 |  |  | alloc_task_pool(struct spdk_vhost_scsi_session *svsession) | 
					
						
							| 
									
										
										
										
											2017-07-07 14:10:41 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-12-13 15:34:15 +00:00
										 |  |  | 	struct spdk_vhost_session *vsession = &svsession->vsession; | 
					
						
							| 
									
										
										
										
											2017-10-10 20:19:40 +00:00
										 |  |  | 	struct spdk_vhost_virtqueue *vq; | 
					
						
							| 
									
										
										
										
											2017-07-11 21:36:13 +00:00
										 |  |  | 	struct spdk_vhost_scsi_task *task; | 
					
						
							| 
									
										
										
										
											2017-10-10 20:19:40 +00:00
										 |  |  | 	uint32_t task_cnt; | 
					
						
							| 
									
										
										
										
											2017-07-07 14:10:41 +00:00
										 |  |  | 	uint16_t i; | 
					
						
							| 
									
										
										
										
											2017-10-10 20:19:40 +00:00
										 |  |  | 	uint32_t j; | 
					
						
							| 
									
										
										
										
											2017-07-07 14:10:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-13 10:51:34 +00:00
										 |  |  | 	for (i = 0; i < vsession->max_queues; i++) { | 
					
						
							|  |  |  | 		vq = &vsession->virtqueue[i]; | 
					
						
							| 
									
										
										
										
											2018-04-23 11:04:18 +00:00
										 |  |  | 		if (vq->vring.desc == NULL) { | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-10 20:19:40 +00:00
										 |  |  | 		task_cnt = vq->vring.size; | 
					
						
							|  |  |  | 		if (task_cnt > SPDK_VHOST_MAX_VQ_SIZE) { | 
					
						
							|  |  |  | 			/* sanity check */ | 
					
						
							| 
									
										
										
										
											2021-11-25 01:40:58 +00:00
										 |  |  | 			SPDK_ERRLOG("%s: virtqueue %"PRIu16" is too big. (size = %"PRIu32", max = %"PRIu32")\n", | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 				    vsession->name, i, task_cnt, SPDK_VHOST_MAX_VQ_SIZE); | 
					
						
							| 
									
										
										
										
											2018-12-13 15:34:15 +00:00
										 |  |  | 			free_task_pool(svsession); | 
					
						
							| 
									
										
										
										
											2017-07-11 22:18:36 +00:00
										 |  |  | 			return -1; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-03-15 14:35:49 +00:00
										 |  |  | 		vq->tasks = spdk_zmalloc(sizeof(struct spdk_vhost_scsi_task) * task_cnt, | 
					
						
							|  |  |  | 					 SPDK_CACHE_LINE_SIZE, NULL, | 
					
						
							|  |  |  | 					 SPDK_ENV_LCORE_ID_ANY, SPDK_MALLOC_DMA); | 
					
						
							| 
									
										
										
										
											2017-10-10 20:19:40 +00:00
										 |  |  | 		if (vq->tasks == NULL) { | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 			SPDK_ERRLOG("%s: failed to allocate %"PRIu32" tasks for virtqueue %"PRIu16"\n", | 
					
						
							|  |  |  | 				    vsession->name, task_cnt, i); | 
					
						
							| 
									
										
										
										
											2018-12-13 15:34:15 +00:00
										 |  |  | 			free_task_pool(svsession); | 
					
						
							| 
									
										
										
										
											2017-07-07 14:10:41 +00:00
										 |  |  | 			return -1; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-10-10 20:19:40 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		for (j = 0; j < task_cnt; j++) { | 
					
						
							|  |  |  | 			task = &((struct spdk_vhost_scsi_task *)vq->tasks)[j]; | 
					
						
							| 
									
										
										
										
											2018-12-15 23:54:53 +00:00
										 |  |  | 			task->svsession = svsession; | 
					
						
							| 
									
										
										
										
											2017-10-10 20:19:40 +00:00
										 |  |  | 			task->vq = vq; | 
					
						
							|  |  |  | 			task->req_idx = j; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-07-07 14:10:41 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | vhost_scsi_start_cb(struct spdk_vhost_dev *vdev, | 
					
						
							|  |  |  | 		    struct spdk_vhost_session *vsession, void *unused) | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-07-20 15:03:04 +00:00
										 |  |  | 	struct spdk_vhost_scsi_session *svsession = to_scsi_session(vsession); | 
					
						
							|  |  |  | 	struct spdk_vhost_scsi_dev *svdev = svsession->svdev; | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | 	struct spdk_scsi_dev_vhost_state *state; | 
					
						
							| 
									
										
										
										
											2017-08-04 10:02:03 +00:00
										 |  |  | 	uint32_t i; | 
					
						
							| 
									
										
										
										
											2017-07-25 11:55:42 +00:00
										 |  |  | 	int rc; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-23 11:04:18 +00:00
										 |  |  | 	/* validate all I/O queues are in a contiguous index range */ | 
					
						
							| 
									
										
										
										
											2018-12-13 10:51:34 +00:00
										 |  |  | 	for (i = VIRTIO_SCSI_REQUESTQ; i < vsession->max_queues; i++) { | 
					
						
							|  |  |  | 		if (vsession->virtqueue[i].vring.desc == NULL) { | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 			SPDK_ERRLOG("%s: queue %"PRIu32" is empty\n", vsession->name, i); | 
					
						
							| 
									
										
										
										
											2018-04-23 11:04:18 +00:00
										 |  |  | 			rc = -1; | 
					
						
							|  |  |  | 			goto out; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-13 15:34:15 +00:00
										 |  |  | 	rc = alloc_task_pool(svsession); | 
					
						
							| 
									
										
										
										
											2017-07-05 10:13:22 +00:00
										 |  |  | 	if (rc != 0) { | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 		SPDK_ERRLOG("%s: failed to alloc task pool.\n", vsession->name); | 
					
						
							| 
									
										
										
										
											2017-08-04 10:02:03 +00:00
										 |  |  | 		goto out; | 
					
						
							| 
									
										
										
										
											2017-07-05 10:13:22 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-04 10:02:03 +00:00
										 |  |  | 	for (i = 0; i < SPDK_VHOST_SCSI_CTRLR_MAX_DEVS; i++) { | 
					
						
							| 
									
										
										
										
											2018-12-17 02:09:30 +00:00
										 |  |  | 		state = &svdev->scsi_dev_state[i]; | 
					
						
							| 
									
										
										
										
											2019-03-27 17:23:49 +00:00
										 |  |  | 		if (state->dev == NULL || state->status == VHOST_SCSI_DEV_REMOVING) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:02:03 +00:00
										 |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-03-27 17:23:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-27 16:20:39 +00:00
										 |  |  | 		assert(svsession->scsi_dev_state[i].status == VHOST_SCSI_DEV_EMPTY); | 
					
						
							| 
									
										
										
										
											2019-03-05 08:00:30 +00:00
										 |  |  | 		svsession->scsi_dev_state[i].dev = state->dev; | 
					
						
							| 
									
										
										
										
											2019-04-03 17:13:41 +00:00
										 |  |  | 		svsession->scsi_dev_state[i].status = VHOST_SCSI_DEV_PRESENT; | 
					
						
							| 
									
										
										
										
											2019-02-15 10:29:09 +00:00
										 |  |  | 		rc = spdk_scsi_dev_allocate_io_channels(state->dev); | 
					
						
							|  |  |  | 		if (rc != 0) { | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 			SPDK_ERRLOG("%s: failed to alloc io_channel for SCSI target %"PRIu32"\n", | 
					
						
							|  |  |  | 				    vsession->name, i); | 
					
						
							| 
									
										
										
										
											2019-02-15 10:29:09 +00:00
										 |  |  | 			/* unset the SCSI target so that all I/O to it will be rejected */ | 
					
						
							|  |  |  | 			svsession->scsi_dev_state[i].dev = NULL; | 
					
						
							| 
									
										
										
										
											2019-03-04 15:20:21 +00:00
										 |  |  | 			/* set EMPTY state so that we won't reply with SCSI hotremove
 | 
					
						
							| 
									
										
										
										
											2019-02-15 10:29:09 +00:00
										 |  |  | 			 * sense codes - the device hasn't ever been added. | 
					
						
							|  |  |  | 			 */ | 
					
						
							| 
									
										
										
										
											2019-03-04 15:20:21 +00:00
										 |  |  | 			svsession->scsi_dev_state[i].status = VHOST_SCSI_DEV_EMPTY; | 
					
						
							| 
									
										
										
										
											2019-02-15 10:29:09 +00:00
										 |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-08-04 10:02:03 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | 	SPDK_INFOLOG(vhost, "%s: started poller on lcore %d\n", | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 		     vsession->name, spdk_env_get_current_core()); | 
					
						
							| 
									
										
										
										
											2017-08-04 10:02:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-14 06:49:46 +00:00
										 |  |  | 	svsession->requestq_poller = SPDK_POLLER_REGISTER(vdev_worker, svsession, 0); | 
					
						
							| 
									
										
										
										
											2020-11-18 09:11:17 +00:00
										 |  |  | 	svsession->mgmt_poller = SPDK_POLLER_REGISTER(vdev_mgmt_worker, svsession, | 
					
						
							|  |  |  | 				 MGMT_POLL_PERIOD_US); | 
					
						
							| 
									
										
										
										
											2017-08-04 10:02:03 +00:00
										 |  |  | out: | 
					
						
							| 
									
										
										
										
											2022-01-11 12:11:22 +00:00
										 |  |  | 	vhost_user_session_start_done(vsession, rc); | 
					
						
							| 
									
										
										
										
											2017-08-04 10:02:03 +00:00
										 |  |  | 	return rc; | 
					
						
							| 
									
										
										
										
											2017-05-22 12:53:27 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-22 11:57:09 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | vhost_scsi_start(struct spdk_vhost_session *vsession) | 
					
						
							| 
									
										
										
										
											2019-01-22 11:57:09 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-07-20 15:03:04 +00:00
										 |  |  | 	struct spdk_vhost_scsi_session *svsession = to_scsi_session(vsession); | 
					
						
							| 
									
										
										
										
											2019-01-22 15:25:35 +00:00
										 |  |  | 	struct spdk_vhost_scsi_dev *svdev; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	svdev = to_scsi_dev(vsession->vdev); | 
					
						
							| 
									
										
										
										
											2019-01-25 11:55:37 +00:00
										 |  |  | 	assert(svdev != NULL); | 
					
						
							| 
									
										
										
										
											2019-01-22 15:25:35 +00:00
										 |  |  | 	svsession->svdev = svdev; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-11 12:11:22 +00:00
										 |  |  | 	return vhost_user_session_send_event(vsession, vhost_scsi_start_cb, | 
					
						
							|  |  |  | 					     3, "start session"); | 
					
						
							| 
									
										
										
										
											2019-01-22 11:57:09 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-13 00:16:47 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2018-12-13 15:34:15 +00:00
										 |  |  | destroy_session_poller_cb(void *arg) | 
					
						
							| 
									
										
										
										
											2017-08-04 10:02:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-12-13 15:34:15 +00:00
										 |  |  | 	struct spdk_vhost_scsi_session *svsession = arg; | 
					
						
							|  |  |  | 	struct spdk_vhost_session *vsession = &svsession->vsession; | 
					
						
							| 
									
										
										
										
											2019-03-05 08:00:30 +00:00
										 |  |  | 	struct spdk_scsi_dev_session_state *state; | 
					
						
							| 
									
										
										
										
											2017-06-12 10:17:50 +00:00
										 |  |  | 	uint32_t i; | 
					
						
							| 
									
										
										
										
											2017-05-24 11:16:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-24 20:05:55 +00:00
										 |  |  | 	if (vsession->task_cnt > 0 || spdk_vhost_trylock() != 0) { | 
					
						
							|  |  |  | 		assert(vsession->stop_retry_count > 0); | 
					
						
							|  |  |  | 		vsession->stop_retry_count--; | 
					
						
							|  |  |  | 		if (vsession->stop_retry_count == 0) { | 
					
						
							|  |  |  | 			SPDK_ERRLOG("%s: Timedout when destroy session (task_cnt %d)\n", vsession->name, | 
					
						
							|  |  |  | 				    vsession->task_cnt); | 
					
						
							|  |  |  | 			spdk_poller_unregister(&svsession->stop_poller); | 
					
						
							| 
									
										
										
										
											2022-01-11 12:11:22 +00:00
										 |  |  | 			vhost_user_session_stop_done(vsession, -ETIMEDOUT); | 
					
						
							| 
									
										
										
										
											2021-11-24 20:05:55 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-06-12 10:17:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-04 09:51:27 +00:00
										 |  |  | 		return SPDK_POLLER_BUSY; | 
					
						
							| 
									
										
										
										
											2019-03-16 23:50:51 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-09-11 17:45:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-13 10:51:34 +00:00
										 |  |  | 	for (i = 0; i < vsession->max_queues; i++) { | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | 		vhost_vq_used_signal(vsession, &vsession->virtqueue[i]); | 
					
						
							| 
									
										
										
										
											2017-09-11 17:45:56 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-04 10:02:03 +00:00
										 |  |  | 	for (i = 0; i < SPDK_VHOST_SCSI_CTRLR_MAX_DEVS; i++) { | 
					
						
							| 
									
										
										
										
											2019-03-27 16:20:39 +00:00
										 |  |  | 		enum spdk_scsi_dev_vhost_status prev_status; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-08 00:10:53 +00:00
										 |  |  | 		state = &svsession->scsi_dev_state[i]; | 
					
						
							| 
									
										
										
										
											2019-03-31 22:11:56 +00:00
										 |  |  | 		/* clear the REMOVED status so that we won't send hotremove events anymore */ | 
					
						
							|  |  |  | 		prev_status = state->status; | 
					
						
							|  |  |  | 		state->status = VHOST_SCSI_DEV_EMPTY; | 
					
						
							| 
									
										
										
										
											2019-01-08 00:10:53 +00:00
										 |  |  | 		if (state->dev == NULL) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:02:03 +00:00
										 |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-12-13 15:34:15 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-08 00:10:53 +00:00
										 |  |  | 		spdk_scsi_dev_free_io_channels(state->dev); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-27 16:20:39 +00:00
										 |  |  | 		state->dev = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (prev_status == VHOST_SCSI_DEV_REMOVING) { | 
					
						
							| 
									
										
										
										
											2019-01-08 00:10:53 +00:00
										 |  |  | 			/* try to detach it globally */ | 
					
						
							| 
									
										
										
										
											2022-01-11 12:11:22 +00:00
										 |  |  | 			vhost_user_dev_foreach_session(vsession->vdev, | 
					
						
							|  |  |  | 						       vhost_scsi_session_process_removed, | 
					
						
							|  |  |  | 						       vhost_scsi_dev_process_removed_cpl_cb, | 
					
						
							|  |  |  | 						       (void *)(uintptr_t)i); | 
					
						
							| 
									
										
										
										
											2019-01-08 00:10:53 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-06-12 10:17:50 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | 	SPDK_INFOLOG(vhost, "%s: stopping poller on lcore %d\n", | 
					
						
							| 
									
										
										
										
											2019-07-20 13:47:06 +00:00
										 |  |  | 		     vsession->name, spdk_env_get_current_core()); | 
					
						
							| 
									
										
										
										
											2017-05-24 11:16:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-13 15:34:15 +00:00
										 |  |  | 	free_task_pool(svsession); | 
					
						
							| 
									
										
										
										
											2017-08-04 10:02:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-16 23:31:45 +00:00
										 |  |  | 	spdk_poller_unregister(&svsession->stop_poller); | 
					
						
							| 
									
										
										
										
											2022-01-11 12:11:22 +00:00
										 |  |  | 	vhost_user_session_stop_done(vsession, 0); | 
					
						
							| 
									
										
										
										
											2018-03-13 00:16:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-16 23:50:51 +00:00
										 |  |  | 	spdk_vhost_unlock(); | 
					
						
							| 
									
										
										
										
											2020-05-04 09:51:27 +00:00
										 |  |  | 	return SPDK_POLLER_BUSY; | 
					
						
							| 
									
										
										
										
											2017-08-04 10:02:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | vhost_scsi_stop_cb(struct spdk_vhost_dev *vdev, | 
					
						
							|  |  |  | 		   struct spdk_vhost_session *vsession, void *unused) | 
					
						
							| 
									
										
										
										
											2017-08-04 10:02:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-07-20 15:03:04 +00:00
										 |  |  | 	struct spdk_vhost_scsi_session *svsession = to_scsi_session(vsession); | 
					
						
							| 
									
										
										
										
											2019-01-08 00:10:53 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Stop receiving new I/O requests */ | 
					
						
							| 
									
										
										
										
											2018-12-13 15:34:15 +00:00
										 |  |  | 	spdk_poller_unregister(&svsession->requestq_poller); | 
					
						
							| 
									
										
										
										
											2019-01-08 00:10:53 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Stop receiving controlq requests, also stop processing the
 | 
					
						
							|  |  |  | 	 * asynchronous hotremove events. All the remaining events | 
					
						
							|  |  |  | 	 * will be finalized by the stop_poller below. | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2018-12-13 15:34:15 +00:00
										 |  |  | 	spdk_poller_unregister(&svsession->mgmt_poller); | 
					
						
							| 
									
										
										
										
											2019-01-08 00:10:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-11 12:11:22 +00:00
										 |  |  | 	/* vhost_user_session_send_event timeout is 3 seconds, here set retry within 4 seconds */ | 
					
						
							| 
									
										
										
										
											2021-11-24 20:05:55 +00:00
										 |  |  | 	svsession->vsession.stop_retry_count = 4000; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-08 00:10:53 +00:00
										 |  |  | 	/* Wait for all pending I/Os to complete, then process all the
 | 
					
						
							|  |  |  | 	 * remaining hotremove events one last time. | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2020-04-14 06:49:46 +00:00
										 |  |  | 	svsession->stop_poller = SPDK_POLLER_REGISTER(destroy_session_poller_cb, | 
					
						
							| 
									
										
										
										
											2019-03-16 23:31:45 +00:00
										 |  |  | 				 svsession, 1000); | 
					
						
							| 
									
										
										
										
											2017-08-04 10:02:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-01 13:46:51 +00:00
										 |  |  | 	return 0; | 
					
						
							| 
									
										
										
										
											2017-05-24 11:16:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-22 11:57:09 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | vhost_scsi_stop(struct spdk_vhost_session *vsession) | 
					
						
							| 
									
										
										
										
											2019-01-22 11:57:09 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2022-01-11 12:11:22 +00:00
										 |  |  | 	return vhost_user_session_send_event(vsession, vhost_scsi_stop_cb, | 
					
						
							|  |  |  | 					     3, "stop session"); | 
					
						
							| 
									
										
										
										
											2019-01-22 11:57:09 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-13 14:37:50 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | vhost_scsi_dump_info_json(struct spdk_vhost_dev *vdev, struct spdk_json_write_ctx *w) | 
					
						
							| 
									
										
										
										
											2017-07-13 14:37:50 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct spdk_scsi_dev *sdev; | 
					
						
							|  |  |  | 	struct spdk_scsi_lun *lun; | 
					
						
							|  |  |  | 	uint32_t dev_idx; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	assert(vdev != NULL); | 
					
						
							| 
									
										
										
										
											2019-02-01 05:45:08 +00:00
										 |  |  | 	spdk_json_write_named_array_begin(w, "scsi"); | 
					
						
							| 
									
										
										
										
											2017-07-13 14:37:50 +00:00
										 |  |  | 	for (dev_idx = 0; dev_idx < SPDK_VHOST_SCSI_CTRLR_MAX_DEVS; dev_idx++) { | 
					
						
							| 
									
										
										
										
											2017-11-22 11:20:03 +00:00
										 |  |  | 		sdev = spdk_vhost_scsi_dev_get_tgt(vdev, dev_idx); | 
					
						
							| 
									
										
										
										
											2017-07-13 14:37:50 +00:00
										 |  |  | 		if (!sdev) { | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-13 16:42:09 +00:00
										 |  |  | 		spdk_json_write_object_begin(w); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-01 05:45:08 +00:00
										 |  |  | 		spdk_json_write_named_uint32(w, "scsi_dev_num", dev_idx); | 
					
						
							| 
									
										
										
										
											2017-07-13 14:37:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-01 05:45:08 +00:00
										 |  |  | 		spdk_json_write_named_uint32(w, "id", spdk_scsi_dev_get_id(sdev)); | 
					
						
							| 
									
										
										
										
											2017-07-13 14:37:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-01 05:45:08 +00:00
										 |  |  | 		spdk_json_write_named_string(w, "target_name", spdk_scsi_dev_get_name(sdev)); | 
					
						
							| 
									
										
										
										
											2017-07-13 14:37:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-01 05:45:08 +00:00
										 |  |  | 		spdk_json_write_named_array_begin(w, "luns"); | 
					
						
							| 
									
										
										
										
											2017-07-13 14:37:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-20 17:31:14 +00:00
										 |  |  | 		for (lun = spdk_scsi_dev_get_first_lun(sdev); lun != NULL; | 
					
						
							|  |  |  | 		     lun = spdk_scsi_dev_get_next_lun(lun)) { | 
					
						
							| 
									
										
										
										
											2017-07-13 14:37:50 +00:00
										 |  |  | 			spdk_json_write_object_begin(w); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-01 05:45:08 +00:00
										 |  |  | 			spdk_json_write_named_int32(w, "id", spdk_scsi_lun_get_id(lun)); | 
					
						
							| 
									
										
										
										
											2017-07-13 14:37:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-01 05:45:08 +00:00
										 |  |  | 			spdk_json_write_named_string(w, "bdev_name", spdk_scsi_lun_get_bdev_name(lun)); | 
					
						
							| 
									
										
										
										
											2017-07-13 14:37:50 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			spdk_json_write_object_end(w); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		spdk_json_write_array_end(w); | 
					
						
							| 
									
										
										
										
											2017-12-13 16:42:09 +00:00
										 |  |  | 		spdk_json_write_object_end(w); | 
					
						
							| 
									
										
										
										
											2017-07-13 14:37:50 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-13 16:42:09 +00:00
										 |  |  | 	spdk_json_write_array_end(w); | 
					
						
							| 
									
										
										
										
											2017-07-13 14:37:50 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-19 18:01:57 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2019-07-20 21:06:19 +00:00
										 |  |  | vhost_scsi_write_config_json(struct spdk_vhost_dev *vdev, struct spdk_json_write_ctx *w) | 
					
						
							| 
									
										
										
										
											2018-03-19 18:01:57 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-03-04 11:32:18 +00:00
										 |  |  | 	struct spdk_scsi_dev *scsi_dev; | 
					
						
							| 
									
										
										
										
											2018-03-19 18:01:57 +00:00
										 |  |  | 	struct spdk_scsi_lun *lun; | 
					
						
							|  |  |  | 	uint32_t i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spdk_json_write_object_begin(w); | 
					
						
							| 
									
										
										
										
											2019-09-24 14:29:10 +00:00
										 |  |  | 	spdk_json_write_named_string(w, "method", "vhost_create_scsi_controller"); | 
					
						
							| 
									
										
										
										
											2018-03-19 18:01:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	spdk_json_write_named_object_begin(w, "params"); | 
					
						
							|  |  |  | 	spdk_json_write_named_string(w, "ctrlr", vdev->name); | 
					
						
							| 
									
										
										
										
											2020-03-05 21:36:13 +00:00
										 |  |  | 	spdk_json_write_named_string(w, "cpumask", | 
					
						
							|  |  |  | 				     spdk_cpuset_fmt(spdk_thread_get_cpumask(vdev->thread))); | 
					
						
							| 
									
										
										
										
											2018-03-19 18:01:57 +00:00
										 |  |  | 	spdk_json_write_object_end(w); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spdk_json_write_object_end(w); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-04 11:32:18 +00:00
										 |  |  | 	for (i = 0; i < SPDK_VHOST_SCSI_CTRLR_MAX_DEVS; i++) { | 
					
						
							|  |  |  | 		scsi_dev = spdk_vhost_scsi_dev_get_tgt(vdev, i); | 
					
						
							|  |  |  | 		if (scsi_dev == NULL) { | 
					
						
							| 
									
										
										
										
											2018-03-19 18:01:57 +00:00
										 |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-04 11:32:18 +00:00
										 |  |  | 		lun = spdk_scsi_dev_get_lun(scsi_dev, 0); | 
					
						
							| 
									
										
										
										
											2020-04-22 23:36:58 +00:00
										 |  |  | 		assert(lun != NULL); | 
					
						
							| 
									
										
										
										
											2018-03-19 18:01:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		spdk_json_write_object_begin(w); | 
					
						
							| 
									
										
										
										
											2019-09-27 20:53:41 +00:00
										 |  |  | 		spdk_json_write_named_string(w, "method", "vhost_scsi_controller_add_target"); | 
					
						
							| 
									
										
										
										
											2018-03-19 18:01:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		spdk_json_write_named_object_begin(w, "params"); | 
					
						
							|  |  |  | 		spdk_json_write_named_string(w, "ctrlr", vdev->name); | 
					
						
							|  |  |  | 		spdk_json_write_named_uint32(w, "scsi_target_num", i); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		spdk_json_write_named_string(w, "bdev_name", spdk_scsi_lun_get_bdev_name(lun)); | 
					
						
							|  |  |  | 		spdk_json_write_object_end(w); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		spdk_json_write_object_end(w); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | SPDK_LOG_REGISTER_COMPONENT(vhost_scsi) | 
					
						
							|  |  |  | SPDK_LOG_REGISTER_COMPONENT(vhost_scsi_queue) | 
					
						
							|  |  |  | SPDK_LOG_REGISTER_COMPONENT(vhost_scsi_data) |