/* * Copyright (c) 2010 Kip Macy. All rights reserved. * Copyright (C) 2017-2021 THL A29 Limited, a Tencent company. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, this * list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. * * Derived in part from libplebnet's pn_veth.c. * */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "ff_veth.h" #include "ff_config.h" #include "ff_dpdk_if.h" struct ff_veth_softc { struct ifnet *ifp; uint8_t mac[ETHER_ADDR_LEN]; char host_ifname[IF_NAMESIZE]; in_addr_t ip; in_addr_t netmask; in_addr_t broadcast; in_addr_t gateway; uint8_t nb_vip; in_addr_t vip[VIP_MAX_NUM]; #ifdef INET6 struct in6_addr ip6; struct in6_addr gateway6; uint8_t prefix_length; uint8_t nb_vip6; uint8_t vip_prefix_length; struct in6_addr vip6[VIP_MAX_NUM]; #endif /* INET6 */ struct ff_dpdk_if_context *host_ctx; }; static int ff_veth_config(struct ff_veth_softc *sc, struct ff_port_cfg *cfg) { int i, j; memcpy(sc->mac, cfg->mac, ETHER_ADDR_LEN); inet_pton(AF_INET, cfg->addr, &sc->ip); inet_pton(AF_INET, cfg->netmask, &sc->netmask); inet_pton(AF_INET, cfg->broadcast, &sc->broadcast); inet_pton(AF_INET, cfg->gateway, &sc->gateway); if (cfg->nb_vip) { for (i = 0, j = 0; i < cfg->nb_vip; ++i) { if (inet_pton(AF_INET, cfg->vip_addr_array[i], &sc->vip[j])) { j++; } else { printf("ff_veth_config inet_pton vip %s failed.\n", cfg->vip_addr_array[i]); } } sc->nb_vip = j; } #ifdef INET6 if (cfg->addr6_str) { inet_pton(AF_INET6_LINUX, cfg->addr6_str, &sc->ip6); printf("%s: Addr6: %s\n", sc->host_ifname, cfg->addr6_str); if (cfg->gateway6_str) { inet_pton(AF_INET6_LINUX, cfg->gateway6_str, &sc->gateway6); printf("%s: Gateway6: %s\n", sc->host_ifname, cfg->gateway6_str); } else { printf("%s: No gateway6 config found.\n", sc->host_ifname); } sc->prefix_length = cfg->prefix_len == 0 ? 64 : cfg->prefix_len; } else { printf("%s: No addr6 config found.\n", sc->host_ifname); } if (cfg->nb_vip6) { for (i = 0, j = 0; i < cfg->nb_vip6; ++i) { if (inet_pton(AF_INET6_LINUX, cfg->vip_addr6_array[i], &sc->vip6[j])) { j++; } else { printf("ff_veth_config inet_pton vip6 %s failed.\n", cfg->vip_addr6_array[i]); } } sc->nb_vip6 = j; sc->vip_prefix_length = cfg->vip_prefix_len == 0 ? 64 : cfg->vip_prefix_len; } #endif /* INET6 */ return 0; } static void ff_veth_init(void *arg) { struct ff_veth_softc *sc = arg; struct ifnet *ifp = sc->ifp; ifp->if_drv_flags |= IFF_DRV_RUNNING; ifp->if_drv_flags &= ~IFF_DRV_OACTIVE; } static void ff_veth_start(struct ifnet *ifp) { /* nothing to do */ } static void ff_veth_stop(struct ff_veth_softc *sc) { struct ifnet *ifp = sc->ifp; ifp->if_drv_flags &= ~(IFF_DRV_RUNNING|IFF_DRV_OACTIVE); } static int ff_veth_ioctl(struct ifnet *ifp, u_long cmd, caddr_t data) { int error = 0; struct ff_veth_softc *sc = ifp->if_softc; switch (cmd) { case SIOCSIFFLAGS: if (ifp->if_flags & IFF_UP) { ff_veth_init(sc); } else if (ifp->if_drv_flags & IFF_DRV_RUNNING) ff_veth_stop(sc); break; default: error = ether_ioctl(ifp, cmd, data); break; } return (error); } int ff_mbuf_copydata(void *m, void *data, int off, int len) { int ret; struct mbuf *mb = (struct mbuf *)m; if (off + len > mb->m_pkthdr.len) { return -1; } m_copydata(mb, off, len, data); return 0; } void ff_mbuf_tx_offload(void *m, struct ff_tx_offload *offload) { struct mbuf *mb = (struct mbuf *)m; if (mb->m_pkthdr.csum_flags & CSUM_IP) { offload->ip_csum = 1; } if (mb->m_pkthdr.csum_flags & CSUM_TCP) { offload->tcp_csum = 1; } if (mb->m_pkthdr.csum_flags & CSUM_UDP) { offload->udp_csum = 1; } if (mb->m_pkthdr.csum_flags & CSUM_SCTP) { offload->sctp_csum = 1; } if (mb->m_pkthdr.csum_flags & CSUM_TSO) { offload->tso_seg_size = mb->m_pkthdr.tso_segsz; } } void ff_mbuf_free(void *m) { m_freem((struct mbuf *)m); } static void ff_mbuf_ext_free(struct mbuf *m) { ff_dpdk_pktmbuf_free(ff_rte_frm_extcl(m)); } int ff_zc_mbuf_get(struct ff_zc_mbuf *m, int len) { struct mbuf *mb; if (m == NULL) { return -1; } mb = m_getm2(NULL, max(len, 1), M_WAITOK, MT_DATA, 0); if (mb == NULL) { return -1; } m->bsd_mbuf = m->bsd_mbuf_off = mb; m->off = 0; m->len = len; return 0; } int ff_zc_mbuf_write(struct ff_zc_mbuf *zm, const char *data, int len) { int ret, length, progress = 0; struct mbuf *m, *mb; if (zm == NULL) { return -1; } m = (struct mbuf *)zm->bsd_mbuf_off; if (zm->off + len > zm->len) { return -1; } for (mb = m; mb != NULL; mb = mb->m_next) { length = min(M_TRAILINGSPACE(mb), len - progress); bcopy(data + progress, mtod(mb, char *) + mb->m_len, length); mb->m_len += length; progress += length; if (len == progress) { break; } //if (flags & M_PKTHDR) // m->m_pkthdr.len += length; } zm->off += len; zm->bsd_mbuf_off = mb; return len; } int ff_zc_mbuf_read(struct ff_zc_mbuf *m, const char *data, int len) { // DOTO: Support read zero copy return 0; } void * ff_mbuf_gethdr(void *pkt, uint16_t total, void *data, uint16_t len, uint8_t rx_csum) { struct mbuf *m = m_gethdr(M_NOWAIT, MT_DATA); if (m == NULL) { return NULL; } if (m_pkthdr_init(m, M_NOWAIT) != 0) { return NULL; } m_extadd(m, data, len, ff_mbuf_ext_free, pkt, NULL, 0, EXT_DISPOSABLE); m->m_pkthdr.len = total; m->m_len = len; m->m_next = NULL; m->m_nextpkt = NULL; if (rx_csum) { m->m_pkthdr.csum_flags = CSUM_IP_CHECKED | CSUM_IP_VALID | CSUM_DATA_VALID | CSUM_PSEUDO_HDR; m->m_pkthdr.csum_data = 0xffff; } return (void *)m; } void * ff_mbuf_get(void *p, void *m, void *data, uint16_t len) { struct mbuf *prev = (struct mbuf *)p; struct mbuf *mb = m_get(M_NOWAIT, MT_DATA); if (mb == NULL) { return NULL; } m_extadd(mb, data, len, ff_mbuf_ext_free, m, NULL, 0, EXT_DISPOSABLE); mb->m_next = NULL; mb->m_nextpkt = NULL; mb->m_len = len; if (prev != NULL) { prev->m_next = mb; } return (void *)mb; } void ff_veth_process_packet(void *arg, void *m) { struct ifnet *ifp = (struct ifnet *)arg; struct mbuf *mb = (struct mbuf *)m; mb->m_pkthdr.rcvif = ifp; ifp->if_input(ifp, mb); } static int ff_veth_transmit(struct ifnet *ifp, struct mbuf *m) { struct ff_veth_softc *sc = (struct ff_veth_softc *)ifp->if_softc; return ff_dpdk_if_send(sc->host_ctx, (void*)m, m->m_pkthdr.len); } static void ff_veth_qflush(struct ifnet *ifp) { } static int ff_veth_setaddr(struct ff_veth_softc *sc) { struct in_aliasreq req; bzero(&req, sizeof req); strcpy(req.ifra_name, sc->ifp->if_dname); struct sockaddr_in sa; bzero(&sa, sizeof(sa)); sa.sin_len = sizeof(sa); sa.sin_family = AF_INET; sa.sin_addr.s_addr = sc->ip; bcopy(&sa, &req.ifra_addr, sizeof(sa)); sa.sin_addr.s_addr = sc->netmask; bcopy(&sa, &req.ifra_mask, sizeof(sa)); sa.sin_addr.s_addr = sc->broadcast; bcopy(&sa, &req.ifra_broadaddr, sizeof(sa)); struct socket *so = NULL; socreate(AF_INET, &so, SOCK_DGRAM, 0, curthread->td_ucred, curthread); int ret = ifioctl(so, SIOCAIFADDR, (caddr_t)&req, curthread); sofree(so); return ret; } static int ff_veth_set_gateway(struct ff_veth_softc *sc) { struct rt_addrinfo info; struct rib_cmd_info rci; bzero((caddr_t)&info, sizeof(info)); info.rti_flags = RTF_GATEWAY; struct sockaddr_in gw; bzero(&gw, sizeof(gw)); gw.sin_len = sizeof(gw); gw.sin_family = AF_INET; gw.sin_addr.s_addr = sc->gateway; info.rti_info[RTAX_GATEWAY] = (struct sockaddr *)&gw; struct sockaddr_in dst; bzero(&dst, sizeof(dst)); dst.sin_len = sizeof(dst); dst.sin_family = AF_INET; dst.sin_addr.s_addr = 0; info.rti_info[RTAX_DST] = (struct sockaddr *)&dst; struct sockaddr_in nm; bzero(&nm, sizeof(nm)); nm.sin_len = sizeof(nm); nm.sin_family = AF_INET; nm.sin_addr.s_addr = 0; info.rti_info[RTAX_NETMASK] = (struct sockaddr *)&nm; return rib_action(RT_DEFAULT_FIB, RTM_ADD, &info, &rci); } static int ff_veth_setvaddr(struct ff_veth_softc *sc, struct ff_port_cfg *cfg) { struct in_aliasreq req; bzero(&req, sizeof req); if (cfg->vip_ifname) { strlcpy(req.ifra_name, cfg->vip_ifname, IFNAMSIZ); } else { strlcpy(req.ifra_name, sc->ifp->if_dname, IFNAMSIZ); } struct sockaddr_in sa; bzero(&sa, sizeof(sa)); sa.sin_len = sizeof(sa); sa.sin_family = AF_INET; int i, ret; struct socket *so = NULL; socreate(AF_INET, &so, SOCK_DGRAM, 0, curthread->td_ucred, curthread); for (i = 0; i < sc->nb_vip; ++i) { sa.sin_addr.s_addr = sc->vip[i]; bcopy(&sa, &req.ifra_addr, sizeof(sa)); // Only support '255.255.255.255' netmask now sa.sin_addr.s_addr = 0xFFFFFFFF; bcopy(&sa, &req.ifra_mask, sizeof(sa)); // Only support 'x.x.x.255' broadaddr now sa.sin_addr.s_addr = sc->vip[i] | 0xFF000000; bcopy(&sa, &req.ifra_broadaddr, sizeof(sa)); ret = ifioctl(so, SIOCAIFADDR, (caddr_t)&req, curthread); if (ret < 0) { printf("ff_veth_setvaddr ifioctl SIOCAIFADDR error\n"); goto done; } } done: sofree(so); return ret; } #ifdef INET6 static int ff_veth_setaddr6(struct ff_veth_softc *sc) { struct in6_aliasreq ifr6; bzero(&ifr6, sizeof(ifr6)); strcpy(ifr6.ifra_name, sc->ifp->if_dname); ifr6.ifra_addr.sin6_len = sizeof ifr6.ifra_addr; ifr6.ifra_addr.sin6_family = AF_INET6; ifr6.ifra_addr.sin6_addr = sc->ip6; ifr6.ifra_prefixmask.sin6_len = sizeof ifr6.ifra_prefixmask; memset(&ifr6.ifra_prefixmask.sin6_addr, 0xff, sc->prefix_length / 8); uint8_t mask_size_mod = sc->prefix_length % 8; if (mask_size_mod) { ifr6.ifra_prefixmask.sin6_addr.__u6_addr.__u6_addr8[sc->prefix_length / 8] = \ ((1 << mask_size_mod) - 1) << (8 - mask_size_mod); } ifr6.ifra_lifetime.ia6t_pltime = ifr6.ifra_lifetime.ia6t_vltime = ND6_INFINITE_LIFETIME; ifr6.ifra_flags |= IN6_IFF_AUTOCONF; struct socket *so = NULL; socreate(AF_INET6, &so, SOCK_DGRAM, 0, curthread->td_ucred, curthread); int ret = ifioctl(so, SIOCAIFADDR_IN6, (caddr_t)&ifr6, curthread); sofree(so); return ret; } static int ff_veth_set_gateway6(struct ff_veth_softc *sc) { struct sockaddr_in6 gw, dst, nm;; struct rt_addrinfo info; struct rib_cmd_info rci; bzero((caddr_t)&info, sizeof(info)); info.rti_flags = RTF_GATEWAY; bzero(&gw, sizeof(gw)); bzero(&dst, sizeof(dst)); bzero(&nm, sizeof(nm)); gw.sin6_len = dst.sin6_len = nm.sin6_len = sizeof(struct sockaddr_in6); gw.sin6_family = dst.sin6_family = nm.sin6_family = AF_INET6; gw.sin6_addr = sc->gateway6; //dst.sin6_addr = nm.sin6_addr = 0; info.rti_info[RTAX_GATEWAY] = (struct sockaddr *)&gw; info.rti_info[RTAX_DST] = (struct sockaddr *)&dst; info.rti_info[RTAX_NETMASK] = (struct sockaddr *)&nm; return rib_action(RT_DEFAULT_FIB, RTM_ADD, &info, &rci); } static int ff_veth_setvaddr6(struct ff_veth_softc *sc, struct ff_port_cfg *cfg) { struct in6_aliasreq ifr6; bzero(&ifr6, sizeof(ifr6)); if (cfg->vip_ifname) { strlcpy(ifr6.ifra_name, cfg->vip_ifname, IFNAMSIZ); } else { strlcpy(ifr6.ifra_name, sc->ifp->if_dname, IFNAMSIZ); } ifr6.ifra_addr.sin6_len = sizeof ifr6.ifra_addr; ifr6.ifra_addr.sin6_family = AF_INET6; ifr6.ifra_prefixmask.sin6_len = sizeof ifr6.ifra_prefixmask; memset(&ifr6.ifra_prefixmask.sin6_addr, 0xff, sc->prefix_length / 8); uint8_t mask_size_mod = sc->prefix_length % 8; if (mask_size_mod) { ifr6.ifra_prefixmask.sin6_addr.__u6_addr.__u6_addr8[sc->prefix_length / 8] = \ ((1 << mask_size_mod) - 1) << (8 - mask_size_mod); } ifr6.ifra_lifetime.ia6t_pltime = ifr6.ifra_lifetime.ia6t_vltime = ND6_INFINITE_LIFETIME; ifr6.ifra_flags |= IN6_IFF_AUTOCONF; struct socket *so = NULL; socreate(AF_INET6, &so, SOCK_DGRAM, 0, curthread->td_ucred, curthread); int i, ret; for (i = 0; i < sc->nb_vip6; ++i) { ifr6.ifra_addr.sin6_addr = sc->vip6[i]; ret = ifioctl(so, SIOCAIFADDR_IN6, (caddr_t)&ifr6, curthread); if (ret < 0) { printf("ff_veth_setvaddr6 ifioctl SIOCAIFADDR error\n"); goto done; } } done: sofree(so); return ret; } #endif /* INET6 */ static int ff_veth_setup_interface(struct ff_veth_softc *sc, struct ff_port_cfg *cfg) { struct ifnet *ifp; ifp = sc->ifp = if_alloc(IFT_ETHER); ifp->if_init = ff_veth_init; ifp->if_softc = sc; if_initname(ifp, sc->host_ifname, IF_DUNIT_NONE); ifp->if_flags = IFF_BROADCAST | IFF_SIMPLEX | IFF_MULTICAST; ifp->if_ioctl = ff_veth_ioctl; ifp->if_start = ff_veth_start; ifp->if_transmit = ff_veth_transmit; ifp->if_qflush = ff_veth_qflush; ether_ifattach(ifp, sc->mac); if (cfg->hw_features.rx_csum) { ifp->if_capabilities |= IFCAP_RXCSUM; } if (cfg->hw_features.tx_csum_ip) { ifp->if_capabilities |= IFCAP_TXCSUM; ifp->if_hwassist |= CSUM_IP; } if (cfg->hw_features.tx_csum_l4) { ifp->if_hwassist |= CSUM_DELAY_DATA; } if (cfg->hw_features.tx_tso) { ifp->if_capabilities |= IFCAP_TSO; ifp->if_hwassist |= CSUM_TSO; } ifp->if_capenable = ifp->if_capabilities; sc->host_ctx = ff_dpdk_register_if((void *)sc, (void *)sc->ifp, cfg); if (sc->host_ctx == NULL) { printf("%s: Failed to register dpdk interface\n", sc->host_ifname); return -1; } // Set ip int ret = ff_veth_setaddr(sc); if (ret != 0) { printf("ff_veth_setaddr failed\n"); } ret = ff_veth_set_gateway(sc); if (ret != 0) { printf("ff_veth_set_gateway failed\n"); } if (sc->nb_vip) { ret = ff_veth_setvaddr(sc, cfg); } #ifdef INET6 // Set IPv6 if (cfg->addr6_str) { ret = ff_veth_setaddr6(sc); if (ret != 0) { printf("ff_veth_setaddr6 failed\n"); } if (cfg->gateway6_str) { ret = ff_veth_set_gateway6(sc); if (ret != 0) { printf("ff_veth_set_gateway6 failed\n"); } } } if (sc->nb_vip6) { ret = ff_veth_setvaddr6(sc, cfg); } #endif /* INET6 */ return (0); } void * ff_veth_attach(struct ff_port_cfg *cfg) { struct ff_veth_softc *sc = NULL; int error; sc = malloc(sizeof(struct ff_veth_softc), M_DEVBUF, M_WAITOK); if (NULL == sc) { printf("ff_veth_softc allocation failed\n"); goto fail; } memset(sc, 0, sizeof(struct ff_veth_softc)); if(cfg->ifname){ snprintf(sc->host_ifname, sizeof(sc->host_ifname), "%s", cfg->ifname); } else { snprintf(sc->host_ifname, sizeof(sc->host_ifname), ff_IF_NAME, cfg->port_id); } error = ff_veth_config(sc, cfg); if (0 != error) { goto fail; } if (0 != ff_veth_setup_interface(sc, cfg)) { goto fail; } return sc->host_ctx; fail: if (sc) { if (sc->host_ctx) ff_dpdk_deregister_if(sc->host_ctx); free(sc, M_DEVBUF); } return NULL; } int ff_veth_detach(void *arg) { struct ff_veth_softc *sc = (struct ff_veth_softc *)arg; if (sc) { ff_dpdk_deregister_if(sc->host_ctx); free(sc, M_DEVBUF); } return (0); } void * ff_veth_softc_to_hostc(void *softc) { struct ff_veth_softc *sc = (struct ff_veth_softc *)softc; return (void *)sc->host_ctx; } /******************** * get next mbuf's addr, current mbuf's data and datalen. * ********************/ int ff_next_mbuf(void **mbuf_bsd, void **data, unsigned *len) { struct mbuf *mb = *(struct mbuf **)mbuf_bsd; *len = mb->m_len; *data = mb->m_data; if (mb->m_next) *mbuf_bsd = mb->m_next; else *mbuf_bsd = NULL; return 0; } void * ff_mbuf_mtod(void* bsd_mbuf) { if ( !bsd_mbuf ) return NULL; return (void*)((struct mbuf *)bsd_mbuf)->m_data; } // get source rte_mbuf from ext cluster, which carry rte_mbuf while recving pkt, such as arp. void* ff_rte_frm_extcl(void* mbuf) { struct mbuf *bsd_mbuf = mbuf; if ( (bsd_mbuf->m_flags & M_EXT) && bsd_mbuf->m_ext.ext_type == EXT_DISPOSABLE && bsd_mbuf->m_ext.ext_free == ff_mbuf_ext_free ) { return bsd_mbuf->m_ext.ext_arg1; } else return NULL; } void ff_mbuf_set_vlan_info(void *hdr, uint16_t vlan_tci) { struct mbuf *m = (struct mbuf *)hdr; m->m_pkthdr.ether_vtag = vlan_tci; m->m_flags |= M_VLANTAG; return; }