14ab7403bSDmitry Chagin /*- 24ab7403bSDmitry Chagin * Copyright (c) 2015 Dmitry Chagin 34ab7403bSDmitry Chagin * All rights reserved. 44ab7403bSDmitry Chagin * 54ab7403bSDmitry Chagin * Redistribution and use in source and binary forms, with or without 64ab7403bSDmitry Chagin * modification, are permitted provided that the following conditions 74ab7403bSDmitry Chagin * are met: 84ab7403bSDmitry Chagin * 1. Redistributions of source code must retain the above copyright 94ab7403bSDmitry Chagin * notice, this list of conditions and the following disclaimer. 104ab7403bSDmitry Chagin * 2. Redistributions in binary form must reproduce the above copyright 114ab7403bSDmitry Chagin * notice, this list of conditions and the following disclaimer in the 124ab7403bSDmitry Chagin * documentation and/or other materials provided with the distribution. 134ab7403bSDmitry Chagin * 144ab7403bSDmitry Chagin * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 154ab7403bSDmitry Chagin * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 164ab7403bSDmitry Chagin * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 174ab7403bSDmitry Chagin * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 184ab7403bSDmitry Chagin * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 194ab7403bSDmitry Chagin * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 204ab7403bSDmitry Chagin * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 214ab7403bSDmitry Chagin * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 224ab7403bSDmitry Chagin * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 234ab7403bSDmitry Chagin * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 244ab7403bSDmitry Chagin * SUCH DAMAGE. 254ab7403bSDmitry Chagin */ 264ab7403bSDmitry Chagin 274ab7403bSDmitry Chagin #include <sys/cdefs.h> 284ab7403bSDmitry Chagin __FBSDID("$FreeBSD$"); 294ab7403bSDmitry Chagin 304ab7403bSDmitry Chagin #include <sys/param.h> 314ab7403bSDmitry Chagin #include <sys/systm.h> 32d151344dSDmitry Chagin #include <sys/ctype.h> 33d151344dSDmitry Chagin #include <sys/jail.h> 34d151344dSDmitry Chagin #include <sys/lock.h> 354ab7403bSDmitry Chagin #include <sys/signalvar.h> 36d151344dSDmitry Chagin #include <sys/socket.h> 37d151344dSDmitry Chagin 38d151344dSDmitry Chagin #include <net/if.h> 39d151344dSDmitry Chagin #include <net/if_var.h> 40d151344dSDmitry Chagin #include <net/if_dl.h> 41d151344dSDmitry Chagin #include <net/if_types.h> 424ab7403bSDmitry Chagin 434ab7403bSDmitry Chagin #include <compat/linux/linux.h> 44d151344dSDmitry Chagin #include <compat/linux/linux_common.h> 454ab7403bSDmitry Chagin 46d151344dSDmitry Chagin CTASSERT(LINUX_IFNAMSIZ == IFNAMSIZ); 474ab7403bSDmitry Chagin 484ab7403bSDmitry Chagin static int bsd_to_linux_sigtbl[LINUX_SIGTBLSZ] = { 494ab7403bSDmitry Chagin LINUX_SIGHUP, /* SIGHUP */ 504ab7403bSDmitry Chagin LINUX_SIGINT, /* SIGINT */ 514ab7403bSDmitry Chagin LINUX_SIGQUIT, /* SIGQUIT */ 524ab7403bSDmitry Chagin LINUX_SIGILL, /* SIGILL */ 534ab7403bSDmitry Chagin LINUX_SIGTRAP, /* SIGTRAP */ 544ab7403bSDmitry Chagin LINUX_SIGABRT, /* SIGABRT */ 554ab7403bSDmitry Chagin 0, /* SIGEMT */ 564ab7403bSDmitry Chagin LINUX_SIGFPE, /* SIGFPE */ 574ab7403bSDmitry Chagin LINUX_SIGKILL, /* SIGKILL */ 584ab7403bSDmitry Chagin LINUX_SIGBUS, /* SIGBUS */ 594ab7403bSDmitry Chagin LINUX_SIGSEGV, /* SIGSEGV */ 604ab7403bSDmitry Chagin LINUX_SIGSYS, /* SIGSYS */ 614ab7403bSDmitry Chagin LINUX_SIGPIPE, /* SIGPIPE */ 624ab7403bSDmitry Chagin LINUX_SIGALRM, /* SIGALRM */ 634ab7403bSDmitry Chagin LINUX_SIGTERM, /* SIGTERM */ 644ab7403bSDmitry Chagin LINUX_SIGURG, /* SIGURG */ 654ab7403bSDmitry Chagin LINUX_SIGSTOP, /* SIGSTOP */ 664ab7403bSDmitry Chagin LINUX_SIGTSTP, /* SIGTSTP */ 674ab7403bSDmitry Chagin LINUX_SIGCONT, /* SIGCONT */ 684ab7403bSDmitry Chagin LINUX_SIGCHLD, /* SIGCHLD */ 694ab7403bSDmitry Chagin LINUX_SIGTTIN, /* SIGTTIN */ 704ab7403bSDmitry Chagin LINUX_SIGTTOU, /* SIGTTOU */ 714ab7403bSDmitry Chagin LINUX_SIGIO, /* SIGIO */ 724ab7403bSDmitry Chagin LINUX_SIGXCPU, /* SIGXCPU */ 734ab7403bSDmitry Chagin LINUX_SIGXFSZ, /* SIGXFSZ */ 744ab7403bSDmitry Chagin LINUX_SIGVTALRM,/* SIGVTALRM */ 754ab7403bSDmitry Chagin LINUX_SIGPROF, /* SIGPROF */ 764ab7403bSDmitry Chagin LINUX_SIGWINCH, /* SIGWINCH */ 774ab7403bSDmitry Chagin 0, /* SIGINFO */ 784ab7403bSDmitry Chagin LINUX_SIGUSR1, /* SIGUSR1 */ 794ab7403bSDmitry Chagin LINUX_SIGUSR2 /* SIGUSR2 */ 804ab7403bSDmitry Chagin }; 814ab7403bSDmitry Chagin 824ab7403bSDmitry Chagin static int linux_to_bsd_sigtbl[LINUX_SIGTBLSZ] = { 834ab7403bSDmitry Chagin SIGHUP, /* LINUX_SIGHUP */ 844ab7403bSDmitry Chagin SIGINT, /* LINUX_SIGINT */ 854ab7403bSDmitry Chagin SIGQUIT, /* LINUX_SIGQUIT */ 864ab7403bSDmitry Chagin SIGILL, /* LINUX_SIGILL */ 874ab7403bSDmitry Chagin SIGTRAP, /* LINUX_SIGTRAP */ 884ab7403bSDmitry Chagin SIGABRT, /* LINUX_SIGABRT */ 894ab7403bSDmitry Chagin SIGBUS, /* LINUX_SIGBUS */ 904ab7403bSDmitry Chagin SIGFPE, /* LINUX_SIGFPE */ 914ab7403bSDmitry Chagin SIGKILL, /* LINUX_SIGKILL */ 924ab7403bSDmitry Chagin SIGUSR1, /* LINUX_SIGUSR1 */ 934ab7403bSDmitry Chagin SIGSEGV, /* LINUX_SIGSEGV */ 944ab7403bSDmitry Chagin SIGUSR2, /* LINUX_SIGUSR2 */ 954ab7403bSDmitry Chagin SIGPIPE, /* LINUX_SIGPIPE */ 964ab7403bSDmitry Chagin SIGALRM, /* LINUX_SIGALRM */ 974ab7403bSDmitry Chagin SIGTERM, /* LINUX_SIGTERM */ 984ab7403bSDmitry Chagin SIGBUS, /* LINUX_SIGSTKFLT */ 994ab7403bSDmitry Chagin SIGCHLD, /* LINUX_SIGCHLD */ 1004ab7403bSDmitry Chagin SIGCONT, /* LINUX_SIGCONT */ 1014ab7403bSDmitry Chagin SIGSTOP, /* LINUX_SIGSTOP */ 1024ab7403bSDmitry Chagin SIGTSTP, /* LINUX_SIGTSTP */ 1034ab7403bSDmitry Chagin SIGTTIN, /* LINUX_SIGTTIN */ 1044ab7403bSDmitry Chagin SIGTTOU, /* LINUX_SIGTTOU */ 1054ab7403bSDmitry Chagin SIGURG, /* LINUX_SIGURG */ 1064ab7403bSDmitry Chagin SIGXCPU, /* LINUX_SIGXCPU */ 1074ab7403bSDmitry Chagin SIGXFSZ, /* LINUX_SIGXFSZ */ 1084ab7403bSDmitry Chagin SIGVTALRM, /* LINUX_SIGVTALARM */ 1094ab7403bSDmitry Chagin SIGPROF, /* LINUX_SIGPROF */ 1104ab7403bSDmitry Chagin SIGWINCH, /* LINUX_SIGWINCH */ 1114ab7403bSDmitry Chagin SIGIO, /* LINUX_SIGIO */ 1124ab7403bSDmitry Chagin /* 1134ab7403bSDmitry Chagin * FreeBSD does not have SIGPWR signal, map Linux SIGPWR signal 1144ab7403bSDmitry Chagin * to the first unused FreeBSD signal number. Since Linux supports 1154ab7403bSDmitry Chagin * signals from 1 to 64 we are ok here as our SIGRTMIN = 65. 1164ab7403bSDmitry Chagin */ 1174ab7403bSDmitry Chagin SIGRTMIN, /* LINUX_SIGPWR */ 1184ab7403bSDmitry Chagin SIGSYS /* LINUX_SIGSYS */ 1194ab7403bSDmitry Chagin }; 1204ab7403bSDmitry Chagin 1214ab7403bSDmitry Chagin /* 1224ab7403bSDmitry Chagin * Map Linux RT signals to the FreeBSD RT signals. 1234ab7403bSDmitry Chagin */ 1244ab7403bSDmitry Chagin static inline int 1254ab7403bSDmitry Chagin linux_to_bsd_rt_signal(int sig) 1264ab7403bSDmitry Chagin { 1274ab7403bSDmitry Chagin 1284ab7403bSDmitry Chagin return (SIGRTMIN + 1 + sig - LINUX_SIGRTMIN); 1294ab7403bSDmitry Chagin } 1304ab7403bSDmitry Chagin 1314ab7403bSDmitry Chagin static inline int 1324ab7403bSDmitry Chagin bsd_to_linux_rt_signal(int sig) 1334ab7403bSDmitry Chagin { 1344ab7403bSDmitry Chagin 1354ab7403bSDmitry Chagin return (sig - SIGRTMIN - 1 + LINUX_SIGRTMIN); 1364ab7403bSDmitry Chagin } 1374ab7403bSDmitry Chagin 1384ab7403bSDmitry Chagin int 1394ab7403bSDmitry Chagin linux_to_bsd_signal(int sig) 1404ab7403bSDmitry Chagin { 1414ab7403bSDmitry Chagin 1428081c6ceSEdward Tomasz Napierala KASSERT(sig > 0 && sig <= LINUX_SIGRTMAX, ("invalid Linux signal %d\n", sig)); 1434ab7403bSDmitry Chagin 1444ab7403bSDmitry Chagin if (sig < LINUX_SIGRTMIN) 1454ab7403bSDmitry Chagin return (linux_to_bsd_sigtbl[_SIG_IDX(sig)]); 1464ab7403bSDmitry Chagin 1474ab7403bSDmitry Chagin return (linux_to_bsd_rt_signal(sig)); 1484ab7403bSDmitry Chagin } 1494ab7403bSDmitry Chagin 1504ab7403bSDmitry Chagin int 1514ab7403bSDmitry Chagin bsd_to_linux_signal(int sig) 1524ab7403bSDmitry Chagin { 1534ab7403bSDmitry Chagin 1544ab7403bSDmitry Chagin if (sig <= LINUX_SIGTBLSZ) 1554ab7403bSDmitry Chagin return (bsd_to_linux_sigtbl[_SIG_IDX(sig)]); 1564ab7403bSDmitry Chagin if (sig == SIGRTMIN) 1574ab7403bSDmitry Chagin return (LINUX_SIGPWR); 1584ab7403bSDmitry Chagin 1594ab7403bSDmitry Chagin return (bsd_to_linux_rt_signal(sig)); 1604ab7403bSDmitry Chagin } 1614ab7403bSDmitry Chagin 1624ab7403bSDmitry Chagin int 1634ab7403bSDmitry Chagin linux_to_bsd_sigaltstack(int lsa) 1644ab7403bSDmitry Chagin { 1654ab7403bSDmitry Chagin int bsa = 0; 1664ab7403bSDmitry Chagin 1674ab7403bSDmitry Chagin if (lsa & LINUX_SS_DISABLE) 1684ab7403bSDmitry Chagin bsa |= SS_DISABLE; 1694ab7403bSDmitry Chagin /* 1704ab7403bSDmitry Chagin * Linux ignores SS_ONSTACK flag for ss 1714ab7403bSDmitry Chagin * parameter while FreeBSD prohibits it. 1724ab7403bSDmitry Chagin */ 1734ab7403bSDmitry Chagin return (bsa); 1744ab7403bSDmitry Chagin } 1754ab7403bSDmitry Chagin 1764ab7403bSDmitry Chagin int 1774ab7403bSDmitry Chagin bsd_to_linux_sigaltstack(int bsa) 1784ab7403bSDmitry Chagin { 1794ab7403bSDmitry Chagin int lsa = 0; 1804ab7403bSDmitry Chagin 1814ab7403bSDmitry Chagin if (bsa & SS_DISABLE) 1824ab7403bSDmitry Chagin lsa |= LINUX_SS_DISABLE; 1834ab7403bSDmitry Chagin if (bsa & SS_ONSTACK) 1844ab7403bSDmitry Chagin lsa |= LINUX_SS_ONSTACK; 1854ab7403bSDmitry Chagin return (lsa); 1864ab7403bSDmitry Chagin } 1874ab7403bSDmitry Chagin 1884ab7403bSDmitry Chagin void 1894ab7403bSDmitry Chagin linux_to_bsd_sigset(l_sigset_t *lss, sigset_t *bss) 1904ab7403bSDmitry Chagin { 1914ab7403bSDmitry Chagin int b, l; 1924ab7403bSDmitry Chagin 1934ab7403bSDmitry Chagin SIGEMPTYSET(*bss); 1944ab7403bSDmitry Chagin for (l = 1; l <= LINUX_SIGRTMAX; l++) { 1954ab7403bSDmitry Chagin if (LINUX_SIGISMEMBER(*lss, l)) { 1964ab7403bSDmitry Chagin b = linux_to_bsd_signal(l); 1974ab7403bSDmitry Chagin if (b) 1984ab7403bSDmitry Chagin SIGADDSET(*bss, b); 1994ab7403bSDmitry Chagin } 2004ab7403bSDmitry Chagin } 2014ab7403bSDmitry Chagin } 2024ab7403bSDmitry Chagin 2034ab7403bSDmitry Chagin void 2044ab7403bSDmitry Chagin bsd_to_linux_sigset(sigset_t *bss, l_sigset_t *lss) 2054ab7403bSDmitry Chagin { 2064ab7403bSDmitry Chagin int b, l; 2074ab7403bSDmitry Chagin 2084ab7403bSDmitry Chagin LINUX_SIGEMPTYSET(*lss); 2094ab7403bSDmitry Chagin for (b = 1; b <= SIGRTMAX; b++) { 2104ab7403bSDmitry Chagin if (SIGISMEMBER(*bss, b)) { 2114ab7403bSDmitry Chagin l = bsd_to_linux_signal(b); 2124ab7403bSDmitry Chagin if (l) 2134ab7403bSDmitry Chagin LINUX_SIGADDSET(*lss, l); 2144ab7403bSDmitry Chagin } 2154ab7403bSDmitry Chagin } 2164ab7403bSDmitry Chagin } 217d151344dSDmitry Chagin 218d151344dSDmitry Chagin /* 219d151344dSDmitry Chagin * Translate a Linux interface name to a FreeBSD interface name, 220d151344dSDmitry Chagin * and return the associated ifnet structure 221d151344dSDmitry Chagin * bsdname and lxname need to be least IFNAMSIZ bytes long, but 222d151344dSDmitry Chagin * can point to the same buffer. 223d151344dSDmitry Chagin */ 224d151344dSDmitry Chagin struct ifnet * 225d151344dSDmitry Chagin ifname_linux_to_bsd(struct thread *td, const char *lxname, char *bsdname) 226d151344dSDmitry Chagin { 227d151344dSDmitry Chagin struct ifnet *ifp; 228d151344dSDmitry Chagin int len, unit; 229d151344dSDmitry Chagin char *ep; 2309c1437aeSDmitry Chagin int index; 2319c1437aeSDmitry Chagin bool is_eth, is_lo; 232d151344dSDmitry Chagin 233d151344dSDmitry Chagin for (len = 0; len < LINUX_IFNAMSIZ; ++len) 2349c1437aeSDmitry Chagin if (!isalpha(lxname[len]) || lxname[len] == '\0') 235d151344dSDmitry Chagin break; 236d151344dSDmitry Chagin if (len == 0 || len == LINUX_IFNAMSIZ) 237d151344dSDmitry Chagin return (NULL); 238d151344dSDmitry Chagin /* Linux loopback interface name is lo (not lo0) */ 2399c1437aeSDmitry Chagin is_lo = (len == 2 && strncmp(lxname, "lo", len) == 0); 240d151344dSDmitry Chagin unit = (int)strtoul(lxname + len, &ep, 10); 241d151344dSDmitry Chagin if ((ep == NULL || ep == lxname + len || ep >= lxname + LINUX_IFNAMSIZ) && 242d151344dSDmitry Chagin is_lo == 0) 243d151344dSDmitry Chagin return (NULL); 244d151344dSDmitry Chagin index = 0; 2459c1437aeSDmitry Chagin is_eth = (len == 3 && strncmp(lxname, "eth", len) == 0); 246d151344dSDmitry Chagin 247d151344dSDmitry Chagin CURVNET_SET(TD_TO_VNET(td)); 248d151344dSDmitry Chagin IFNET_RLOCK(); 249d151344dSDmitry Chagin CK_STAILQ_FOREACH(ifp, &V_ifnet, if_link) { 250d151344dSDmitry Chagin /* 251d151344dSDmitry Chagin * Allow Linux programs to use FreeBSD names. Don't presume 252d151344dSDmitry Chagin * we never have an interface named "eth", so don't make 253d151344dSDmitry Chagin * the test optional based on is_eth. 254d151344dSDmitry Chagin */ 255d151344dSDmitry Chagin if (strncmp(ifp->if_xname, lxname, LINUX_IFNAMSIZ) == 0) 256d151344dSDmitry Chagin break; 257d151344dSDmitry Chagin if (is_eth && IFP_IS_ETH(ifp) && unit == index++) 258d151344dSDmitry Chagin break; 259d151344dSDmitry Chagin if (is_lo && IFP_IS_LOOP(ifp)) 260d151344dSDmitry Chagin break; 261d151344dSDmitry Chagin } 262d151344dSDmitry Chagin IFNET_RUNLOCK(); 263d151344dSDmitry Chagin CURVNET_RESTORE(); 264d151344dSDmitry Chagin if (ifp != NULL && bsdname != NULL) 265d151344dSDmitry Chagin strlcpy(bsdname, ifp->if_xname, IFNAMSIZ); 266d151344dSDmitry Chagin return (ifp); 267d151344dSDmitry Chagin } 268d151344dSDmitry Chagin 269d151344dSDmitry Chagin void 270d151344dSDmitry Chagin linux_ifflags(struct ifnet *ifp, short *flags) 271d151344dSDmitry Chagin { 272*bbac65c7SDmitry Chagin unsigned short fl; 273d151344dSDmitry Chagin 274*bbac65c7SDmitry Chagin fl = (ifp->if_flags | ifp->if_drv_flags) & 0xffff; 275*bbac65c7SDmitry Chagin *flags = 0; 276*bbac65c7SDmitry Chagin if (fl & IFF_UP) 277*bbac65c7SDmitry Chagin *flags |= LINUX_IFF_UP; 278*bbac65c7SDmitry Chagin if (fl & IFF_BROADCAST) 279*bbac65c7SDmitry Chagin *flags |= LINUX_IFF_BROADCAST; 280*bbac65c7SDmitry Chagin if (fl & IFF_DEBUG) 281*bbac65c7SDmitry Chagin *flags |= LINUX_IFF_DEBUG; 282*bbac65c7SDmitry Chagin if (fl & IFF_LOOPBACK) 283*bbac65c7SDmitry Chagin *flags |= LINUX_IFF_LOOPBACK; 284*bbac65c7SDmitry Chagin if (fl & IFF_POINTOPOINT) 285*bbac65c7SDmitry Chagin *flags |= LINUX_IFF_POINTOPOINT; 286*bbac65c7SDmitry Chagin if (fl & IFF_DRV_RUNNING) 287*bbac65c7SDmitry Chagin *flags |= LINUX_IFF_RUNNING; 288*bbac65c7SDmitry Chagin if (fl & IFF_NOARP) 289*bbac65c7SDmitry Chagin *flags |= LINUX_IFF_NOARP; 290*bbac65c7SDmitry Chagin if (fl & IFF_PROMISC) 291*bbac65c7SDmitry Chagin *flags |= LINUX_IFF_PROMISC; 292*bbac65c7SDmitry Chagin if (fl & IFF_ALLMULTI) 293*bbac65c7SDmitry Chagin *flags |= LINUX_IFF_ALLMULTI; 294*bbac65c7SDmitry Chagin if (fl & IFF_MULTICAST) 295*bbac65c7SDmitry Chagin *flags |= LINUX_IFF_MULTICAST; 296d151344dSDmitry Chagin } 297d151344dSDmitry Chagin 298d151344dSDmitry Chagin int 299d151344dSDmitry Chagin linux_ifhwaddr(struct ifnet *ifp, struct l_sockaddr *lsa) 300d151344dSDmitry Chagin { 301d151344dSDmitry Chagin struct ifaddr *ifa; 302d151344dSDmitry Chagin struct sockaddr_dl *sdl; 303d151344dSDmitry Chagin 304d151344dSDmitry Chagin if (IFP_IS_LOOP(ifp)) { 305d151344dSDmitry Chagin bzero(lsa, sizeof(*lsa)); 306d151344dSDmitry Chagin lsa->sa_family = LINUX_ARPHRD_LOOPBACK; 307d151344dSDmitry Chagin return (0); 308d151344dSDmitry Chagin } 309d151344dSDmitry Chagin 310d151344dSDmitry Chagin if (!IFP_IS_ETH(ifp)) 311d151344dSDmitry Chagin return (ENOENT); 312d151344dSDmitry Chagin 313d151344dSDmitry Chagin CK_STAILQ_FOREACH(ifa, &ifp->if_addrhead, ifa_link) { 314d151344dSDmitry Chagin sdl = (struct sockaddr_dl*)ifa->ifa_addr; 315d151344dSDmitry Chagin if (sdl != NULL && (sdl->sdl_family == AF_LINK) && 316d151344dSDmitry Chagin (sdl->sdl_type == IFT_ETHER)) { 317d151344dSDmitry Chagin bzero(lsa, sizeof(*lsa)); 318d151344dSDmitry Chagin lsa->sa_family = LINUX_ARPHRD_ETHER; 319d151344dSDmitry Chagin bcopy(LLADDR(sdl), lsa->sa_data, LINUX_IFHWADDRLEN); 320d151344dSDmitry Chagin return (0); 321d151344dSDmitry Chagin } 322d151344dSDmitry Chagin } 323d151344dSDmitry Chagin 324d151344dSDmitry Chagin return (ENOENT); 325d151344dSDmitry Chagin } 326