1 // SPDX-License-Identifier: GPL-2.0-only
2 /* Copyright (C) 2009 Red Hat, Inc.
3 * Author: Michael S. Tsirkin <mst@redhat.com>
5 * test virtio server in host kernel.
8 #include <linux/compat.h>
9 #include <linux/eventfd.h>
10 #include <linux/vhost.h>
11 #include <linux/miscdevice.h>
12 #include <linux/module.h>
13 #include <linux/mutex.h>
14 #include <linux/workqueue.h>
15 #include <linux/file.h>
16 #include <linux/slab.h>
21 /* Max number of bytes transferred before requeueing the job.
22 * Using this limit prevents one virtqueue from starving others. */
23 #define VHOST_TEST_WEIGHT 0x80000
25 /* Max number of packets transferred before requeueing the job.
26 * Using this limit prevents one virtqueue from starving others with
29 #define VHOST_TEST_PKT_WEIGHT 256
33 VHOST_TEST_VQ_MAX
= 1,
38 struct vhost_virtqueue vqs
[VHOST_TEST_VQ_MAX
];
41 /* Expects to be always run from workqueue - which acts as
42 * read-size critical section for our kind of RCU. */
43 static void handle_vq(struct vhost_test
*n
)
45 struct vhost_virtqueue
*vq
= &n
->vqs
[VHOST_TEST_VQ
];
48 size_t len
, total_len
= 0;
51 mutex_lock(&vq
->mutex
);
52 private = vhost_vq_get_backend(vq
);
54 mutex_unlock(&vq
->mutex
);
58 vhost_disable_notify(&n
->dev
, vq
);
61 head
= vhost_get_vq_desc(vq
, vq
->iov
,
65 /* On error, stop handling until the next kick. */
66 if (unlikely(head
< 0))
68 /* Nothing new? Wait for eventfd to tell us they refilled. */
69 if (head
== vq
->num
) {
70 if (unlikely(vhost_enable_notify(&n
->dev
, vq
))) {
71 vhost_disable_notify(&n
->dev
, vq
);
77 vq_err(vq
, "Unexpected descriptor format for TX: "
78 "out %d, int %d\n", out
, in
);
81 len
= iov_length(vq
->iov
, out
);
84 vq_err(vq
, "Unexpected 0 len for TX\n");
87 vhost_add_used_and_signal(&n
->dev
, vq
, head
, 0);
89 if (unlikely(vhost_exceeds_weight(vq
, 0, total_len
)))
93 mutex_unlock(&vq
->mutex
);
96 static void handle_vq_kick(struct vhost_work
*work
)
98 struct vhost_virtqueue
*vq
= container_of(work
, struct vhost_virtqueue
,
100 struct vhost_test
*n
= container_of(vq
->dev
, struct vhost_test
, dev
);
105 static int vhost_test_open(struct inode
*inode
, struct file
*f
)
107 struct vhost_test
*n
= kmalloc(sizeof *n
, GFP_KERNEL
);
108 struct vhost_dev
*dev
;
109 struct vhost_virtqueue
**vqs
;
113 vqs
= kmalloc_array(VHOST_TEST_VQ_MAX
, sizeof(*vqs
), GFP_KERNEL
);
120 vqs
[VHOST_TEST_VQ
] = &n
->vqs
[VHOST_TEST_VQ
];
121 n
->vqs
[VHOST_TEST_VQ
].handle_kick
= handle_vq_kick
;
122 vhost_dev_init(dev
, vqs
, VHOST_TEST_VQ_MAX
, UIO_MAXIOV
,
123 VHOST_TEST_PKT_WEIGHT
, VHOST_TEST_WEIGHT
, true, NULL
);
130 static void *vhost_test_stop_vq(struct vhost_test
*n
,
131 struct vhost_virtqueue
*vq
)
135 mutex_lock(&vq
->mutex
);
136 private = vhost_vq_get_backend(vq
);
137 vhost_vq_set_backend(vq
, NULL
);
138 mutex_unlock(&vq
->mutex
);
142 static void vhost_test_stop(struct vhost_test
*n
, void **privatep
)
144 *privatep
= vhost_test_stop_vq(n
, n
->vqs
+ VHOST_TEST_VQ
);
147 static void vhost_test_flush(struct vhost_test
*n
)
149 vhost_dev_flush(&n
->dev
);
152 static int vhost_test_release(struct inode
*inode
, struct file
*f
)
154 struct vhost_test
*n
= f
->private_data
;
157 vhost_test_stop(n
, &private);
159 vhost_dev_stop(&n
->dev
);
160 vhost_dev_cleanup(&n
->dev
);
166 static long vhost_test_run(struct vhost_test
*n
, int test
)
168 void *priv
, *oldpriv
;
169 struct vhost_virtqueue
*vq
;
172 if (test
< 0 || test
> 1)
175 mutex_lock(&n
->dev
.mutex
);
176 r
= vhost_dev_check_owner(&n
->dev
);
180 for (index
= 0; index
< n
->dev
.nvqs
; ++index
) {
181 /* Verify that ring has been setup correctly. */
182 if (!vhost_vq_access_ok(&n
->vqs
[index
])) {
188 for (index
= 0; index
< n
->dev
.nvqs
; ++index
) {
190 mutex_lock(&vq
->mutex
);
191 priv
= test
? n
: NULL
;
193 /* start polling new socket */
194 oldpriv
= vhost_vq_get_backend(vq
);
195 vhost_vq_set_backend(vq
, priv
);
197 r
= vhost_vq_init_access(&n
->vqs
[index
]);
199 mutex_unlock(&vq
->mutex
);
209 mutex_unlock(&n
->dev
.mutex
);
213 mutex_unlock(&n
->dev
.mutex
);
217 static long vhost_test_reset_owner(struct vhost_test
*n
)
221 struct vhost_iotlb
*umem
;
223 mutex_lock(&n
->dev
.mutex
);
224 err
= vhost_dev_check_owner(&n
->dev
);
227 umem
= vhost_dev_reset_owner_prepare();
232 vhost_test_stop(n
, &priv
);
234 vhost_dev_stop(&n
->dev
);
235 vhost_dev_reset_owner(&n
->dev
, umem
);
237 mutex_unlock(&n
->dev
.mutex
);
241 static int vhost_test_set_features(struct vhost_test
*n
, u64 features
)
243 struct vhost_virtqueue
*vq
;
245 mutex_lock(&n
->dev
.mutex
);
246 if ((features
& (1 << VHOST_F_LOG_ALL
)) &&
247 !vhost_log_access_ok(&n
->dev
)) {
248 mutex_unlock(&n
->dev
.mutex
);
251 vq
= &n
->vqs
[VHOST_TEST_VQ
];
252 mutex_lock(&vq
->mutex
);
253 vq
->acked_features
= features
;
254 mutex_unlock(&vq
->mutex
);
255 mutex_unlock(&n
->dev
.mutex
);
259 static long vhost_test_set_backend(struct vhost_test
*n
, unsigned index
, int fd
)
261 static void *backend
;
263 const bool enable
= fd
!= -1;
264 struct vhost_virtqueue
*vq
;
267 mutex_lock(&n
->dev
.mutex
);
268 r
= vhost_dev_check_owner(&n
->dev
);
272 if (index
>= VHOST_TEST_VQ_MAX
) {
277 mutex_lock(&vq
->mutex
);
279 /* Verify that ring has been setup correctly. */
280 if (!vhost_vq_access_ok(vq
)) {
285 vhost_poll_stop(&vq
->poll
);
286 backend
= vhost_vq_get_backend(vq
);
287 vhost_vq_set_backend(vq
, NULL
);
289 vhost_vq_set_backend(vq
, backend
);
290 r
= vhost_vq_init_access(vq
);
292 r
= vhost_poll_start(&vq
->poll
, vq
->kick
);
295 mutex_unlock(&vq
->mutex
);
301 mutex_unlock(&n
->dev
.mutex
);
305 mutex_unlock(&vq
->mutex
);
307 mutex_unlock(&n
->dev
.mutex
);
311 static long vhost_test_ioctl(struct file
*f
, unsigned int ioctl
,
314 struct vhost_vring_file backend
;
315 struct vhost_test
*n
= f
->private_data
;
316 void __user
*argp
= (void __user
*)arg
;
317 u64 __user
*featurep
= argp
;
323 if (copy_from_user(&test
, argp
, sizeof test
))
325 return vhost_test_run(n
, test
);
326 case VHOST_TEST_SET_BACKEND
:
327 if (copy_from_user(&backend
, argp
, sizeof backend
))
329 return vhost_test_set_backend(n
, backend
.index
, backend
.fd
);
330 case VHOST_GET_FEATURES
:
331 features
= VHOST_FEATURES
;
332 if (copy_to_user(featurep
, &features
, sizeof features
))
335 case VHOST_SET_FEATURES
:
336 if (copy_from_user(&features
, featurep
, sizeof features
))
338 if (features
& ~VHOST_FEATURES
)
340 return vhost_test_set_features(n
, features
);
341 case VHOST_RESET_OWNER
:
342 return vhost_test_reset_owner(n
);
344 mutex_lock(&n
->dev
.mutex
);
345 r
= vhost_dev_ioctl(&n
->dev
, ioctl
, argp
);
346 if (r
== -ENOIOCTLCMD
)
347 r
= vhost_vring_ioctl(&n
->dev
, ioctl
, argp
);
349 mutex_unlock(&n
->dev
.mutex
);
354 static const struct file_operations vhost_test_fops
= {
355 .owner
= THIS_MODULE
,
356 .release
= vhost_test_release
,
357 .unlocked_ioctl
= vhost_test_ioctl
,
358 .compat_ioctl
= compat_ptr_ioctl
,
359 .open
= vhost_test_open
,
360 .llseek
= noop_llseek
,
363 static struct miscdevice vhost_test_misc
= {
368 module_misc_device(vhost_test_misc
);
370 MODULE_VERSION("0.0.1");
371 MODULE_LICENSE("GPL v2");
372 MODULE_AUTHOR("Michael S. Tsirkin");
373 MODULE_DESCRIPTION("Host kernel side for virtio simulator");