xref: /qemu/hw/virtio/vhost-backend.c (revision 299e6f19b3e285f3ab00b780e3b48f5e58b5d4ed)
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 
119b8bfe21SPeter 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"
1518658a3cSPaolo Bonzini #include "standard-headers/linux/vhost_types.h"
1618658a3cSPaolo Bonzini 
17*299e6f19SPaolo Bonzini #ifdef CONFIG_VHOST_KERNEL
1818658a3cSPaolo Bonzini #include <linux/vhost.h>
1918658a3cSPaolo Bonzini #include <sys/ioctl.h>
201a1bfac9SNikolay Nikolaev 
211a1bfac9SNikolay Nikolaev static int vhost_kernel_call(struct vhost_dev *dev, unsigned long int request,
221a1bfac9SNikolay Nikolaev                              void *arg)
231a1bfac9SNikolay Nikolaev {
241a1bfac9SNikolay Nikolaev     int fd = (uintptr_t) dev->opaque;
251a1bfac9SNikolay Nikolaev 
261a1bfac9SNikolay Nikolaev     assert(dev->vhost_ops->backend_type == VHOST_BACKEND_TYPE_KERNEL);
271a1bfac9SNikolay Nikolaev 
281a1bfac9SNikolay Nikolaev     return ioctl(fd, request, arg);
291a1bfac9SNikolay Nikolaev }
301a1bfac9SNikolay Nikolaev 
311a1bfac9SNikolay Nikolaev static int vhost_kernel_init(struct vhost_dev *dev, void *opaque)
321a1bfac9SNikolay Nikolaev {
331a1bfac9SNikolay Nikolaev     assert(dev->vhost_ops->backend_type == VHOST_BACKEND_TYPE_KERNEL);
341a1bfac9SNikolay Nikolaev 
351a1bfac9SNikolay Nikolaev     dev->opaque = opaque;
361a1bfac9SNikolay Nikolaev 
371a1bfac9SNikolay Nikolaev     return 0;
381a1bfac9SNikolay Nikolaev }
391a1bfac9SNikolay Nikolaev 
401a1bfac9SNikolay Nikolaev static int vhost_kernel_cleanup(struct vhost_dev *dev)
411a1bfac9SNikolay Nikolaev {
421a1bfac9SNikolay Nikolaev     int fd = (uintptr_t) dev->opaque;
431a1bfac9SNikolay Nikolaev 
441a1bfac9SNikolay Nikolaev     assert(dev->vhost_ops->backend_type == VHOST_BACKEND_TYPE_KERNEL);
451a1bfac9SNikolay Nikolaev 
461a1bfac9SNikolay Nikolaev     return close(fd);
471a1bfac9SNikolay Nikolaev }
481a1bfac9SNikolay Nikolaev 
492ce68e4cSIgor Mammedov static int vhost_kernel_memslots_limit(struct vhost_dev *dev)
502ce68e4cSIgor Mammedov {
512ce68e4cSIgor Mammedov     int limit = 64;
522ce68e4cSIgor Mammedov     char *s;
532ce68e4cSIgor Mammedov 
542ce68e4cSIgor Mammedov     if (g_file_get_contents("/sys/module/vhost/parameters/max_mem_regions",
552ce68e4cSIgor Mammedov                             &s, NULL, NULL)) {
562ce68e4cSIgor Mammedov         uint64_t val = g_ascii_strtoull(s, NULL, 10);
572ce68e4cSIgor Mammedov         if (!((val == G_MAXUINT64 || !val) && errno)) {
5808b9e0baSPeng Hao             g_free(s);
592ce68e4cSIgor Mammedov             return val;
602ce68e4cSIgor Mammedov         }
612ce68e4cSIgor Mammedov         error_report("ignoring invalid max_mem_regions value in vhost module:"
622ce68e4cSIgor Mammedov                      " %s", s);
632ce68e4cSIgor Mammedov     }
6408b9e0baSPeng Hao     g_free(s);
652ce68e4cSIgor Mammedov     return limit;
662ce68e4cSIgor Mammedov }
672ce68e4cSIgor Mammedov 
6821e70425SMarc-André Lureau static int vhost_kernel_net_set_backend(struct vhost_dev *dev,
6921e70425SMarc-André Lureau                                         struct vhost_vring_file *file)
7021e70425SMarc-André Lureau {
7121e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_NET_SET_BACKEND, file);
7221e70425SMarc-André Lureau }
7321e70425SMarc-André Lureau 
7421e70425SMarc-André Lureau static int vhost_kernel_scsi_set_endpoint(struct vhost_dev *dev,
7521e70425SMarc-André Lureau                                           struct vhost_scsi_target *target)
7621e70425SMarc-André Lureau {
7721e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_SCSI_SET_ENDPOINT, target);
7821e70425SMarc-André Lureau }
7921e70425SMarc-André Lureau 
8021e70425SMarc-André Lureau static int vhost_kernel_scsi_clear_endpoint(struct vhost_dev *dev,
8121e70425SMarc-André Lureau                                             struct vhost_scsi_target *target)
8221e70425SMarc-André Lureau {
8321e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_SCSI_CLEAR_ENDPOINT, target);
8421e70425SMarc-André Lureau }
8521e70425SMarc-André Lureau 
8621e70425SMarc-André Lureau static int vhost_kernel_scsi_get_abi_version(struct vhost_dev *dev, int *version)
8721e70425SMarc-André Lureau {
8821e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_SCSI_GET_ABI_VERSION, version);
8921e70425SMarc-André Lureau }
9021e70425SMarc-André Lureau 
9121e70425SMarc-André Lureau static int vhost_kernel_set_log_base(struct vhost_dev *dev, uint64_t base,
929a78a5ddSMarc-André Lureau                                      struct vhost_log *log)
93c2bea314SMarc-André Lureau {
94c2bea314SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_SET_LOG_BASE, &base);
95c2bea314SMarc-André Lureau }
96c2bea314SMarc-André Lureau 
9721e70425SMarc-André Lureau static int vhost_kernel_set_mem_table(struct vhost_dev *dev,
9821e70425SMarc-André Lureau                                       struct vhost_memory *mem)
9921e70425SMarc-André Lureau {
10021e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_SET_MEM_TABLE, mem);
10121e70425SMarc-André Lureau }
10221e70425SMarc-André Lureau 
10321e70425SMarc-André Lureau static int vhost_kernel_set_vring_addr(struct vhost_dev *dev,
10421e70425SMarc-André Lureau                                        struct vhost_vring_addr *addr)
10521e70425SMarc-André Lureau {
10621e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_SET_VRING_ADDR, addr);
10721e70425SMarc-André Lureau }
10821e70425SMarc-André Lureau 
10921e70425SMarc-André Lureau static int vhost_kernel_set_vring_endian(struct vhost_dev *dev,
11021e70425SMarc-André Lureau                                          struct vhost_vring_state *ring)
11121e70425SMarc-André Lureau {
11221e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_SET_VRING_ENDIAN, ring);
11321e70425SMarc-André Lureau }
11421e70425SMarc-André Lureau 
11521e70425SMarc-André Lureau static int vhost_kernel_set_vring_num(struct vhost_dev *dev,
11621e70425SMarc-André Lureau                                       struct vhost_vring_state *ring)
11721e70425SMarc-André Lureau {
11821e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_SET_VRING_NUM, ring);
11921e70425SMarc-André Lureau }
12021e70425SMarc-André Lureau 
12121e70425SMarc-André Lureau static int vhost_kernel_set_vring_base(struct vhost_dev *dev,
12221e70425SMarc-André Lureau                                        struct vhost_vring_state *ring)
12321e70425SMarc-André Lureau {
12421e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_SET_VRING_BASE, ring);
12521e70425SMarc-André Lureau }
12621e70425SMarc-André Lureau 
12721e70425SMarc-André Lureau static int vhost_kernel_get_vring_base(struct vhost_dev *dev,
12821e70425SMarc-André Lureau                                        struct vhost_vring_state *ring)
12921e70425SMarc-André Lureau {
13021e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_GET_VRING_BASE, ring);
13121e70425SMarc-André Lureau }
13221e70425SMarc-André Lureau 
13321e70425SMarc-André Lureau static int vhost_kernel_set_vring_kick(struct vhost_dev *dev,
13421e70425SMarc-André Lureau                                        struct vhost_vring_file *file)
13521e70425SMarc-André Lureau {
13621e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_SET_VRING_KICK, file);
13721e70425SMarc-André Lureau }
13821e70425SMarc-André Lureau 
13921e70425SMarc-André Lureau static int vhost_kernel_set_vring_call(struct vhost_dev *dev,
14021e70425SMarc-André Lureau                                        struct vhost_vring_file *file)
14121e70425SMarc-André Lureau {
14221e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_SET_VRING_CALL, file);
14321e70425SMarc-André Lureau }
14421e70425SMarc-André Lureau 
14569e87b32SJason Wang static int vhost_kernel_set_vring_busyloop_timeout(struct vhost_dev *dev,
14669e87b32SJason Wang                                                    struct vhost_vring_state *s)
14769e87b32SJason Wang {
14869e87b32SJason Wang     return vhost_kernel_call(dev, VHOST_SET_VRING_BUSYLOOP_TIMEOUT, s);
14969e87b32SJason Wang }
15069e87b32SJason Wang 
15121e70425SMarc-André Lureau static int vhost_kernel_set_features(struct vhost_dev *dev,
15221e70425SMarc-André Lureau                                      uint64_t features)
15321e70425SMarc-André Lureau {
15421e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_SET_FEATURES, &features);
15521e70425SMarc-André Lureau }
15621e70425SMarc-André Lureau 
15721e70425SMarc-André Lureau static int vhost_kernel_get_features(struct vhost_dev *dev,
15821e70425SMarc-André Lureau                                      uint64_t *features)
15921e70425SMarc-André Lureau {
16021e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_GET_FEATURES, features);
16121e70425SMarc-André Lureau }
16221e70425SMarc-André Lureau 
16321e70425SMarc-André Lureau static int vhost_kernel_set_owner(struct vhost_dev *dev)
16421e70425SMarc-André Lureau {
16521e70425SMarc-André Lureau     return vhost_kernel_call(dev, VHOST_SET_OWNER, NULL);
16621e70425SMarc-André Lureau }
16721e70425SMarc-André Lureau 
16821e70425SMarc-André Lureau static int vhost_kernel_reset_device(struct vhost_dev *dev)
16921e70425SMarc-André Lureau {
17060915dc4SYuanhan Liu     return vhost_kernel_call(dev, VHOST_RESET_OWNER, NULL);
17121e70425SMarc-André Lureau }
17221e70425SMarc-André Lureau 
17321e70425SMarc-André Lureau static int vhost_kernel_get_vq_index(struct vhost_dev *dev, int idx)
17421e70425SMarc-André Lureau {
17521e70425SMarc-André Lureau     assert(idx >= dev->vq_index && idx < dev->vq_index + dev->nvqs);
17621e70425SMarc-André Lureau 
17721e70425SMarc-André Lureau     return idx - dev->vq_index;
17821e70425SMarc-André Lureau }
17921e70425SMarc-André Lureau 
180fc0b9b0eSStefan Hajnoczi #ifdef CONFIG_VHOST_VSOCK
181fc0b9b0eSStefan Hajnoczi static int vhost_kernel_vsock_set_guest_cid(struct vhost_dev *dev,
182fc0b9b0eSStefan Hajnoczi                                             uint64_t guest_cid)
183fc0b9b0eSStefan Hajnoczi {
184fc0b9b0eSStefan Hajnoczi     return vhost_kernel_call(dev, VHOST_VSOCK_SET_GUEST_CID, &guest_cid);
185fc0b9b0eSStefan Hajnoczi }
186fc0b9b0eSStefan Hajnoczi 
187fc0b9b0eSStefan Hajnoczi static int vhost_kernel_vsock_set_running(struct vhost_dev *dev, int start)
188fc0b9b0eSStefan Hajnoczi {
189fc0b9b0eSStefan Hajnoczi     return vhost_kernel_call(dev, VHOST_VSOCK_SET_RUNNING, &start);
190fc0b9b0eSStefan Hajnoczi }
191fc0b9b0eSStefan Hajnoczi #endif /* CONFIG_VHOST_VSOCK */
192fc0b9b0eSStefan Hajnoczi 
193c471ad0eSJason Wang static void vhost_kernel_iotlb_read(void *opaque)
194c471ad0eSJason Wang {
195c471ad0eSJason Wang     struct vhost_dev *dev = opaque;
196c471ad0eSJason Wang     struct vhost_msg msg;
197c471ad0eSJason Wang     ssize_t len;
198c471ad0eSJason Wang 
199c471ad0eSJason Wang     while ((len = read((uintptr_t)dev->opaque, &msg, sizeof msg)) > 0) {
200c471ad0eSJason Wang         if (len < sizeof msg) {
201c471ad0eSJason Wang             error_report("Wrong vhost message len: %d", (int)len);
202c471ad0eSJason Wang             break;
203c471ad0eSJason Wang         }
204c471ad0eSJason Wang         if (msg.type != VHOST_IOTLB_MSG) {
205c471ad0eSJason Wang             error_report("Unknown vhost iotlb message type");
206c471ad0eSJason Wang             break;
207c471ad0eSJason Wang         }
208020e571bSMaxime Coquelin 
209020e571bSMaxime Coquelin         vhost_backend_handle_iotlb_msg(dev, &msg.iotlb);
210c471ad0eSJason Wang     }
211c471ad0eSJason Wang }
212c471ad0eSJason Wang 
213020e571bSMaxime Coquelin static int vhost_kernel_send_device_iotlb_msg(struct vhost_dev *dev,
214020e571bSMaxime Coquelin                                               struct vhost_iotlb_msg *imsg)
215c471ad0eSJason Wang {
216c471ad0eSJason Wang     struct vhost_msg msg;
217c471ad0eSJason Wang 
218020e571bSMaxime Coquelin     msg.type = VHOST_IOTLB_MSG;
219020e571bSMaxime Coquelin     msg.iotlb = *imsg;
220c471ad0eSJason Wang 
221c471ad0eSJason Wang     if (write((uintptr_t)dev->opaque, &msg, sizeof msg) != sizeof msg) {
222c471ad0eSJason Wang         error_report("Fail to update device iotlb");
223c471ad0eSJason Wang         return -EFAULT;
224c471ad0eSJason Wang     }
225c471ad0eSJason Wang 
226c471ad0eSJason Wang     return 0;
227c471ad0eSJason Wang }
228c471ad0eSJason Wang 
229c471ad0eSJason Wang static void vhost_kernel_set_iotlb_callback(struct vhost_dev *dev,
230c471ad0eSJason Wang                                            int enabled)
231c471ad0eSJason Wang {
232c471ad0eSJason Wang     if (enabled)
233c471ad0eSJason Wang         qemu_set_fd_handler((uintptr_t)dev->opaque,
234c471ad0eSJason Wang                             vhost_kernel_iotlb_read, NULL, dev);
235c471ad0eSJason Wang     else
236c471ad0eSJason Wang         qemu_set_fd_handler((uintptr_t)dev->opaque, NULL, NULL, NULL);
237c471ad0eSJason Wang }
238c471ad0eSJason Wang 
2391a1bfac9SNikolay Nikolaev static const VhostOps kernel_ops = {
2401a1bfac9SNikolay Nikolaev         .backend_type = VHOST_BACKEND_TYPE_KERNEL,
2411a1bfac9SNikolay Nikolaev         .vhost_backend_init = vhost_kernel_init,
242fc57fd99SYuanhan Liu         .vhost_backend_cleanup = vhost_kernel_cleanup,
2432ce68e4cSIgor Mammedov         .vhost_backend_memslots_limit = vhost_kernel_memslots_limit,
24421e70425SMarc-André Lureau         .vhost_net_set_backend = vhost_kernel_net_set_backend,
24521e70425SMarc-André Lureau         .vhost_scsi_set_endpoint = vhost_kernel_scsi_set_endpoint,
24621e70425SMarc-André Lureau         .vhost_scsi_clear_endpoint = vhost_kernel_scsi_clear_endpoint,
24721e70425SMarc-André Lureau         .vhost_scsi_get_abi_version = vhost_kernel_scsi_get_abi_version,
24821e70425SMarc-André Lureau         .vhost_set_log_base = vhost_kernel_set_log_base,
24921e70425SMarc-André Lureau         .vhost_set_mem_table = vhost_kernel_set_mem_table,
25021e70425SMarc-André Lureau         .vhost_set_vring_addr = vhost_kernel_set_vring_addr,
25121e70425SMarc-André Lureau         .vhost_set_vring_endian = vhost_kernel_set_vring_endian,
25221e70425SMarc-André Lureau         .vhost_set_vring_num = vhost_kernel_set_vring_num,
25321e70425SMarc-André Lureau         .vhost_set_vring_base = vhost_kernel_set_vring_base,
25421e70425SMarc-André Lureau         .vhost_get_vring_base = vhost_kernel_get_vring_base,
25521e70425SMarc-André Lureau         .vhost_set_vring_kick = vhost_kernel_set_vring_kick,
25621e70425SMarc-André Lureau         .vhost_set_vring_call = vhost_kernel_set_vring_call,
25769e87b32SJason Wang         .vhost_set_vring_busyloop_timeout =
25869e87b32SJason Wang                                 vhost_kernel_set_vring_busyloop_timeout,
25921e70425SMarc-André Lureau         .vhost_set_features = vhost_kernel_set_features,
26021e70425SMarc-André Lureau         .vhost_get_features = vhost_kernel_get_features,
26121e70425SMarc-André Lureau         .vhost_set_owner = vhost_kernel_set_owner,
26221e70425SMarc-André Lureau         .vhost_reset_device = vhost_kernel_reset_device,
26321e70425SMarc-André Lureau         .vhost_get_vq_index = vhost_kernel_get_vq_index,
264fc0b9b0eSStefan Hajnoczi #ifdef CONFIG_VHOST_VSOCK
265fc0b9b0eSStefan Hajnoczi         .vhost_vsock_set_guest_cid = vhost_kernel_vsock_set_guest_cid,
266fc0b9b0eSStefan Hajnoczi         .vhost_vsock_set_running = vhost_kernel_vsock_set_running,
267fc0b9b0eSStefan Hajnoczi #endif /* CONFIG_VHOST_VSOCK */
268c471ad0eSJason Wang         .vhost_set_iotlb_callback = vhost_kernel_set_iotlb_callback,
269020e571bSMaxime Coquelin         .vhost_send_device_iotlb_msg = vhost_kernel_send_device_iotlb_msg,
2701a1bfac9SNikolay Nikolaev };
27118658a3cSPaolo Bonzini #endif
2721a1bfac9SNikolay Nikolaev 
2731a1bfac9SNikolay Nikolaev int vhost_set_backend_type(struct vhost_dev *dev, VhostBackendType backend_type)
2741a1bfac9SNikolay Nikolaev {
2751a1bfac9SNikolay Nikolaev     int r = 0;
2761a1bfac9SNikolay Nikolaev 
2771a1bfac9SNikolay Nikolaev     switch (backend_type) {
278*299e6f19SPaolo Bonzini #ifdef CONFIG_VHOST_KERNEL
2791a1bfac9SNikolay Nikolaev     case VHOST_BACKEND_TYPE_KERNEL:
2801a1bfac9SNikolay Nikolaev         dev->vhost_ops = &kernel_ops;
2811a1bfac9SNikolay Nikolaev         break;
28218658a3cSPaolo Bonzini #endif
28318658a3cSPaolo Bonzini #ifdef CONFIG_VHOST_USER
2845f6f6664SNikolay Nikolaev     case VHOST_BACKEND_TYPE_USER:
2855f6f6664SNikolay Nikolaev         dev->vhost_ops = &user_ops;
2865f6f6664SNikolay Nikolaev         break;
28718658a3cSPaolo Bonzini #endif
2881a1bfac9SNikolay Nikolaev     default:
289c6547661SGonglei         error_report("Unknown vhost backend type");
2901a1bfac9SNikolay Nikolaev         r = -1;
2911a1bfac9SNikolay Nikolaev     }
2921a1bfac9SNikolay Nikolaev 
2931a1bfac9SNikolay Nikolaev     return r;
2941a1bfac9SNikolay Nikolaev }
295020e571bSMaxime Coquelin 
296020e571bSMaxime Coquelin int vhost_backend_update_device_iotlb(struct vhost_dev *dev,
297020e571bSMaxime Coquelin                                              uint64_t iova, uint64_t uaddr,
298020e571bSMaxime Coquelin                                              uint64_t len,
299020e571bSMaxime Coquelin                                              IOMMUAccessFlags perm)
300020e571bSMaxime Coquelin {
301020e571bSMaxime Coquelin     struct vhost_iotlb_msg imsg;
302020e571bSMaxime Coquelin 
303020e571bSMaxime Coquelin     imsg.iova =  iova;
304020e571bSMaxime Coquelin     imsg.uaddr = uaddr;
305020e571bSMaxime Coquelin     imsg.size = len;
306020e571bSMaxime Coquelin     imsg.type = VHOST_IOTLB_UPDATE;
307020e571bSMaxime Coquelin 
308020e571bSMaxime Coquelin     switch (perm) {
309020e571bSMaxime Coquelin     case IOMMU_RO:
310020e571bSMaxime Coquelin         imsg.perm = VHOST_ACCESS_RO;
311020e571bSMaxime Coquelin         break;
312020e571bSMaxime Coquelin     case IOMMU_WO:
313020e571bSMaxime Coquelin         imsg.perm = VHOST_ACCESS_WO;
314020e571bSMaxime Coquelin         break;
315020e571bSMaxime Coquelin     case IOMMU_RW:
316020e571bSMaxime Coquelin         imsg.perm = VHOST_ACCESS_RW;
317020e571bSMaxime Coquelin         break;
318020e571bSMaxime Coquelin     default:
319020e571bSMaxime Coquelin         return -EINVAL;
320020e571bSMaxime Coquelin     }
321020e571bSMaxime Coquelin 
322384b557dSMaxime Coquelin     if (dev->vhost_ops && dev->vhost_ops->vhost_send_device_iotlb_msg)
323020e571bSMaxime Coquelin         return dev->vhost_ops->vhost_send_device_iotlb_msg(dev, &imsg);
324384b557dSMaxime Coquelin 
325384b557dSMaxime Coquelin     return -ENODEV;
326020e571bSMaxime Coquelin }
327020e571bSMaxime Coquelin 
328020e571bSMaxime Coquelin int vhost_backend_invalidate_device_iotlb(struct vhost_dev *dev,
329020e571bSMaxime Coquelin                                                  uint64_t iova, uint64_t len)
330020e571bSMaxime Coquelin {
331020e571bSMaxime Coquelin     struct vhost_iotlb_msg imsg;
332020e571bSMaxime Coquelin 
333020e571bSMaxime Coquelin     imsg.iova = iova;
334020e571bSMaxime Coquelin     imsg.size = len;
335020e571bSMaxime Coquelin     imsg.type = VHOST_IOTLB_INVALIDATE;
336020e571bSMaxime Coquelin 
337384b557dSMaxime Coquelin     if (dev->vhost_ops && dev->vhost_ops->vhost_send_device_iotlb_msg)
338020e571bSMaxime Coquelin         return dev->vhost_ops->vhost_send_device_iotlb_msg(dev, &imsg);
339384b557dSMaxime Coquelin 
340384b557dSMaxime Coquelin     return -ENODEV;
341020e571bSMaxime Coquelin }
342020e571bSMaxime Coquelin 
343020e571bSMaxime Coquelin int vhost_backend_handle_iotlb_msg(struct vhost_dev *dev,
344020e571bSMaxime Coquelin                                           struct vhost_iotlb_msg *imsg)
345020e571bSMaxime Coquelin {
346020e571bSMaxime Coquelin     int ret = 0;
347020e571bSMaxime Coquelin 
348020e571bSMaxime Coquelin     switch (imsg->type) {
349020e571bSMaxime Coquelin     case VHOST_IOTLB_MISS:
350020e571bSMaxime Coquelin         ret = vhost_device_iotlb_miss(dev, imsg->iova,
351020e571bSMaxime Coquelin                                       imsg->perm != VHOST_ACCESS_RO);
352020e571bSMaxime Coquelin         break;
353020e571bSMaxime Coquelin     case VHOST_IOTLB_ACCESS_FAIL:
354020e571bSMaxime Coquelin         /* FIXME: report device iotlb error */
355020e571bSMaxime Coquelin         error_report("Access failure IOTLB message type not supported");
356020e571bSMaxime Coquelin         ret = -ENOTSUP;
357020e571bSMaxime Coquelin         break;
358020e571bSMaxime Coquelin     case VHOST_IOTLB_UPDATE:
359020e571bSMaxime Coquelin     case VHOST_IOTLB_INVALIDATE:
360020e571bSMaxime Coquelin     default:
361020e571bSMaxime Coquelin         error_report("Unexpected IOTLB message type");
362020e571bSMaxime Coquelin         ret = -EINVAL;
363020e571bSMaxime Coquelin         break;
364020e571bSMaxime Coquelin     }
365020e571bSMaxime Coquelin 
366020e571bSMaxime Coquelin     return ret;
367020e571bSMaxime Coquelin }
368