131638bcaSCyrill Gorcunov #include "kvm/virtio-pci-dev.h" 2b5ee1ea7SAsias He #include "kvm/virtio-net.h" 34f56d42cSAsias He #include "kvm/virtio.h" 44f56d42cSAsias He #include "kvm/types.h" 54f56d42cSAsias He #include "kvm/mutex.h" 64f56d42cSAsias He #include "kvm/util.h" 74f56d42cSAsias He #include "kvm/kvm.h" 82449f6e3SSasha Levin #include "kvm/irq.h" 9b5ee1ea7SAsias He #include "kvm/uip.h" 1027ab67f5SSasha Levin #include "kvm/ioeventfd.h" 11cb83de6fSSasha Levin #include "kvm/guest_compat.h" 12*78a2a3e2SSasha Levin #include "kvm/virtio-pci.h" 134f56d42cSAsias He 144f56d42cSAsias He #include <linux/virtio_net.h> 154f56d42cSAsias He #include <linux/if_tun.h> 16*78a2a3e2SSasha Levin #include <linux/types.h> 17c229370aSIngo Molnar 18c229370aSIngo Molnar #include <arpa/inet.h> 194f56d42cSAsias He #include <net/if.h> 20c229370aSIngo Molnar 21c229370aSIngo Molnar #include <unistd.h> 224f56d42cSAsias He #include <assert.h> 234f56d42cSAsias He #include <fcntl.h> 24c229370aSIngo Molnar 25cb7202c1SSasha Levin #include <sys/socket.h> 26c229370aSIngo Molnar #include <sys/ioctl.h> 27c229370aSIngo Molnar #include <sys/types.h> 2873b7d038SAmos Kong #include <sys/wait.h> 294f56d42cSAsias He 304f56d42cSAsias He #define VIRTIO_NET_QUEUE_SIZE 128 314f56d42cSAsias He #define VIRTIO_NET_NUM_QUEUES 2 324f56d42cSAsias He #define VIRTIO_NET_RX_QUEUE 0 334f56d42cSAsias He #define VIRTIO_NET_TX_QUEUE 1 344f56d42cSAsias He 35b4fdde6dSAsias He struct net_dev; 36b4fdde6dSAsias He 37*78a2a3e2SSasha Levin extern struct kvm *kvm; 38*78a2a3e2SSasha Levin 39b4fdde6dSAsias He struct net_dev_operations { 40b4fdde6dSAsias He int (*rx)(struct iovec *iov, u16 in, struct net_dev *ndev); 41b4fdde6dSAsias He int (*tx)(struct iovec *iov, u16 in, struct net_dev *ndev); 42b4fdde6dSAsias He }; 43b4fdde6dSAsias He 448626798bSAsias He struct net_dev { 454f56d42cSAsias He pthread_mutex_t mutex; 46*78a2a3e2SSasha Levin struct virtio_pci vpci; 474f56d42cSAsias He 484f56d42cSAsias He struct virt_queue vqs[VIRTIO_NET_NUM_QUEUES]; 49c229370aSIngo Molnar struct virtio_net_config config; 50*78a2a3e2SSasha Levin u32 features; 51cb83de6fSSasha Levin int compat_id; 524f56d42cSAsias He 53c4aa7c02SPekka Enberg pthread_t io_rx_thread; 54c229370aSIngo Molnar pthread_mutex_t io_rx_lock; 55c4aa7c02SPekka Enberg pthread_cond_t io_rx_cond; 56c4aa7c02SPekka Enberg 57c4aa7c02SPekka Enberg pthread_t io_tx_thread; 58c229370aSIngo Molnar pthread_mutex_t io_tx_lock; 59c4aa7c02SPekka Enberg pthread_cond_t io_tx_cond; 60c4aa7c02SPekka Enberg 614f56d42cSAsias He int tap_fd; 624f56d42cSAsias He char tap_name[IFNAMSIZ]; 63bb1a32f1SAsias He 64bb1a32f1SAsias He int mode; 65bb1a32f1SAsias He 66b5ee1ea7SAsias He struct uip_info info; 67b4fdde6dSAsias He struct net_dev_operations *ops; 684f56d42cSAsias He }; 694f56d42cSAsias He 708626798bSAsias He static struct net_dev ndev = { 714f56d42cSAsias He .mutex = PTHREAD_MUTEX_INITIALIZER, 724f56d42cSAsias He 73c229370aSIngo Molnar .config = { 744f56d42cSAsias He .status = VIRTIO_NET_S_LINK_UP, 754f56d42cSAsias He }, 76b5ee1ea7SAsias He .info = { 77b5ee1ea7SAsias He .buf_nr = 20, 78b5ee1ea7SAsias He } 794f56d42cSAsias He }; 804f56d42cSAsias He 81c4aa7c02SPekka Enberg static void *virtio_net_rx_thread(void *p) 824f56d42cSAsias He { 834f56d42cSAsias He struct iovec iov[VIRTIO_NET_QUEUE_SIZE]; 844f56d42cSAsias He struct virt_queue *vq; 8543835ac9SSasha Levin struct kvm *kvm; 863fdf659dSSasha Levin u16 out, in; 873fdf659dSSasha Levin u16 head; 884f56d42cSAsias He int len; 894f56d42cSAsias He 9043835ac9SSasha Levin kvm = p; 91c229370aSIngo Molnar vq = &ndev.vqs[VIRTIO_NET_RX_QUEUE]; 92c4aa7c02SPekka Enberg 93c4aa7c02SPekka Enberg while (1) { 94b5ee1ea7SAsias He 95c229370aSIngo Molnar mutex_lock(&ndev.io_rx_lock); 96c4aa7c02SPekka Enberg if (!virt_queue__available(vq)) 97c229370aSIngo Molnar pthread_cond_wait(&ndev.io_rx_cond, &ndev.io_rx_lock); 98c229370aSIngo Molnar mutex_unlock(&ndev.io_rx_lock); 994f56d42cSAsias He 1004f56d42cSAsias He while (virt_queue__available(vq)) { 101b5ee1ea7SAsias He 10243835ac9SSasha Levin head = virt_queue__get_iov(vq, iov, &out, &in, kvm); 103b5ee1ea7SAsias He 104b4fdde6dSAsias He len = ndev.ops->rx(iov, in, &ndev); 105b5ee1ea7SAsias He 106246c8347SAsias He virt_queue__set_used_elem(vq, head, len); 1077f5ffaf5SAsias He 108c4aa7c02SPekka Enberg /* We should interrupt guest right now, otherwise latency is huge. */ 109*78a2a3e2SSasha Levin virtio_pci__signal_vq(kvm, &ndev.vpci, VIRTIO_NET_RX_QUEUE); 1104f56d42cSAsias He } 1114f56d42cSAsias He 112c4aa7c02SPekka Enberg } 113c4aa7c02SPekka Enberg 114c4aa7c02SPekka Enberg pthread_exit(NULL); 115c4aa7c02SPekka Enberg return NULL; 116c4aa7c02SPekka Enberg 117c4aa7c02SPekka Enberg } 118c4aa7c02SPekka Enberg 119c4aa7c02SPekka Enberg static void *virtio_net_tx_thread(void *p) 1204f56d42cSAsias He { 1214f56d42cSAsias He struct iovec iov[VIRTIO_NET_QUEUE_SIZE]; 1224f56d42cSAsias He struct virt_queue *vq; 12343835ac9SSasha Levin struct kvm *kvm; 1243fdf659dSSasha Levin u16 out, in; 1253fdf659dSSasha Levin u16 head; 1264f56d42cSAsias He int len; 1274f56d42cSAsias He 12843835ac9SSasha Levin kvm = p; 129c229370aSIngo Molnar vq = &ndev.vqs[VIRTIO_NET_TX_QUEUE]; 130c4aa7c02SPekka Enberg 131c4aa7c02SPekka Enberg while (1) { 132c229370aSIngo Molnar mutex_lock(&ndev.io_tx_lock); 133c4aa7c02SPekka Enberg if (!virt_queue__available(vq)) 134c229370aSIngo Molnar pthread_cond_wait(&ndev.io_tx_cond, &ndev.io_tx_lock); 135c229370aSIngo Molnar mutex_unlock(&ndev.io_tx_lock); 1364f56d42cSAsias He 1374f56d42cSAsias He while (virt_queue__available(vq)) { 138b5ee1ea7SAsias He 13943835ac9SSasha Levin head = virt_queue__get_iov(vq, iov, &out, &in, kvm); 140b5ee1ea7SAsias He 141b4fdde6dSAsias He len = ndev.ops->tx(iov, out, &ndev); 142b5ee1ea7SAsias He 1434f56d42cSAsias He virt_queue__set_used_elem(vq, head, len); 1444f56d42cSAsias He } 1454f56d42cSAsias He 146*78a2a3e2SSasha Levin virtio_pci__signal_vq(kvm, &ndev.vpci, VIRTIO_NET_TX_QUEUE); 1474f56d42cSAsias He } 1484f56d42cSAsias He 149c4aa7c02SPekka Enberg pthread_exit(NULL); 150407475bfSPekka Enberg 151c4aa7c02SPekka Enberg return NULL; 152c4aa7c02SPekka Enberg 153c4aa7c02SPekka Enberg } 154407475bfSPekka Enberg 15543835ac9SSasha Levin static void virtio_net_handle_callback(struct kvm *kvm, u16 queue_index) 1564f56d42cSAsias He { 157407475bfSPekka Enberg switch (queue_index) { 158b5ee1ea7SAsias He case VIRTIO_NET_TX_QUEUE: 159c229370aSIngo Molnar mutex_lock(&ndev.io_tx_lock); 160c229370aSIngo Molnar pthread_cond_signal(&ndev.io_tx_cond); 161c229370aSIngo Molnar mutex_unlock(&ndev.io_tx_lock); 162407475bfSPekka Enberg break; 163b5ee1ea7SAsias He case VIRTIO_NET_RX_QUEUE: 164c229370aSIngo Molnar mutex_lock(&ndev.io_rx_lock); 165c229370aSIngo Molnar pthread_cond_signal(&ndev.io_rx_cond); 166c229370aSIngo Molnar mutex_unlock(&ndev.io_rx_lock); 167407475bfSPekka Enberg break; 168407475bfSPekka Enberg default: 1694542f276SCyrill Gorcunov pr_warning("Unknown queue index %u", queue_index); 170c4aa7c02SPekka Enberg } 1714f56d42cSAsias He } 1724f56d42cSAsias He 17327ab67f5SSasha Levin static void ioevent_callback(struct kvm *kvm, void *param) 17427ab67f5SSasha Levin { 175926e0e2fSIngo Molnar virtio_net_handle_callback(kvm, (u64)(long)param); 17627ab67f5SSasha Levin } 17727ab67f5SSasha Levin 1783b02f580SSasha Levin static bool virtio_net__tap_init(const struct virtio_net_parameters *params) 1794f56d42cSAsias He { 180cb7202c1SSasha Levin int sock = socket(AF_INET, SOCK_STREAM, 0); 181f715177dSAsias He int pid, status, offload, hdr_len; 182cb7202c1SSasha Levin struct sockaddr_in sin = {0}; 183246c8347SAsias He struct ifreq ifr; 1844f56d42cSAsias He 185c229370aSIngo Molnar ndev.tap_fd = open("/dev/net/tun", O_RDWR); 186c229370aSIngo Molnar if (ndev.tap_fd < 0) { 1874542f276SCyrill Gorcunov pr_warning("Unable to open /dev/net/tun"); 1883b02f580SSasha Levin goto fail; 1893b02f580SSasha Levin } 1904f56d42cSAsias He 1914f56d42cSAsias He memset(&ifr, 0, sizeof(ifr)); 192246c8347SAsias He ifr.ifr_flags = IFF_TAP | IFF_NO_PI | IFF_VNET_HDR; 193c229370aSIngo Molnar if (ioctl(ndev.tap_fd, TUNSETIFF, &ifr) < 0) { 1944542f276SCyrill Gorcunov pr_warning("Config tap device error. Are you root?"); 1953b02f580SSasha Levin goto fail; 1963b02f580SSasha Levin } 1974f56d42cSAsias He 198c229370aSIngo Molnar strncpy(ndev.tap_name, ifr.ifr_name, sizeof(ndev.tap_name)); 1994f56d42cSAsias He 200c229370aSIngo Molnar if (ioctl(ndev.tap_fd, TUNSETNOCSUM, 1) < 0) { 2014542f276SCyrill Gorcunov pr_warning("Config tap device TUNSETNOCSUM error"); 202246c8347SAsias He goto fail; 203246c8347SAsias He } 204246c8347SAsias He 205246c8347SAsias He hdr_len = sizeof(struct virtio_net_hdr); 206c229370aSIngo Molnar if (ioctl(ndev.tap_fd, TUNSETVNETHDRSZ, &hdr_len) < 0) { 2074542f276SCyrill Gorcunov pr_warning("Config tap device TUNSETVNETHDRSZ error"); 208246c8347SAsias He } 209246c8347SAsias He 210246c8347SAsias He offload = TUN_F_CSUM | TUN_F_TSO4 | TUN_F_TSO6 | TUN_F_UFO; 211c229370aSIngo Molnar if (ioctl(ndev.tap_fd, TUNSETOFFLOAD, offload) < 0) { 2124542f276SCyrill Gorcunov pr_warning("Config tap device TUNSETOFFLOAD error"); 213246c8347SAsias He goto fail; 214246c8347SAsias He } 2154f56d42cSAsias He 21673b7d038SAmos Kong if (strcmp(params->script, "none")) { 21773b7d038SAmos Kong pid = fork(); 21873b7d038SAmos Kong if (pid == 0) { 219c229370aSIngo Molnar execl(params->script, params->script, ndev.tap_name, NULL); 22073b7d038SAmos Kong _exit(1); 22173b7d038SAmos Kong } else { 22273b7d038SAmos Kong waitpid(pid, &status, 0); 22373b7d038SAmos Kong if (WIFEXITED(status) && WEXITSTATUS(status) != 0) { 2244542f276SCyrill Gorcunov pr_warning("Fail to setup tap by %s", params->script); 22573b7d038SAmos Kong goto fail; 22673b7d038SAmos Kong } 22773b7d038SAmos Kong } 22873b7d038SAmos Kong } else { 229cb7202c1SSasha Levin memset(&ifr, 0, sizeof(ifr)); 230c229370aSIngo Molnar strncpy(ifr.ifr_name, ndev.tap_name, sizeof(ndev.tap_name)); 231bdfcfca6SSasha Levin sin.sin_addr.s_addr = inet_addr(params->host_ip); 232cb7202c1SSasha Levin memcpy(&(ifr.ifr_addr), &sin, sizeof(ifr.ifr_addr)); 233cb7202c1SSasha Levin ifr.ifr_addr.sa_family = AF_INET; 2343b02f580SSasha Levin if (ioctl(sock, SIOCSIFADDR, &ifr) < 0) { 2354542f276SCyrill Gorcunov pr_warning("Could not set ip address on tap device"); 2363b02f580SSasha Levin goto fail; 2373b02f580SSasha Levin } 23873b7d038SAmos Kong } 239cb7202c1SSasha Levin 240cb7202c1SSasha Levin memset(&ifr, 0, sizeof(ifr)); 241c229370aSIngo Molnar strncpy(ifr.ifr_name, ndev.tap_name, sizeof(ndev.tap_name)); 242cb7202c1SSasha Levin ioctl(sock, SIOCGIFFLAGS, &ifr); 243cb7202c1SSasha Levin ifr.ifr_flags |= IFF_UP | IFF_RUNNING; 244cb7202c1SSasha Levin if (ioctl(sock, SIOCSIFFLAGS, &ifr) < 0) 2454542f276SCyrill Gorcunov pr_warning("Could not bring tap device up"); 246cb7202c1SSasha Levin 247cb7202c1SSasha Levin close(sock); 2483b02f580SSasha Levin 2493b02f580SSasha Levin return 1; 2503b02f580SSasha Levin 2513b02f580SSasha Levin fail: 2523b02f580SSasha Levin if (sock >= 0) 2533b02f580SSasha Levin close(sock); 254c229370aSIngo Molnar if (ndev.tap_fd >= 0) 255c229370aSIngo Molnar close(ndev.tap_fd); 2563b02f580SSasha Levin 2573b02f580SSasha Levin return 0; 2584f56d42cSAsias He } 2594f56d42cSAsias He 26043835ac9SSasha Levin static void virtio_net__io_thread_init(struct kvm *kvm) 261c4aa7c02SPekka Enberg { 262c229370aSIngo Molnar pthread_mutex_init(&ndev.io_rx_lock, NULL); 263c229370aSIngo Molnar pthread_cond_init(&ndev.io_tx_cond, NULL); 264c4aa7c02SPekka Enberg 265c229370aSIngo Molnar pthread_mutex_init(&ndev.io_rx_lock, NULL); 266c229370aSIngo Molnar pthread_cond_init(&ndev.io_tx_cond, NULL); 267c4aa7c02SPekka Enberg 268c229370aSIngo Molnar pthread_create(&ndev.io_rx_thread, NULL, virtio_net_rx_thread, (void *)kvm); 269c229370aSIngo Molnar pthread_create(&ndev.io_tx_thread, NULL, virtio_net_tx_thread, (void *)kvm); 270c4aa7c02SPekka Enberg } 271c4aa7c02SPekka Enberg 272b4fdde6dSAsias He static inline int tap_ops_tx(struct iovec *iov, u16 out, struct net_dev *ndev) 273b4fdde6dSAsias He { 274b4fdde6dSAsias He return writev(ndev->tap_fd, iov, out); 275b4fdde6dSAsias He } 276b4fdde6dSAsias He 277b4fdde6dSAsias He static inline int tap_ops_rx(struct iovec *iov, u16 in, struct net_dev *ndev) 278b4fdde6dSAsias He { 279b4fdde6dSAsias He return readv(ndev->tap_fd, iov, in); 280b4fdde6dSAsias He } 281b4fdde6dSAsias He 282b4fdde6dSAsias He static inline int uip_ops_tx(struct iovec *iov, u16 out, struct net_dev *ndev) 283b4fdde6dSAsias He { 284b4fdde6dSAsias He return uip_tx(iov, out, &ndev->info); 285b4fdde6dSAsias He } 286b4fdde6dSAsias He 287b4fdde6dSAsias He static inline int uip_ops_rx(struct iovec *iov, u16 in, struct net_dev *ndev) 288b4fdde6dSAsias He { 289b4fdde6dSAsias He return uip_rx(iov, in, &ndev->info); 290b4fdde6dSAsias He } 291b4fdde6dSAsias He 292b4fdde6dSAsias He static struct net_dev_operations tap_ops = { 293b4fdde6dSAsias He .rx = tap_ops_rx, 294b4fdde6dSAsias He .tx = tap_ops_tx, 295b4fdde6dSAsias He }; 296b4fdde6dSAsias He 297b4fdde6dSAsias He static struct net_dev_operations uip_ops = { 298b4fdde6dSAsias He .rx = uip_ops_rx, 299b4fdde6dSAsias He .tx = uip_ops_tx, 300b4fdde6dSAsias He }; 301b4fdde6dSAsias He 302*78a2a3e2SSasha Levin static void set_config(struct kvm *kvm, void *dev, u8 data, u32 offset) 303*78a2a3e2SSasha Levin { 304*78a2a3e2SSasha Levin struct net_dev *ndev = dev; 305*78a2a3e2SSasha Levin 306*78a2a3e2SSasha Levin ((u8 *)(&ndev->config))[offset] = data; 307*78a2a3e2SSasha Levin } 308*78a2a3e2SSasha Levin 309*78a2a3e2SSasha Levin static u8 get_config(struct kvm *kvm, void *dev, u32 offset) 310*78a2a3e2SSasha Levin { 311*78a2a3e2SSasha Levin struct net_dev *ndev = dev; 312*78a2a3e2SSasha Levin 313*78a2a3e2SSasha Levin return ((u8 *)(&ndev->config))[offset]; 314*78a2a3e2SSasha Levin } 315*78a2a3e2SSasha Levin 316*78a2a3e2SSasha Levin static u32 get_host_features(struct kvm *kvm, void *dev) 317*78a2a3e2SSasha Levin { 318*78a2a3e2SSasha Levin return 1UL << VIRTIO_NET_F_MAC 319*78a2a3e2SSasha Levin | 1UL << VIRTIO_NET_F_CSUM 320*78a2a3e2SSasha Levin | 1UL << VIRTIO_NET_F_HOST_UFO 321*78a2a3e2SSasha Levin | 1UL << VIRTIO_NET_F_HOST_TSO4 322*78a2a3e2SSasha Levin | 1UL << VIRTIO_NET_F_HOST_TSO6 323*78a2a3e2SSasha Levin | 1UL << VIRTIO_NET_F_GUEST_UFO 324*78a2a3e2SSasha Levin | 1UL << VIRTIO_NET_F_GUEST_TSO4 325*78a2a3e2SSasha Levin | 1UL << VIRTIO_NET_F_GUEST_TSO6; 326*78a2a3e2SSasha Levin } 327*78a2a3e2SSasha Levin 328*78a2a3e2SSasha Levin static void set_guest_features(struct kvm *kvm, void *dev, u32 features) 329*78a2a3e2SSasha Levin { 330*78a2a3e2SSasha Levin struct net_dev *ndev = dev; 331*78a2a3e2SSasha Levin 332*78a2a3e2SSasha Levin ndev->features = features; 333*78a2a3e2SSasha Levin } 334*78a2a3e2SSasha Levin 335*78a2a3e2SSasha Levin static int init_vq(struct kvm *kvm, void *dev, u32 vq, u32 pfn) 336*78a2a3e2SSasha Levin { 337*78a2a3e2SSasha Levin struct net_dev *ndev = dev; 338*78a2a3e2SSasha Levin struct virt_queue *queue; 339*78a2a3e2SSasha Levin void *p; 340*78a2a3e2SSasha Levin struct ioevent ioevent; 341*78a2a3e2SSasha Levin 342*78a2a3e2SSasha Levin compat__remove_message(ndev->compat_id); 343*78a2a3e2SSasha Levin 344*78a2a3e2SSasha Levin queue = &ndev->vqs[vq]; 345*78a2a3e2SSasha Levin queue->pfn = pfn; 346*78a2a3e2SSasha Levin p = guest_pfn_to_host(kvm, queue->pfn); 347*78a2a3e2SSasha Levin 348*78a2a3e2SSasha Levin vring_init(&queue->vring, VIRTIO_NET_QUEUE_SIZE, p, VIRTIO_PCI_VRING_ALIGN); 349*78a2a3e2SSasha Levin 350*78a2a3e2SSasha Levin ioevent = (struct ioevent) { 351*78a2a3e2SSasha Levin .io_addr = ndev->vpci.base_addr + VIRTIO_PCI_QUEUE_NOTIFY, 352*78a2a3e2SSasha Levin .io_len = sizeof(u16), 353*78a2a3e2SSasha Levin .fn = ioevent_callback, 354*78a2a3e2SSasha Levin .fn_ptr = (void *)(u64)vq, 355*78a2a3e2SSasha Levin .datamatch = vq, 356*78a2a3e2SSasha Levin .fn_kvm = kvm, 357*78a2a3e2SSasha Levin .fd = eventfd(0, 0), 358*78a2a3e2SSasha Levin }; 359*78a2a3e2SSasha Levin 360*78a2a3e2SSasha Levin ioeventfd__add_event(&ioevent); 361*78a2a3e2SSasha Levin 362*78a2a3e2SSasha Levin return 0; 363*78a2a3e2SSasha Levin } 364*78a2a3e2SSasha Levin 365*78a2a3e2SSasha Levin static int notify_vq(struct kvm *kvm, void *dev, u32 vq) 366*78a2a3e2SSasha Levin { 367*78a2a3e2SSasha Levin virtio_net_handle_callback(kvm, vq); 368*78a2a3e2SSasha Levin 369*78a2a3e2SSasha Levin return 0; 370*78a2a3e2SSasha Levin } 371*78a2a3e2SSasha Levin 372*78a2a3e2SSasha Levin static int get_pfn_vq(struct kvm *kvm, void *dev, u32 vq) 373*78a2a3e2SSasha Levin { 374*78a2a3e2SSasha Levin struct net_dev *ndev = dev; 375*78a2a3e2SSasha Levin 376*78a2a3e2SSasha Levin return ndev->vqs[vq].pfn; 377*78a2a3e2SSasha Levin } 378*78a2a3e2SSasha Levin 379*78a2a3e2SSasha Levin static int get_size_vq(struct kvm *kvm, void *dev, u32 vq) 380*78a2a3e2SSasha Levin { 381*78a2a3e2SSasha Levin return VIRTIO_NET_QUEUE_SIZE; 382*78a2a3e2SSasha Levin } 383*78a2a3e2SSasha Levin 384bdfcfca6SSasha Levin void virtio_net__init(const struct virtio_net_parameters *params) 3854f56d42cSAsias He { 386b5ee1ea7SAsias He int i; 3872449f6e3SSasha Levin 3880c54698eSAsias He for (i = 0 ; i < 6 ; i++) { 389f715177dSAsias He ndev.config.mac[i] = params->guest_mac[i]; 3900c54698eSAsias He ndev.info.guest_mac.addr[i] = params->guest_mac[i]; 3910c54698eSAsias He ndev.info.host_mac.addr[i] = params->host_mac[i]; 3920c54698eSAsias He } 393f715177dSAsias He 394b5ee1ea7SAsias He ndev.mode = params->mode; 395b4fdde6dSAsias He if (ndev.mode == NET_MODE_TAP) { 396b5ee1ea7SAsias He virtio_net__tap_init(params); 397b4fdde6dSAsias He ndev.ops = &tap_ops; 398b4fdde6dSAsias He } else { 3990c54698eSAsias He ndev.info.host_ip = ntohl(inet_addr(params->host_ip)); 4000c54698eSAsias He ndev.info.guest_ip = ntohl(inet_addr(params->guest_ip)); 4010c54698eSAsias He ndev.info.guest_netmask = ntohl(inet_addr("255.255.255.0")); 402b5ee1ea7SAsias He uip_init(&ndev.info); 403b4fdde6dSAsias He ndev.ops = &uip_ops; 404b4fdde6dSAsias He } 405b5ee1ea7SAsias He 406*78a2a3e2SSasha Levin virtio_pci__init(kvm, &ndev.vpci, &ndev, PCI_DEVICE_ID_VIRTIO_NET, VIRTIO_ID_NET); 407*78a2a3e2SSasha Levin ndev.vpci.ops = (struct virtio_pci_ops) { 408*78a2a3e2SSasha Levin .set_config = set_config, 409*78a2a3e2SSasha Levin .get_config = get_config, 410*78a2a3e2SSasha Levin .get_host_features = get_host_features, 411*78a2a3e2SSasha Levin .set_guest_features = set_guest_features, 412*78a2a3e2SSasha Levin .init_vq = init_vq, 413*78a2a3e2SSasha Levin .notify_vq = notify_vq, 414*78a2a3e2SSasha Levin .get_pfn_vq = get_pfn_vq, 415*78a2a3e2SSasha Levin .get_size_vq = get_size_vq, 41627ab67f5SSasha Levin }; 41727ab67f5SSasha Levin 418*78a2a3e2SSasha Levin virtio_net__io_thread_init(params->kvm); 419cb83de6fSSasha Levin 420cb83de6fSSasha Levin ndev.compat_id = compat__add_message("virtio-net device was not detected", 421cb83de6fSSasha Levin "While you have requested a virtio-net device, " 422cb83de6fSSasha Levin "the guest kernel didn't seem to detect it.\n" 423cb83de6fSSasha Levin "Please make sure that the kernel was compiled" 424cb83de6fSSasha Levin "with CONFIG_VIRTIO_NET."); 4254f56d42cSAsias He } 426