/* SPDX-License-Identifier: GPL-2.0 */ /* * Copyright(c) 2010-2014 Intel Corporation. */ #ifndef _KNI_DEV_H_ #define _KNI_DEV_H_ #ifdef pr_fmt #undef pr_fmt #endif #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt #define KNI_VERSION "1.0" #include "compat.h" #include #include #ifdef HAVE_SIGNAL_FUNCTIONS_OWN_HEADER #include #else #include #endif #include #include #include #include #define KNI_KTHREAD_RESCHEDULE_INTERVAL 5 /* us */ #define MBUF_BURST_SZ 32 /* Default carrier state for created KNI network interfaces */ extern uint32_t kni_dflt_carrier; /* Request processing support for bifurcated drivers. */ extern uint32_t bifurcated_support; /** * A structure describing the private information for a kni device. */ struct kni_dev { /* kni list */ struct list_head list; uint8_t iova_mode; uint32_t core_id; /* Core ID to bind */ char name[RTE_KNI_NAMESIZE]; /* Network device name */ struct task_struct *pthread; /* wait queue for req/resp */ wait_queue_head_t wq; struct mutex sync_lock; /* kni device */ struct net_device *net_dev; /* queue for packets to be sent out */ struct rte_kni_fifo *tx_q; /* queue for the packets received */ struct rte_kni_fifo *rx_q; /* queue for the allocated mbufs those can be used to save sk buffs */ struct rte_kni_fifo *alloc_q; /* free queue for the mbufs to be freed */ struct rte_kni_fifo *free_q; /* request queue */ struct rte_kni_fifo *req_q; /* response queue */ struct rte_kni_fifo *resp_q; void *sync_kva; void *sync_va; void *mbuf_kva; void *mbuf_va; /* mbuf size */ uint32_t mbuf_size; /* buffers */ void *pa[MBUF_BURST_SZ]; void *va[MBUF_BURST_SZ]; void *alloc_pa[MBUF_BURST_SZ]; void *alloc_va[MBUF_BURST_SZ]; struct task_struct *usr_tsk; }; #ifdef HAVE_IOVA_TO_KVA_MAPPING_SUPPORT static inline phys_addr_t iova_to_phys(struct task_struct *tsk, unsigned long iova) { phys_addr_t offset, phys_addr; struct page *page = NULL; long ret; offset = iova & (PAGE_SIZE - 1); /* Read one page struct info */ #ifdef HAVE_TSK_IN_GUP ret = get_user_pages_remote(tsk, tsk->mm, iova, 1, FOLL_TOUCH, &page, NULL, NULL); #else ret = get_user_pages_remote(tsk->mm, iova, 1, FOLL_TOUCH, &page, NULL, NULL); #endif if (ret < 0) return 0; phys_addr = page_to_phys(page) | offset; put_page(page); return phys_addr; } static inline void *iova_to_kva(struct task_struct *tsk, unsigned long iova) { return phys_to_virt(iova_to_phys(tsk, iova)); } #endif void kni_net_release_fifo_phy(struct kni_dev *kni); void kni_net_rx(struct kni_dev *kni); void kni_net_init(struct net_device *dev); void kni_net_config_lo_mode(char *lo_str); void kni_net_poll_resp(struct kni_dev *kni); #endif