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/ioport.h" 54f56d42cSAsias He #include "kvm/types.h" 64f56d42cSAsias He #include "kvm/mutex.h" 74f56d42cSAsias He #include "kvm/util.h" 84f56d42cSAsias He #include "kvm/kvm.h" 94f56d42cSAsias He #include "kvm/pci.h" 102449f6e3SSasha Levin #include "kvm/irq.h" 11b5ee1ea7SAsias He #include "kvm/uip.h" 1227ab67f5SSasha Levin #include "kvm/ioeventfd.h" 134f56d42cSAsias He 144f56d42cSAsias He #include <linux/virtio_net.h> 154f56d42cSAsias He #include <linux/if_tun.h> 16c229370aSIngo Molnar 17c229370aSIngo Molnar #include <arpa/inet.h> 184f56d42cSAsias He #include <net/if.h> 19c229370aSIngo Molnar 20c229370aSIngo Molnar #include <unistd.h> 214f56d42cSAsias He #include <assert.h> 224f56d42cSAsias He #include <fcntl.h> 23c229370aSIngo Molnar 24cb7202c1SSasha Levin #include <sys/socket.h> 25c229370aSIngo Molnar #include <sys/ioctl.h> 26c229370aSIngo Molnar #include <sys/types.h> 2773b7d038SAmos Kong #include <sys/wait.h> 284f56d42cSAsias He 294f56d42cSAsias He #define VIRTIO_NET_QUEUE_SIZE 128 304f56d42cSAsias He #define VIRTIO_NET_NUM_QUEUES 2 314f56d42cSAsias He #define VIRTIO_NET_RX_QUEUE 0 324f56d42cSAsias He #define VIRTIO_NET_TX_QUEUE 1 334f56d42cSAsias He 34c229370aSIngo Molnar static struct pci_device_header pci_header = { 352449f6e3SSasha Levin .vendor_id = PCI_VENDOR_ID_REDHAT_QUMRANET, 362449f6e3SSasha Levin .device_id = PCI_DEVICE_ID_VIRTIO_NET, 372449f6e3SSasha Levin .header_type = PCI_HEADER_TYPE_NORMAL, 382449f6e3SSasha Levin .revision_id = 0, 392449f6e3SSasha Levin .class = 0x020000, 402449f6e3SSasha Levin .subsys_vendor_id = PCI_SUBSYSTEM_VENDOR_ID_REDHAT_QUMRANET, 410a7ab0c6SSasha Levin .subsys_id = VIRTIO_ID_NET, 422449f6e3SSasha Levin }; 432449f6e3SSasha Levin 44b4fdde6dSAsias He struct net_dev; 45b4fdde6dSAsias He 46b4fdde6dSAsias He struct net_dev_operations { 47b4fdde6dSAsias He int (*rx)(struct iovec *iov, u16 in, struct net_dev *ndev); 48b4fdde6dSAsias He int (*tx)(struct iovec *iov, u16 in, struct net_dev *ndev); 49b4fdde6dSAsias He }; 50b4fdde6dSAsias He 518626798bSAsias He struct net_dev { 524f56d42cSAsias He pthread_mutex_t mutex; 534f56d42cSAsias He 544f56d42cSAsias He struct virt_queue vqs[VIRTIO_NET_NUM_QUEUES]; 55c229370aSIngo Molnar struct virtio_net_config config; 563fdf659dSSasha Levin u32 host_features; 573fdf659dSSasha Levin u32 guest_features; 583fdf659dSSasha Levin u16 config_vector; 593fdf659dSSasha Levin u8 status; 607f5ffaf5SAsias He u8 isr; 613fdf659dSSasha Levin u16 queue_selector; 6207f90696SSasha Levin u16 base_addr; 634f56d42cSAsias He 64c4aa7c02SPekka Enberg pthread_t io_rx_thread; 65c229370aSIngo Molnar pthread_mutex_t io_rx_lock; 66c4aa7c02SPekka Enberg pthread_cond_t io_rx_cond; 67c4aa7c02SPekka Enberg 68c4aa7c02SPekka Enberg pthread_t io_tx_thread; 69c229370aSIngo Molnar pthread_mutex_t io_tx_lock; 70c4aa7c02SPekka Enberg pthread_cond_t io_tx_cond; 71c4aa7c02SPekka Enberg 724f56d42cSAsias He int tap_fd; 734f56d42cSAsias He char tap_name[IFNAMSIZ]; 74bb1a32f1SAsias He 75bb1a32f1SAsias He int mode; 76bb1a32f1SAsias He 77b5ee1ea7SAsias He struct uip_info info; 78b4fdde6dSAsias He struct net_dev_operations *ops; 794f56d42cSAsias He }; 804f56d42cSAsias He 818626798bSAsias He static struct net_dev ndev = { 824f56d42cSAsias He .mutex = PTHREAD_MUTEX_INITIALIZER, 834f56d42cSAsias He 84c229370aSIngo Molnar .config = { 854f56d42cSAsias He .status = VIRTIO_NET_S_LINK_UP, 864f56d42cSAsias He }, 87407475bfSPekka Enberg .host_features = 1UL << VIRTIO_NET_F_MAC 88407475bfSPekka Enberg | 1UL << VIRTIO_NET_F_CSUM 89407475bfSPekka Enberg | 1UL << VIRTIO_NET_F_HOST_UFO 90407475bfSPekka Enberg | 1UL << VIRTIO_NET_F_HOST_TSO4 91407475bfSPekka Enberg | 1UL << VIRTIO_NET_F_HOST_TSO6 92407475bfSPekka Enberg | 1UL << VIRTIO_NET_F_GUEST_UFO 93407475bfSPekka Enberg | 1UL << VIRTIO_NET_F_GUEST_TSO4 94407475bfSPekka Enberg | 1UL << VIRTIO_NET_F_GUEST_TSO6, 95b5ee1ea7SAsias He .info = { 96b5ee1ea7SAsias He .buf_nr = 20, 97b5ee1ea7SAsias He } 984f56d42cSAsias He }; 994f56d42cSAsias He 100c4aa7c02SPekka Enberg static void *virtio_net_rx_thread(void *p) 1014f56d42cSAsias He { 1024f56d42cSAsias He struct iovec iov[VIRTIO_NET_QUEUE_SIZE]; 1034f56d42cSAsias He struct virt_queue *vq; 10443835ac9SSasha Levin struct kvm *kvm; 1053fdf659dSSasha Levin u16 out, in; 1063fdf659dSSasha Levin u16 head; 1074f56d42cSAsias He int len; 1084f56d42cSAsias He 10943835ac9SSasha Levin kvm = p; 110c229370aSIngo Molnar vq = &ndev.vqs[VIRTIO_NET_RX_QUEUE]; 111c4aa7c02SPekka Enberg 112c4aa7c02SPekka Enberg while (1) { 113b5ee1ea7SAsias He 114c229370aSIngo Molnar mutex_lock(&ndev.io_rx_lock); 115c4aa7c02SPekka Enberg if (!virt_queue__available(vq)) 116c229370aSIngo Molnar pthread_cond_wait(&ndev.io_rx_cond, &ndev.io_rx_lock); 117c229370aSIngo Molnar mutex_unlock(&ndev.io_rx_lock); 1184f56d42cSAsias He 1194f56d42cSAsias He while (virt_queue__available(vq)) { 120b5ee1ea7SAsias He 12143835ac9SSasha Levin head = virt_queue__get_iov(vq, iov, &out, &in, kvm); 122b5ee1ea7SAsias He 123b4fdde6dSAsias He len = ndev.ops->rx(iov, in, &ndev); 124b5ee1ea7SAsias He 125246c8347SAsias He virt_queue__set_used_elem(vq, head, len); 1267f5ffaf5SAsias He 127c4aa7c02SPekka Enberg /* We should interrupt guest right now, otherwise latency is huge. */ 128c229370aSIngo Molnar virt_queue__trigger_irq(vq, pci_header.irq_line, &ndev.isr, kvm); 1294f56d42cSAsias He } 1304f56d42cSAsias He 131c4aa7c02SPekka Enberg } 132c4aa7c02SPekka Enberg 133c4aa7c02SPekka Enberg pthread_exit(NULL); 134c4aa7c02SPekka Enberg return NULL; 135c4aa7c02SPekka Enberg 136c4aa7c02SPekka Enberg } 137c4aa7c02SPekka Enberg 138c4aa7c02SPekka Enberg static void *virtio_net_tx_thread(void *p) 1394f56d42cSAsias He { 1404f56d42cSAsias He struct iovec iov[VIRTIO_NET_QUEUE_SIZE]; 1414f56d42cSAsias He struct virt_queue *vq; 14243835ac9SSasha Levin struct kvm *kvm; 1433fdf659dSSasha Levin u16 out, in; 1443fdf659dSSasha Levin u16 head; 1454f56d42cSAsias He int len; 1464f56d42cSAsias He 14743835ac9SSasha Levin kvm = p; 148c229370aSIngo Molnar vq = &ndev.vqs[VIRTIO_NET_TX_QUEUE]; 149c4aa7c02SPekka Enberg 150c4aa7c02SPekka Enberg while (1) { 151c229370aSIngo Molnar mutex_lock(&ndev.io_tx_lock); 152c4aa7c02SPekka Enberg if (!virt_queue__available(vq)) 153c229370aSIngo Molnar pthread_cond_wait(&ndev.io_tx_cond, &ndev.io_tx_lock); 154c229370aSIngo Molnar mutex_unlock(&ndev.io_tx_lock); 1554f56d42cSAsias He 1564f56d42cSAsias He while (virt_queue__available(vq)) { 157b5ee1ea7SAsias He 15843835ac9SSasha Levin head = virt_queue__get_iov(vq, iov, &out, &in, kvm); 159b5ee1ea7SAsias He 160b4fdde6dSAsias He len = ndev.ops->tx(iov, out, &ndev); 161b5ee1ea7SAsias He 1624f56d42cSAsias He virt_queue__set_used_elem(vq, head, len); 1634f56d42cSAsias He } 1644f56d42cSAsias He 165c229370aSIngo Molnar virt_queue__trigger_irq(vq, pci_header.irq_line, &ndev.isr, kvm); 1667f5ffaf5SAsias He 1674f56d42cSAsias He } 1684f56d42cSAsias He 169c4aa7c02SPekka Enberg pthread_exit(NULL); 170407475bfSPekka Enberg 171c4aa7c02SPekka Enberg return NULL; 172c4aa7c02SPekka Enberg 173c4aa7c02SPekka Enberg } 174407475bfSPekka Enberg 1753fdf659dSSasha Levin static bool virtio_net_pci_io_device_specific_in(void *data, unsigned long offset, int size, u32 count) 1764f56d42cSAsias He { 177c229370aSIngo Molnar u8 *config_space = (u8 *)&ndev.config; 1784f56d42cSAsias He 1794f56d42cSAsias He if (size != 1 || count != 1) 1804f56d42cSAsias He return false; 1814f56d42cSAsias He 182b8f43678SSasha Levin if ((offset - VIRTIO_MSI_CONFIG_VECTOR) > sizeof(struct virtio_net_config)) 1834542f276SCyrill Gorcunov pr_error("config offset is too big: %li", offset - VIRTIO_MSI_CONFIG_VECTOR); 1844f56d42cSAsias He 185b8f43678SSasha Levin ioport__write8(data, config_space[offset - VIRTIO_MSI_CONFIG_VECTOR]); 1864f56d42cSAsias He 1874f56d42cSAsias He return true; 1884f56d42cSAsias He } 1894f56d42cSAsias He 1903d62dea6SSasha Levin static bool virtio_net_pci_io_in(struct ioport *ioport, struct kvm *kvm, u16 port, void *data, int size, u32 count) 1914f56d42cSAsias He { 19207f90696SSasha Levin unsigned long offset = port - ndev.base_addr; 1934f56d42cSAsias He bool ret = true; 1944f56d42cSAsias He 195c229370aSIngo Molnar mutex_lock(&ndev.mutex); 1964f56d42cSAsias He 1974f56d42cSAsias He switch (offset) { 1984f56d42cSAsias He case VIRTIO_PCI_HOST_FEATURES: 199c229370aSIngo Molnar ioport__write32(data, ndev.host_features); 2004f56d42cSAsias He break; 2014f56d42cSAsias He case VIRTIO_PCI_GUEST_FEATURES: 2024f56d42cSAsias He ret = false; 2034f56d42cSAsias He break; 2044f56d42cSAsias He case VIRTIO_PCI_QUEUE_PFN: 205c229370aSIngo Molnar ioport__write32(data, ndev.vqs[ndev.queue_selector].pfn); 2064f56d42cSAsias He break; 2074f56d42cSAsias He case VIRTIO_PCI_QUEUE_NUM: 2084f56d42cSAsias He ioport__write16(data, VIRTIO_NET_QUEUE_SIZE); 2094f56d42cSAsias He break; 2104f56d42cSAsias He case VIRTIO_PCI_QUEUE_SEL: 2114f56d42cSAsias He case VIRTIO_PCI_QUEUE_NOTIFY: 2124f56d42cSAsias He ret = false; 2134f56d42cSAsias He break; 2144f56d42cSAsias He case VIRTIO_PCI_STATUS: 215c229370aSIngo Molnar ioport__write8(data, ndev.status); 2164f56d42cSAsias He break; 2174f56d42cSAsias He case VIRTIO_PCI_ISR: 218c229370aSIngo Molnar ioport__write8(data, ndev.isr); 219c229370aSIngo Molnar kvm__irq_line(kvm, pci_header.irq_line, VIRTIO_IRQ_LOW); 220c229370aSIngo Molnar ndev.isr = VIRTIO_IRQ_LOW; 2214f56d42cSAsias He break; 2224f56d42cSAsias He case VIRTIO_MSI_CONFIG_VECTOR: 223c229370aSIngo Molnar ioport__write16(data, ndev.config_vector); 2244f56d42cSAsias He break; 2254f56d42cSAsias He default: 2264f56d42cSAsias He ret = virtio_net_pci_io_device_specific_in(data, offset, size, count); 2274f56d42cSAsias He }; 2284f56d42cSAsias He 229c229370aSIngo Molnar mutex_unlock(&ndev.mutex); 2304f56d42cSAsias He 2314f56d42cSAsias He return ret; 2324f56d42cSAsias He } 2334f56d42cSAsias He 23443835ac9SSasha Levin static void virtio_net_handle_callback(struct kvm *kvm, u16 queue_index) 2354f56d42cSAsias He { 236407475bfSPekka Enberg switch (queue_index) { 237b5ee1ea7SAsias He case VIRTIO_NET_TX_QUEUE: 238c229370aSIngo Molnar mutex_lock(&ndev.io_tx_lock); 239c229370aSIngo Molnar pthread_cond_signal(&ndev.io_tx_cond); 240c229370aSIngo Molnar mutex_unlock(&ndev.io_tx_lock); 241407475bfSPekka Enberg break; 242b5ee1ea7SAsias He case VIRTIO_NET_RX_QUEUE: 243c229370aSIngo Molnar mutex_lock(&ndev.io_rx_lock); 244c229370aSIngo Molnar pthread_cond_signal(&ndev.io_rx_cond); 245c229370aSIngo Molnar mutex_unlock(&ndev.io_rx_lock); 246407475bfSPekka Enberg break; 247407475bfSPekka Enberg default: 2484542f276SCyrill Gorcunov pr_warning("Unknown queue index %u", queue_index); 249c4aa7c02SPekka Enberg } 2504f56d42cSAsias He } 2514f56d42cSAsias He 2523d62dea6SSasha Levin static bool virtio_net_pci_io_out(struct ioport *ioport, struct kvm *kvm, u16 port, void *data, int size, u32 count) 2534f56d42cSAsias He { 25407f90696SSasha Levin unsigned long offset = port - ndev.base_addr; 2554f56d42cSAsias He bool ret = true; 2564f56d42cSAsias He 257c229370aSIngo Molnar mutex_lock(&ndev.mutex); 2584f56d42cSAsias He 2594f56d42cSAsias He switch (offset) { 2604f56d42cSAsias He case VIRTIO_PCI_GUEST_FEATURES: 261c229370aSIngo Molnar ndev.guest_features = ioport__read32(data); 2624f56d42cSAsias He break; 2634f56d42cSAsias He case VIRTIO_PCI_QUEUE_PFN: { 2644f56d42cSAsias He struct virt_queue *queue; 2654f56d42cSAsias He void *p; 2664f56d42cSAsias He 267c229370aSIngo Molnar assert(ndev.queue_selector < VIRTIO_NET_NUM_QUEUES); 2684f56d42cSAsias He 269c229370aSIngo Molnar queue = &ndev.vqs[ndev.queue_selector]; 2704f56d42cSAsias He queue->pfn = ioport__read32(data); 27143835ac9SSasha Levin p = guest_pfn_to_host(kvm, queue->pfn); 2724f56d42cSAsias He 273b8f43678SSasha Levin vring_init(&queue->vring, VIRTIO_NET_QUEUE_SIZE, p, VIRTIO_PCI_VRING_ALIGN); 2744f56d42cSAsias He 2754f56d42cSAsias He break; 2764f56d42cSAsias He } 2774f56d42cSAsias He case VIRTIO_PCI_QUEUE_SEL: 278c229370aSIngo Molnar ndev.queue_selector = ioport__read16(data); 2794f56d42cSAsias He break; 2804f56d42cSAsias He case VIRTIO_PCI_QUEUE_NOTIFY: { 2813fdf659dSSasha Levin u16 queue_index; 282c229370aSIngo Molnar 2834f56d42cSAsias He queue_index = ioport__read16(data); 28443835ac9SSasha Levin virtio_net_handle_callback(kvm, queue_index); 2854f56d42cSAsias He break; 2864f56d42cSAsias He } 2874f56d42cSAsias He case VIRTIO_PCI_STATUS: 288c229370aSIngo Molnar ndev.status = ioport__read8(data); 2894f56d42cSAsias He break; 2904f56d42cSAsias He case VIRTIO_MSI_CONFIG_VECTOR: 291c229370aSIngo Molnar ndev.config_vector = VIRTIO_MSI_NO_VECTOR; 2924f56d42cSAsias He break; 2934f56d42cSAsias He case VIRTIO_MSI_QUEUE_VECTOR: 2944f56d42cSAsias He break; 2954f56d42cSAsias He default: 2964f56d42cSAsias He ret = false; 2974f56d42cSAsias He }; 2984f56d42cSAsias He 299c229370aSIngo Molnar mutex_unlock(&ndev.mutex); 300407475bfSPekka Enberg 3014f56d42cSAsias He return ret; 3024f56d42cSAsias He } 3034f56d42cSAsias He 30427ab67f5SSasha Levin static void ioevent_callback(struct kvm *kvm, void *param) 30527ab67f5SSasha Levin { 306926e0e2fSIngo Molnar virtio_net_handle_callback(kvm, (u64)(long)param); 30727ab67f5SSasha Levin } 30827ab67f5SSasha Levin 3094f56d42cSAsias He static struct ioport_operations virtio_net_io_ops = { 3104f56d42cSAsias He .io_in = virtio_net_pci_io_in, 3114f56d42cSAsias He .io_out = virtio_net_pci_io_out, 3124f56d42cSAsias He }; 3134f56d42cSAsias He 3143b02f580SSasha Levin static bool virtio_net__tap_init(const struct virtio_net_parameters *params) 3154f56d42cSAsias He { 316cb7202c1SSasha Levin int sock = socket(AF_INET, SOCK_STREAM, 0); 317f715177dSAsias He int pid, status, offload, hdr_len; 318cb7202c1SSasha Levin struct sockaddr_in sin = {0}; 319246c8347SAsias He struct ifreq ifr; 3204f56d42cSAsias He 321c229370aSIngo Molnar ndev.tap_fd = open("/dev/net/tun", O_RDWR); 322c229370aSIngo Molnar if (ndev.tap_fd < 0) { 3234542f276SCyrill Gorcunov pr_warning("Unable to open /dev/net/tun"); 3243b02f580SSasha Levin goto fail; 3253b02f580SSasha Levin } 3264f56d42cSAsias He 3274f56d42cSAsias He memset(&ifr, 0, sizeof(ifr)); 328246c8347SAsias He ifr.ifr_flags = IFF_TAP | IFF_NO_PI | IFF_VNET_HDR; 329c229370aSIngo Molnar if (ioctl(ndev.tap_fd, TUNSETIFF, &ifr) < 0) { 3304542f276SCyrill Gorcunov pr_warning("Config tap device error. Are you root?"); 3313b02f580SSasha Levin goto fail; 3323b02f580SSasha Levin } 3334f56d42cSAsias He 334c229370aSIngo Molnar strncpy(ndev.tap_name, ifr.ifr_name, sizeof(ndev.tap_name)); 3354f56d42cSAsias He 336c229370aSIngo Molnar if (ioctl(ndev.tap_fd, TUNSETNOCSUM, 1) < 0) { 3374542f276SCyrill Gorcunov pr_warning("Config tap device TUNSETNOCSUM error"); 338246c8347SAsias He goto fail; 339246c8347SAsias He } 340246c8347SAsias He 341246c8347SAsias He hdr_len = sizeof(struct virtio_net_hdr); 342c229370aSIngo Molnar if (ioctl(ndev.tap_fd, TUNSETVNETHDRSZ, &hdr_len) < 0) { 3434542f276SCyrill Gorcunov pr_warning("Config tap device TUNSETVNETHDRSZ error"); 344246c8347SAsias He } 345246c8347SAsias He 346246c8347SAsias He offload = TUN_F_CSUM | TUN_F_TSO4 | TUN_F_TSO6 | TUN_F_UFO; 347c229370aSIngo Molnar if (ioctl(ndev.tap_fd, TUNSETOFFLOAD, offload) < 0) { 3484542f276SCyrill Gorcunov pr_warning("Config tap device TUNSETOFFLOAD error"); 349246c8347SAsias He goto fail; 350246c8347SAsias He } 3514f56d42cSAsias He 35273b7d038SAmos Kong if (strcmp(params->script, "none")) { 35373b7d038SAmos Kong pid = fork(); 35473b7d038SAmos Kong if (pid == 0) { 355c229370aSIngo Molnar execl(params->script, params->script, ndev.tap_name, NULL); 35673b7d038SAmos Kong _exit(1); 35773b7d038SAmos Kong } else { 35873b7d038SAmos Kong waitpid(pid, &status, 0); 35973b7d038SAmos Kong if (WIFEXITED(status) && WEXITSTATUS(status) != 0) { 3604542f276SCyrill Gorcunov pr_warning("Fail to setup tap by %s", params->script); 36173b7d038SAmos Kong goto fail; 36273b7d038SAmos Kong } 36373b7d038SAmos Kong } 36473b7d038SAmos Kong } else { 365cb7202c1SSasha Levin memset(&ifr, 0, sizeof(ifr)); 366c229370aSIngo Molnar strncpy(ifr.ifr_name, ndev.tap_name, sizeof(ndev.tap_name)); 367bdfcfca6SSasha Levin sin.sin_addr.s_addr = inet_addr(params->host_ip); 368cb7202c1SSasha Levin memcpy(&(ifr.ifr_addr), &sin, sizeof(ifr.ifr_addr)); 369cb7202c1SSasha Levin ifr.ifr_addr.sa_family = AF_INET; 3703b02f580SSasha Levin if (ioctl(sock, SIOCSIFADDR, &ifr) < 0) { 3714542f276SCyrill Gorcunov pr_warning("Could not set ip address on tap device"); 3723b02f580SSasha Levin goto fail; 3733b02f580SSasha Levin } 37473b7d038SAmos Kong } 375cb7202c1SSasha Levin 376cb7202c1SSasha Levin memset(&ifr, 0, sizeof(ifr)); 377c229370aSIngo Molnar strncpy(ifr.ifr_name, ndev.tap_name, sizeof(ndev.tap_name)); 378cb7202c1SSasha Levin ioctl(sock, SIOCGIFFLAGS, &ifr); 379cb7202c1SSasha Levin ifr.ifr_flags |= IFF_UP | IFF_RUNNING; 380cb7202c1SSasha Levin if (ioctl(sock, SIOCSIFFLAGS, &ifr) < 0) 3814542f276SCyrill Gorcunov pr_warning("Could not bring tap device up"); 382cb7202c1SSasha Levin 383cb7202c1SSasha Levin close(sock); 3843b02f580SSasha Levin 3853b02f580SSasha Levin return 1; 3863b02f580SSasha Levin 3873b02f580SSasha Levin fail: 3883b02f580SSasha Levin if (sock >= 0) 3893b02f580SSasha Levin close(sock); 390c229370aSIngo Molnar if (ndev.tap_fd >= 0) 391c229370aSIngo Molnar close(ndev.tap_fd); 3923b02f580SSasha Levin 3933b02f580SSasha Levin return 0; 3944f56d42cSAsias He } 3954f56d42cSAsias He 39643835ac9SSasha Levin static void virtio_net__io_thread_init(struct kvm *kvm) 397c4aa7c02SPekka Enberg { 398c229370aSIngo Molnar pthread_mutex_init(&ndev.io_rx_lock, NULL); 399c229370aSIngo Molnar pthread_cond_init(&ndev.io_tx_cond, NULL); 400c4aa7c02SPekka Enberg 401c229370aSIngo Molnar pthread_mutex_init(&ndev.io_rx_lock, NULL); 402c229370aSIngo Molnar pthread_cond_init(&ndev.io_tx_cond, NULL); 403c4aa7c02SPekka Enberg 404c229370aSIngo Molnar pthread_create(&ndev.io_rx_thread, NULL, virtio_net_rx_thread, (void *)kvm); 405c229370aSIngo Molnar pthread_create(&ndev.io_tx_thread, NULL, virtio_net_tx_thread, (void *)kvm); 406c4aa7c02SPekka Enberg } 407c4aa7c02SPekka Enberg 408b4fdde6dSAsias He static inline int tap_ops_tx(struct iovec *iov, u16 out, struct net_dev *ndev) 409b4fdde6dSAsias He { 410b4fdde6dSAsias He return writev(ndev->tap_fd, iov, out); 411b4fdde6dSAsias He } 412b4fdde6dSAsias He 413b4fdde6dSAsias He static inline int tap_ops_rx(struct iovec *iov, u16 in, struct net_dev *ndev) 414b4fdde6dSAsias He { 415b4fdde6dSAsias He return readv(ndev->tap_fd, iov, in); 416b4fdde6dSAsias He } 417b4fdde6dSAsias He 418b4fdde6dSAsias He static inline int uip_ops_tx(struct iovec *iov, u16 out, struct net_dev *ndev) 419b4fdde6dSAsias He { 420b4fdde6dSAsias He return uip_tx(iov, out, &ndev->info); 421b4fdde6dSAsias He } 422b4fdde6dSAsias He 423b4fdde6dSAsias He static inline int uip_ops_rx(struct iovec *iov, u16 in, struct net_dev *ndev) 424b4fdde6dSAsias He { 425b4fdde6dSAsias He return uip_rx(iov, in, &ndev->info); 426b4fdde6dSAsias He } 427b4fdde6dSAsias He 428b4fdde6dSAsias He static struct net_dev_operations tap_ops = { 429b4fdde6dSAsias He .rx = tap_ops_rx, 430b4fdde6dSAsias He .tx = tap_ops_tx, 431b4fdde6dSAsias He }; 432b4fdde6dSAsias He 433b4fdde6dSAsias He static struct net_dev_operations uip_ops = { 434b4fdde6dSAsias He .rx = uip_ops_rx, 435b4fdde6dSAsias He .tx = uip_ops_tx, 436b4fdde6dSAsias He }; 437b4fdde6dSAsias He 438bdfcfca6SSasha Levin void virtio_net__init(const struct virtio_net_parameters *params) 4394f56d42cSAsias He { 440b5ee1ea7SAsias He struct ioevent ioevent; 4412449f6e3SSasha Levin u8 dev, line, pin; 44207f90696SSasha Levin u16 net_base_addr; 443b5ee1ea7SAsias He int i; 4442449f6e3SSasha Levin 4450a7ab0c6SSasha Levin if (irq__register_device(VIRTIO_ID_NET, &dev, &pin, &line) < 0) 4462449f6e3SSasha Levin return; 4472449f6e3SSasha Levin 448c229370aSIngo Molnar pci_header.irq_pin = pin; 449c229370aSIngo Molnar pci_header.irq_line = line; 45007f90696SSasha Levin net_base_addr = ioport__register(IOPORT_EMPTY, &virtio_net_io_ops, IOPORT_SIZE, NULL); 45107f90696SSasha Levin pci_header.bar[0] = net_base_addr | PCI_BASE_ADDRESS_SPACE_IO; 45207f90696SSasha Levin ndev.base_addr = net_base_addr; 453c229370aSIngo Molnar pci__register(&pci_header, dev); 454c4aa7c02SPekka Enberg 455*0c54698eSAsias He for (i = 0 ; i < 6 ; i++) { 456f715177dSAsias He ndev.config.mac[i] = params->guest_mac[i]; 457*0c54698eSAsias He ndev.info.guest_mac.addr[i] = params->guest_mac[i]; 458*0c54698eSAsias He ndev.info.host_mac.addr[i] = params->host_mac[i]; 459*0c54698eSAsias He } 460f715177dSAsias He 461b5ee1ea7SAsias He ndev.mode = params->mode; 462b4fdde6dSAsias He if (ndev.mode == NET_MODE_TAP) { 463b5ee1ea7SAsias He virtio_net__tap_init(params); 464b4fdde6dSAsias He ndev.ops = &tap_ops; 465b4fdde6dSAsias He } else { 466*0c54698eSAsias He ndev.info.host_ip = ntohl(inet_addr(params->host_ip)); 467*0c54698eSAsias He ndev.info.guest_ip = ntohl(inet_addr(params->guest_ip)); 468*0c54698eSAsias He ndev.info.guest_netmask = ntohl(inet_addr("255.255.255.0")); 469b5ee1ea7SAsias He uip_init(&ndev.info); 470b4fdde6dSAsias He ndev.ops = &uip_ops; 471b4fdde6dSAsias He } 472b5ee1ea7SAsias He 47343835ac9SSasha Levin virtio_net__io_thread_init(params->kvm); 47427ab67f5SSasha Levin 47527ab67f5SSasha Levin for (i = 0; i < VIRTIO_NET_NUM_QUEUES; i++) { 47627ab67f5SSasha Levin ioevent = (struct ioevent) { 47727ab67f5SSasha Levin .io_addr = net_base_addr + VIRTIO_PCI_QUEUE_NOTIFY, 47827ab67f5SSasha Levin .io_len = sizeof(u16), 47927ab67f5SSasha Levin .fn = ioevent_callback, 48027ab67f5SSasha Levin .datamatch = i, 481926e0e2fSIngo Molnar .fn_ptr = (void *)(long)i, 48227ab67f5SSasha Levin .fn_kvm = params->kvm, 48327ab67f5SSasha Levin .fd = eventfd(0, 0), 48427ab67f5SSasha Levin }; 48527ab67f5SSasha Levin 48627ab67f5SSasha Levin ioeventfd__add_event(&ioevent); 48727ab67f5SSasha Levin } 4884f56d42cSAsias He } 489