xref: /qemu/hw/virtio/vhost-backend.c (revision 9b8bfe21bee37ced21aa57c89388c2ebea8c3f31)
11a1bfac9SNikolay Nikolaev /*
21a1bfac9SNikolay Nikolaev  * vhost-backend
31a1bfac9SNikolay Nikolaev  *
41a1bfac9SNikolay Nikolaev  * Copyright (c) 2013 Virtual Open Systems Sarl.
51a1bfac9SNikolay Nikolaev  *
61a1bfac9SNikolay Nikolaev  * This work is licensed under the terms of the GNU GPL, version 2 or later.
71a1bfac9SNikolay Nikolaev  * See the COPYING file in the top-level directory.
81a1bfac9SNikolay Nikolaev  *
91a1bfac9SNikolay Nikolaev  */
101a1bfac9SNikolay Nikolaev 
11*9b8bfe21SPeter Maydell #include "qemu/osdep.h"
121a1bfac9SNikolay Nikolaev #include "hw/virtio/vhost.h"
131a1bfac9SNikolay Nikolaev #include "hw/virtio/vhost-backend.h"
141a1bfac9SNikolay Nikolaev #include "qemu/error-report.h"
152ce68e4cSIgor Mammedov #include "linux/vhost.h"
161a1bfac9SNikolay Nikolaev 
171a1bfac9SNikolay Nikolaev #include <sys/ioctl.h>
181a1bfac9SNikolay Nikolaev 
191a1bfac9SNikolay Nikolaev static int vhost_kernel_call(struct vhost_dev *dev, unsigned long int request,
201a1bfac9SNikolay Nikolaev                              void *arg)
211a1bfac9SNikolay Nikolaev {
221a1bfac9SNikolay Nikolaev     int fd = (uintptr_t) dev->opaque;
231a1bfac9SNikolay Nikolaev 
241a1bfac9SNikolay Nikolaev     assert(dev->vhost_ops->backend_type == VHOST_BACKEND_TYPE_KERNEL);
251a1bfac9SNikolay Nikolaev 
261a1bfac9SNikolay Nikolaev     return ioctl(fd, request, arg);
271a1bfac9SNikolay Nikolaev }
281a1bfac9SNikolay Nikolaev 
291a1bfac9SNikolay Nikolaev static int vhost_kernel_init(struct vhost_dev *dev, void *opaque)
301a1bfac9SNikolay Nikolaev {
311a1bfac9SNikolay Nikolaev     assert(dev->vhost_ops->backend_type == VHOST_BACKEND_TYPE_KERNEL);
321a1bfac9SNikolay Nikolaev 
331a1bfac9SNikolay Nikolaev     dev->opaque = opaque;
341a1bfac9SNikolay Nikolaev 
351a1bfac9SNikolay Nikolaev     return 0;
361a1bfac9SNikolay Nikolaev }
371a1bfac9SNikolay Nikolaev 
381a1bfac9SNikolay Nikolaev static int vhost_kernel_cleanup(struct vhost_dev *dev)
391a1bfac9SNikolay Nikolaev {
401a1bfac9SNikolay Nikolaev     int fd = (uintptr_t) dev->opaque;
411a1bfac9SNikolay Nikolaev 
421a1bfac9SNikolay Nikolaev     assert(dev->vhost_ops->backend_type == VHOST_BACKEND_TYPE_KERNEL);
431a1bfac9SNikolay Nikolaev 
441a1bfac9SNikolay Nikolaev     return close(fd);
451a1bfac9SNikolay Nikolaev }
461a1bfac9SNikolay Nikolaev 
472ce68e4cSIgor Mammedov static int vhost_kernel_memslots_limit(struct vhost_dev *dev)
482ce68e4cSIgor Mammedov {
492ce68e4cSIgor Mammedov     int limit = 64;
502ce68e4cSIgor Mammedov     char *s;
512ce68e4cSIgor Mammedov 
522ce68e4cSIgor Mammedov     if (g_file_get_contents("/sys/module/vhost/parameters/max_mem_regions",
532ce68e4cSIgor Mammedov                             &s, NULL, NULL)) {
542ce68e4cSIgor Mammedov         uint64_t val = g_ascii_strtoull(s, NULL, 10);
552ce68e4cSIgor Mammedov         if (!((val == G_MAXUINT64 || !val) && errno)) {
562ce68e4cSIgor Mammedov             return val;
572ce68e4cSIgor Mammedov         }
582ce68e4cSIgor Mammedov         error_report("ignoring invalid max_mem_regions value in vhost module:"
592ce68e4cSIgor Mammedov                      " %s", s);
602ce68e4cSIgor Mammedov     }
612ce68e4cSIgor Mammedov     return limit;
622ce68e4cSIgor Mammedov }
632ce68e4cSIgor Mammedov 
6421e70425SMarc-André Lureau static int vhost_kernel_net_set_backend(struct vhost_dev *dev,
6521e70425SMarc-André Lureau                                         struct vhost_vring_file *file)
6621e70425SMarc-André Lureau {
6721e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_NET_SET_BACKEND, file);
6821e70425SMarc-André Lureau }
6921e70425SMarc-André Lureau 
7021e70425SMarc-André Lureau static int vhost_kernel_scsi_set_endpoint(struct vhost_dev *dev,
7121e70425SMarc-André Lureau                                           struct vhost_scsi_target *target)
7221e70425SMarc-André Lureau {
7321e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_SCSI_SET_ENDPOINT, target);
7421e70425SMarc-André Lureau }
7521e70425SMarc-André Lureau 
7621e70425SMarc-André Lureau static int vhost_kernel_scsi_clear_endpoint(struct vhost_dev *dev,
7721e70425SMarc-André Lureau                                             struct vhost_scsi_target *target)
7821e70425SMarc-André Lureau {
7921e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_SCSI_CLEAR_ENDPOINT, target);
8021e70425SMarc-André Lureau }
8121e70425SMarc-André Lureau 
8221e70425SMarc-André Lureau static int vhost_kernel_scsi_get_abi_version(struct vhost_dev *dev, int *version)
8321e70425SMarc-André Lureau {
8421e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_SCSI_GET_ABI_VERSION, version);
8521e70425SMarc-André Lureau }
8621e70425SMarc-André Lureau 
8721e70425SMarc-André Lureau static int vhost_kernel_set_log_base(struct vhost_dev *dev, uint64_t base,
889a78a5ddSMarc-André Lureau                                      struct vhost_log *log)
89c2bea314SMarc-André Lureau {
90c2bea314SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_SET_LOG_BASE, &base);
91c2bea314SMarc-André Lureau }
92c2bea314SMarc-André Lureau 
9321e70425SMarc-André Lureau static int vhost_kernel_set_mem_table(struct vhost_dev *dev,
9421e70425SMarc-André Lureau                                       struct vhost_memory *mem)
9521e70425SMarc-André Lureau {
9621e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_SET_MEM_TABLE, mem);
9721e70425SMarc-André Lureau }
9821e70425SMarc-André Lureau 
9921e70425SMarc-André Lureau static int vhost_kernel_set_vring_addr(struct vhost_dev *dev,
10021e70425SMarc-André Lureau                                        struct vhost_vring_addr *addr)
10121e70425SMarc-André Lureau {
10221e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_SET_VRING_ADDR, addr);
10321e70425SMarc-André Lureau }
10421e70425SMarc-André Lureau 
10521e70425SMarc-André Lureau static int vhost_kernel_set_vring_endian(struct vhost_dev *dev,
10621e70425SMarc-André Lureau                                          struct vhost_vring_state *ring)
10721e70425SMarc-André Lureau {
10821e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_SET_VRING_ENDIAN, ring);
10921e70425SMarc-André Lureau }
11021e70425SMarc-André Lureau 
11121e70425SMarc-André Lureau static int vhost_kernel_set_vring_num(struct vhost_dev *dev,
11221e70425SMarc-André Lureau                                       struct vhost_vring_state *ring)
11321e70425SMarc-André Lureau {
11421e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_SET_VRING_NUM, ring);
11521e70425SMarc-André Lureau }
11621e70425SMarc-André Lureau 
11721e70425SMarc-André Lureau static int vhost_kernel_set_vring_base(struct vhost_dev *dev,
11821e70425SMarc-André Lureau                                        struct vhost_vring_state *ring)
11921e70425SMarc-André Lureau {
12021e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_SET_VRING_BASE, ring);
12121e70425SMarc-André Lureau }
12221e70425SMarc-André Lureau 
12321e70425SMarc-André Lureau static int vhost_kernel_get_vring_base(struct vhost_dev *dev,
12421e70425SMarc-André Lureau                                        struct vhost_vring_state *ring)
12521e70425SMarc-André Lureau {
12621e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_GET_VRING_BASE, ring);
12721e70425SMarc-André Lureau }
12821e70425SMarc-André Lureau 
12921e70425SMarc-André Lureau static int vhost_kernel_set_vring_kick(struct vhost_dev *dev,
13021e70425SMarc-André Lureau                                        struct vhost_vring_file *file)
13121e70425SMarc-André Lureau {
13221e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_SET_VRING_KICK, file);
13321e70425SMarc-André Lureau }
13421e70425SMarc-André Lureau 
13521e70425SMarc-André Lureau static int vhost_kernel_set_vring_call(struct vhost_dev *dev,
13621e70425SMarc-André Lureau                                        struct vhost_vring_file *file)
13721e70425SMarc-André Lureau {
13821e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_SET_VRING_CALL, file);
13921e70425SMarc-André Lureau }
14021e70425SMarc-André Lureau 
14121e70425SMarc-André Lureau static int vhost_kernel_set_features(struct vhost_dev *dev,
14221e70425SMarc-André Lureau                                      uint64_t features)
14321e70425SMarc-André Lureau {
14421e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_SET_FEATURES, &features);
14521e70425SMarc-André Lureau }
14621e70425SMarc-André Lureau 
14721e70425SMarc-André Lureau static int vhost_kernel_get_features(struct vhost_dev *dev,
14821e70425SMarc-André Lureau                                      uint64_t *features)
14921e70425SMarc-André Lureau {
15021e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_GET_FEATURES, features);
15121e70425SMarc-André Lureau }
15221e70425SMarc-André Lureau 
15321e70425SMarc-André Lureau static int vhost_kernel_set_owner(struct vhost_dev *dev)
15421e70425SMarc-André Lureau {
15521e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_SET_OWNER, NULL);
15621e70425SMarc-André Lureau }
15721e70425SMarc-André Lureau 
15821e70425SMarc-André Lureau static int vhost_kernel_reset_device(struct vhost_dev *dev)
15921e70425SMarc-André Lureau {
16060915dc4SYuanhan Liu     return vhost_kernel_call(dev, VHOST_RESET_OWNER, NULL);
16121e70425SMarc-André Lureau }
16221e70425SMarc-André Lureau 
16321e70425SMarc-André Lureau static int vhost_kernel_get_vq_index(struct vhost_dev *dev, int idx)
16421e70425SMarc-André Lureau {
16521e70425SMarc-André Lureau     assert(idx >= dev->vq_index && idx < dev->vq_index + dev->nvqs);
16621e70425SMarc-André Lureau 
16721e70425SMarc-André Lureau     return idx - dev->vq_index;
16821e70425SMarc-André Lureau }
16921e70425SMarc-André Lureau 
1701a1bfac9SNikolay Nikolaev static const VhostOps kernel_ops = {
1711a1bfac9SNikolay Nikolaev         .backend_type = VHOST_BACKEND_TYPE_KERNEL,
1721a1bfac9SNikolay Nikolaev         .vhost_backend_init = vhost_kernel_init,
173fc57fd99SYuanhan Liu         .vhost_backend_cleanup = vhost_kernel_cleanup,
1742ce68e4cSIgor Mammedov         .vhost_backend_memslots_limit = vhost_kernel_memslots_limit,
17521e70425SMarc-André Lureau         .vhost_net_set_backend = vhost_kernel_net_set_backend,
17621e70425SMarc-André Lureau         .vhost_scsi_set_endpoint = vhost_kernel_scsi_set_endpoint,
17721e70425SMarc-André Lureau         .vhost_scsi_clear_endpoint = vhost_kernel_scsi_clear_endpoint,
17821e70425SMarc-André Lureau         .vhost_scsi_get_abi_version = vhost_kernel_scsi_get_abi_version,
17921e70425SMarc-André Lureau         .vhost_set_log_base = vhost_kernel_set_log_base,
18021e70425SMarc-André Lureau         .vhost_set_mem_table = vhost_kernel_set_mem_table,
18121e70425SMarc-André Lureau         .vhost_set_vring_addr = vhost_kernel_set_vring_addr,
18221e70425SMarc-André Lureau         .vhost_set_vring_endian = vhost_kernel_set_vring_endian,
18321e70425SMarc-André Lureau         .vhost_set_vring_num = vhost_kernel_set_vring_num,
18421e70425SMarc-André Lureau         .vhost_set_vring_base = vhost_kernel_set_vring_base,
18521e70425SMarc-André Lureau         .vhost_get_vring_base = vhost_kernel_get_vring_base,
18621e70425SMarc-André Lureau         .vhost_set_vring_kick = vhost_kernel_set_vring_kick,
18721e70425SMarc-André Lureau         .vhost_set_vring_call = vhost_kernel_set_vring_call,
18821e70425SMarc-André Lureau         .vhost_set_features = vhost_kernel_set_features,
18921e70425SMarc-André Lureau         .vhost_get_features = vhost_kernel_get_features,
19021e70425SMarc-André Lureau         .vhost_set_owner = vhost_kernel_set_owner,
19121e70425SMarc-André Lureau         .vhost_reset_device = vhost_kernel_reset_device,
19221e70425SMarc-André Lureau         .vhost_get_vq_index = vhost_kernel_get_vq_index,
1931a1bfac9SNikolay Nikolaev };
1941a1bfac9SNikolay Nikolaev 
1951a1bfac9SNikolay Nikolaev int vhost_set_backend_type(struct vhost_dev *dev, VhostBackendType backend_type)
1961a1bfac9SNikolay Nikolaev {
1971a1bfac9SNikolay Nikolaev     int r = 0;
1981a1bfac9SNikolay Nikolaev 
1991a1bfac9SNikolay Nikolaev     switch (backend_type) {
2001a1bfac9SNikolay Nikolaev     case VHOST_BACKEND_TYPE_KERNEL:
2011a1bfac9SNikolay Nikolaev         dev->vhost_ops = &kernel_ops;
2021a1bfac9SNikolay Nikolaev         break;
2035f6f6664SNikolay Nikolaev     case VHOST_BACKEND_TYPE_USER:
2045f6f6664SNikolay Nikolaev         dev->vhost_ops = &user_ops;
2055f6f6664SNikolay Nikolaev         break;
2061a1bfac9SNikolay Nikolaev     default:
207c6547661SGonglei         error_report("Unknown vhost backend type");
2081a1bfac9SNikolay Nikolaev         r = -1;
2091a1bfac9SNikolay Nikolaev     }
2101a1bfac9SNikolay Nikolaev 
2111a1bfac9SNikolay Nikolaev     return r;
2121a1bfac9SNikolay Nikolaev }
213