15566a3e3SBruce Richardson /* SPDX-License-Identifier: BSD-3-Clause 25566a3e3SBruce Richardson * Copyright(c) 2016 Intel Corporation 3e3b43481SJianfeng Tan */ 4e3b43481SJianfeng Tan 5e3b43481SJianfeng Tan #include <sys/types.h> 6e3b43481SJianfeng Tan #include <sys/stat.h> 7e3b43481SJianfeng Tan #include <fcntl.h> 8e3b43481SJianfeng Tan #include <unistd.h> 9e3b43481SJianfeng Tan 10e3b43481SJianfeng Tan #include <rte_memory.h> 11e3b43481SJianfeng Tan #include <rte_eal_memconfig.h> 12e3b43481SJianfeng Tan 13e3b43481SJianfeng Tan #include "vhost.h" 14e3b43481SJianfeng Tan #include "virtio_user_dev.h" 15e3b43481SJianfeng Tan #include "vhost_kernel_tap.h" 16e3b43481SJianfeng Tan 17e3b43481SJianfeng Tan struct vhost_memory_kernel { 18e3b43481SJianfeng Tan uint32_t nregions; 19e3b43481SJianfeng Tan uint32_t padding; 20e3b43481SJianfeng Tan struct vhost_memory_region regions[0]; 21e3b43481SJianfeng Tan }; 22e3b43481SJianfeng Tan 23e3b43481SJianfeng Tan /* vhost kernel ioctls */ 24e3b43481SJianfeng Tan #define VHOST_VIRTIO 0xAF 25e3b43481SJianfeng Tan #define VHOST_GET_FEATURES _IOR(VHOST_VIRTIO, 0x00, __u64) 26e3b43481SJianfeng Tan #define VHOST_SET_FEATURES _IOW(VHOST_VIRTIO, 0x00, __u64) 27e3b43481SJianfeng Tan #define VHOST_SET_OWNER _IO(VHOST_VIRTIO, 0x01) 28e3b43481SJianfeng Tan #define VHOST_RESET_OWNER _IO(VHOST_VIRTIO, 0x02) 29e3b43481SJianfeng Tan #define VHOST_SET_MEM_TABLE _IOW(VHOST_VIRTIO, 0x03, struct vhost_memory_kernel) 30e3b43481SJianfeng Tan #define VHOST_SET_LOG_BASE _IOW(VHOST_VIRTIO, 0x04, __u64) 31e3b43481SJianfeng Tan #define VHOST_SET_LOG_FD _IOW(VHOST_VIRTIO, 0x07, int) 32e3b43481SJianfeng Tan #define VHOST_SET_VRING_NUM _IOW(VHOST_VIRTIO, 0x10, struct vhost_vring_state) 33e3b43481SJianfeng Tan #define VHOST_SET_VRING_ADDR _IOW(VHOST_VIRTIO, 0x11, struct vhost_vring_addr) 34e3b43481SJianfeng Tan #define VHOST_SET_VRING_BASE _IOW(VHOST_VIRTIO, 0x12, struct vhost_vring_state) 35e3b43481SJianfeng Tan #define VHOST_GET_VRING_BASE _IOWR(VHOST_VIRTIO, 0x12, struct vhost_vring_state) 36e3b43481SJianfeng Tan #define VHOST_SET_VRING_KICK _IOW(VHOST_VIRTIO, 0x20, struct vhost_vring_file) 37e3b43481SJianfeng Tan #define VHOST_SET_VRING_CALL _IOW(VHOST_VIRTIO, 0x21, struct vhost_vring_file) 38e3b43481SJianfeng Tan #define VHOST_SET_VRING_ERR _IOW(VHOST_VIRTIO, 0x22, struct vhost_vring_file) 39e3b43481SJianfeng Tan #define VHOST_NET_SET_BACKEND _IOW(VHOST_VIRTIO, 0x30, struct vhost_vring_file) 40e3b43481SJianfeng Tan 41e3b43481SJianfeng Tan static uint64_t max_regions = 64; 42e3b43481SJianfeng Tan 43e3b43481SJianfeng Tan static void 44e3b43481SJianfeng Tan get_vhost_kernel_max_regions(void) 45e3b43481SJianfeng Tan { 46e3b43481SJianfeng Tan int fd; 47e3b43481SJianfeng Tan char buf[20] = {'\0'}; 48e3b43481SJianfeng Tan 49e3b43481SJianfeng Tan fd = open("/sys/module/vhost/parameters/max_mem_regions", O_RDONLY); 50e3b43481SJianfeng Tan if (fd < 0) 51e3b43481SJianfeng Tan return; 52e3b43481SJianfeng Tan 53e3b43481SJianfeng Tan if (read(fd, buf, sizeof(buf) - 1) > 0) 54e3b43481SJianfeng Tan max_regions = strtoull(buf, NULL, 10); 55e3b43481SJianfeng Tan 56e3b43481SJianfeng Tan close(fd); 57e3b43481SJianfeng Tan } 58e3b43481SJianfeng Tan 59e3b43481SJianfeng Tan static uint64_t vhost_req_user_to_kernel[] = { 60e3b43481SJianfeng Tan [VHOST_USER_SET_OWNER] = VHOST_SET_OWNER, 61e3b43481SJianfeng Tan [VHOST_USER_RESET_OWNER] = VHOST_RESET_OWNER, 62e3b43481SJianfeng Tan [VHOST_USER_SET_FEATURES] = VHOST_SET_FEATURES, 63e3b43481SJianfeng Tan [VHOST_USER_GET_FEATURES] = VHOST_GET_FEATURES, 64e3b43481SJianfeng Tan [VHOST_USER_SET_VRING_CALL] = VHOST_SET_VRING_CALL, 65e3b43481SJianfeng Tan [VHOST_USER_SET_VRING_NUM] = VHOST_SET_VRING_NUM, 66e3b43481SJianfeng Tan [VHOST_USER_SET_VRING_BASE] = VHOST_SET_VRING_BASE, 67e3b43481SJianfeng Tan [VHOST_USER_GET_VRING_BASE] = VHOST_GET_VRING_BASE, 68e3b43481SJianfeng Tan [VHOST_USER_SET_VRING_ADDR] = VHOST_SET_VRING_ADDR, 69e3b43481SJianfeng Tan [VHOST_USER_SET_VRING_KICK] = VHOST_SET_VRING_KICK, 70e3b43481SJianfeng Tan [VHOST_USER_SET_MEM_TABLE] = VHOST_SET_MEM_TABLE, 71e3b43481SJianfeng Tan }; 72e3b43481SJianfeng Tan 73746c346dSAnatoly Burakov struct walk_arg { 74746c346dSAnatoly Burakov struct vhost_memory_kernel *vm; 75746c346dSAnatoly Burakov uint32_t region_nr; 76746c346dSAnatoly Burakov }; 77746c346dSAnatoly Burakov static int 7866cc45e2SAnatoly Burakov add_memory_region(const struct rte_memseg_list *msl __rte_unused, 7966cc45e2SAnatoly Burakov const struct rte_memseg *ms, size_t len, void *arg) 80746c346dSAnatoly Burakov { 81746c346dSAnatoly Burakov struct walk_arg *wa = arg; 82746c346dSAnatoly Burakov struct vhost_memory_region *mr; 83746c346dSAnatoly Burakov void *start_addr; 84746c346dSAnatoly Burakov 85746c346dSAnatoly Burakov if (wa->region_nr >= max_regions) 86746c346dSAnatoly Burakov return -1; 87746c346dSAnatoly Burakov 88746c346dSAnatoly Burakov mr = &wa->vm->regions[wa->region_nr++]; 89746c346dSAnatoly Burakov start_addr = ms->addr; 90746c346dSAnatoly Burakov 91746c346dSAnatoly Burakov mr->guest_phys_addr = (uint64_t)(uintptr_t)start_addr; 92746c346dSAnatoly Burakov mr->userspace_addr = (uint64_t)(uintptr_t)start_addr; 93746c346dSAnatoly Burakov mr->memory_size = len; 94746c346dSAnatoly Burakov mr->mmap_offset = 0; 95746c346dSAnatoly Burakov 96746c346dSAnatoly Burakov return 0; 97746c346dSAnatoly Burakov } 98746c346dSAnatoly Burakov 99e3b43481SJianfeng Tan /* By default, vhost kernel module allows 64 regions, but DPDK allows 100e3b43481SJianfeng Tan * 256 segments. As a relief, below function merges those virtually 101e3b43481SJianfeng Tan * adjacent memsegs into one region. 102e3b43481SJianfeng Tan */ 103e3b43481SJianfeng Tan static struct vhost_memory_kernel * 104e3b43481SJianfeng Tan prepare_vhost_memory_kernel(void) 105e3b43481SJianfeng Tan { 106e3b43481SJianfeng Tan struct vhost_memory_kernel *vm; 107746c346dSAnatoly Burakov struct walk_arg wa; 108e3b43481SJianfeng Tan 109e3b43481SJianfeng Tan vm = malloc(sizeof(struct vhost_memory_kernel) + 110e3b43481SJianfeng Tan max_regions * 111e3b43481SJianfeng Tan sizeof(struct vhost_memory_region)); 1121e9057a9SJianfeng Tan if (!vm) 1131e9057a9SJianfeng Tan return NULL; 114e3b43481SJianfeng Tan 115746c346dSAnatoly Burakov wa.region_nr = 0; 116746c346dSAnatoly Burakov wa.vm = vm; 117e3b43481SJianfeng Tan 118746c346dSAnatoly Burakov if (rte_memseg_contig_walk(add_memory_region, &wa) < 0) { 119e3b43481SJianfeng Tan free(vm); 120e3b43481SJianfeng Tan return NULL; 121e3b43481SJianfeng Tan } 122e3b43481SJianfeng Tan 123746c346dSAnatoly Burakov vm->nregions = wa.region_nr; 124e3b43481SJianfeng Tan vm->padding = 0; 125e3b43481SJianfeng Tan return vm; 126e3b43481SJianfeng Tan } 127e3b43481SJianfeng Tan 1285e97e420SJianfeng Tan /* with below features, vhost kernel does not need to do the checksum and TSO, 1295e97e420SJianfeng Tan * these info will be passed to virtio_user through virtio net header. 1305e97e420SJianfeng Tan */ 1315e97e420SJianfeng Tan #define VHOST_KERNEL_GUEST_OFFLOADS_MASK \ 1325e97e420SJianfeng Tan ((1ULL << VIRTIO_NET_F_GUEST_CSUM) | \ 1335e97e420SJianfeng Tan (1ULL << VIRTIO_NET_F_GUEST_TSO4) | \ 1345e97e420SJianfeng Tan (1ULL << VIRTIO_NET_F_GUEST_TSO6) | \ 1355e97e420SJianfeng Tan (1ULL << VIRTIO_NET_F_GUEST_ECN) | \ 1365e97e420SJianfeng Tan (1ULL << VIRTIO_NET_F_GUEST_UFO)) 1375e97e420SJianfeng Tan 1385e97e420SJianfeng Tan /* with below features, when flows from virtio_user to vhost kernel 1395e97e420SJianfeng Tan * (1) if flows goes up through the kernel networking stack, it does not need 1405e97e420SJianfeng Tan * to verify checksum, which can save CPU cycles; 1415e97e420SJianfeng Tan * (2) if flows goes through a Linux bridge and outside from an interface 1425e97e420SJianfeng Tan * (kernel driver), checksum and TSO will be done by GSO in kernel or even 1435e97e420SJianfeng Tan * offloaded into real physical device. 1445e97e420SJianfeng Tan */ 1455e97e420SJianfeng Tan #define VHOST_KERNEL_HOST_OFFLOADS_MASK \ 1465e97e420SJianfeng Tan ((1ULL << VIRTIO_NET_F_HOST_TSO4) | \ 1475e97e420SJianfeng Tan (1ULL << VIRTIO_NET_F_HOST_TSO6) | \ 1485e97e420SJianfeng Tan (1ULL << VIRTIO_NET_F_CSUM)) 1495e97e420SJianfeng Tan 150*1db4d233SEric Zhang static unsigned int 151*1db4d233SEric Zhang tap_support_features(void) 152be7a4707SJianfeng Tan { 153be7a4707SJianfeng Tan int tapfd; 154be7a4707SJianfeng Tan unsigned int tap_features; 155be7a4707SJianfeng Tan 156be7a4707SJianfeng Tan tapfd = open(PATH_NET_TUN, O_RDWR); 157be7a4707SJianfeng Tan if (tapfd < 0) { 158be7a4707SJianfeng Tan PMD_DRV_LOG(ERR, "fail to open %s: %s", 159be7a4707SJianfeng Tan PATH_NET_TUN, strerror(errno)); 160be7a4707SJianfeng Tan return -1; 161be7a4707SJianfeng Tan } 162be7a4707SJianfeng Tan 163be7a4707SJianfeng Tan if (ioctl(tapfd, TUNGETFEATURES, &tap_features) == -1) { 164be7a4707SJianfeng Tan PMD_DRV_LOG(ERR, "TUNGETFEATURES failed: %s", strerror(errno)); 165be7a4707SJianfeng Tan close(tapfd); 166be7a4707SJianfeng Tan return -1; 167be7a4707SJianfeng Tan } 168be7a4707SJianfeng Tan 169be7a4707SJianfeng Tan close(tapfd); 170*1db4d233SEric Zhang return tap_features; 171be7a4707SJianfeng Tan } 172be7a4707SJianfeng Tan 173be7a4707SJianfeng Tan static int 174e3b43481SJianfeng Tan vhost_kernel_ioctl(struct virtio_user_dev *dev, 175e3b43481SJianfeng Tan enum vhost_user_request req, 176e3b43481SJianfeng Tan void *arg) 177e3b43481SJianfeng Tan { 178e3b43481SJianfeng Tan int ret = -1; 179e3b43481SJianfeng Tan unsigned int i; 180e3b43481SJianfeng Tan uint64_t req_kernel; 181e3b43481SJianfeng Tan struct vhost_memory_kernel *vm = NULL; 182be7a4707SJianfeng Tan int vhostfd; 183be7a4707SJianfeng Tan unsigned int queue_sel; 184*1db4d233SEric Zhang unsigned int features; 185e3b43481SJianfeng Tan 186e3b43481SJianfeng Tan PMD_DRV_LOG(INFO, "%s", vhost_msg_strings[req]); 187e3b43481SJianfeng Tan 188e3b43481SJianfeng Tan req_kernel = vhost_req_user_to_kernel[req]; 189e3b43481SJianfeng Tan 190e3b43481SJianfeng Tan if (req_kernel == VHOST_SET_MEM_TABLE) { 191e3b43481SJianfeng Tan vm = prepare_vhost_memory_kernel(); 192e3b43481SJianfeng Tan if (!vm) 193e3b43481SJianfeng Tan return -1; 194e3b43481SJianfeng Tan arg = (void *)vm; 195e3b43481SJianfeng Tan } 196e3b43481SJianfeng Tan 1975e97e420SJianfeng Tan if (req_kernel == VHOST_SET_FEATURES) { 198e3b43481SJianfeng Tan /* We don't need memory protection here */ 199e3b43481SJianfeng Tan *(uint64_t *)arg &= ~(1ULL << VIRTIO_F_IOMMU_PLATFORM); 200e3b43481SJianfeng Tan 2015e97e420SJianfeng Tan /* VHOST kernel does not know about below flags */ 2025e97e420SJianfeng Tan *(uint64_t *)arg &= ~VHOST_KERNEL_GUEST_OFFLOADS_MASK; 2035e97e420SJianfeng Tan *(uint64_t *)arg &= ~VHOST_KERNEL_HOST_OFFLOADS_MASK; 204be7a4707SJianfeng Tan 205be7a4707SJianfeng Tan *(uint64_t *)arg &= ~(1ULL << VIRTIO_NET_F_MQ); 2065e97e420SJianfeng Tan } 2075e97e420SJianfeng Tan 208be7a4707SJianfeng Tan switch (req_kernel) { 209be7a4707SJianfeng Tan case VHOST_SET_VRING_NUM: 210be7a4707SJianfeng Tan case VHOST_SET_VRING_ADDR: 211be7a4707SJianfeng Tan case VHOST_SET_VRING_BASE: 212be7a4707SJianfeng Tan case VHOST_GET_VRING_BASE: 213be7a4707SJianfeng Tan case VHOST_SET_VRING_KICK: 214be7a4707SJianfeng Tan case VHOST_SET_VRING_CALL: 215be7a4707SJianfeng Tan queue_sel = *(unsigned int *)arg; 216be7a4707SJianfeng Tan vhostfd = dev->vhostfds[queue_sel / 2]; 217be7a4707SJianfeng Tan *(unsigned int *)arg = queue_sel % 2; 218be7a4707SJianfeng Tan PMD_DRV_LOG(DEBUG, "vhostfd=%d, index=%u", 219be7a4707SJianfeng Tan vhostfd, *(unsigned int *)arg); 220be7a4707SJianfeng Tan break; 221be7a4707SJianfeng Tan default: 222be7a4707SJianfeng Tan vhostfd = -1; 223be7a4707SJianfeng Tan } 224be7a4707SJianfeng Tan if (vhostfd == -1) { 225e3b43481SJianfeng Tan for (i = 0; i < dev->max_queue_pairs; ++i) { 226e3b43481SJianfeng Tan if (dev->vhostfds[i] < 0) 227e3b43481SJianfeng Tan continue; 228e3b43481SJianfeng Tan 229e3b43481SJianfeng Tan ret = ioctl(dev->vhostfds[i], req_kernel, arg); 230e3b43481SJianfeng Tan if (ret < 0) 231e3b43481SJianfeng Tan break; 232e3b43481SJianfeng Tan } 233be7a4707SJianfeng Tan } else { 234be7a4707SJianfeng Tan ret = ioctl(vhostfd, req_kernel, arg); 235be7a4707SJianfeng Tan } 236e3b43481SJianfeng Tan 2375e97e420SJianfeng Tan if (!ret && req_kernel == VHOST_GET_FEATURES) { 238*1db4d233SEric Zhang features = tap_support_features(); 2395e97e420SJianfeng Tan /* with tap as the backend, all these features are supported 2405e97e420SJianfeng Tan * but not claimed by vhost-net, so we add them back when 2415e97e420SJianfeng Tan * reporting to upper layer. 2425e97e420SJianfeng Tan */ 243*1db4d233SEric Zhang if (features & IFF_VNET_HDR) { 2445e97e420SJianfeng Tan *((uint64_t *)arg) |= VHOST_KERNEL_GUEST_OFFLOADS_MASK; 2455e97e420SJianfeng Tan *((uint64_t *)arg) |= VHOST_KERNEL_HOST_OFFLOADS_MASK; 246*1db4d233SEric Zhang } 247be7a4707SJianfeng Tan 248be7a4707SJianfeng Tan /* vhost_kernel will not declare this feature, but it does 249be7a4707SJianfeng Tan * support multi-queue. 250be7a4707SJianfeng Tan */ 251*1db4d233SEric Zhang if (features & IFF_MULTI_QUEUE) 252be7a4707SJianfeng Tan *(uint64_t *)arg |= (1ull << VIRTIO_NET_F_MQ); 2535e97e420SJianfeng Tan } 2545e97e420SJianfeng Tan 255e3b43481SJianfeng Tan if (vm) 256e3b43481SJianfeng Tan free(vm); 257e3b43481SJianfeng Tan 258e3b43481SJianfeng Tan if (ret < 0) 259e3b43481SJianfeng Tan PMD_DRV_LOG(ERR, "%s failed: %s", 260e3b43481SJianfeng Tan vhost_msg_strings[req], strerror(errno)); 261e3b43481SJianfeng Tan 262e3b43481SJianfeng Tan return ret; 263e3b43481SJianfeng Tan } 264e3b43481SJianfeng Tan 265e3b43481SJianfeng Tan /** 266e3b43481SJianfeng Tan * Set up environment to talk with a vhost kernel backend. 267e3b43481SJianfeng Tan * 268e3b43481SJianfeng Tan * @return 269e3b43481SJianfeng Tan * - (-1) if fail to set up; 270e3b43481SJianfeng Tan * - (>=0) if successful. 271e3b43481SJianfeng Tan */ 272e3b43481SJianfeng Tan static int 273e3b43481SJianfeng Tan vhost_kernel_setup(struct virtio_user_dev *dev) 274e3b43481SJianfeng Tan { 275e3b43481SJianfeng Tan int vhostfd; 276e3b43481SJianfeng Tan uint32_t i; 277e3b43481SJianfeng Tan 278e3b43481SJianfeng Tan get_vhost_kernel_max_regions(); 279e3b43481SJianfeng Tan 280e3b43481SJianfeng Tan for (i = 0; i < dev->max_queue_pairs; ++i) { 281e3b43481SJianfeng Tan vhostfd = open(dev->path, O_RDWR); 282e3b43481SJianfeng Tan if (vhostfd < 0) { 283e3b43481SJianfeng Tan PMD_DRV_LOG(ERR, "fail to open %s, %s", 284e3b43481SJianfeng Tan dev->path, strerror(errno)); 285e3b43481SJianfeng Tan return -1; 286e3b43481SJianfeng Tan } 287e3b43481SJianfeng Tan 288e3b43481SJianfeng Tan dev->vhostfds[i] = vhostfd; 289e3b43481SJianfeng Tan } 290e3b43481SJianfeng Tan 291e3b43481SJianfeng Tan return 0; 292e3b43481SJianfeng Tan } 293e3b43481SJianfeng Tan 294e3b43481SJianfeng Tan static int 295e3b43481SJianfeng Tan vhost_kernel_set_backend(int vhostfd, int tapfd) 296e3b43481SJianfeng Tan { 297e3b43481SJianfeng Tan struct vhost_vring_file f; 298e3b43481SJianfeng Tan 299e3b43481SJianfeng Tan f.fd = tapfd; 300e3b43481SJianfeng Tan f.index = 0; 301e3b43481SJianfeng Tan if (ioctl(vhostfd, VHOST_NET_SET_BACKEND, &f) < 0) { 302e3b43481SJianfeng Tan PMD_DRV_LOG(ERR, "VHOST_NET_SET_BACKEND fails, %s", 303e3b43481SJianfeng Tan strerror(errno)); 304e3b43481SJianfeng Tan return -1; 305e3b43481SJianfeng Tan } 306e3b43481SJianfeng Tan 307e3b43481SJianfeng Tan f.index = 1; 308e3b43481SJianfeng Tan if (ioctl(vhostfd, VHOST_NET_SET_BACKEND, &f) < 0) { 309e3b43481SJianfeng Tan PMD_DRV_LOG(ERR, "VHOST_NET_SET_BACKEND fails, %s", 310e3b43481SJianfeng Tan strerror(errno)); 311e3b43481SJianfeng Tan return -1; 312e3b43481SJianfeng Tan } 313e3b43481SJianfeng Tan 314e3b43481SJianfeng Tan return 0; 315e3b43481SJianfeng Tan } 316e3b43481SJianfeng Tan 317e3b43481SJianfeng Tan static int 318e3b43481SJianfeng Tan vhost_kernel_enable_queue_pair(struct virtio_user_dev *dev, 319e3b43481SJianfeng Tan uint16_t pair_idx, 320e3b43481SJianfeng Tan int enable) 321e3b43481SJianfeng Tan { 322e3b43481SJianfeng Tan int hdr_size; 323e3b43481SJianfeng Tan int vhostfd; 324e3b43481SJianfeng Tan int tapfd; 325be7a4707SJianfeng Tan int req_mq = (dev->max_queue_pairs > 1); 326e3b43481SJianfeng Tan 327e3b43481SJianfeng Tan vhostfd = dev->vhostfds[pair_idx]; 328e3b43481SJianfeng Tan 329e3b43481SJianfeng Tan if (!enable) { 330250c9965SWenfeng Liu if (dev->tapfds[pair_idx] >= 0) { 331e3b43481SJianfeng Tan close(dev->tapfds[pair_idx]); 332e3b43481SJianfeng Tan dev->tapfds[pair_idx] = -1; 333e3b43481SJianfeng Tan } 334e3b43481SJianfeng Tan return vhost_kernel_set_backend(vhostfd, -1); 335e3b43481SJianfeng Tan } else if (dev->tapfds[pair_idx] >= 0) { 336e3b43481SJianfeng Tan return 0; 337e3b43481SJianfeng Tan } 338e3b43481SJianfeng Tan 339e3b43481SJianfeng Tan if ((dev->features & (1ULL << VIRTIO_NET_F_MRG_RXBUF)) || 340e3b43481SJianfeng Tan (dev->features & (1ULL << VIRTIO_F_VERSION_1))) 341e3b43481SJianfeng Tan hdr_size = sizeof(struct virtio_net_hdr_mrg_rxbuf); 342e3b43481SJianfeng Tan else 343e3b43481SJianfeng Tan hdr_size = sizeof(struct virtio_net_hdr); 344e3b43481SJianfeng Tan 345791b43e0SNing Li tapfd = vhost_kernel_open_tap(&dev->ifname, hdr_size, req_mq, 346*1db4d233SEric Zhang (char *)dev->mac_addr, dev->features); 347e3b43481SJianfeng Tan if (tapfd < 0) { 348e3b43481SJianfeng Tan PMD_DRV_LOG(ERR, "fail to open tap for vhost kernel"); 349e3b43481SJianfeng Tan return -1; 350e3b43481SJianfeng Tan } 351e3b43481SJianfeng Tan 352e3b43481SJianfeng Tan if (vhost_kernel_set_backend(vhostfd, tapfd) < 0) { 353e3b43481SJianfeng Tan PMD_DRV_LOG(ERR, "fail to set backend for vhost kernel"); 354e3b43481SJianfeng Tan close(tapfd); 355e3b43481SJianfeng Tan return -1; 356e3b43481SJianfeng Tan } 357e3b43481SJianfeng Tan 358e3b43481SJianfeng Tan dev->tapfds[pair_idx] = tapfd; 359e3b43481SJianfeng Tan return 0; 360e3b43481SJianfeng Tan } 361e3b43481SJianfeng Tan 362e3b43481SJianfeng Tan struct virtio_user_backend_ops ops_kernel = { 363e3b43481SJianfeng Tan .setup = vhost_kernel_setup, 364e3b43481SJianfeng Tan .send_request = vhost_kernel_ioctl, 365e3b43481SJianfeng Tan .enable_qp = vhost_kernel_enable_queue_pair 366e3b43481SJianfeng Tan }; 367