| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | /*-
 | 
					
						
							|  |  |  |  *   BSD LICENSE | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *   Copyright(c) 2010-2016 Intel Corporation. All rights reserved. | 
					
						
							|  |  |  |  *   All rights reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *   Redistribution and use in source and binary forms, with or without | 
					
						
							|  |  |  |  *   modification, are permitted provided that the following conditions | 
					
						
							|  |  |  |  *   are met: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *     * Redistributions of source code must retain the above copyright | 
					
						
							|  |  |  |  *       notice, this list of conditions and the following disclaimer. | 
					
						
							|  |  |  |  *     * Redistributions in binary form must reproduce the above copyright | 
					
						
							|  |  |  |  *       notice, this list of conditions and the following disclaimer in | 
					
						
							|  |  |  |  *       the documentation and/or other materials provided with the | 
					
						
							|  |  |  |  *       distribution. | 
					
						
							|  |  |  |  *     * Neither the name of Intel Corporation nor the names of its | 
					
						
							|  |  |  |  *       contributors may be used to endorse or promote products derived | 
					
						
							|  |  |  |  *       from this software without specific prior written permission. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *   THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | 
					
						
							|  |  |  |  *   "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | 
					
						
							|  |  |  |  *   LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR | 
					
						
							|  |  |  |  *   A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT | 
					
						
							|  |  |  |  *   OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | 
					
						
							|  |  |  |  *   SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | 
					
						
							|  |  |  |  *   LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, | 
					
						
							|  |  |  |  *   DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY | 
					
						
							|  |  |  |  *   THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | 
					
						
							|  |  |  |  *   (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | 
					
						
							|  |  |  |  *   OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-18 17:14:30 +00:00
										 |  |  | #include "spdk/stdinc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-20 15:50:08 +00:00
										 |  |  | #include <sys/eventfd.h>
 | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-03 07:52:12 +00:00
										 |  |  | #include "vhost_user.h"
 | 
					
						
							| 
									
										
										
										
											2017-10-16 21:46:01 +00:00
										 |  |  | #include "spdk/string.h"
 | 
					
						
							| 
									
										
										
										
											2019-03-17 13:20:05 +00:00
										 |  |  | #include "spdk/config.h"
 | 
					
						
							| 
									
										
										
										
											2021-03-31 21:50:14 +00:00
										 |  |  | #include "spdk/util.h"
 | 
					
						
							| 
									
										
										
										
											2017-10-16 21:46:01 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-20 12:17:52 +00:00
										 |  |  | #include "spdk_internal/virtio.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-02 10:43:05 +00:00
										 |  |  | #define VIRTIO_USER_SUPPORTED_PROTOCOL_FEATURES \
 | 
					
						
							| 
									
										
										
										
											2018-07-02 11:24:43 +00:00
										 |  |  | 	((1ULL << VHOST_USER_PROTOCOL_F_MQ) | \ | 
					
						
							|  |  |  | 	(1ULL << VHOST_USER_PROTOCOL_F_CONFIG)) | 
					
						
							| 
									
										
										
										
											2018-07-02 10:43:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-03 11:44:41 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | virtio_user_create_queue(struct virtio_dev *vdev, uint32_t queue_sel) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2017-11-03 14:39:45 +00:00
										 |  |  | 	struct virtio_user_dev *dev = vdev->ctx; | 
					
						
							| 
									
										
										
										
											2017-11-03 11:44:41 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Of all per virtqueue MSGs, make sure VHOST_SET_VRING_CALL come
 | 
					
						
							|  |  |  | 	 * firstly because vhost depends on this msg to allocate virtqueue | 
					
						
							|  |  |  | 	 * pair. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	struct vhost_vring_file file; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	file.index = queue_sel; | 
					
						
							|  |  |  | 	file.fd = dev->callfds[queue_sel]; | 
					
						
							| 
									
										
										
										
											2018-03-30 18:17:23 +00:00
										 |  |  | 	return dev->ops->send_request(dev, VHOST_USER_SET_VRING_CALL, &file); | 
					
						
							| 
									
										
										
										
											2017-11-03 11:44:41 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2019-01-07 21:26:14 +00:00
										 |  |  | virtio_user_set_vring_addr(struct virtio_dev *vdev, uint32_t queue_sel) | 
					
						
							| 
									
										
										
										
											2017-11-03 11:44:41 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-11-03 14:39:45 +00:00
										 |  |  | 	struct virtio_user_dev *dev = vdev->ctx; | 
					
						
							| 
									
										
										
										
											2017-11-03 11:44:41 +00:00
										 |  |  | 	struct vring *vring = &dev->vrings[queue_sel]; | 
					
						
							|  |  |  | 	struct vhost_vring_addr addr = { | 
					
						
							|  |  |  | 		.index = queue_sel, | 
					
						
							|  |  |  | 		.desc_user_addr = (uint64_t)(uintptr_t)vring->desc, | 
					
						
							|  |  |  | 		.avail_user_addr = (uint64_t)(uintptr_t)vring->avail, | 
					
						
							|  |  |  | 		.used_user_addr = (uint64_t)(uintptr_t)vring->used, | 
					
						
							|  |  |  | 		.log_guest_addr = 0, | 
					
						
							|  |  |  | 		.flags = 0, /* disable log */ | 
					
						
							|  |  |  | 	}; | 
					
						
							| 
									
										
										
										
											2019-01-07 21:26:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return dev->ops->send_request(dev, VHOST_USER_SET_VRING_ADDR, &addr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | virtio_user_kick_queue(struct virtio_dev *vdev, uint32_t queue_sel) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct virtio_user_dev *dev = vdev->ctx; | 
					
						
							|  |  |  | 	struct vhost_vring_file file; | 
					
						
							|  |  |  | 	struct vhost_vring_state state; | 
					
						
							|  |  |  | 	struct vring *vring = &dev->vrings[queue_sel]; | 
					
						
							| 
									
										
										
										
											2018-03-30 18:17:23 +00:00
										 |  |  | 	int rc; | 
					
						
							| 
									
										
										
										
											2017-11-03 11:44:41 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	state.index = queue_sel; | 
					
						
							|  |  |  | 	state.num = vring->num; | 
					
						
							| 
									
										
										
										
											2018-03-30 18:17:23 +00:00
										 |  |  | 	rc = dev->ops->send_request(dev, VHOST_USER_SET_VRING_NUM, &state); | 
					
						
							|  |  |  | 	if (rc < 0) { | 
					
						
							|  |  |  | 		return rc; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-11-03 11:44:41 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	state.index = queue_sel; | 
					
						
							|  |  |  | 	state.num = 0; /* no reservation */ | 
					
						
							| 
									
										
										
										
											2018-03-30 18:17:23 +00:00
										 |  |  | 	rc = dev->ops->send_request(dev, VHOST_USER_SET_VRING_BASE, &state); | 
					
						
							|  |  |  | 	if (rc < 0) { | 
					
						
							|  |  |  | 		return rc; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-11-03 11:44:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-07 21:26:14 +00:00
										 |  |  | 	virtio_user_set_vring_addr(vdev, queue_sel); | 
					
						
							| 
									
										
										
										
											2017-11-03 11:44:41 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Of all per virtqueue MSGs, make sure VHOST_USER_SET_VRING_KICK comes
 | 
					
						
							|  |  |  | 	 * lastly because vhost depends on this msg to judge if | 
					
						
							|  |  |  | 	 * virtio is ready. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	file.index = queue_sel; | 
					
						
							|  |  |  | 	file.fd = dev->kickfds[queue_sel]; | 
					
						
							| 
									
										
										
										
											2018-03-30 18:17:23 +00:00
										 |  |  | 	return dev->ops->send_request(dev, VHOST_USER_SET_VRING_KICK, &file); | 
					
						
							| 
									
										
										
										
											2017-11-03 11:44:41 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | virtio_user_stop_queue(struct virtio_dev *vdev, uint32_t queue_sel) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2017-11-03 14:39:45 +00:00
										 |  |  | 	struct virtio_user_dev *dev = vdev->ctx; | 
					
						
							| 
									
										
										
										
											2017-11-03 11:44:41 +00:00
										 |  |  | 	struct vhost_vring_state state; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	state.index = queue_sel; | 
					
						
							|  |  |  | 	state.num = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-30 18:17:23 +00:00
										 |  |  | 	return dev->ops->send_request(dev, VHOST_USER_GET_VRING_BASE, &state); | 
					
						
							| 
									
										
										
										
											2017-11-03 11:44:41 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | virtio_user_queue_setup(struct virtio_dev *vdev, | 
					
						
							|  |  |  | 			int (*fn)(struct virtio_dev *, uint32_t)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	uint32_t i; | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 	int rc; | 
					
						
							| 
									
										
										
										
											2017-11-03 11:44:41 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for (i = 0; i < vdev->max_queues; ++i) { | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 		rc = fn(vdev, i); | 
					
						
							|  |  |  | 		if (rc < 0) { | 
					
						
							| 
									
										
										
										
											2017-11-03 11:44:41 +00:00
										 |  |  | 			SPDK_ERRLOG("setup tx vq fails: %"PRIu32".\n", i); | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 			return rc; | 
					
						
							| 
									
										
										
										
											2017-11-03 11:44:41 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-16 16:10:18 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | virtio_user_map_notify(void *cb_ctx, struct spdk_mem_map *map, | 
					
						
							|  |  |  | 		       enum spdk_mem_map_notify_action action, | 
					
						
							|  |  |  | 		       void *vaddr, size_t size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct virtio_dev *vdev = cb_ctx; | 
					
						
							|  |  |  | 	struct virtio_user_dev *dev = vdev->ctx; | 
					
						
							|  |  |  | 	uint64_t features; | 
					
						
							|  |  |  | 	int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* We have to resend all mappings anyway, so don't bother with any
 | 
					
						
							|  |  |  | 	 * page tracking. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	ret = dev->ops->send_request(dev, VHOST_USER_SET_MEM_TABLE, NULL); | 
					
						
							|  |  |  | 	if (ret < 0) { | 
					
						
							|  |  |  | 		return ret; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Since we might want to use that mapping straight away, we have to
 | 
					
						
							|  |  |  | 	 * make sure the guest has already processed our SET_MEM_TABLE message. | 
					
						
							|  |  |  | 	 * F_REPLY_ACK is just a feature and the host is not obliged to | 
					
						
							|  |  |  | 	 * support it, so we send a simple message that always has a response | 
					
						
							|  |  |  | 	 * and we wait for that response. Messages are always processed in order. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	return dev->ops->send_request(dev, VHOST_USER_GET_FEATURES, &features); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | virtio_user_register_mem(struct virtio_dev *vdev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct virtio_user_dev *dev = vdev->ctx; | 
					
						
							|  |  |  | 	const struct spdk_mem_map_ops virtio_user_map_ops = { | 
					
						
							|  |  |  | 		.notify_cb = virtio_user_map_notify, | 
					
						
							|  |  |  | 		.are_contiguous = NULL | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	dev->mem_map = spdk_mem_map_alloc(0, &virtio_user_map_ops, vdev); | 
					
						
							|  |  |  | 	if (dev->mem_map == NULL) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("spdk_mem_map_alloc() failed\n"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | virtio_user_unregister_mem(struct virtio_dev *vdev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct virtio_user_dev *dev = vdev->ctx; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spdk_mem_map_free(&dev->mem_map); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-03 11:44:41 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | virtio_user_start_device(struct virtio_dev *vdev) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2017-11-03 14:39:45 +00:00
										 |  |  | 	struct virtio_user_dev *dev = vdev->ctx; | 
					
						
							| 
									
										
										
										
											2017-12-27 14:51:17 +00:00
										 |  |  | 	uint64_t host_max_queues; | 
					
						
							| 
									
										
										
										
											2017-11-03 11:44:41 +00:00
										 |  |  | 	int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-02 10:54:11 +00:00
										 |  |  | 	if ((dev->protocol_features & (1ULL << VHOST_USER_PROTOCOL_F_MQ)) == 0 && | 
					
						
							|  |  |  | 	    vdev->max_queues > 1 + vdev->fixed_queues_num) { | 
					
						
							|  |  |  | 		SPDK_WARNLOG("%s: requested %"PRIu16" request queues, but the " | 
					
						
							|  |  |  | 			     "host doesn't support VHOST_USER_PROTOCOL_F_MQ. " | 
					
						
							|  |  |  | 			     "Only one request queue will be used.\n", | 
					
						
							|  |  |  | 			     vdev->name, vdev->max_queues - vdev->fixed_queues_num); | 
					
						
							|  |  |  | 		vdev->max_queues = 1 + vdev->fixed_queues_num; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-27 14:51:17 +00:00
										 |  |  | 	/* negotiate the number of I/O queues. */ | 
					
						
							|  |  |  | 	ret = dev->ops->send_request(dev, VHOST_USER_GET_QUEUE_NUM, &host_max_queues); | 
					
						
							|  |  |  | 	if (ret < 0) { | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 		return ret; | 
					
						
							| 
									
										
										
										
											2017-12-27 14:51:17 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (vdev->max_queues > host_max_queues + vdev->fixed_queues_num) { | 
					
						
							|  |  |  | 		SPDK_WARNLOG("%s: requested %"PRIu16" request queues" | 
					
						
							|  |  |  | 			     "but only %"PRIu64" available\n", | 
					
						
							|  |  |  | 			     vdev->name, vdev->max_queues - vdev->fixed_queues_num, | 
					
						
							|  |  |  | 			     host_max_queues); | 
					
						
							|  |  |  | 		vdev->max_queues = host_max_queues; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-03 11:44:41 +00:00
										 |  |  | 	/* tell vhost to create queues */ | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 	ret = virtio_user_queue_setup(vdev, virtio_user_create_queue); | 
					
						
							|  |  |  | 	if (ret < 0) { | 
					
						
							|  |  |  | 		return ret; | 
					
						
							| 
									
										
										
										
											2017-12-07 23:23:48 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-11-03 11:44:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-16 16:10:18 +00:00
										 |  |  | 	ret = virtio_user_register_mem(vdev); | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 	if (ret < 0) { | 
					
						
							|  |  |  | 		return ret; | 
					
						
							| 
									
										
										
										
											2017-12-07 23:23:48 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-11-03 11:44:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-07 21:26:14 +00:00
										 |  |  | 	return virtio_user_queue_setup(vdev, virtio_user_kick_queue); | 
					
						
							| 
									
										
										
										
											2017-11-03 11:44:41 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-16 16:10:18 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | virtio_user_stop_device(struct virtio_dev *vdev) | 
					
						
							| 
									
										
										
										
											2017-11-03 11:44:41 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-08-16 16:10:18 +00:00
										 |  |  | 	int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ret = virtio_user_queue_setup(vdev, virtio_user_stop_queue); | 
					
						
							|  |  |  | 	/* a queue might fail to stop for various reasons, e.g. socket
 | 
					
						
							|  |  |  | 	 * connection going down, but this mustn't prevent us from freeing | 
					
						
							|  |  |  | 	 * the mem map. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	virtio_user_unregister_mem(vdev); | 
					
						
							|  |  |  | 	return ret; | 
					
						
							| 
									
										
										
										
											2017-11-03 11:44:41 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | virtio_user_dev_setup(struct virtio_dev *vdev) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2017-11-03 14:39:45 +00:00
										 |  |  | 	struct virtio_user_dev *dev = vdev->ctx; | 
					
						
							| 
									
										
										
										
											2017-11-03 11:44:41 +00:00
										 |  |  | 	uint16_t i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	dev->vhostfd = -1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-20 22:22:02 +00:00
										 |  |  | 	for (i = 0; i < SPDK_VIRTIO_MAX_VIRTQUEUES; ++i) { | 
					
						
							| 
									
										
										
										
											2017-11-03 11:44:41 +00:00
										 |  |  | 		dev->callfds[i] = -1; | 
					
						
							|  |  |  | 		dev->kickfds[i] = -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	dev->ops = &ops_user; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 	return dev->ops->setup(dev); | 
					
						
							| 
									
										
										
										
											2017-11-03 11:44:41 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-02 11:13:52 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2017-09-01 17:33:53 +00:00
										 |  |  | virtio_user_read_dev_config(struct virtio_dev *vdev, size_t offset, | 
					
						
							| 
									
										
										
										
											2017-10-18 17:06:51 +00:00
										 |  |  | 			    void *dst, int length) | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-02-07 15:08:13 +00:00
										 |  |  | 	struct virtio_user_dev *dev = vdev->ctx; | 
					
						
							|  |  |  | 	struct vhost_user_config cfg = {0}; | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 	int rc; | 
					
						
							| 
									
										
										
										
											2018-02-07 15:08:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-03 07:26:33 +00:00
										 |  |  | 	if ((dev->protocol_features & (1ULL << VHOST_USER_PROTOCOL_F_CONFIG)) == 0) { | 
					
						
							|  |  |  | 		return -ENOTSUP; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-07 15:08:13 +00:00
										 |  |  | 	cfg.offset = 0; | 
					
						
							|  |  |  | 	cfg.size = VHOST_USER_MAX_CONFIG_SIZE; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 	rc = dev->ops->send_request(dev, VHOST_USER_GET_CONFIG, &cfg); | 
					
						
							|  |  |  | 	if (rc < 0) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("get_config failed: %s\n", spdk_strerror(-rc)); | 
					
						
							|  |  |  | 		return rc; | 
					
						
							| 
									
										
										
										
											2018-02-07 15:08:13 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	memcpy(dst, cfg.region + offset, length); | 
					
						
							| 
									
										
										
										
											2018-07-02 11:13:52 +00:00
										 |  |  | 	return 0; | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-02 11:13:52 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2017-09-01 17:33:53 +00:00
										 |  |  | virtio_user_write_dev_config(struct virtio_dev *vdev, size_t offset, | 
					
						
							| 
									
										
										
										
											2017-10-18 17:06:51 +00:00
										 |  |  | 			     const void *src, int length) | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-02-07 15:08:13 +00:00
										 |  |  | 	struct virtio_user_dev *dev = vdev->ctx; | 
					
						
							|  |  |  | 	struct vhost_user_config cfg = {0}; | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 	int rc; | 
					
						
							| 
									
										
										
										
											2018-02-07 15:08:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-03 07:26:33 +00:00
										 |  |  | 	if ((dev->protocol_features & (1ULL << VHOST_USER_PROTOCOL_F_CONFIG)) == 0) { | 
					
						
							|  |  |  | 		return -ENOTSUP; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-07 15:08:13 +00:00
										 |  |  | 	cfg.offset = offset; | 
					
						
							|  |  |  | 	cfg.size = length; | 
					
						
							|  |  |  | 	memcpy(cfg.region, src, length); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 	rc = dev->ops->send_request(dev, VHOST_USER_SET_CONFIG, &cfg); | 
					
						
							|  |  |  | 	if (rc < 0) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("set_config failed: %s\n", spdk_strerror(-rc)); | 
					
						
							|  |  |  | 		return rc; | 
					
						
							| 
									
										
										
										
											2018-02-07 15:08:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-07-02 11:13:52 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2017-09-01 17:33:53 +00:00
										 |  |  | virtio_user_set_status(struct virtio_dev *vdev, uint8_t status) | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-11-03 14:39:45 +00:00
										 |  |  | 	struct virtio_user_dev *dev = vdev->ctx; | 
					
						
							| 
									
										
										
										
											2018-03-30 17:24:17 +00:00
										 |  |  | 	int rc = 0; | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-30 17:24:17 +00:00
										 |  |  | 	if ((dev->status & VIRTIO_CONFIG_S_NEEDS_RESET) && | 
					
						
							|  |  |  | 	    status != VIRTIO_CONFIG_S_RESET) { | 
					
						
							|  |  |  | 		rc = -1; | 
					
						
							|  |  |  | 	} else if (status & VIRTIO_CONFIG_S_DRIVER_OK) { | 
					
						
							|  |  |  | 		rc = virtio_user_start_device(vdev); | 
					
						
							| 
									
										
										
										
											2017-10-04 10:19:04 +00:00
										 |  |  | 	} else if (status == VIRTIO_CONFIG_S_RESET && | 
					
						
							| 
									
										
										
										
											2017-10-18 17:06:51 +00:00
										 |  |  | 		   (dev->status & VIRTIO_CONFIG_S_DRIVER_OK)) { | 
					
						
							| 
									
										
										
										
											2018-03-30 17:24:17 +00:00
										 |  |  | 		rc = virtio_user_stop_device(vdev); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (rc != 0) { | 
					
						
							|  |  |  | 		dev->status |= VIRTIO_CONFIG_S_NEEDS_RESET; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		dev->status = status; | 
					
						
							| 
									
										
										
										
											2017-09-20 16:06:28 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static uint8_t | 
					
						
							| 
									
										
										
										
											2017-09-01 17:33:53 +00:00
										 |  |  | virtio_user_get_status(struct virtio_dev *vdev) | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-11-03 14:39:45 +00:00
										 |  |  | 	struct virtio_user_dev *dev = vdev->ctx; | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return dev->status; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static uint64_t | 
					
						
							| 
									
										
										
										
											2017-09-01 17:33:53 +00:00
										 |  |  | virtio_user_get_features(struct virtio_dev *vdev) | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-11-03 14:39:45 +00:00
										 |  |  | 	struct virtio_user_dev *dev = vdev->ctx; | 
					
						
							| 
									
										
										
										
											2017-10-05 12:46:24 +00:00
										 |  |  | 	uint64_t features; | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 	int rc; | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 	rc = dev->ops->send_request(dev, VHOST_USER_GET_FEATURES, &features); | 
					
						
							|  |  |  | 	if (rc < 0) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("get_features failed: %s\n", spdk_strerror(-rc)); | 
					
						
							| 
									
										
										
										
											2017-10-05 12:46:24 +00:00
										 |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return features; | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-05 12:46:24 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2017-09-01 17:33:53 +00:00
										 |  |  | virtio_user_set_features(struct virtio_dev *vdev, uint64_t features) | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-11-03 14:39:45 +00:00
										 |  |  | 	struct virtio_user_dev *dev = vdev->ctx; | 
					
						
							| 
									
										
										
										
											2018-07-02 10:43:05 +00:00
										 |  |  | 	uint64_t protocol_features; | 
					
						
							| 
									
										
										
										
											2017-10-05 12:46:24 +00:00
										 |  |  | 	int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ret = dev->ops->send_request(dev, VHOST_USER_SET_FEATURES, &features); | 
					
						
							|  |  |  | 	if (ret < 0) { | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 		return ret; | 
					
						
							| 
									
										
										
										
											2017-10-05 12:46:24 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	vdev->negotiated_features = features; | 
					
						
							| 
									
										
										
										
											2017-11-02 16:10:56 +00:00
										 |  |  | 	vdev->modern = virtio_dev_has_feature(vdev, VIRTIO_F_VERSION_1); | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-02 10:43:05 +00:00
										 |  |  | 	if (!virtio_dev_has_feature(vdev, VHOST_USER_F_PROTOCOL_FEATURES)) { | 
					
						
							|  |  |  | 		/* nothing else to do */ | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ret = dev->ops->send_request(dev, VHOST_USER_GET_PROTOCOL_FEATURES, &protocol_features); | 
					
						
							|  |  |  | 	if (ret < 0) { | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 		return ret; | 
					
						
							| 
									
										
										
										
											2018-07-02 10:43:05 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	protocol_features &= VIRTIO_USER_SUPPORTED_PROTOCOL_FEATURES; | 
					
						
							|  |  |  | 	ret = dev->ops->send_request(dev, VHOST_USER_SET_PROTOCOL_FEATURES, &protocol_features); | 
					
						
							|  |  |  | 	if (ret < 0) { | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 		return ret; | 
					
						
							| 
									
										
										
										
											2018-07-02 10:43:05 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	dev->protocol_features = protocol_features; | 
					
						
							| 
									
										
										
										
											2017-10-05 12:46:24 +00:00
										 |  |  | 	return 0; | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static uint16_t | 
					
						
							| 
									
										
										
										
											2018-04-18 11:37:48 +00:00
										 |  |  | virtio_user_get_queue_size(struct virtio_dev *vdev, uint16_t queue_id) | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-11-03 14:39:45 +00:00
										 |  |  | 	struct virtio_user_dev *dev = vdev->ctx; | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-18 11:37:48 +00:00
										 |  |  | 	/* Currently each queue has same queue size */ | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | 	return dev->queue_size; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2017-09-01 17:33:53 +00:00
										 |  |  | virtio_user_setup_queue(struct virtio_dev *vdev, struct virtqueue *vq) | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-11-03 14:39:45 +00:00
										 |  |  | 	struct virtio_user_dev *dev = vdev->ctx; | 
					
						
							| 
									
										
										
										
											2018-07-02 10:43:05 +00:00
										 |  |  | 	struct vhost_vring_state state; | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | 	uint16_t queue_idx = vq->vq_queue_index; | 
					
						
							| 
									
										
										
										
											2018-06-27 10:33:19 +00:00
										 |  |  | 	void *queue_mem; | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | 	uint64_t desc_addr, avail_addr, used_addr; | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 	int callfd, kickfd, rc; | 
					
						
							| 
									
										
										
										
											2017-09-20 15:50:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (dev->callfds[queue_idx] != -1 || dev->kickfds[queue_idx] != -1) { | 
					
						
							| 
									
										
										
										
											2017-10-14 10:43:58 +00:00
										 |  |  | 		SPDK_ERRLOG("queue %"PRIu16" already exists\n", queue_idx); | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 		return -EEXIST; | 
					
						
							| 
									
										
										
										
											2017-09-20 15:50:08 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* May use invalid flag, but some backend uses kickfd and
 | 
					
						
							|  |  |  | 	 * callfd as criteria to judge if dev is alive. so finally we | 
					
						
							|  |  |  | 	 * use real event_fd. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	callfd = eventfd(0, EFD_CLOEXEC | EFD_NONBLOCK); | 
					
						
							|  |  |  | 	if (callfd < 0) { | 
					
						
							| 
									
										
										
										
											2017-11-07 12:56:52 +00:00
										 |  |  | 		SPDK_ERRLOG("callfd error, %s\n", spdk_strerror(errno)); | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 		return -errno; | 
					
						
							| 
									
										
										
										
											2017-09-20 15:50:08 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	kickfd = eventfd(0, EFD_CLOEXEC | EFD_NONBLOCK); | 
					
						
							|  |  |  | 	if (kickfd < 0) { | 
					
						
							| 
									
										
										
										
											2017-11-07 12:56:52 +00:00
										 |  |  | 		SPDK_ERRLOG("kickfd error, %s\n", spdk_strerror(errno)); | 
					
						
							| 
									
										
										
										
											2017-09-20 15:50:08 +00:00
										 |  |  | 		close(callfd); | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 		return -errno; | 
					
						
							| 
									
										
										
										
											2017-09-20 15:50:08 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-08 21:35:25 +00:00
										 |  |  | 	queue_mem = spdk_zmalloc(vq->vq_ring_size, VIRTIO_PCI_VRING_ALIGN, NULL, | 
					
						
							|  |  |  | 				 SPDK_ENV_LCORE_ID_ANY, SPDK_MALLOC_DMA); | 
					
						
							| 
									
										
										
										
											2018-07-11 15:43:15 +00:00
										 |  |  | 	if (queue_mem == NULL) { | 
					
						
							|  |  |  | 		close(kickfd); | 
					
						
							|  |  |  | 		close(callfd); | 
					
						
							|  |  |  | 		return -ENOMEM; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	vq->vq_ring_mem = SPDK_VTOPHYS_ERROR; | 
					
						
							|  |  |  | 	vq->vq_ring_virt_mem = queue_mem; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-02 10:43:05 +00:00
										 |  |  | 	state.index = vq->vq_queue_index; | 
					
						
							| 
									
										
										
										
											2020-09-01 01:15:07 +00:00
										 |  |  | 	state.num = vq->vq_nentries; | 
					
						
							| 
									
										
										
										
											2018-07-02 10:43:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 	if (virtio_dev_has_feature(vdev, VHOST_USER_F_PROTOCOL_FEATURES)) { | 
					
						
							|  |  |  | 		rc = dev->ops->send_request(dev, VHOST_USER_SET_VRING_ENABLE, &state); | 
					
						
							|  |  |  | 		if (rc < 0) { | 
					
						
							|  |  |  | 			SPDK_ERRLOG("failed to send VHOST_USER_SET_VRING_ENABLE: %s\n", | 
					
						
							|  |  |  | 				    spdk_strerror(-rc)); | 
					
						
							| 
									
										
										
										
											2020-01-02 16:33:12 +00:00
										 |  |  | 			close(kickfd); | 
					
						
							|  |  |  | 			close(callfd); | 
					
						
							| 
									
										
										
										
											2019-04-08 21:35:25 +00:00
										 |  |  | 			spdk_free(queue_mem); | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 			return -rc; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-07-02 10:43:05 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-20 15:50:08 +00:00
										 |  |  | 	dev->callfds[queue_idx] = callfd; | 
					
						
							|  |  |  | 	dev->kickfds[queue_idx] = kickfd; | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	desc_addr = (uintptr_t)vq->vq_ring_virt_mem; | 
					
						
							|  |  |  | 	avail_addr = desc_addr + vq->vq_nentries * sizeof(struct vring_desc); | 
					
						
							| 
									
										
										
										
											2019-12-19 07:33:53 +00:00
										 |  |  | 	used_addr = SPDK_ALIGN_CEIL(avail_addr + offsetof(struct vring_avail, | 
					
						
							|  |  |  | 				    ring[vq->vq_nentries]), | 
					
						
							|  |  |  | 				    VIRTIO_PCI_VRING_ALIGN); | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	dev->vrings[queue_idx].num = vq->vq_nentries; | 
					
						
							|  |  |  | 	dev->vrings[queue_idx].desc = (void *)(uintptr_t)desc_addr; | 
					
						
							|  |  |  | 	dev->vrings[queue_idx].avail = (void *)(uintptr_t)avail_addr; | 
					
						
							|  |  |  | 	dev->vrings[queue_idx].used = (void *)(uintptr_t)used_addr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2017-09-01 17:33:53 +00:00
										 |  |  | virtio_user_del_queue(struct virtio_dev *vdev, struct virtqueue *vq) | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* For legacy devices, write 0 to VIRTIO_PCI_QUEUE_PFN port, QEMU
 | 
					
						
							|  |  |  | 	 * correspondingly stops the ioeventfds, and reset the status of | 
					
						
							|  |  |  | 	 * the device. | 
					
						
							|  |  |  | 	 * For modern devices, set queue desc, avail, used in PCI bar to 0, | 
					
						
							|  |  |  | 	 * not see any more behavior in QEMU. | 
					
						
							|  |  |  | 	 * | 
					
						
							| 
									
										
										
										
											2018-07-02 09:33:33 +00:00
										 |  |  | 	 * Here we just care about what information to deliver to vhost-user. | 
					
						
							|  |  |  | 	 * So we just close ioeventfd for now. | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2017-11-03 14:39:45 +00:00
										 |  |  | 	struct virtio_user_dev *dev = vdev->ctx; | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	close(dev->callfds[vq->vq_queue_index]); | 
					
						
							|  |  |  | 	close(dev->kickfds[vq->vq_queue_index]); | 
					
						
							| 
									
										
										
										
											2017-09-20 15:50:08 +00:00
										 |  |  | 	dev->callfds[vq->vq_queue_index] = -1; | 
					
						
							|  |  |  | 	dev->kickfds[vq->vq_queue_index] = -1; | 
					
						
							| 
									
										
										
										
											2018-06-27 10:33:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-08 21:35:25 +00:00
										 |  |  | 	spdk_free(vq->vq_ring_virt_mem); | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2017-09-01 17:33:53 +00:00
										 |  |  | virtio_user_notify_queue(struct virtio_dev *vdev, struct virtqueue *vq) | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	uint64_t buf = 1; | 
					
						
							| 
									
										
										
										
											2017-11-03 14:39:45 +00:00
										 |  |  | 	struct virtio_user_dev *dev = vdev->ctx; | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-18 11:30:18 +00:00
										 |  |  | 	if (write(dev->kickfds[vq->vq_queue_index], &buf, sizeof(buf)) < 0) { | 
					
						
							| 
									
										
										
										
											2017-11-07 12:56:52 +00:00
										 |  |  | 		SPDK_ERRLOG("failed to kick backend: %s.\n", spdk_strerror(errno)); | 
					
						
							| 
									
										
										
										
											2017-10-18 11:30:18 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-29 11:03:45 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2017-11-20 18:24:24 +00:00
										 |  |  | virtio_user_destroy(struct virtio_dev *vdev) | 
					
						
							| 
									
										
										
										
											2017-09-29 11:03:45 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-11-03 14:39:45 +00:00
										 |  |  | 	struct virtio_user_dev *dev = vdev->ctx; | 
					
						
							| 
									
										
										
										
											2017-09-29 11:03:45 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-03 12:28:07 +00:00
										 |  |  | 	close(dev->vhostfd); | 
					
						
							|  |  |  | 	free(dev); | 
					
						
							| 
									
										
										
										
											2017-09-29 11:03:45 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-09 18:45:13 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2018-03-27 13:45:10 +00:00
										 |  |  | virtio_user_dump_json_info(struct virtio_dev *vdev, struct spdk_json_write_ctx *w) | 
					
						
							| 
									
										
										
										
											2017-10-09 18:45:13 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-11-03 14:39:45 +00:00
										 |  |  | 	struct virtio_user_dev *dev = vdev->ctx; | 
					
						
							| 
									
										
										
										
											2017-10-09 18:45:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-01 05:41:15 +00:00
										 |  |  | 	spdk_json_write_named_string(w, "type", "user"); | 
					
						
							|  |  |  | 	spdk_json_write_named_string(w, "socket", dev->path); | 
					
						
							| 
									
										
										
										
											2017-10-09 18:45:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-27 13:45:10 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | virtio_user_write_json_config(struct virtio_dev *vdev, struct spdk_json_write_ctx *w) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct virtio_user_dev *dev = vdev->ctx; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spdk_json_write_named_string(w, "trtype", "user"); | 
					
						
							|  |  |  | 	spdk_json_write_named_string(w, "traddr", dev->path); | 
					
						
							|  |  |  | 	spdk_json_write_named_uint32(w, "vq_count", vdev->max_queues - vdev->fixed_queues_num); | 
					
						
							| 
									
										
										
										
											2018-04-18 11:37:48 +00:00
										 |  |  | 	spdk_json_write_named_uint32(w, "vq_size", virtio_dev_backend_ops(vdev)->get_queue_size(vdev, 0)); | 
					
						
							| 
									
										
										
										
											2018-03-27 13:45:10 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-03 15:55:38 +00:00
										 |  |  | static const struct virtio_dev_ops virtio_user_ops = { | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | 	.read_dev_cfg	= virtio_user_read_dev_config, | 
					
						
							|  |  |  | 	.write_dev_cfg	= virtio_user_write_dev_config, | 
					
						
							|  |  |  | 	.get_status	= virtio_user_get_status, | 
					
						
							|  |  |  | 	.set_status	= virtio_user_set_status, | 
					
						
							|  |  |  | 	.get_features	= virtio_user_get_features, | 
					
						
							|  |  |  | 	.set_features	= virtio_user_set_features, | 
					
						
							| 
									
										
										
										
											2017-11-20 18:24:24 +00:00
										 |  |  | 	.destruct_dev	= virtio_user_destroy, | 
					
						
							| 
									
										
										
										
											2018-04-18 11:37:48 +00:00
										 |  |  | 	.get_queue_size	= virtio_user_get_queue_size, | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | 	.setup_queue	= virtio_user_setup_queue, | 
					
						
							|  |  |  | 	.del_queue	= virtio_user_del_queue, | 
					
						
							|  |  |  | 	.notify_queue	= virtio_user_notify_queue, | 
					
						
							| 
									
										
										
										
											2018-03-27 13:45:10 +00:00
										 |  |  | 	.dump_json_info = virtio_user_dump_json_info, | 
					
						
							|  |  |  | 	.write_json_config = virtio_user_write_json_config, | 
					
						
							| 
									
										
										
										
											2017-05-30 21:13:50 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2017-11-03 15:55:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-20 18:24:24 +00:00
										 |  |  | int | 
					
						
							|  |  |  | virtio_user_dev_init(struct virtio_dev *vdev, const char *name, const char *path, | 
					
						
							| 
									
										
										
										
											2017-12-27 14:51:17 +00:00
										 |  |  | 		     uint32_t queue_size) | 
					
						
							| 
									
										
										
										
											2017-11-03 15:55:38 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct virtio_user_dev *dev; | 
					
						
							| 
									
										
										
										
											2017-11-20 18:24:24 +00:00
										 |  |  | 	int rc; | 
					
						
							| 
									
										
										
										
											2017-11-03 15:55:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (name == NULL) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("No name gived for controller: %s\n", path); | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 		return -EINVAL; | 
					
						
							| 
									
										
										
										
											2017-11-03 15:55:38 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	dev = calloc(1, sizeof(*dev)); | 
					
						
							|  |  |  | 	if (dev == NULL) { | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 		return -ENOMEM; | 
					
						
							| 
									
										
										
										
											2017-11-03 15:55:38 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-15 12:06:58 +00:00
										 |  |  | 	rc = virtio_dev_construct(vdev, name, &virtio_user_ops, dev); | 
					
						
							| 
									
										
										
										
											2017-11-20 18:24:24 +00:00
										 |  |  | 	if (rc != 0) { | 
					
						
							| 
									
										
										
										
											2017-11-03 15:55:38 +00:00
										 |  |  | 		SPDK_ERRLOG("Failed to init device: %s\n", path); | 
					
						
							|  |  |  | 		free(dev); | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 		return rc; | 
					
						
							| 
									
										
										
										
											2017-11-03 15:55:38 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	vdev->is_hw = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	snprintf(dev->path, PATH_MAX, "%s", path); | 
					
						
							|  |  |  | 	dev->queue_size = queue_size; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 	rc = virtio_user_dev_setup(vdev); | 
					
						
							|  |  |  | 	if (rc < 0) { | 
					
						
							| 
									
										
										
										
											2017-11-03 15:55:38 +00:00
										 |  |  | 		SPDK_ERRLOG("backend set up fails\n"); | 
					
						
							|  |  |  | 		goto err; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 	rc = dev->ops->send_request(dev, VHOST_USER_SET_OWNER, NULL); | 
					
						
							|  |  |  | 	if (rc < 0) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("set_owner fails: %s\n", spdk_strerror(-rc)); | 
					
						
							| 
									
										
										
										
											2017-11-03 15:55:38 +00:00
										 |  |  | 		goto err; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-20 18:24:24 +00:00
										 |  |  | 	return 0; | 
					
						
							| 
									
										
										
										
											2017-11-03 15:55:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | err: | 
					
						
							| 
									
										
										
										
											2017-11-20 18:24:24 +00:00
										 |  |  | 	virtio_dev_destruct(vdev); | 
					
						
							| 
									
										
										
										
											2018-07-11 16:54:35 +00:00
										 |  |  | 	return rc; | 
					
						
							| 
									
										
										
										
											2017-11-03 15:55:38 +00:00
										 |  |  | } |