14ab7403bSDmitry Chagin /*- 21ca6b15bSDmitry Chagin * Copyright (c) 2015 Dmitry Chagin <dchagin@FreeBSD.org> 34ab7403bSDmitry Chagin * 44ab7403bSDmitry Chagin * Redistribution and use in source and binary forms, with or without 54ab7403bSDmitry Chagin * modification, are permitted provided that the following conditions 64ab7403bSDmitry Chagin * are met: 74ab7403bSDmitry Chagin * 1. Redistributions of source code must retain the above copyright 84ab7403bSDmitry Chagin * notice, this list of conditions and the following disclaimer. 94ab7403bSDmitry Chagin * 2. Redistributions in binary form must reproduce the above copyright 104ab7403bSDmitry Chagin * notice, this list of conditions and the following disclaimer in the 114ab7403bSDmitry Chagin * documentation and/or other materials provided with the distribution. 124ab7403bSDmitry Chagin * 134ab7403bSDmitry Chagin * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 144ab7403bSDmitry Chagin * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 154ab7403bSDmitry Chagin * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 164ab7403bSDmitry Chagin * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 174ab7403bSDmitry Chagin * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 184ab7403bSDmitry Chagin * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 194ab7403bSDmitry Chagin * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 204ab7403bSDmitry Chagin * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 214ab7403bSDmitry Chagin * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 224ab7403bSDmitry Chagin * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 234ab7403bSDmitry Chagin * SUCH DAMAGE. 244ab7403bSDmitry Chagin */ 254ab7403bSDmitry Chagin 264ab7403bSDmitry Chagin #include <sys/cdefs.h> 274ab7403bSDmitry Chagin __FBSDID("$FreeBSD$"); 284ab7403bSDmitry Chagin 2977f66834SDmitry Chagin #include "opt_inet6.h" 30d5368bf3SDmitry Chagin 314ab7403bSDmitry Chagin #include <sys/param.h> 32044ab55eSEdward Tomasz Napierala #include <sys/conf.h> 33d151344dSDmitry Chagin #include <sys/ctype.h> 3426795a03SDmitry Chagin #include <sys/file.h> 3526795a03SDmitry Chagin #include <sys/filedesc.h> 36d151344dSDmitry Chagin #include <sys/jail.h> 37d151344dSDmitry Chagin #include <sys/lock.h> 38d5368bf3SDmitry Chagin #include <sys/malloc.h> 3926795a03SDmitry Chagin #include <sys/poll.h> 4026795a03SDmitry Chagin #include <sys/proc.h> 414ab7403bSDmitry Chagin #include <sys/signalvar.h> 42d151344dSDmitry Chagin #include <sys/socket.h> 43d5368bf3SDmitry Chagin #include <sys/socketvar.h> 44d151344dSDmitry Chagin 45d151344dSDmitry Chagin #include <net/if.h> 46d151344dSDmitry Chagin #include <net/if_var.h> 47d151344dSDmitry Chagin #include <net/if_dl.h> 48d151344dSDmitry Chagin #include <net/if_types.h> 497c40e2d5SAlexander V. Chernikov #include <netlink/netlink.h> 504ab7403bSDmitry Chagin 51d5368bf3SDmitry Chagin #include <sys/un.h> 52d5368bf3SDmitry Chagin #include <netinet/in.h> 53d5368bf3SDmitry Chagin 544ab7403bSDmitry Chagin #include <compat/linux/linux.h> 55d151344dSDmitry Chagin #include <compat/linux/linux_common.h> 5626795a03SDmitry Chagin #include <compat/linux/linux_mib.h> 57d5368bf3SDmitry Chagin #include <compat/linux/linux_util.h> 584ab7403bSDmitry Chagin 59d151344dSDmitry Chagin CTASSERT(LINUX_IFNAMSIZ == IFNAMSIZ); 604ab7403bSDmitry Chagin 6132fdc75fSDmitry Chagin static bool use_real_ifnames = false; 6232fdc75fSDmitry Chagin SYSCTL_BOOL(_compat_linux, OID_AUTO, use_real_ifnames, CTLFLAG_RWTUN, 6332fdc75fSDmitry Chagin &use_real_ifnames, 0, 6432fdc75fSDmitry Chagin "Use FreeBSD interface names instead of generating ethN aliases"); 6532fdc75fSDmitry Chagin 664ab7403bSDmitry Chagin static int bsd_to_linux_sigtbl[LINUX_SIGTBLSZ] = { 674ab7403bSDmitry Chagin LINUX_SIGHUP, /* SIGHUP */ 684ab7403bSDmitry Chagin LINUX_SIGINT, /* SIGINT */ 694ab7403bSDmitry Chagin LINUX_SIGQUIT, /* SIGQUIT */ 704ab7403bSDmitry Chagin LINUX_SIGILL, /* SIGILL */ 714ab7403bSDmitry Chagin LINUX_SIGTRAP, /* SIGTRAP */ 724ab7403bSDmitry Chagin LINUX_SIGABRT, /* SIGABRT */ 734ab7403bSDmitry Chagin 0, /* SIGEMT */ 744ab7403bSDmitry Chagin LINUX_SIGFPE, /* SIGFPE */ 754ab7403bSDmitry Chagin LINUX_SIGKILL, /* SIGKILL */ 764ab7403bSDmitry Chagin LINUX_SIGBUS, /* SIGBUS */ 774ab7403bSDmitry Chagin LINUX_SIGSEGV, /* SIGSEGV */ 784ab7403bSDmitry Chagin LINUX_SIGSYS, /* SIGSYS */ 794ab7403bSDmitry Chagin LINUX_SIGPIPE, /* SIGPIPE */ 804ab7403bSDmitry Chagin LINUX_SIGALRM, /* SIGALRM */ 814ab7403bSDmitry Chagin LINUX_SIGTERM, /* SIGTERM */ 824ab7403bSDmitry Chagin LINUX_SIGURG, /* SIGURG */ 834ab7403bSDmitry Chagin LINUX_SIGSTOP, /* SIGSTOP */ 844ab7403bSDmitry Chagin LINUX_SIGTSTP, /* SIGTSTP */ 854ab7403bSDmitry Chagin LINUX_SIGCONT, /* SIGCONT */ 864ab7403bSDmitry Chagin LINUX_SIGCHLD, /* SIGCHLD */ 874ab7403bSDmitry Chagin LINUX_SIGTTIN, /* SIGTTIN */ 884ab7403bSDmitry Chagin LINUX_SIGTTOU, /* SIGTTOU */ 894ab7403bSDmitry Chagin LINUX_SIGIO, /* SIGIO */ 904ab7403bSDmitry Chagin LINUX_SIGXCPU, /* SIGXCPU */ 914ab7403bSDmitry Chagin LINUX_SIGXFSZ, /* SIGXFSZ */ 924ab7403bSDmitry Chagin LINUX_SIGVTALRM,/* SIGVTALRM */ 934ab7403bSDmitry Chagin LINUX_SIGPROF, /* SIGPROF */ 944ab7403bSDmitry Chagin LINUX_SIGWINCH, /* SIGWINCH */ 954ab7403bSDmitry Chagin 0, /* SIGINFO */ 964ab7403bSDmitry Chagin LINUX_SIGUSR1, /* SIGUSR1 */ 974ab7403bSDmitry Chagin LINUX_SIGUSR2 /* SIGUSR2 */ 984ab7403bSDmitry Chagin }; 994ab7403bSDmitry Chagin 1007a7cee55SDmitry Chagin #define LINUX_SIGPWREMU (SIGRTMIN + (LINUX_SIGRTMAX - LINUX_SIGRTMIN) + 1) 1017a7cee55SDmitry Chagin 1024ab7403bSDmitry Chagin static int linux_to_bsd_sigtbl[LINUX_SIGTBLSZ] = { 1034ab7403bSDmitry Chagin SIGHUP, /* LINUX_SIGHUP */ 1044ab7403bSDmitry Chagin SIGINT, /* LINUX_SIGINT */ 1054ab7403bSDmitry Chagin SIGQUIT, /* LINUX_SIGQUIT */ 1064ab7403bSDmitry Chagin SIGILL, /* LINUX_SIGILL */ 1074ab7403bSDmitry Chagin SIGTRAP, /* LINUX_SIGTRAP */ 1084ab7403bSDmitry Chagin SIGABRT, /* LINUX_SIGABRT */ 1094ab7403bSDmitry Chagin SIGBUS, /* LINUX_SIGBUS */ 1104ab7403bSDmitry Chagin SIGFPE, /* LINUX_SIGFPE */ 1114ab7403bSDmitry Chagin SIGKILL, /* LINUX_SIGKILL */ 1124ab7403bSDmitry Chagin SIGUSR1, /* LINUX_SIGUSR1 */ 1134ab7403bSDmitry Chagin SIGSEGV, /* LINUX_SIGSEGV */ 1144ab7403bSDmitry Chagin SIGUSR2, /* LINUX_SIGUSR2 */ 1154ab7403bSDmitry Chagin SIGPIPE, /* LINUX_SIGPIPE */ 1164ab7403bSDmitry Chagin SIGALRM, /* LINUX_SIGALRM */ 1174ab7403bSDmitry Chagin SIGTERM, /* LINUX_SIGTERM */ 1184ab7403bSDmitry Chagin SIGBUS, /* LINUX_SIGSTKFLT */ 1194ab7403bSDmitry Chagin SIGCHLD, /* LINUX_SIGCHLD */ 1204ab7403bSDmitry Chagin SIGCONT, /* LINUX_SIGCONT */ 1214ab7403bSDmitry Chagin SIGSTOP, /* LINUX_SIGSTOP */ 1224ab7403bSDmitry Chagin SIGTSTP, /* LINUX_SIGTSTP */ 1234ab7403bSDmitry Chagin SIGTTIN, /* LINUX_SIGTTIN */ 1244ab7403bSDmitry Chagin SIGTTOU, /* LINUX_SIGTTOU */ 1254ab7403bSDmitry Chagin SIGURG, /* LINUX_SIGURG */ 1264ab7403bSDmitry Chagin SIGXCPU, /* LINUX_SIGXCPU */ 1274ab7403bSDmitry Chagin SIGXFSZ, /* LINUX_SIGXFSZ */ 1284ab7403bSDmitry Chagin SIGVTALRM, /* LINUX_SIGVTALARM */ 1294ab7403bSDmitry Chagin SIGPROF, /* LINUX_SIGPROF */ 1304ab7403bSDmitry Chagin SIGWINCH, /* LINUX_SIGWINCH */ 1314ab7403bSDmitry Chagin SIGIO, /* LINUX_SIGIO */ 1324ab7403bSDmitry Chagin /* 1334ab7403bSDmitry Chagin * FreeBSD does not have SIGPWR signal, map Linux SIGPWR signal 1344ab7403bSDmitry Chagin * to the first unused FreeBSD signal number. Since Linux supports 1354ab7403bSDmitry Chagin * signals from 1 to 64 we are ok here as our SIGRTMIN = 65. 1364ab7403bSDmitry Chagin */ 1377a7cee55SDmitry Chagin LINUX_SIGPWREMU,/* LINUX_SIGPWR */ 1384ab7403bSDmitry Chagin SIGSYS /* LINUX_SIGSYS */ 1394ab7403bSDmitry Chagin }; 1404ab7403bSDmitry Chagin 141044ab55eSEdward Tomasz Napierala static struct cdev *dev_shm_cdev; 142044ab55eSEdward Tomasz Napierala static struct cdevsw dev_shm_cdevsw = { 143044ab55eSEdward Tomasz Napierala .d_version = D_VERSION, 144044ab55eSEdward Tomasz Napierala .d_name = "dev_shm", 145044ab55eSEdward Tomasz Napierala }; 146044ab55eSEdward Tomasz Napierala 1474ab7403bSDmitry Chagin /* 1484ab7403bSDmitry Chagin * Map Linux RT signals to the FreeBSD RT signals. 1494ab7403bSDmitry Chagin */ 1504ab7403bSDmitry Chagin static inline int 1514ab7403bSDmitry Chagin linux_to_bsd_rt_signal(int sig) 1524ab7403bSDmitry Chagin { 1534ab7403bSDmitry Chagin 1547a7cee55SDmitry Chagin return (SIGRTMIN + sig - LINUX_SIGRTMIN); 1554ab7403bSDmitry Chagin } 1564ab7403bSDmitry Chagin 1574ab7403bSDmitry Chagin static inline int 1584ab7403bSDmitry Chagin bsd_to_linux_rt_signal(int sig) 1594ab7403bSDmitry Chagin { 1604ab7403bSDmitry Chagin 1617a7cee55SDmitry Chagin return (sig - SIGRTMIN + LINUX_SIGRTMIN); 1624ab7403bSDmitry Chagin } 1634ab7403bSDmitry Chagin 1644ab7403bSDmitry Chagin int 1654ab7403bSDmitry Chagin linux_to_bsd_signal(int sig) 1664ab7403bSDmitry Chagin { 1674ab7403bSDmitry Chagin 1688081c6ceSEdward Tomasz Napierala KASSERT(sig > 0 && sig <= LINUX_SIGRTMAX, ("invalid Linux signal %d\n", sig)); 1694ab7403bSDmitry Chagin 1704ab7403bSDmitry Chagin if (sig < LINUX_SIGRTMIN) 1714ab7403bSDmitry Chagin return (linux_to_bsd_sigtbl[_SIG_IDX(sig)]); 1724ab7403bSDmitry Chagin 1734ab7403bSDmitry Chagin return (linux_to_bsd_rt_signal(sig)); 1744ab7403bSDmitry Chagin } 1754ab7403bSDmitry Chagin 1764ab7403bSDmitry Chagin int 1774ab7403bSDmitry Chagin bsd_to_linux_signal(int sig) 1784ab7403bSDmitry Chagin { 1794ab7403bSDmitry Chagin 1804ab7403bSDmitry Chagin if (sig <= LINUX_SIGTBLSZ) 1814ab7403bSDmitry Chagin return (bsd_to_linux_sigtbl[_SIG_IDX(sig)]); 1827a7cee55SDmitry Chagin if (sig == LINUX_SIGPWREMU) 1834ab7403bSDmitry Chagin return (LINUX_SIGPWR); 1844ab7403bSDmitry Chagin 1854ab7403bSDmitry Chagin return (bsd_to_linux_rt_signal(sig)); 1864ab7403bSDmitry Chagin } 1874ab7403bSDmitry Chagin 1884ab7403bSDmitry Chagin int 1894ab7403bSDmitry Chagin linux_to_bsd_sigaltstack(int lsa) 1904ab7403bSDmitry Chagin { 1914ab7403bSDmitry Chagin int bsa = 0; 1924ab7403bSDmitry Chagin 1934ab7403bSDmitry Chagin if (lsa & LINUX_SS_DISABLE) 1944ab7403bSDmitry Chagin bsa |= SS_DISABLE; 1954ab7403bSDmitry Chagin /* 1964ab7403bSDmitry Chagin * Linux ignores SS_ONSTACK flag for ss 1974ab7403bSDmitry Chagin * parameter while FreeBSD prohibits it. 1984ab7403bSDmitry Chagin */ 1994ab7403bSDmitry Chagin return (bsa); 2004ab7403bSDmitry Chagin } 2014ab7403bSDmitry Chagin 2024ab7403bSDmitry Chagin int 2034ab7403bSDmitry Chagin bsd_to_linux_sigaltstack(int bsa) 2044ab7403bSDmitry Chagin { 2054ab7403bSDmitry Chagin int lsa = 0; 2064ab7403bSDmitry Chagin 2074ab7403bSDmitry Chagin if (bsa & SS_DISABLE) 2084ab7403bSDmitry Chagin lsa |= LINUX_SS_DISABLE; 2094ab7403bSDmitry Chagin if (bsa & SS_ONSTACK) 2104ab7403bSDmitry Chagin lsa |= LINUX_SS_ONSTACK; 2114ab7403bSDmitry Chagin return (lsa); 2124ab7403bSDmitry Chagin } 2134ab7403bSDmitry Chagin 2144ab7403bSDmitry Chagin void 2154ab7403bSDmitry Chagin linux_to_bsd_sigset(l_sigset_t *lss, sigset_t *bss) 2164ab7403bSDmitry Chagin { 2174ab7403bSDmitry Chagin int b, l; 2184ab7403bSDmitry Chagin 2194ab7403bSDmitry Chagin SIGEMPTYSET(*bss); 2204ab7403bSDmitry Chagin for (l = 1; l <= LINUX_SIGRTMAX; l++) { 2214ab7403bSDmitry Chagin if (LINUX_SIGISMEMBER(*lss, l)) { 2224ab7403bSDmitry Chagin b = linux_to_bsd_signal(l); 2234ab7403bSDmitry Chagin if (b) 2244ab7403bSDmitry Chagin SIGADDSET(*bss, b); 2254ab7403bSDmitry Chagin } 2264ab7403bSDmitry Chagin } 2274ab7403bSDmitry Chagin } 2284ab7403bSDmitry Chagin 2294ab7403bSDmitry Chagin void 2304ab7403bSDmitry Chagin bsd_to_linux_sigset(sigset_t *bss, l_sigset_t *lss) 2314ab7403bSDmitry Chagin { 2324ab7403bSDmitry Chagin int b, l; 2334ab7403bSDmitry Chagin 2344ab7403bSDmitry Chagin LINUX_SIGEMPTYSET(*lss); 2354ab7403bSDmitry Chagin for (b = 1; b <= SIGRTMAX; b++) { 2364ab7403bSDmitry Chagin if (SIGISMEMBER(*bss, b)) { 2374ab7403bSDmitry Chagin l = bsd_to_linux_signal(b); 2384ab7403bSDmitry Chagin if (l) 2394ab7403bSDmitry Chagin LINUX_SIGADDSET(*lss, l); 2404ab7403bSDmitry Chagin } 2414ab7403bSDmitry Chagin } 2424ab7403bSDmitry Chagin } 243d151344dSDmitry Chagin 244d151344dSDmitry Chagin /* 245*3ab3c9c2SDmitry Chagin * Translate a FreeBSD interface name to a Linux interface name 246*3ab3c9c2SDmitry Chagin * by interface name, and return the number of bytes copied to lxname. 247*3ab3c9c2SDmitry Chagin */ 248*3ab3c9c2SDmitry Chagin int 249*3ab3c9c2SDmitry Chagin ifname_bsd_to_linux_name(const char *bsdname, char *lxname, size_t len) 250*3ab3c9c2SDmitry Chagin { 251*3ab3c9c2SDmitry Chagin struct epoch_tracker et; 252*3ab3c9c2SDmitry Chagin struct ifnet *ifp; 253*3ab3c9c2SDmitry Chagin int ret; 254*3ab3c9c2SDmitry Chagin 255*3ab3c9c2SDmitry Chagin ret = 0; 256*3ab3c9c2SDmitry Chagin CURVNET_SET(TD_TO_VNET(curthread)); 257*3ab3c9c2SDmitry Chagin NET_EPOCH_ENTER(et); 258*3ab3c9c2SDmitry Chagin ifp = ifunit(bsdname); 259*3ab3c9c2SDmitry Chagin if (ifp != NULL) 260*3ab3c9c2SDmitry Chagin ret = ifname_bsd_to_linux_ifp(ifp, lxname, len); 261*3ab3c9c2SDmitry Chagin NET_EPOCH_EXIT(et); 262*3ab3c9c2SDmitry Chagin CURVNET_RESTORE(); 263*3ab3c9c2SDmitry Chagin return (ret); 264*3ab3c9c2SDmitry Chagin } 265*3ab3c9c2SDmitry Chagin 266*3ab3c9c2SDmitry Chagin /* 267*3ab3c9c2SDmitry Chagin * Translate a FreeBSD interface name to a Linux interface name 268*3ab3c9c2SDmitry Chagin * by interface index, and return the number of bytes copied to lxname. 269*3ab3c9c2SDmitry Chagin */ 270*3ab3c9c2SDmitry Chagin int 271*3ab3c9c2SDmitry Chagin ifname_bsd_to_linux_idx(u_int idx, char *lxname, size_t len) 272*3ab3c9c2SDmitry Chagin { 273*3ab3c9c2SDmitry Chagin struct epoch_tracker et; 274*3ab3c9c2SDmitry Chagin struct ifnet *ifp; 275*3ab3c9c2SDmitry Chagin int ret; 276*3ab3c9c2SDmitry Chagin 277*3ab3c9c2SDmitry Chagin ret = 0; 278*3ab3c9c2SDmitry Chagin CURVNET_SET(TD_TO_VNET(curthread)); 279*3ab3c9c2SDmitry Chagin NET_EPOCH_ENTER(et); 280*3ab3c9c2SDmitry Chagin ifp = ifnet_byindex(idx); 281*3ab3c9c2SDmitry Chagin if (ifp != NULL) 282*3ab3c9c2SDmitry Chagin ret = ifname_bsd_to_linux_ifp(ifp, lxname, len); 283*3ab3c9c2SDmitry Chagin NET_EPOCH_EXIT(et); 284*3ab3c9c2SDmitry Chagin CURVNET_RESTORE(); 285*3ab3c9c2SDmitry Chagin return (ret); 286*3ab3c9c2SDmitry Chagin } 287*3ab3c9c2SDmitry Chagin 288*3ab3c9c2SDmitry Chagin /* 289*3ab3c9c2SDmitry Chagin * Translate a FreeBSD interface name to a Linux interface name, 290*3ab3c9c2SDmitry Chagin * and return the number of bytes copied to lxname. 291*3ab3c9c2SDmitry Chagin */ 292*3ab3c9c2SDmitry Chagin int 293*3ab3c9c2SDmitry Chagin ifname_bsd_to_linux_ifp(struct ifnet *ifp, char *lxname, size_t len) 294*3ab3c9c2SDmitry Chagin { 295*3ab3c9c2SDmitry Chagin struct ifnet *ifscan; 296*3ab3c9c2SDmitry Chagin int unit; 297*3ab3c9c2SDmitry Chagin 298*3ab3c9c2SDmitry Chagin NET_EPOCH_ASSERT(); 299*3ab3c9c2SDmitry Chagin 300*3ab3c9c2SDmitry Chagin /* 301*3ab3c9c2SDmitry Chagin * Linux loopback interface name is lo (not lo0), 302*3ab3c9c2SDmitry Chagin * we translate lo to lo0, loX to loX. 303*3ab3c9c2SDmitry Chagin */ 304*3ab3c9c2SDmitry Chagin if (IFP_IS_LOOP(ifp) && strncmp(ifp->if_xname, "lo0", IFNAMSIZ) == 0) 305*3ab3c9c2SDmitry Chagin return (strlcpy(lxname, "lo", len)); 306*3ab3c9c2SDmitry Chagin 307*3ab3c9c2SDmitry Chagin /* Short-circuit non ethernet interfaces. */ 308*3ab3c9c2SDmitry Chagin if (!IFP_IS_ETH(ifp) || linux_use_real_ifname(ifp)) 309*3ab3c9c2SDmitry Chagin return (strlcpy(lxname, ifp->if_xname, len)); 310*3ab3c9c2SDmitry Chagin 311*3ab3c9c2SDmitry Chagin /* Determine the (relative) unit number for ethernet interfaces. */ 312*3ab3c9c2SDmitry Chagin unit = 0; 313*3ab3c9c2SDmitry Chagin CK_STAILQ_FOREACH(ifscan, &V_ifnet, if_link) { 314*3ab3c9c2SDmitry Chagin if (ifscan == ifp) 315*3ab3c9c2SDmitry Chagin return (snprintf(lxname, len, "eth%d", unit)); 316*3ab3c9c2SDmitry Chagin if (IFP_IS_ETH(ifscan)) 317*3ab3c9c2SDmitry Chagin unit++; 318*3ab3c9c2SDmitry Chagin } 319*3ab3c9c2SDmitry Chagin return (0); 320*3ab3c9c2SDmitry Chagin } 321*3ab3c9c2SDmitry Chagin 322*3ab3c9c2SDmitry Chagin /* 323d151344dSDmitry Chagin * Translate a Linux interface name to a FreeBSD interface name, 324d151344dSDmitry Chagin * and return the associated ifnet structure 325d151344dSDmitry Chagin * bsdname and lxname need to be least IFNAMSIZ bytes long, but 326d151344dSDmitry Chagin * can point to the same buffer. 327d151344dSDmitry Chagin */ 328d151344dSDmitry Chagin struct ifnet * 329d151344dSDmitry Chagin ifname_linux_to_bsd(struct thread *td, const char *lxname, char *bsdname) 330d151344dSDmitry Chagin { 331e9e637bfSJustin Hibbits struct ifnet *ifp; 332e9e637bfSJustin Hibbits int len, unit; 333d151344dSDmitry Chagin char *ep; 334e9e637bfSJustin Hibbits int index; 335e9e637bfSJustin Hibbits bool is_eth, is_lo; 336d151344dSDmitry Chagin 337d151344dSDmitry Chagin for (len = 0; len < LINUX_IFNAMSIZ; ++len) 3389c1437aeSDmitry Chagin if (!isalpha(lxname[len]) || lxname[len] == '\0') 339d151344dSDmitry Chagin break; 340d151344dSDmitry Chagin if (len == 0 || len == LINUX_IFNAMSIZ) 341d151344dSDmitry Chagin return (NULL); 342*3ab3c9c2SDmitry Chagin /* 343*3ab3c9c2SDmitry Chagin * Linux loopback interface name is lo (not lo0), 344*3ab3c9c2SDmitry Chagin * we translate lo to lo0, loX to loX. 345*3ab3c9c2SDmitry Chagin */ 346*3ab3c9c2SDmitry Chagin is_lo = (len == 2 && strncmp(lxname, "lo", LINUX_IFNAMSIZ) == 0); 347e9e637bfSJustin Hibbits unit = (int)strtoul(lxname + len, &ep, 10); 348d151344dSDmitry Chagin if ((ep == NULL || ep == lxname + len || ep >= lxname + LINUX_IFNAMSIZ) && 349e9e637bfSJustin Hibbits is_lo == 0) 350d151344dSDmitry Chagin return (NULL); 351e9e637bfSJustin Hibbits index = 0; 352e9e637bfSJustin Hibbits is_eth = (len == 3 && strncmp(lxname, "eth", len) == 0); 353d151344dSDmitry Chagin 354d151344dSDmitry Chagin CURVNET_SET(TD_TO_VNET(td)); 355d151344dSDmitry Chagin IFNET_RLOCK(); 356e9e637bfSJustin Hibbits CK_STAILQ_FOREACH(ifp, &V_ifnet, if_link) { 357e9e637bfSJustin Hibbits /* 358e9e637bfSJustin Hibbits * Allow Linux programs to use FreeBSD names. Don't presume 359e9e637bfSJustin Hibbits * we never have an interface named "eth", so don't make 360e9e637bfSJustin Hibbits * the test optional based on is_eth. 361e9e637bfSJustin Hibbits */ 362e9e637bfSJustin Hibbits if (strncmp(ifp->if_xname, lxname, LINUX_IFNAMSIZ) == 0) 363e9e637bfSJustin Hibbits break; 364e9e637bfSJustin Hibbits if (is_eth && IFP_IS_ETH(ifp) && unit == index++) 365e9e637bfSJustin Hibbits break; 366e9e637bfSJustin Hibbits if (is_lo && IFP_IS_LOOP(ifp)) 367e9e637bfSJustin Hibbits break; 368e9e637bfSJustin Hibbits } 369d151344dSDmitry Chagin IFNET_RUNLOCK(); 370d151344dSDmitry Chagin CURVNET_RESTORE(); 371e9e637bfSJustin Hibbits if (ifp != NULL && bsdname != NULL) 372e9e637bfSJustin Hibbits strlcpy(bsdname, ifp->if_xname, IFNAMSIZ); 373e9e637bfSJustin Hibbits return (ifp); 374d151344dSDmitry Chagin } 375d151344dSDmitry Chagin 376d151344dSDmitry Chagin void 377d151344dSDmitry Chagin linux_ifflags(struct ifnet *ifp, short *flags) 378d151344dSDmitry Chagin { 379bbac65c7SDmitry Chagin unsigned short fl; 380d151344dSDmitry Chagin 381e9e637bfSJustin Hibbits fl = (ifp->if_flags | ifp->if_drv_flags) & 0xffff; 382bbac65c7SDmitry Chagin *flags = 0; 383bbac65c7SDmitry Chagin if (fl & IFF_UP) 384bbac65c7SDmitry Chagin *flags |= LINUX_IFF_UP; 385bbac65c7SDmitry Chagin if (fl & IFF_BROADCAST) 386bbac65c7SDmitry Chagin *flags |= LINUX_IFF_BROADCAST; 387bbac65c7SDmitry Chagin if (fl & IFF_DEBUG) 388bbac65c7SDmitry Chagin *flags |= LINUX_IFF_DEBUG; 389bbac65c7SDmitry Chagin if (fl & IFF_LOOPBACK) 390bbac65c7SDmitry Chagin *flags |= LINUX_IFF_LOOPBACK; 391bbac65c7SDmitry Chagin if (fl & IFF_POINTOPOINT) 392bbac65c7SDmitry Chagin *flags |= LINUX_IFF_POINTOPOINT; 393bbac65c7SDmitry Chagin if (fl & IFF_DRV_RUNNING) 394bbac65c7SDmitry Chagin *flags |= LINUX_IFF_RUNNING; 395bbac65c7SDmitry Chagin if (fl & IFF_NOARP) 396bbac65c7SDmitry Chagin *flags |= LINUX_IFF_NOARP; 397bbac65c7SDmitry Chagin if (fl & IFF_PROMISC) 398bbac65c7SDmitry Chagin *flags |= LINUX_IFF_PROMISC; 399bbac65c7SDmitry Chagin if (fl & IFF_ALLMULTI) 400bbac65c7SDmitry Chagin *flags |= LINUX_IFF_ALLMULTI; 401bbac65c7SDmitry Chagin if (fl & IFF_MULTICAST) 402bbac65c7SDmitry Chagin *flags |= LINUX_IFF_MULTICAST; 403d151344dSDmitry Chagin } 404d151344dSDmitry Chagin 405d151344dSDmitry Chagin int 406d151344dSDmitry Chagin linux_ifhwaddr(struct ifnet *ifp, struct l_sockaddr *lsa) 407d151344dSDmitry Chagin { 408e9e637bfSJustin Hibbits struct ifaddr *ifa; 409e9e637bfSJustin Hibbits struct sockaddr_dl *sdl; 410d151344dSDmitry Chagin 411d151344dSDmitry Chagin if (IFP_IS_LOOP(ifp)) { 412d151344dSDmitry Chagin bzero(lsa, sizeof(*lsa)); 413d151344dSDmitry Chagin lsa->sa_family = LINUX_ARPHRD_LOOPBACK; 414d151344dSDmitry Chagin return (0); 415d151344dSDmitry Chagin } 416d151344dSDmitry Chagin 417d151344dSDmitry Chagin if (!IFP_IS_ETH(ifp)) 418d151344dSDmitry Chagin return (ENOENT); 419d151344dSDmitry Chagin 420e9e637bfSJustin Hibbits CK_STAILQ_FOREACH(ifa, &ifp->if_addrhead, ifa_link) { 421e9e637bfSJustin Hibbits sdl = (struct sockaddr_dl*)ifa->ifa_addr; 422e9e637bfSJustin Hibbits if (sdl != NULL && (sdl->sdl_family == AF_LINK) && 423e9e637bfSJustin Hibbits (sdl->sdl_type == IFT_ETHER)) { 424e9e637bfSJustin Hibbits bzero(lsa, sizeof(*lsa)); 425e9e637bfSJustin Hibbits lsa->sa_family = LINUX_ARPHRD_ETHER; 426e9e637bfSJustin Hibbits bcopy(LLADDR(sdl), lsa->sa_data, LINUX_IFHWADDRLEN); 427d151344dSDmitry Chagin return (0); 428e9e637bfSJustin Hibbits } 429e9e637bfSJustin Hibbits } 430d151344dSDmitry Chagin 431d151344dSDmitry Chagin return (ENOENT); 432d151344dSDmitry Chagin } 433d5368bf3SDmitry Chagin 434d5368bf3SDmitry Chagin int 435d5368bf3SDmitry Chagin linux_to_bsd_domain(int domain) 436d5368bf3SDmitry Chagin { 437d5368bf3SDmitry Chagin 438d5368bf3SDmitry Chagin switch (domain) { 439d5368bf3SDmitry Chagin case LINUX_AF_UNSPEC: 440d5368bf3SDmitry Chagin return (AF_UNSPEC); 441d5368bf3SDmitry Chagin case LINUX_AF_UNIX: 442d5368bf3SDmitry Chagin return (AF_LOCAL); 443d5368bf3SDmitry Chagin case LINUX_AF_INET: 444d5368bf3SDmitry Chagin return (AF_INET); 445d5368bf3SDmitry Chagin case LINUX_AF_INET6: 446d5368bf3SDmitry Chagin return (AF_INET6); 447d5368bf3SDmitry Chagin case LINUX_AF_AX25: 448d5368bf3SDmitry Chagin return (AF_CCITT); 449d5368bf3SDmitry Chagin case LINUX_AF_IPX: 450d5368bf3SDmitry Chagin return (AF_IPX); 451d5368bf3SDmitry Chagin case LINUX_AF_APPLETALK: 452d5368bf3SDmitry Chagin return (AF_APPLETALK); 4537c40e2d5SAlexander V. Chernikov case LINUX_AF_NETLINK: 4547c40e2d5SAlexander V. Chernikov return (AF_NETLINK); 455d5368bf3SDmitry Chagin } 456d5368bf3SDmitry Chagin return (-1); 457d5368bf3SDmitry Chagin } 458d5368bf3SDmitry Chagin 459d5368bf3SDmitry Chagin int 460d5368bf3SDmitry Chagin bsd_to_linux_domain(int domain) 461d5368bf3SDmitry Chagin { 462d5368bf3SDmitry Chagin 463d5368bf3SDmitry Chagin switch (domain) { 464d5368bf3SDmitry Chagin case AF_UNSPEC: 465d5368bf3SDmitry Chagin return (LINUX_AF_UNSPEC); 466d5368bf3SDmitry Chagin case AF_LOCAL: 467d5368bf3SDmitry Chagin return (LINUX_AF_UNIX); 468d5368bf3SDmitry Chagin case AF_INET: 469d5368bf3SDmitry Chagin return (LINUX_AF_INET); 470d5368bf3SDmitry Chagin case AF_INET6: 471d5368bf3SDmitry Chagin return (LINUX_AF_INET6); 472d5368bf3SDmitry Chagin case AF_CCITT: 473d5368bf3SDmitry Chagin return (LINUX_AF_AX25); 474d5368bf3SDmitry Chagin case AF_IPX: 475d5368bf3SDmitry Chagin return (LINUX_AF_IPX); 476d5368bf3SDmitry Chagin case AF_APPLETALK: 477d5368bf3SDmitry Chagin return (LINUX_AF_APPLETALK); 4787c40e2d5SAlexander V. Chernikov case AF_NETLINK: 4797c40e2d5SAlexander V. Chernikov return (LINUX_AF_NETLINK); 480d5368bf3SDmitry Chagin } 481d5368bf3SDmitry Chagin return (-1); 482d5368bf3SDmitry Chagin } 483d5368bf3SDmitry Chagin 484d5368bf3SDmitry Chagin /* 485d5368bf3SDmitry Chagin * Based on the fact that: 486d5368bf3SDmitry Chagin * 1. Native and Linux storage of struct sockaddr 487d5368bf3SDmitry Chagin * and struct sockaddr_in6 are equal. 488d5368bf3SDmitry Chagin * 2. On Linux sa_family is the first member of all struct sockaddr. 489d5368bf3SDmitry Chagin */ 490d5368bf3SDmitry Chagin int 491d5368bf3SDmitry Chagin bsd_to_linux_sockaddr(const struct sockaddr *sa, struct l_sockaddr **lsa, 492d5368bf3SDmitry Chagin socklen_t len) 493d5368bf3SDmitry Chagin { 494d5368bf3SDmitry Chagin struct l_sockaddr *kosa; 495bbddd588SDmitry Chagin int bdom; 496d5368bf3SDmitry Chagin 497d5368bf3SDmitry Chagin *lsa = NULL; 498d5368bf3SDmitry Chagin if (len < 2 || len > UCHAR_MAX) 499d5368bf3SDmitry Chagin return (EINVAL); 500bbddd588SDmitry Chagin bdom = bsd_to_linux_domain(sa->sa_family); 501bbddd588SDmitry Chagin if (bdom == -1) 502bbddd588SDmitry Chagin return (EAFNOSUPPORT); 503d5368bf3SDmitry Chagin 50463355839SDmitry Chagin kosa = malloc(len, M_LINUX, M_WAITOK); 505d5368bf3SDmitry Chagin bcopy(sa, kosa, len); 506d5368bf3SDmitry Chagin kosa->sa_family = bdom; 507d5368bf3SDmitry Chagin *lsa = kosa; 508d5368bf3SDmitry Chagin return (0); 509d5368bf3SDmitry Chagin } 510d5368bf3SDmitry Chagin 511d5368bf3SDmitry Chagin int 512d5368bf3SDmitry Chagin linux_to_bsd_sockaddr(const struct l_sockaddr *osa, struct sockaddr **sap, 513d5368bf3SDmitry Chagin socklen_t *len) 514d5368bf3SDmitry Chagin { 515d5368bf3SDmitry Chagin struct sockaddr *sa; 516d5368bf3SDmitry Chagin struct l_sockaddr *kosa; 517d5368bf3SDmitry Chagin #ifdef INET6 518d5368bf3SDmitry Chagin struct sockaddr_in6 *sin6; 519d5368bf3SDmitry Chagin bool oldv6size; 520d5368bf3SDmitry Chagin #endif 521d5368bf3SDmitry Chagin char *name; 522d5368bf3SDmitry Chagin int salen, bdom, error, hdrlen, namelen; 523d5368bf3SDmitry Chagin 524d5368bf3SDmitry Chagin if (*len < 2 || *len > UCHAR_MAX) 525d5368bf3SDmitry Chagin return (EINVAL); 526d5368bf3SDmitry Chagin 527d5368bf3SDmitry Chagin salen = *len; 528d5368bf3SDmitry Chagin 529d5368bf3SDmitry Chagin #ifdef INET6 530d5368bf3SDmitry Chagin oldv6size = false; 531d5368bf3SDmitry Chagin /* 532d5368bf3SDmitry Chagin * Check for old (pre-RFC2553) sockaddr_in6. We may accept it 533d5368bf3SDmitry Chagin * if it's a v4-mapped address, so reserve the proper space 534d5368bf3SDmitry Chagin * for it. 535d5368bf3SDmitry Chagin */ 536d5368bf3SDmitry Chagin if (salen == sizeof(struct sockaddr_in6) - sizeof(uint32_t)) { 537d5368bf3SDmitry Chagin salen += sizeof(uint32_t); 538d5368bf3SDmitry Chagin oldv6size = true; 539d5368bf3SDmitry Chagin } 540d5368bf3SDmitry Chagin #endif 541d5368bf3SDmitry Chagin 542d5368bf3SDmitry Chagin kosa = malloc(salen, M_SONAME, M_WAITOK); 543d5368bf3SDmitry Chagin 544d5368bf3SDmitry Chagin if ((error = copyin(osa, kosa, *len))) 545d5368bf3SDmitry Chagin goto out; 546d5368bf3SDmitry Chagin 547d5368bf3SDmitry Chagin bdom = linux_to_bsd_domain(kosa->sa_family); 548d5368bf3SDmitry Chagin if (bdom == -1) { 549d5368bf3SDmitry Chagin error = EAFNOSUPPORT; 550d5368bf3SDmitry Chagin goto out; 551d5368bf3SDmitry Chagin } 552d5368bf3SDmitry Chagin 553d5368bf3SDmitry Chagin #ifdef INET6 554d5368bf3SDmitry Chagin /* 555d5368bf3SDmitry Chagin * Older Linux IPv6 code uses obsolete RFC2133 struct sockaddr_in6, 556d5368bf3SDmitry Chagin * which lacks the scope id compared with RFC2553 one. If we detect 557d5368bf3SDmitry Chagin * the situation, reject the address and write a message to system log. 558d5368bf3SDmitry Chagin * 559d5368bf3SDmitry Chagin * Still accept addresses for which the scope id is not used. 560d5368bf3SDmitry Chagin */ 561d5368bf3SDmitry Chagin if (oldv6size) { 562d5368bf3SDmitry Chagin if (bdom == AF_INET6) { 563d5368bf3SDmitry Chagin sin6 = (struct sockaddr_in6 *)kosa; 564d5368bf3SDmitry Chagin if (IN6_IS_ADDR_V4MAPPED(&sin6->sin6_addr) || 565d5368bf3SDmitry Chagin (!IN6_IS_ADDR_LINKLOCAL(&sin6->sin6_addr) && 566d5368bf3SDmitry Chagin !IN6_IS_ADDR_SITELOCAL(&sin6->sin6_addr) && 567d5368bf3SDmitry Chagin !IN6_IS_ADDR_V4COMPAT(&sin6->sin6_addr) && 568d5368bf3SDmitry Chagin !IN6_IS_ADDR_UNSPECIFIED(&sin6->sin6_addr) && 569d5368bf3SDmitry Chagin !IN6_IS_ADDR_MULTICAST(&sin6->sin6_addr))) { 570d5368bf3SDmitry Chagin sin6->sin6_scope_id = 0; 571d5368bf3SDmitry Chagin } else { 572d5368bf3SDmitry Chagin linux_msg(curthread, 57386e794ebSEdward Tomasz Napierala "obsolete pre-RFC2553 sockaddr_in6 rejected"); 574d5368bf3SDmitry Chagin error = EINVAL; 575d5368bf3SDmitry Chagin goto out; 576d5368bf3SDmitry Chagin } 577d5368bf3SDmitry Chagin } else 578d5368bf3SDmitry Chagin salen -= sizeof(uint32_t); 579d5368bf3SDmitry Chagin } 580d5368bf3SDmitry Chagin #endif 581d5368bf3SDmitry Chagin if (bdom == AF_INET) { 582d5368bf3SDmitry Chagin if (salen < sizeof(struct sockaddr_in)) { 583d5368bf3SDmitry Chagin error = EINVAL; 584d5368bf3SDmitry Chagin goto out; 585d5368bf3SDmitry Chagin } 586d5368bf3SDmitry Chagin salen = sizeof(struct sockaddr_in); 587d5368bf3SDmitry Chagin } 588d5368bf3SDmitry Chagin 589d5368bf3SDmitry Chagin if (bdom == AF_LOCAL && salen > sizeof(struct sockaddr_un)) { 590d5368bf3SDmitry Chagin hdrlen = offsetof(struct sockaddr_un, sun_path); 591d5368bf3SDmitry Chagin name = ((struct sockaddr_un *)kosa)->sun_path; 592d5368bf3SDmitry Chagin if (*name == '\0') { 593d5368bf3SDmitry Chagin /* 594d5368bf3SDmitry Chagin * Linux abstract namespace starts with a NULL byte. 595d5368bf3SDmitry Chagin * XXX We do not support abstract namespace yet. 596d5368bf3SDmitry Chagin */ 597d5368bf3SDmitry Chagin namelen = strnlen(name + 1, salen - hdrlen - 1) + 1; 598d5368bf3SDmitry Chagin } else 599d5368bf3SDmitry Chagin namelen = strnlen(name, salen - hdrlen); 600d5368bf3SDmitry Chagin salen = hdrlen + namelen; 601d5368bf3SDmitry Chagin if (salen > sizeof(struct sockaddr_un)) { 602d5368bf3SDmitry Chagin error = ENAMETOOLONG; 603d5368bf3SDmitry Chagin goto out; 604d5368bf3SDmitry Chagin } 605d5368bf3SDmitry Chagin } 606d5368bf3SDmitry Chagin 6077c40e2d5SAlexander V. Chernikov if (bdom == AF_NETLINK) { 6087c40e2d5SAlexander V. Chernikov if (salen < sizeof(struct sockaddr_nl)) { 6097c40e2d5SAlexander V. Chernikov error = EINVAL; 6107c40e2d5SAlexander V. Chernikov goto out; 6117c40e2d5SAlexander V. Chernikov } 6127c40e2d5SAlexander V. Chernikov salen = sizeof(struct sockaddr_nl); 6137c40e2d5SAlexander V. Chernikov } 6147c40e2d5SAlexander V. Chernikov 615d5368bf3SDmitry Chagin sa = (struct sockaddr *)kosa; 616d5368bf3SDmitry Chagin sa->sa_family = bdom; 617d5368bf3SDmitry Chagin sa->sa_len = salen; 618d5368bf3SDmitry Chagin 619d5368bf3SDmitry Chagin *sap = sa; 620d5368bf3SDmitry Chagin *len = salen; 621d5368bf3SDmitry Chagin return (0); 622d5368bf3SDmitry Chagin 623d5368bf3SDmitry Chagin out: 624d5368bf3SDmitry Chagin free(kosa, M_SONAME); 625d5368bf3SDmitry Chagin return (error); 626d5368bf3SDmitry Chagin } 627044ab55eSEdward Tomasz Napierala 628044ab55eSEdward Tomasz Napierala void 629044ab55eSEdward Tomasz Napierala linux_dev_shm_create(void) 630044ab55eSEdward Tomasz Napierala { 631044ab55eSEdward Tomasz Napierala int error; 632044ab55eSEdward Tomasz Napierala 633044ab55eSEdward Tomasz Napierala error = make_dev_p(MAKEDEV_CHECKNAME | MAKEDEV_WAITOK, &dev_shm_cdev, 634044ab55eSEdward Tomasz Napierala &dev_shm_cdevsw, NULL, UID_ROOT, GID_WHEEL, 0, "shm/.mountpoint"); 635044ab55eSEdward Tomasz Napierala if (error != 0) { 636044ab55eSEdward Tomasz Napierala printf("%s: failed to create device node, error %d\n", 637044ab55eSEdward Tomasz Napierala __func__, error); 638044ab55eSEdward Tomasz Napierala } 639044ab55eSEdward Tomasz Napierala } 640044ab55eSEdward Tomasz Napierala 641044ab55eSEdward Tomasz Napierala void 642044ab55eSEdward Tomasz Napierala linux_dev_shm_destroy(void) 643044ab55eSEdward Tomasz Napierala { 644044ab55eSEdward Tomasz Napierala 645044ab55eSEdward Tomasz Napierala destroy_dev(dev_shm_cdev); 646044ab55eSEdward Tomasz Napierala } 6475403f186SKyle Evans 6485403f186SKyle Evans int 6495403f186SKyle Evans bsd_to_linux_bits_(int value, struct bsd_to_linux_bitmap *bitmap, 6505403f186SKyle Evans size_t mapcnt, int no_value) 6515403f186SKyle Evans { 6525403f186SKyle Evans int bsd_mask, bsd_value, linux_mask, linux_value; 6535403f186SKyle Evans int linux_ret; 6545403f186SKyle Evans size_t i; 6555403f186SKyle Evans bool applied; 6565403f186SKyle Evans 6575403f186SKyle Evans applied = false; 6585403f186SKyle Evans linux_ret = 0; 6595403f186SKyle Evans for (i = 0; i < mapcnt; ++i) { 6605403f186SKyle Evans bsd_mask = bitmap[i].bsd_mask; 6615403f186SKyle Evans bsd_value = bitmap[i].bsd_value; 6625403f186SKyle Evans if (bsd_mask == 0) 6635403f186SKyle Evans bsd_mask = bsd_value; 6645403f186SKyle Evans 6655403f186SKyle Evans linux_mask = bitmap[i].linux_mask; 6665403f186SKyle Evans linux_value = bitmap[i].linux_value; 6675403f186SKyle Evans if (linux_mask == 0) 6685403f186SKyle Evans linux_mask = linux_value; 6695403f186SKyle Evans 6705403f186SKyle Evans /* 6715403f186SKyle Evans * If a mask larger than just the value is set, we explicitly 6725403f186SKyle Evans * want to make sure that only this bit we mapped within that 6735403f186SKyle Evans * mask is set. 6745403f186SKyle Evans */ 6755403f186SKyle Evans if ((value & bsd_mask) == bsd_value) { 6765403f186SKyle Evans linux_ret = (linux_ret & ~linux_mask) | linux_value; 6775403f186SKyle Evans applied = true; 6785403f186SKyle Evans } 6795403f186SKyle Evans } 6805403f186SKyle Evans 6815403f186SKyle Evans if (!applied) 6825403f186SKyle Evans return (no_value); 6835403f186SKyle Evans return (linux_ret); 6845403f186SKyle Evans } 6855403f186SKyle Evans 6865403f186SKyle Evans int 6875403f186SKyle Evans linux_to_bsd_bits_(int value, struct bsd_to_linux_bitmap *bitmap, 6885403f186SKyle Evans size_t mapcnt, int no_value) 6895403f186SKyle Evans { 6905403f186SKyle Evans int bsd_mask, bsd_value, linux_mask, linux_value; 6915403f186SKyle Evans int bsd_ret; 6925403f186SKyle Evans size_t i; 6935403f186SKyle Evans bool applied; 6945403f186SKyle Evans 6955403f186SKyle Evans applied = false; 6965403f186SKyle Evans bsd_ret = 0; 6975403f186SKyle Evans for (i = 0; i < mapcnt; ++i) { 6985403f186SKyle Evans bsd_mask = bitmap[i].bsd_mask; 6995403f186SKyle Evans bsd_value = bitmap[i].bsd_value; 7005403f186SKyle Evans if (bsd_mask == 0) 7015403f186SKyle Evans bsd_mask = bsd_value; 7025403f186SKyle Evans 7035403f186SKyle Evans linux_mask = bitmap[i].linux_mask; 7045403f186SKyle Evans linux_value = bitmap[i].linux_value; 7055403f186SKyle Evans if (linux_mask == 0) 7065403f186SKyle Evans linux_mask = linux_value; 7075403f186SKyle Evans 7085403f186SKyle Evans /* 7095403f186SKyle Evans * If a mask larger than just the value is set, we explicitly 7105403f186SKyle Evans * want to make sure that only this bit we mapped within that 7115403f186SKyle Evans * mask is set. 7125403f186SKyle Evans */ 7135403f186SKyle Evans if ((value & linux_mask) == linux_value) { 7145403f186SKyle Evans bsd_ret = (bsd_ret & ~bsd_mask) | bsd_value; 7155403f186SKyle Evans applied = true; 7165403f186SKyle Evans } 7175403f186SKyle Evans } 7185403f186SKyle Evans 7195403f186SKyle Evans if (!applied) 7205403f186SKyle Evans return (no_value); 7215403f186SKyle Evans return (bsd_ret); 7225403f186SKyle Evans } 72326795a03SDmitry Chagin 72426795a03SDmitry Chagin void 72526795a03SDmitry Chagin linux_to_bsd_poll_events(struct thread *td, int fd, short lev, 72626795a03SDmitry Chagin short *bev) 72726795a03SDmitry Chagin { 72826795a03SDmitry Chagin struct file *fp; 72926795a03SDmitry Chagin int error; 73026795a03SDmitry Chagin short bits = 0; 73126795a03SDmitry Chagin 73226795a03SDmitry Chagin if (lev & LINUX_POLLIN) 73326795a03SDmitry Chagin bits |= POLLIN; 73426795a03SDmitry Chagin if (lev & LINUX_POLLPRI) 73526795a03SDmitry Chagin bits |= POLLPRI; 73626795a03SDmitry Chagin if (lev & LINUX_POLLOUT) 73726795a03SDmitry Chagin bits |= POLLOUT; 73826795a03SDmitry Chagin if (lev & LINUX_POLLERR) 73926795a03SDmitry Chagin bits |= POLLERR; 74026795a03SDmitry Chagin if (lev & LINUX_POLLHUP) 74126795a03SDmitry Chagin bits |= POLLHUP; 74226795a03SDmitry Chagin if (lev & LINUX_POLLNVAL) 74326795a03SDmitry Chagin bits |= POLLNVAL; 74426795a03SDmitry Chagin if (lev & LINUX_POLLRDNORM) 74526795a03SDmitry Chagin bits |= POLLRDNORM; 74626795a03SDmitry Chagin if (lev & LINUX_POLLRDBAND) 74726795a03SDmitry Chagin bits |= POLLRDBAND; 74826795a03SDmitry Chagin if (lev & LINUX_POLLWRBAND) 74926795a03SDmitry Chagin bits |= POLLWRBAND; 75026795a03SDmitry Chagin if (lev & LINUX_POLLWRNORM) 75126795a03SDmitry Chagin bits |= POLLWRNORM; 75226795a03SDmitry Chagin 75326795a03SDmitry Chagin if (lev & LINUX_POLLRDHUP) { 75426795a03SDmitry Chagin /* 75526795a03SDmitry Chagin * It seems that the Linux silencly ignores POLLRDHUP 75626795a03SDmitry Chagin * on non-socket file descriptors unlike FreeBSD, where 75726795a03SDmitry Chagin * events bits is more strictly checked (POLLSTANDARD). 75826795a03SDmitry Chagin */ 759513c7a6eSMateusz Guzik error = fget_unlocked(td, fd, &cap_no_rights, &fp); 76026795a03SDmitry Chagin if (error == 0) { 76126795a03SDmitry Chagin /* 76226795a03SDmitry Chagin * XXX. On FreeBSD POLLRDHUP applies only to 76326795a03SDmitry Chagin * stream sockets. 76426795a03SDmitry Chagin */ 76526795a03SDmitry Chagin if (fp->f_type == DTYPE_SOCKET) 76626795a03SDmitry Chagin bits |= POLLRDHUP; 76726795a03SDmitry Chagin fdrop(fp, td); 76826795a03SDmitry Chagin } 76926795a03SDmitry Chagin } 77026795a03SDmitry Chagin 77126795a03SDmitry Chagin if (lev & LINUX_POLLMSG) 77226795a03SDmitry Chagin LINUX_RATELIMIT_MSG_OPT1("unsupported POLLMSG, events(%d)", lev); 77326795a03SDmitry Chagin if (lev & LINUX_POLLREMOVE) 77426795a03SDmitry Chagin LINUX_RATELIMIT_MSG_OPT1("unsupported POLLREMOVE, events(%d)", lev); 77526795a03SDmitry Chagin 77626795a03SDmitry Chagin *bev = bits; 77726795a03SDmitry Chagin } 77826795a03SDmitry Chagin 77926795a03SDmitry Chagin void 78026795a03SDmitry Chagin bsd_to_linux_poll_events(short bev, short *lev) 78126795a03SDmitry Chagin { 78226795a03SDmitry Chagin short bits = 0; 78326795a03SDmitry Chagin 78426795a03SDmitry Chagin if (bev & POLLIN) 78526795a03SDmitry Chagin bits |= LINUX_POLLIN; 78626795a03SDmitry Chagin if (bev & POLLPRI) 78726795a03SDmitry Chagin bits |= LINUX_POLLPRI; 78826795a03SDmitry Chagin if (bev & (POLLOUT | POLLWRNORM)) 78926795a03SDmitry Chagin /* 79026795a03SDmitry Chagin * POLLWRNORM is equal to POLLOUT on FreeBSD, 79126795a03SDmitry Chagin * but not on Linux 79226795a03SDmitry Chagin */ 79326795a03SDmitry Chagin bits |= LINUX_POLLOUT; 79426795a03SDmitry Chagin if (bev & POLLERR) 79526795a03SDmitry Chagin bits |= LINUX_POLLERR; 79626795a03SDmitry Chagin if (bev & POLLHUP) 79726795a03SDmitry Chagin bits |= LINUX_POLLHUP; 79826795a03SDmitry Chagin if (bev & POLLNVAL) 79926795a03SDmitry Chagin bits |= LINUX_POLLNVAL; 80026795a03SDmitry Chagin if (bev & POLLRDNORM) 80126795a03SDmitry Chagin bits |= LINUX_POLLRDNORM; 80226795a03SDmitry Chagin if (bev & POLLRDBAND) 80326795a03SDmitry Chagin bits |= LINUX_POLLRDBAND; 80426795a03SDmitry Chagin if (bev & POLLWRBAND) 80526795a03SDmitry Chagin bits |= LINUX_POLLWRBAND; 80626795a03SDmitry Chagin if (bev & POLLRDHUP) 80726795a03SDmitry Chagin bits |= LINUX_POLLRDHUP; 80826795a03SDmitry Chagin 80926795a03SDmitry Chagin *lev = bits; 81026795a03SDmitry Chagin } 81132fdc75fSDmitry Chagin 81232fdc75fSDmitry Chagin bool 81332fdc75fSDmitry Chagin linux_use_real_ifname(const struct ifnet *ifp) 81432fdc75fSDmitry Chagin { 81532fdc75fSDmitry Chagin 816*3ab3c9c2SDmitry Chagin return (use_real_ifnames); 81732fdc75fSDmitry Chagin } 818