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 599883961eSDmitry Chagin _Static_assert(LINUX_IFNAMSIZ == IFNAMSIZ, "Linux IFNAMSIZ"); 60945a7f0dSDmitry Chagin _Static_assert(sizeof(struct sockaddr) == sizeof(struct l_sockaddr), 61945a7f0dSDmitry Chagin "Linux struct sockaddr size"); 62945a7f0dSDmitry Chagin _Static_assert(offsetof(struct sockaddr, sa_data) == 63945a7f0dSDmitry Chagin offsetof(struct l_sockaddr, sa_data), "Linux struct sockaddr layout"); 644ab7403bSDmitry Chagin 6532fdc75fSDmitry Chagin static bool use_real_ifnames = false; 6632fdc75fSDmitry Chagin SYSCTL_BOOL(_compat_linux, OID_AUTO, use_real_ifnames, CTLFLAG_RWTUN, 6732fdc75fSDmitry Chagin &use_real_ifnames, 0, 6832fdc75fSDmitry Chagin "Use FreeBSD interface names instead of generating ethN aliases"); 6932fdc75fSDmitry Chagin 704ab7403bSDmitry Chagin static int bsd_to_linux_sigtbl[LINUX_SIGTBLSZ] = { 714ab7403bSDmitry Chagin LINUX_SIGHUP, /* SIGHUP */ 724ab7403bSDmitry Chagin LINUX_SIGINT, /* SIGINT */ 734ab7403bSDmitry Chagin LINUX_SIGQUIT, /* SIGQUIT */ 744ab7403bSDmitry Chagin LINUX_SIGILL, /* SIGILL */ 754ab7403bSDmitry Chagin LINUX_SIGTRAP, /* SIGTRAP */ 764ab7403bSDmitry Chagin LINUX_SIGABRT, /* SIGABRT */ 774ab7403bSDmitry Chagin 0, /* SIGEMT */ 784ab7403bSDmitry Chagin LINUX_SIGFPE, /* SIGFPE */ 794ab7403bSDmitry Chagin LINUX_SIGKILL, /* SIGKILL */ 804ab7403bSDmitry Chagin LINUX_SIGBUS, /* SIGBUS */ 814ab7403bSDmitry Chagin LINUX_SIGSEGV, /* SIGSEGV */ 824ab7403bSDmitry Chagin LINUX_SIGSYS, /* SIGSYS */ 834ab7403bSDmitry Chagin LINUX_SIGPIPE, /* SIGPIPE */ 844ab7403bSDmitry Chagin LINUX_SIGALRM, /* SIGALRM */ 854ab7403bSDmitry Chagin LINUX_SIGTERM, /* SIGTERM */ 864ab7403bSDmitry Chagin LINUX_SIGURG, /* SIGURG */ 874ab7403bSDmitry Chagin LINUX_SIGSTOP, /* SIGSTOP */ 884ab7403bSDmitry Chagin LINUX_SIGTSTP, /* SIGTSTP */ 894ab7403bSDmitry Chagin LINUX_SIGCONT, /* SIGCONT */ 904ab7403bSDmitry Chagin LINUX_SIGCHLD, /* SIGCHLD */ 914ab7403bSDmitry Chagin LINUX_SIGTTIN, /* SIGTTIN */ 924ab7403bSDmitry Chagin LINUX_SIGTTOU, /* SIGTTOU */ 934ab7403bSDmitry Chagin LINUX_SIGIO, /* SIGIO */ 944ab7403bSDmitry Chagin LINUX_SIGXCPU, /* SIGXCPU */ 954ab7403bSDmitry Chagin LINUX_SIGXFSZ, /* SIGXFSZ */ 964ab7403bSDmitry Chagin LINUX_SIGVTALRM,/* SIGVTALRM */ 974ab7403bSDmitry Chagin LINUX_SIGPROF, /* SIGPROF */ 984ab7403bSDmitry Chagin LINUX_SIGWINCH, /* SIGWINCH */ 994ab7403bSDmitry Chagin 0, /* SIGINFO */ 1004ab7403bSDmitry Chagin LINUX_SIGUSR1, /* SIGUSR1 */ 1014ab7403bSDmitry Chagin LINUX_SIGUSR2 /* SIGUSR2 */ 1024ab7403bSDmitry Chagin }; 1034ab7403bSDmitry Chagin 1047a7cee55SDmitry Chagin #define LINUX_SIGPWREMU (SIGRTMIN + (LINUX_SIGRTMAX - LINUX_SIGRTMIN) + 1) 1057a7cee55SDmitry Chagin 1064ab7403bSDmitry Chagin static int linux_to_bsd_sigtbl[LINUX_SIGTBLSZ] = { 1074ab7403bSDmitry Chagin SIGHUP, /* LINUX_SIGHUP */ 1084ab7403bSDmitry Chagin SIGINT, /* LINUX_SIGINT */ 1094ab7403bSDmitry Chagin SIGQUIT, /* LINUX_SIGQUIT */ 1104ab7403bSDmitry Chagin SIGILL, /* LINUX_SIGILL */ 1114ab7403bSDmitry Chagin SIGTRAP, /* LINUX_SIGTRAP */ 1124ab7403bSDmitry Chagin SIGABRT, /* LINUX_SIGABRT */ 1134ab7403bSDmitry Chagin SIGBUS, /* LINUX_SIGBUS */ 1144ab7403bSDmitry Chagin SIGFPE, /* LINUX_SIGFPE */ 1154ab7403bSDmitry Chagin SIGKILL, /* LINUX_SIGKILL */ 1164ab7403bSDmitry Chagin SIGUSR1, /* LINUX_SIGUSR1 */ 1174ab7403bSDmitry Chagin SIGSEGV, /* LINUX_SIGSEGV */ 1184ab7403bSDmitry Chagin SIGUSR2, /* LINUX_SIGUSR2 */ 1194ab7403bSDmitry Chagin SIGPIPE, /* LINUX_SIGPIPE */ 1204ab7403bSDmitry Chagin SIGALRM, /* LINUX_SIGALRM */ 1214ab7403bSDmitry Chagin SIGTERM, /* LINUX_SIGTERM */ 1224ab7403bSDmitry Chagin SIGBUS, /* LINUX_SIGSTKFLT */ 1234ab7403bSDmitry Chagin SIGCHLD, /* LINUX_SIGCHLD */ 1244ab7403bSDmitry Chagin SIGCONT, /* LINUX_SIGCONT */ 1254ab7403bSDmitry Chagin SIGSTOP, /* LINUX_SIGSTOP */ 1264ab7403bSDmitry Chagin SIGTSTP, /* LINUX_SIGTSTP */ 1274ab7403bSDmitry Chagin SIGTTIN, /* LINUX_SIGTTIN */ 1284ab7403bSDmitry Chagin SIGTTOU, /* LINUX_SIGTTOU */ 1294ab7403bSDmitry Chagin SIGURG, /* LINUX_SIGURG */ 1304ab7403bSDmitry Chagin SIGXCPU, /* LINUX_SIGXCPU */ 1314ab7403bSDmitry Chagin SIGXFSZ, /* LINUX_SIGXFSZ */ 1324ab7403bSDmitry Chagin SIGVTALRM, /* LINUX_SIGVTALARM */ 1334ab7403bSDmitry Chagin SIGPROF, /* LINUX_SIGPROF */ 1344ab7403bSDmitry Chagin SIGWINCH, /* LINUX_SIGWINCH */ 1354ab7403bSDmitry Chagin SIGIO, /* LINUX_SIGIO */ 1364ab7403bSDmitry Chagin /* 1374ab7403bSDmitry Chagin * FreeBSD does not have SIGPWR signal, map Linux SIGPWR signal 1384ab7403bSDmitry Chagin * to the first unused FreeBSD signal number. Since Linux supports 1394ab7403bSDmitry Chagin * signals from 1 to 64 we are ok here as our SIGRTMIN = 65. 1404ab7403bSDmitry Chagin */ 1417a7cee55SDmitry Chagin LINUX_SIGPWREMU,/* LINUX_SIGPWR */ 1424ab7403bSDmitry Chagin SIGSYS /* LINUX_SIGSYS */ 1434ab7403bSDmitry Chagin }; 1444ab7403bSDmitry Chagin 145044ab55eSEdward Tomasz Napierala static struct cdev *dev_shm_cdev; 146044ab55eSEdward Tomasz Napierala static struct cdevsw dev_shm_cdevsw = { 147044ab55eSEdward Tomasz Napierala .d_version = D_VERSION, 148044ab55eSEdward Tomasz Napierala .d_name = "dev_shm", 149044ab55eSEdward Tomasz Napierala }; 150044ab55eSEdward Tomasz Napierala 1514ab7403bSDmitry Chagin /* 1524ab7403bSDmitry Chagin * Map Linux RT signals to the FreeBSD RT signals. 1534ab7403bSDmitry Chagin */ 1544ab7403bSDmitry Chagin static inline int 1554ab7403bSDmitry Chagin linux_to_bsd_rt_signal(int sig) 1564ab7403bSDmitry Chagin { 1574ab7403bSDmitry Chagin 1587a7cee55SDmitry Chagin return (SIGRTMIN + sig - LINUX_SIGRTMIN); 1594ab7403bSDmitry Chagin } 1604ab7403bSDmitry Chagin 1614ab7403bSDmitry Chagin static inline int 1624ab7403bSDmitry Chagin bsd_to_linux_rt_signal(int sig) 1634ab7403bSDmitry Chagin { 1644ab7403bSDmitry Chagin 1657a7cee55SDmitry Chagin return (sig - SIGRTMIN + LINUX_SIGRTMIN); 1664ab7403bSDmitry Chagin } 1674ab7403bSDmitry Chagin 1684ab7403bSDmitry Chagin int 1694ab7403bSDmitry Chagin linux_to_bsd_signal(int sig) 1704ab7403bSDmitry Chagin { 1714ab7403bSDmitry Chagin 1728081c6ceSEdward Tomasz Napierala KASSERT(sig > 0 && sig <= LINUX_SIGRTMAX, ("invalid Linux signal %d\n", sig)); 1734ab7403bSDmitry Chagin 1744ab7403bSDmitry Chagin if (sig < LINUX_SIGRTMIN) 1754ab7403bSDmitry Chagin return (linux_to_bsd_sigtbl[_SIG_IDX(sig)]); 1764ab7403bSDmitry Chagin 1774ab7403bSDmitry Chagin return (linux_to_bsd_rt_signal(sig)); 1784ab7403bSDmitry Chagin } 1794ab7403bSDmitry Chagin 1804ab7403bSDmitry Chagin int 1814ab7403bSDmitry Chagin bsd_to_linux_signal(int sig) 1824ab7403bSDmitry Chagin { 1834ab7403bSDmitry Chagin 1844ab7403bSDmitry Chagin if (sig <= LINUX_SIGTBLSZ) 1854ab7403bSDmitry Chagin return (bsd_to_linux_sigtbl[_SIG_IDX(sig)]); 1867a7cee55SDmitry Chagin if (sig == LINUX_SIGPWREMU) 1874ab7403bSDmitry Chagin return (LINUX_SIGPWR); 1884ab7403bSDmitry Chagin 1894ab7403bSDmitry Chagin return (bsd_to_linux_rt_signal(sig)); 1904ab7403bSDmitry Chagin } 1914ab7403bSDmitry Chagin 1924ab7403bSDmitry Chagin int 1934ab7403bSDmitry Chagin linux_to_bsd_sigaltstack(int lsa) 1944ab7403bSDmitry Chagin { 1954ab7403bSDmitry Chagin int bsa = 0; 1964ab7403bSDmitry Chagin 1974ab7403bSDmitry Chagin if (lsa & LINUX_SS_DISABLE) 1984ab7403bSDmitry Chagin bsa |= SS_DISABLE; 1994ab7403bSDmitry Chagin /* 2004ab7403bSDmitry Chagin * Linux ignores SS_ONSTACK flag for ss 2014ab7403bSDmitry Chagin * parameter while FreeBSD prohibits it. 2024ab7403bSDmitry Chagin */ 2034ab7403bSDmitry Chagin return (bsa); 2044ab7403bSDmitry Chagin } 2054ab7403bSDmitry Chagin 2064ab7403bSDmitry Chagin int 2074ab7403bSDmitry Chagin bsd_to_linux_sigaltstack(int bsa) 2084ab7403bSDmitry Chagin { 2094ab7403bSDmitry Chagin int lsa = 0; 2104ab7403bSDmitry Chagin 2114ab7403bSDmitry Chagin if (bsa & SS_DISABLE) 2124ab7403bSDmitry Chagin lsa |= LINUX_SS_DISABLE; 2134ab7403bSDmitry Chagin if (bsa & SS_ONSTACK) 2144ab7403bSDmitry Chagin lsa |= LINUX_SS_ONSTACK; 2154ab7403bSDmitry Chagin return (lsa); 2164ab7403bSDmitry Chagin } 2174ab7403bSDmitry Chagin 2184ab7403bSDmitry Chagin void 2194ab7403bSDmitry Chagin linux_to_bsd_sigset(l_sigset_t *lss, sigset_t *bss) 2204ab7403bSDmitry Chagin { 2214ab7403bSDmitry Chagin int b, l; 2224ab7403bSDmitry Chagin 2234ab7403bSDmitry Chagin SIGEMPTYSET(*bss); 2244ab7403bSDmitry Chagin for (l = 1; l <= LINUX_SIGRTMAX; l++) { 2254ab7403bSDmitry Chagin if (LINUX_SIGISMEMBER(*lss, l)) { 2264ab7403bSDmitry Chagin b = linux_to_bsd_signal(l); 2274ab7403bSDmitry Chagin if (b) 2284ab7403bSDmitry Chagin SIGADDSET(*bss, b); 2294ab7403bSDmitry Chagin } 2304ab7403bSDmitry Chagin } 2314ab7403bSDmitry Chagin } 2324ab7403bSDmitry Chagin 2334ab7403bSDmitry Chagin void 2344ab7403bSDmitry Chagin bsd_to_linux_sigset(sigset_t *bss, l_sigset_t *lss) 2354ab7403bSDmitry Chagin { 2364ab7403bSDmitry Chagin int b, l; 2374ab7403bSDmitry Chagin 2384ab7403bSDmitry Chagin LINUX_SIGEMPTYSET(*lss); 2394ab7403bSDmitry Chagin for (b = 1; b <= SIGRTMAX; b++) { 2404ab7403bSDmitry Chagin if (SIGISMEMBER(*bss, b)) { 2414ab7403bSDmitry Chagin l = bsd_to_linux_signal(b); 2424ab7403bSDmitry Chagin if (l) 2434ab7403bSDmitry Chagin LINUX_SIGADDSET(*lss, l); 2444ab7403bSDmitry Chagin } 2454ab7403bSDmitry Chagin } 2464ab7403bSDmitry Chagin } 247d151344dSDmitry Chagin 248d151344dSDmitry Chagin /* 2493ab3c9c2SDmitry Chagin * Translate a FreeBSD interface name to a Linux interface name 2503ab3c9c2SDmitry Chagin * by interface name, and return the number of bytes copied to lxname. 2513ab3c9c2SDmitry Chagin */ 2523ab3c9c2SDmitry Chagin int 2533ab3c9c2SDmitry Chagin ifname_bsd_to_linux_name(const char *bsdname, char *lxname, size_t len) 2543ab3c9c2SDmitry Chagin { 2553ab3c9c2SDmitry Chagin struct epoch_tracker et; 2563ab3c9c2SDmitry Chagin struct ifnet *ifp; 2573ab3c9c2SDmitry Chagin int ret; 2583ab3c9c2SDmitry Chagin 2593ab3c9c2SDmitry Chagin ret = 0; 2603ab3c9c2SDmitry Chagin CURVNET_SET(TD_TO_VNET(curthread)); 2613ab3c9c2SDmitry Chagin NET_EPOCH_ENTER(et); 2623ab3c9c2SDmitry Chagin ifp = ifunit(bsdname); 2633ab3c9c2SDmitry Chagin if (ifp != NULL) 2643ab3c9c2SDmitry Chagin ret = ifname_bsd_to_linux_ifp(ifp, lxname, len); 2653ab3c9c2SDmitry Chagin NET_EPOCH_EXIT(et); 2663ab3c9c2SDmitry Chagin CURVNET_RESTORE(); 2673ab3c9c2SDmitry Chagin return (ret); 2683ab3c9c2SDmitry Chagin } 2693ab3c9c2SDmitry Chagin 2703ab3c9c2SDmitry Chagin /* 2713ab3c9c2SDmitry Chagin * Translate a FreeBSD interface name to a Linux interface name 2723ab3c9c2SDmitry Chagin * by interface index, and return the number of bytes copied to lxname. 2733ab3c9c2SDmitry Chagin */ 2743ab3c9c2SDmitry Chagin int 2753ab3c9c2SDmitry Chagin ifname_bsd_to_linux_idx(u_int idx, char *lxname, size_t len) 2763ab3c9c2SDmitry Chagin { 2773ab3c9c2SDmitry Chagin struct epoch_tracker et; 2783ab3c9c2SDmitry Chagin struct ifnet *ifp; 2793ab3c9c2SDmitry Chagin int ret; 2803ab3c9c2SDmitry Chagin 2813ab3c9c2SDmitry Chagin ret = 0; 2823ab3c9c2SDmitry Chagin CURVNET_SET(TD_TO_VNET(curthread)); 2833ab3c9c2SDmitry Chagin NET_EPOCH_ENTER(et); 2843ab3c9c2SDmitry Chagin ifp = ifnet_byindex(idx); 2853ab3c9c2SDmitry Chagin if (ifp != NULL) 2863ab3c9c2SDmitry Chagin ret = ifname_bsd_to_linux_ifp(ifp, lxname, len); 2873ab3c9c2SDmitry Chagin NET_EPOCH_EXIT(et); 2883ab3c9c2SDmitry Chagin CURVNET_RESTORE(); 2893ab3c9c2SDmitry Chagin return (ret); 2903ab3c9c2SDmitry Chagin } 2913ab3c9c2SDmitry Chagin 2923ab3c9c2SDmitry Chagin /* 2933ab3c9c2SDmitry Chagin * Translate a FreeBSD interface name to a Linux interface name, 2946c5786fdSDmitry Chagin * and return the number of bytes copied to lxname, 0 if interface 2956c5786fdSDmitry Chagin * not found, -1 on error. 2963ab3c9c2SDmitry Chagin */ 2976c5786fdSDmitry Chagin struct ifname_bsd_to_linux_ifp_cb_s { 2986c5786fdSDmitry Chagin struct ifnet *ifp; 2996c5786fdSDmitry Chagin int ethno; 3006c5786fdSDmitry Chagin char *lxname; 3016c5786fdSDmitry Chagin size_t len; 3026c5786fdSDmitry Chagin }; 3036c5786fdSDmitry Chagin 3046c5786fdSDmitry Chagin static int 3056c5786fdSDmitry Chagin ifname_bsd_to_linux_ifp_cb(if_t ifp, void *arg) 3066c5786fdSDmitry Chagin { 3076c5786fdSDmitry Chagin struct ifname_bsd_to_linux_ifp_cb_s *cbs = arg; 3086c5786fdSDmitry Chagin 3096c5786fdSDmitry Chagin if (ifp == cbs->ifp) 3106c5786fdSDmitry Chagin return (snprintf(cbs->lxname, cbs->len, "eth%d", cbs->ethno)); 3116c5786fdSDmitry Chagin if (IFP_IS_ETH(ifp)) 3126c5786fdSDmitry Chagin cbs->ethno++; 3136c5786fdSDmitry Chagin return (0); 3146c5786fdSDmitry Chagin } 3156c5786fdSDmitry Chagin 3163ab3c9c2SDmitry Chagin int 3173ab3c9c2SDmitry Chagin ifname_bsd_to_linux_ifp(struct ifnet *ifp, char *lxname, size_t len) 3183ab3c9c2SDmitry Chagin { 3196c5786fdSDmitry Chagin struct ifname_bsd_to_linux_ifp_cb_s arg = { 3206c5786fdSDmitry Chagin .ifp = ifp, 3216c5786fdSDmitry Chagin .ethno = 0, 3226c5786fdSDmitry Chagin .lxname = lxname, 3236c5786fdSDmitry Chagin .len = len, 3246c5786fdSDmitry Chagin }; 3253ab3c9c2SDmitry Chagin 3263ab3c9c2SDmitry Chagin NET_EPOCH_ASSERT(); 3273ab3c9c2SDmitry Chagin 3283ab3c9c2SDmitry Chagin /* 3293ab3c9c2SDmitry Chagin * Linux loopback interface name is lo (not lo0), 3303ab3c9c2SDmitry Chagin * we translate lo to lo0, loX to loX. 3313ab3c9c2SDmitry Chagin */ 332*56928500SJustin Hibbits if (IFP_IS_LOOP(ifp) && strncmp(if_name(ifp), "lo0", IFNAMSIZ) == 0) 3333ab3c9c2SDmitry Chagin return (strlcpy(lxname, "lo", len)); 3343ab3c9c2SDmitry Chagin 3353ab3c9c2SDmitry Chagin /* Short-circuit non ethernet interfaces. */ 3363ab3c9c2SDmitry Chagin if (!IFP_IS_ETH(ifp) || linux_use_real_ifname(ifp)) 3376c5786fdSDmitry Chagin return (strlcpy(lxname, if_name(ifp), len)); 3383ab3c9c2SDmitry Chagin 3393ab3c9c2SDmitry Chagin /* Determine the (relative) unit number for ethernet interfaces. */ 3406c5786fdSDmitry Chagin return (if_foreach(ifname_bsd_to_linux_ifp_cb, &arg)); 3413ab3c9c2SDmitry Chagin } 3423ab3c9c2SDmitry Chagin 3433ab3c9c2SDmitry Chagin /* 344d151344dSDmitry Chagin * Translate a Linux interface name to a FreeBSD interface name, 345d151344dSDmitry Chagin * and return the associated ifnet structure 346d151344dSDmitry Chagin * bsdname and lxname need to be least IFNAMSIZ bytes long, but 347d151344dSDmitry Chagin * can point to the same buffer. 348d151344dSDmitry Chagin */ 3495d5b633dSDmitry Chagin struct ifname_linux_to_ifp_cb_s { 3506c5786fdSDmitry Chagin bool is_lo; 3516c5786fdSDmitry Chagin bool is_eth; 3526c5786fdSDmitry Chagin int ethno; 3536c5786fdSDmitry Chagin int unit; 3546c5786fdSDmitry Chagin const char *lxname; 3556c5786fdSDmitry Chagin if_t ifp; 3566c5786fdSDmitry Chagin }; 3576c5786fdSDmitry Chagin 3586c5786fdSDmitry Chagin static int 3595d5b633dSDmitry Chagin ifname_linux_to_ifp_cb(if_t ifp, void *arg) 3606c5786fdSDmitry Chagin { 3615d5b633dSDmitry Chagin struct ifname_linux_to_ifp_cb_s *cbs = arg; 3626c5786fdSDmitry Chagin 3636c5786fdSDmitry Chagin NET_EPOCH_ASSERT(); 3646c5786fdSDmitry Chagin 3656c5786fdSDmitry Chagin /* 3666c5786fdSDmitry Chagin * Allow Linux programs to use FreeBSD names. Don't presume 3676c5786fdSDmitry Chagin * we never have an interface named "eth", so don't make 3686c5786fdSDmitry Chagin * the test optional based on is_eth. 3696c5786fdSDmitry Chagin */ 3706c5786fdSDmitry Chagin if (strncmp(if_name(ifp), cbs->lxname, LINUX_IFNAMSIZ) == 0) 3716c5786fdSDmitry Chagin goto out; 3726c5786fdSDmitry Chagin if (cbs->is_eth && IFP_IS_ETH(ifp) && cbs->unit == cbs->ethno) 3736c5786fdSDmitry Chagin goto out; 3746c5786fdSDmitry Chagin if (cbs->is_lo && IFP_IS_LOOP(ifp)) 3756c5786fdSDmitry Chagin goto out; 3766c5786fdSDmitry Chagin if (IFP_IS_ETH(ifp)) 3776c5786fdSDmitry Chagin cbs->ethno++; 3786c5786fdSDmitry Chagin return (0); 3796c5786fdSDmitry Chagin 3806c5786fdSDmitry Chagin out: 3816c5786fdSDmitry Chagin cbs->ifp = ifp; 3826c5786fdSDmitry Chagin return (1); 3836c5786fdSDmitry Chagin } 3846c5786fdSDmitry Chagin 385d151344dSDmitry Chagin struct ifnet * 3865d5b633dSDmitry Chagin ifname_linux_to_ifp(struct thread *td, const char *lxname) 387d151344dSDmitry Chagin { 3885d5b633dSDmitry Chagin struct ifname_linux_to_ifp_cb_s arg = { 3896c5786fdSDmitry Chagin .ethno = 0, 3906c5786fdSDmitry Chagin .lxname = lxname, 3916c5786fdSDmitry Chagin .ifp = NULL, 3926c5786fdSDmitry Chagin }; 3935d5b633dSDmitry Chagin int len; 394d151344dSDmitry Chagin char *ep; 395d151344dSDmitry Chagin 3965d5b633dSDmitry Chagin NET_EPOCH_ASSERT(); 3975d5b633dSDmitry Chagin 398d151344dSDmitry Chagin for (len = 0; len < LINUX_IFNAMSIZ; ++len) 3999c1437aeSDmitry Chagin if (!isalpha(lxname[len]) || lxname[len] == '\0') 400d151344dSDmitry Chagin break; 401d151344dSDmitry Chagin if (len == 0 || len == LINUX_IFNAMSIZ) 402d151344dSDmitry Chagin return (NULL); 4033ab3c9c2SDmitry Chagin /* 4043ab3c9c2SDmitry Chagin * Linux loopback interface name is lo (not lo0), 4053ab3c9c2SDmitry Chagin * we translate lo to lo0, loX to loX. 4063ab3c9c2SDmitry Chagin */ 4076c5786fdSDmitry Chagin arg.is_lo = (len == 2 && strncmp(lxname, "lo", LINUX_IFNAMSIZ) == 0); 4086c5786fdSDmitry Chagin arg.unit = (int)strtoul(lxname + len, &ep, 10); 409d151344dSDmitry Chagin if ((ep == NULL || ep == lxname + len || ep >= lxname + LINUX_IFNAMSIZ) && 4106c5786fdSDmitry Chagin arg.is_lo == 0) 411d151344dSDmitry Chagin return (NULL); 4126c5786fdSDmitry Chagin arg.is_eth = (len == 3 && strncmp(lxname, "eth", len) == 0); 413d151344dSDmitry Chagin 4145d5b633dSDmitry Chagin if_foreach(ifname_linux_to_ifp_cb, &arg); 4155d5b633dSDmitry Chagin return (arg.ifp); 4165d5b633dSDmitry Chagin } 4175d5b633dSDmitry Chagin 418f9b0675bSDmitry Chagin int 4195d5b633dSDmitry Chagin ifname_linux_to_bsd(struct thread *td, const char *lxname, char *bsdname) 4205d5b633dSDmitry Chagin { 4215d5b633dSDmitry Chagin struct epoch_tracker et; 4225d5b633dSDmitry Chagin struct ifnet *ifp; 4235d5b633dSDmitry Chagin 424d151344dSDmitry Chagin CURVNET_SET(TD_TO_VNET(td)); 4256c5786fdSDmitry Chagin NET_EPOCH_ENTER(et); 4265d5b633dSDmitry Chagin ifp = ifname_linux_to_ifp(td, lxname); 4275d5b633dSDmitry Chagin if (ifp != NULL && bsdname != NULL) 4285d5b633dSDmitry Chagin strlcpy(bsdname, if_name(ifp), IFNAMSIZ); 4296c5786fdSDmitry Chagin NET_EPOCH_EXIT(et); 430d151344dSDmitry Chagin CURVNET_RESTORE(); 431f9b0675bSDmitry Chagin return (ifp != NULL ? 0 : EINVAL); 432d151344dSDmitry Chagin } 433d151344dSDmitry Chagin 434130383f2SDmitry Chagin unsigned short 435130383f2SDmitry Chagin linux_ifflags(struct ifnet *ifp) 436d151344dSDmitry Chagin { 4375c32488dSDmitry Chagin unsigned short flags; 438d151344dSDmitry Chagin 4398243e174SDmitry Chagin NET_EPOCH_ASSERT(); 4408243e174SDmitry Chagin 4415c32488dSDmitry Chagin flags = if_getflags(ifp) | if_getdrvflags(ifp); 4425c32488dSDmitry Chagin return (bsd_to_linux_ifflags(flags)); 4435c32488dSDmitry Chagin } 4445c32488dSDmitry Chagin 4455c32488dSDmitry Chagin unsigned short 4465c32488dSDmitry Chagin bsd_to_linux_ifflags(int fl) 4475c32488dSDmitry Chagin { 4485c32488dSDmitry Chagin unsigned short flags = 0; 4495c32488dSDmitry Chagin 450bbac65c7SDmitry Chagin if (fl & IFF_UP) 451130383f2SDmitry Chagin flags |= LINUX_IFF_UP; 452bbac65c7SDmitry Chagin if (fl & IFF_BROADCAST) 453130383f2SDmitry Chagin flags |= LINUX_IFF_BROADCAST; 454bbac65c7SDmitry Chagin if (fl & IFF_DEBUG) 455130383f2SDmitry Chagin flags |= LINUX_IFF_DEBUG; 456bbac65c7SDmitry Chagin if (fl & IFF_LOOPBACK) 457130383f2SDmitry Chagin flags |= LINUX_IFF_LOOPBACK; 458bbac65c7SDmitry Chagin if (fl & IFF_POINTOPOINT) 459130383f2SDmitry Chagin flags |= LINUX_IFF_POINTOPOINT; 460bbac65c7SDmitry Chagin if (fl & IFF_DRV_RUNNING) 461130383f2SDmitry Chagin flags |= LINUX_IFF_RUNNING; 462bbac65c7SDmitry Chagin if (fl & IFF_NOARP) 463130383f2SDmitry Chagin flags |= LINUX_IFF_NOARP; 464bbac65c7SDmitry Chagin if (fl & IFF_PROMISC) 465130383f2SDmitry Chagin flags |= LINUX_IFF_PROMISC; 466bbac65c7SDmitry Chagin if (fl & IFF_ALLMULTI) 467130383f2SDmitry Chagin flags |= LINUX_IFF_ALLMULTI; 468bbac65c7SDmitry Chagin if (fl & IFF_MULTICAST) 469130383f2SDmitry Chagin flags |= LINUX_IFF_MULTICAST; 470130383f2SDmitry Chagin return (flags); 471d151344dSDmitry Chagin } 472d151344dSDmitry Chagin 4736c5786fdSDmitry Chagin static u_int 4746c5786fdSDmitry Chagin linux_ifhwaddr_cb(void *arg, struct ifaddr *ifa, u_int count) 4756c5786fdSDmitry Chagin { 4766c5786fdSDmitry Chagin struct sockaddr_dl *sdl = (struct sockaddr_dl *)ifa->ifa_addr; 4776c5786fdSDmitry Chagin struct l_sockaddr *lsa = arg; 4786c5786fdSDmitry Chagin 4796c5786fdSDmitry Chagin if (count > 0) 4806c5786fdSDmitry Chagin return (0); 4816c5786fdSDmitry Chagin if (sdl->sdl_type != IFT_ETHER) 4826c5786fdSDmitry Chagin return (0); 4836c5786fdSDmitry Chagin bzero(lsa, sizeof(*lsa)); 4846c5786fdSDmitry Chagin lsa->sa_family = LINUX_ARPHRD_ETHER; 4856c5786fdSDmitry Chagin bcopy(LLADDR(sdl), lsa->sa_data, LINUX_IFHWADDRLEN); 4866c5786fdSDmitry Chagin return (1); 4876c5786fdSDmitry Chagin } 4886c5786fdSDmitry Chagin 489d151344dSDmitry Chagin int 490d151344dSDmitry Chagin linux_ifhwaddr(struct ifnet *ifp, struct l_sockaddr *lsa) 491d151344dSDmitry Chagin { 492d151344dSDmitry Chagin 4938d6dd96dSDmitry Chagin NET_EPOCH_ASSERT(); 4948d6dd96dSDmitry Chagin 495d151344dSDmitry Chagin if (IFP_IS_LOOP(ifp)) { 496d151344dSDmitry Chagin bzero(lsa, sizeof(*lsa)); 497d151344dSDmitry Chagin lsa->sa_family = LINUX_ARPHRD_LOOPBACK; 498d151344dSDmitry Chagin return (0); 499d151344dSDmitry Chagin } 500d151344dSDmitry Chagin if (!IFP_IS_ETH(ifp)) 501d151344dSDmitry Chagin return (ENOENT); 5026c5786fdSDmitry Chagin if (if_foreach_addr_type(ifp, AF_LINK, linux_ifhwaddr_cb, lsa) > 0) 503d151344dSDmitry Chagin return (0); 504d151344dSDmitry Chagin return (ENOENT); 505d151344dSDmitry Chagin } 506d5368bf3SDmitry Chagin 507d5368bf3SDmitry Chagin int 508d5368bf3SDmitry Chagin linux_to_bsd_domain(int domain) 509d5368bf3SDmitry Chagin { 510d5368bf3SDmitry Chagin 511d5368bf3SDmitry Chagin switch (domain) { 512d5368bf3SDmitry Chagin case LINUX_AF_UNSPEC: 513d5368bf3SDmitry Chagin return (AF_UNSPEC); 514d5368bf3SDmitry Chagin case LINUX_AF_UNIX: 515d5368bf3SDmitry Chagin return (AF_LOCAL); 516d5368bf3SDmitry Chagin case LINUX_AF_INET: 517d5368bf3SDmitry Chagin return (AF_INET); 518d5368bf3SDmitry Chagin case LINUX_AF_INET6: 519d5368bf3SDmitry Chagin return (AF_INET6); 520d5368bf3SDmitry Chagin case LINUX_AF_AX25: 521d5368bf3SDmitry Chagin return (AF_CCITT); 522d5368bf3SDmitry Chagin case LINUX_AF_IPX: 523d5368bf3SDmitry Chagin return (AF_IPX); 524d5368bf3SDmitry Chagin case LINUX_AF_APPLETALK: 525d5368bf3SDmitry Chagin return (AF_APPLETALK); 5267c40e2d5SAlexander V. Chernikov case LINUX_AF_NETLINK: 5277c40e2d5SAlexander V. Chernikov return (AF_NETLINK); 528d5368bf3SDmitry Chagin } 529d5368bf3SDmitry Chagin return (-1); 530d5368bf3SDmitry Chagin } 531d5368bf3SDmitry Chagin 532d5368bf3SDmitry Chagin int 533d5368bf3SDmitry Chagin bsd_to_linux_domain(int domain) 534d5368bf3SDmitry Chagin { 535d5368bf3SDmitry Chagin 536d5368bf3SDmitry Chagin switch (domain) { 537d5368bf3SDmitry Chagin case AF_UNSPEC: 538d5368bf3SDmitry Chagin return (LINUX_AF_UNSPEC); 539d5368bf3SDmitry Chagin case AF_LOCAL: 540d5368bf3SDmitry Chagin return (LINUX_AF_UNIX); 541d5368bf3SDmitry Chagin case AF_INET: 542d5368bf3SDmitry Chagin return (LINUX_AF_INET); 543d5368bf3SDmitry Chagin case AF_INET6: 544d5368bf3SDmitry Chagin return (LINUX_AF_INET6); 545d5368bf3SDmitry Chagin case AF_CCITT: 546d5368bf3SDmitry Chagin return (LINUX_AF_AX25); 547d5368bf3SDmitry Chagin case AF_IPX: 548d5368bf3SDmitry Chagin return (LINUX_AF_IPX); 549d5368bf3SDmitry Chagin case AF_APPLETALK: 550d5368bf3SDmitry Chagin return (LINUX_AF_APPLETALK); 5517c40e2d5SAlexander V. Chernikov case AF_NETLINK: 5527c40e2d5SAlexander V. Chernikov return (LINUX_AF_NETLINK); 553d5368bf3SDmitry Chagin } 554d5368bf3SDmitry Chagin return (-1); 555d5368bf3SDmitry Chagin } 556d5368bf3SDmitry Chagin 557d5368bf3SDmitry Chagin /* 558d5368bf3SDmitry Chagin * Based on the fact that: 559d5368bf3SDmitry Chagin * 1. Native and Linux storage of struct sockaddr 560d5368bf3SDmitry Chagin * and struct sockaddr_in6 are equal. 561d5368bf3SDmitry Chagin * 2. On Linux sa_family is the first member of all struct sockaddr. 562d5368bf3SDmitry Chagin */ 563d5368bf3SDmitry Chagin int 564d5368bf3SDmitry Chagin bsd_to_linux_sockaddr(const struct sockaddr *sa, struct l_sockaddr **lsa, 565d5368bf3SDmitry Chagin socklen_t len) 566d5368bf3SDmitry Chagin { 567d5368bf3SDmitry Chagin struct l_sockaddr *kosa; 568bbddd588SDmitry Chagin int bdom; 569d5368bf3SDmitry Chagin 570d5368bf3SDmitry Chagin *lsa = NULL; 571d5368bf3SDmitry Chagin if (len < 2 || len > UCHAR_MAX) 572d5368bf3SDmitry Chagin return (EINVAL); 573bbddd588SDmitry Chagin bdom = bsd_to_linux_domain(sa->sa_family); 574bbddd588SDmitry Chagin if (bdom == -1) 575bbddd588SDmitry Chagin return (EAFNOSUPPORT); 576d5368bf3SDmitry Chagin 57763355839SDmitry Chagin kosa = malloc(len, M_LINUX, M_WAITOK); 578d5368bf3SDmitry Chagin bcopy(sa, kosa, len); 579d5368bf3SDmitry Chagin kosa->sa_family = bdom; 580d5368bf3SDmitry Chagin *lsa = kosa; 581d5368bf3SDmitry Chagin return (0); 582d5368bf3SDmitry Chagin } 583d5368bf3SDmitry Chagin 584d5368bf3SDmitry Chagin int 585d5368bf3SDmitry Chagin linux_to_bsd_sockaddr(const struct l_sockaddr *osa, struct sockaddr **sap, 586d5368bf3SDmitry Chagin socklen_t *len) 587d5368bf3SDmitry Chagin { 588d5368bf3SDmitry Chagin struct sockaddr *sa; 589d5368bf3SDmitry Chagin struct l_sockaddr *kosa; 590d5368bf3SDmitry Chagin #ifdef INET6 591d5368bf3SDmitry Chagin struct sockaddr_in6 *sin6; 592d5368bf3SDmitry Chagin bool oldv6size; 593d5368bf3SDmitry Chagin #endif 594d5368bf3SDmitry Chagin char *name; 595d5368bf3SDmitry Chagin int salen, bdom, error, hdrlen, namelen; 596d5368bf3SDmitry Chagin 597d5368bf3SDmitry Chagin if (*len < 2 || *len > UCHAR_MAX) 598d5368bf3SDmitry Chagin return (EINVAL); 599d5368bf3SDmitry Chagin 600d5368bf3SDmitry Chagin salen = *len; 601d5368bf3SDmitry Chagin 602d5368bf3SDmitry Chagin #ifdef INET6 603d5368bf3SDmitry Chagin oldv6size = false; 604d5368bf3SDmitry Chagin /* 605d5368bf3SDmitry Chagin * Check for old (pre-RFC2553) sockaddr_in6. We may accept it 606d5368bf3SDmitry Chagin * if it's a v4-mapped address, so reserve the proper space 607d5368bf3SDmitry Chagin * for it. 608d5368bf3SDmitry Chagin */ 609d5368bf3SDmitry Chagin if (salen == sizeof(struct sockaddr_in6) - sizeof(uint32_t)) { 610d5368bf3SDmitry Chagin salen += sizeof(uint32_t); 611d5368bf3SDmitry Chagin oldv6size = true; 612d5368bf3SDmitry Chagin } 613d5368bf3SDmitry Chagin #endif 614d5368bf3SDmitry Chagin 615d5368bf3SDmitry Chagin kosa = malloc(salen, M_SONAME, M_WAITOK); 616d5368bf3SDmitry Chagin 617d5368bf3SDmitry Chagin if ((error = copyin(osa, kosa, *len))) 618d5368bf3SDmitry Chagin goto out; 619d5368bf3SDmitry Chagin 620d5368bf3SDmitry Chagin bdom = linux_to_bsd_domain(kosa->sa_family); 621d5368bf3SDmitry Chagin if (bdom == -1) { 622d5368bf3SDmitry Chagin error = EAFNOSUPPORT; 623d5368bf3SDmitry Chagin goto out; 624d5368bf3SDmitry Chagin } 625d5368bf3SDmitry Chagin 626d5368bf3SDmitry Chagin #ifdef INET6 627d5368bf3SDmitry Chagin /* 628d5368bf3SDmitry Chagin * Older Linux IPv6 code uses obsolete RFC2133 struct sockaddr_in6, 629d5368bf3SDmitry Chagin * which lacks the scope id compared with RFC2553 one. If we detect 630d5368bf3SDmitry Chagin * the situation, reject the address and write a message to system log. 631d5368bf3SDmitry Chagin * 632d5368bf3SDmitry Chagin * Still accept addresses for which the scope id is not used. 633d5368bf3SDmitry Chagin */ 634d5368bf3SDmitry Chagin if (oldv6size) { 635d5368bf3SDmitry Chagin if (bdom == AF_INET6) { 636d5368bf3SDmitry Chagin sin6 = (struct sockaddr_in6 *)kosa; 637d5368bf3SDmitry Chagin if (IN6_IS_ADDR_V4MAPPED(&sin6->sin6_addr) || 638d5368bf3SDmitry Chagin (!IN6_IS_ADDR_LINKLOCAL(&sin6->sin6_addr) && 639d5368bf3SDmitry Chagin !IN6_IS_ADDR_SITELOCAL(&sin6->sin6_addr) && 640d5368bf3SDmitry Chagin !IN6_IS_ADDR_V4COMPAT(&sin6->sin6_addr) && 641d5368bf3SDmitry Chagin !IN6_IS_ADDR_UNSPECIFIED(&sin6->sin6_addr) && 642d5368bf3SDmitry Chagin !IN6_IS_ADDR_MULTICAST(&sin6->sin6_addr))) { 643d5368bf3SDmitry Chagin sin6->sin6_scope_id = 0; 644d5368bf3SDmitry Chagin } else { 645d5368bf3SDmitry Chagin linux_msg(curthread, 64686e794ebSEdward Tomasz Napierala "obsolete pre-RFC2553 sockaddr_in6 rejected"); 647d5368bf3SDmitry Chagin error = EINVAL; 648d5368bf3SDmitry Chagin goto out; 649d5368bf3SDmitry Chagin } 650d5368bf3SDmitry Chagin } else 651d5368bf3SDmitry Chagin salen -= sizeof(uint32_t); 652d5368bf3SDmitry Chagin } 653d5368bf3SDmitry Chagin #endif 654d5368bf3SDmitry Chagin if (bdom == AF_INET) { 655d5368bf3SDmitry Chagin if (salen < sizeof(struct sockaddr_in)) { 656d5368bf3SDmitry Chagin error = EINVAL; 657d5368bf3SDmitry Chagin goto out; 658d5368bf3SDmitry Chagin } 659d5368bf3SDmitry Chagin salen = sizeof(struct sockaddr_in); 660d5368bf3SDmitry Chagin } 661d5368bf3SDmitry Chagin 662d5368bf3SDmitry Chagin if (bdom == AF_LOCAL && salen > sizeof(struct sockaddr_un)) { 663d5368bf3SDmitry Chagin hdrlen = offsetof(struct sockaddr_un, sun_path); 664d5368bf3SDmitry Chagin name = ((struct sockaddr_un *)kosa)->sun_path; 665d5368bf3SDmitry Chagin if (*name == '\0') { 666d5368bf3SDmitry Chagin /* 667d5368bf3SDmitry Chagin * Linux abstract namespace starts with a NULL byte. 668d5368bf3SDmitry Chagin * XXX We do not support abstract namespace yet. 669d5368bf3SDmitry Chagin */ 670d5368bf3SDmitry Chagin namelen = strnlen(name + 1, salen - hdrlen - 1) + 1; 671d5368bf3SDmitry Chagin } else 672d5368bf3SDmitry Chagin namelen = strnlen(name, salen - hdrlen); 673d5368bf3SDmitry Chagin salen = hdrlen + namelen; 674d5368bf3SDmitry Chagin if (salen > sizeof(struct sockaddr_un)) { 675d5368bf3SDmitry Chagin error = ENAMETOOLONG; 676d5368bf3SDmitry Chagin goto out; 677d5368bf3SDmitry Chagin } 678d5368bf3SDmitry Chagin } 679d5368bf3SDmitry Chagin 6807c40e2d5SAlexander V. Chernikov if (bdom == AF_NETLINK) { 6817c40e2d5SAlexander V. Chernikov if (salen < sizeof(struct sockaddr_nl)) { 6827c40e2d5SAlexander V. Chernikov error = EINVAL; 6837c40e2d5SAlexander V. Chernikov goto out; 6847c40e2d5SAlexander V. Chernikov } 6857c40e2d5SAlexander V. Chernikov salen = sizeof(struct sockaddr_nl); 6867c40e2d5SAlexander V. Chernikov } 6877c40e2d5SAlexander V. Chernikov 688d5368bf3SDmitry Chagin sa = (struct sockaddr *)kosa; 689d5368bf3SDmitry Chagin sa->sa_family = bdom; 690d5368bf3SDmitry Chagin sa->sa_len = salen; 691d5368bf3SDmitry Chagin 692d5368bf3SDmitry Chagin *sap = sa; 693d5368bf3SDmitry Chagin *len = salen; 694d5368bf3SDmitry Chagin return (0); 695d5368bf3SDmitry Chagin 696d5368bf3SDmitry Chagin out: 697d5368bf3SDmitry Chagin free(kosa, M_SONAME); 698d5368bf3SDmitry Chagin return (error); 699d5368bf3SDmitry Chagin } 700044ab55eSEdward Tomasz Napierala 701044ab55eSEdward Tomasz Napierala void 702044ab55eSEdward Tomasz Napierala linux_dev_shm_create(void) 703044ab55eSEdward Tomasz Napierala { 704044ab55eSEdward Tomasz Napierala int error; 705044ab55eSEdward Tomasz Napierala 706044ab55eSEdward Tomasz Napierala error = make_dev_p(MAKEDEV_CHECKNAME | MAKEDEV_WAITOK, &dev_shm_cdev, 707044ab55eSEdward Tomasz Napierala &dev_shm_cdevsw, NULL, UID_ROOT, GID_WHEEL, 0, "shm/.mountpoint"); 708044ab55eSEdward Tomasz Napierala if (error != 0) { 709044ab55eSEdward Tomasz Napierala printf("%s: failed to create device node, error %d\n", 710044ab55eSEdward Tomasz Napierala __func__, error); 711044ab55eSEdward Tomasz Napierala } 712044ab55eSEdward Tomasz Napierala } 713044ab55eSEdward Tomasz Napierala 714044ab55eSEdward Tomasz Napierala void 715044ab55eSEdward Tomasz Napierala linux_dev_shm_destroy(void) 716044ab55eSEdward Tomasz Napierala { 717044ab55eSEdward Tomasz Napierala 718044ab55eSEdward Tomasz Napierala destroy_dev(dev_shm_cdev); 719044ab55eSEdward Tomasz Napierala } 7205403f186SKyle Evans 7215403f186SKyle Evans int 7225403f186SKyle Evans bsd_to_linux_bits_(int value, struct bsd_to_linux_bitmap *bitmap, 7235403f186SKyle Evans size_t mapcnt, int no_value) 7245403f186SKyle Evans { 7255403f186SKyle Evans int bsd_mask, bsd_value, linux_mask, linux_value; 7265403f186SKyle Evans int linux_ret; 7275403f186SKyle Evans size_t i; 7285403f186SKyle Evans bool applied; 7295403f186SKyle Evans 7305403f186SKyle Evans applied = false; 7315403f186SKyle Evans linux_ret = 0; 7325403f186SKyle Evans for (i = 0; i < mapcnt; ++i) { 7335403f186SKyle Evans bsd_mask = bitmap[i].bsd_mask; 7345403f186SKyle Evans bsd_value = bitmap[i].bsd_value; 7355403f186SKyle Evans if (bsd_mask == 0) 7365403f186SKyle Evans bsd_mask = bsd_value; 7375403f186SKyle Evans 7385403f186SKyle Evans linux_mask = bitmap[i].linux_mask; 7395403f186SKyle Evans linux_value = bitmap[i].linux_value; 7405403f186SKyle Evans if (linux_mask == 0) 7415403f186SKyle Evans linux_mask = linux_value; 7425403f186SKyle Evans 7435403f186SKyle Evans /* 7445403f186SKyle Evans * If a mask larger than just the value is set, we explicitly 7455403f186SKyle Evans * want to make sure that only this bit we mapped within that 7465403f186SKyle Evans * mask is set. 7475403f186SKyle Evans */ 7485403f186SKyle Evans if ((value & bsd_mask) == bsd_value) { 7495403f186SKyle Evans linux_ret = (linux_ret & ~linux_mask) | linux_value; 7505403f186SKyle Evans applied = true; 7515403f186SKyle Evans } 7525403f186SKyle Evans } 7535403f186SKyle Evans 7545403f186SKyle Evans if (!applied) 7555403f186SKyle Evans return (no_value); 7565403f186SKyle Evans return (linux_ret); 7575403f186SKyle Evans } 7585403f186SKyle Evans 7595403f186SKyle Evans int 7605403f186SKyle Evans linux_to_bsd_bits_(int value, struct bsd_to_linux_bitmap *bitmap, 7615403f186SKyle Evans size_t mapcnt, int no_value) 7625403f186SKyle Evans { 7635403f186SKyle Evans int bsd_mask, bsd_value, linux_mask, linux_value; 7645403f186SKyle Evans int bsd_ret; 7655403f186SKyle Evans size_t i; 7665403f186SKyle Evans bool applied; 7675403f186SKyle Evans 7685403f186SKyle Evans applied = false; 7695403f186SKyle Evans bsd_ret = 0; 7705403f186SKyle Evans for (i = 0; i < mapcnt; ++i) { 7715403f186SKyle Evans bsd_mask = bitmap[i].bsd_mask; 7725403f186SKyle Evans bsd_value = bitmap[i].bsd_value; 7735403f186SKyle Evans if (bsd_mask == 0) 7745403f186SKyle Evans bsd_mask = bsd_value; 7755403f186SKyle Evans 7765403f186SKyle Evans linux_mask = bitmap[i].linux_mask; 7775403f186SKyle Evans linux_value = bitmap[i].linux_value; 7785403f186SKyle Evans if (linux_mask == 0) 7795403f186SKyle Evans linux_mask = linux_value; 7805403f186SKyle Evans 7815403f186SKyle Evans /* 7825403f186SKyle Evans * If a mask larger than just the value is set, we explicitly 7835403f186SKyle Evans * want to make sure that only this bit we mapped within that 7845403f186SKyle Evans * mask is set. 7855403f186SKyle Evans */ 7865403f186SKyle Evans if ((value & linux_mask) == linux_value) { 7875403f186SKyle Evans bsd_ret = (bsd_ret & ~bsd_mask) | bsd_value; 7885403f186SKyle Evans applied = true; 7895403f186SKyle Evans } 7905403f186SKyle Evans } 7915403f186SKyle Evans 7925403f186SKyle Evans if (!applied) 7935403f186SKyle Evans return (no_value); 7945403f186SKyle Evans return (bsd_ret); 7955403f186SKyle Evans } 79626795a03SDmitry Chagin 79726795a03SDmitry Chagin void 79826795a03SDmitry Chagin linux_to_bsd_poll_events(struct thread *td, int fd, short lev, 79926795a03SDmitry Chagin short *bev) 80026795a03SDmitry Chagin { 80126795a03SDmitry Chagin struct file *fp; 80226795a03SDmitry Chagin int error; 80326795a03SDmitry Chagin short bits = 0; 80426795a03SDmitry Chagin 80526795a03SDmitry Chagin if (lev & LINUX_POLLIN) 80626795a03SDmitry Chagin bits |= POLLIN; 80726795a03SDmitry Chagin if (lev & LINUX_POLLPRI) 80826795a03SDmitry Chagin bits |= POLLPRI; 80926795a03SDmitry Chagin if (lev & LINUX_POLLOUT) 81026795a03SDmitry Chagin bits |= POLLOUT; 81126795a03SDmitry Chagin if (lev & LINUX_POLLERR) 81226795a03SDmitry Chagin bits |= POLLERR; 81326795a03SDmitry Chagin if (lev & LINUX_POLLHUP) 81426795a03SDmitry Chagin bits |= POLLHUP; 81526795a03SDmitry Chagin if (lev & LINUX_POLLNVAL) 81626795a03SDmitry Chagin bits |= POLLNVAL; 81726795a03SDmitry Chagin if (lev & LINUX_POLLRDNORM) 81826795a03SDmitry Chagin bits |= POLLRDNORM; 81926795a03SDmitry Chagin if (lev & LINUX_POLLRDBAND) 82026795a03SDmitry Chagin bits |= POLLRDBAND; 82126795a03SDmitry Chagin if (lev & LINUX_POLLWRBAND) 82226795a03SDmitry Chagin bits |= POLLWRBAND; 82326795a03SDmitry Chagin if (lev & LINUX_POLLWRNORM) 82426795a03SDmitry Chagin bits |= POLLWRNORM; 82526795a03SDmitry Chagin 82626795a03SDmitry Chagin if (lev & LINUX_POLLRDHUP) { 82726795a03SDmitry Chagin /* 82826795a03SDmitry Chagin * It seems that the Linux silencly ignores POLLRDHUP 82926795a03SDmitry Chagin * on non-socket file descriptors unlike FreeBSD, where 83026795a03SDmitry Chagin * events bits is more strictly checked (POLLSTANDARD). 83126795a03SDmitry Chagin */ 832513c7a6eSMateusz Guzik error = fget_unlocked(td, fd, &cap_no_rights, &fp); 83326795a03SDmitry Chagin if (error == 0) { 83426795a03SDmitry Chagin /* 83526795a03SDmitry Chagin * XXX. On FreeBSD POLLRDHUP applies only to 83626795a03SDmitry Chagin * stream sockets. 83726795a03SDmitry Chagin */ 83826795a03SDmitry Chagin if (fp->f_type == DTYPE_SOCKET) 83926795a03SDmitry Chagin bits |= POLLRDHUP; 84026795a03SDmitry Chagin fdrop(fp, td); 84126795a03SDmitry Chagin } 84226795a03SDmitry Chagin } 84326795a03SDmitry Chagin 84426795a03SDmitry Chagin if (lev & LINUX_POLLMSG) 84526795a03SDmitry Chagin LINUX_RATELIMIT_MSG_OPT1("unsupported POLLMSG, events(%d)", lev); 84626795a03SDmitry Chagin if (lev & LINUX_POLLREMOVE) 84726795a03SDmitry Chagin LINUX_RATELIMIT_MSG_OPT1("unsupported POLLREMOVE, events(%d)", lev); 84826795a03SDmitry Chagin 84926795a03SDmitry Chagin *bev = bits; 85026795a03SDmitry Chagin } 85126795a03SDmitry Chagin 85226795a03SDmitry Chagin void 85326795a03SDmitry Chagin bsd_to_linux_poll_events(short bev, short *lev) 85426795a03SDmitry Chagin { 85526795a03SDmitry Chagin short bits = 0; 85626795a03SDmitry Chagin 85726795a03SDmitry Chagin if (bev & POLLIN) 85826795a03SDmitry Chagin bits |= LINUX_POLLIN; 85926795a03SDmitry Chagin if (bev & POLLPRI) 86026795a03SDmitry Chagin bits |= LINUX_POLLPRI; 86126795a03SDmitry Chagin if (bev & (POLLOUT | POLLWRNORM)) 86226795a03SDmitry Chagin /* 86326795a03SDmitry Chagin * POLLWRNORM is equal to POLLOUT on FreeBSD, 86426795a03SDmitry Chagin * but not on Linux 86526795a03SDmitry Chagin */ 86626795a03SDmitry Chagin bits |= LINUX_POLLOUT; 86726795a03SDmitry Chagin if (bev & POLLERR) 86826795a03SDmitry Chagin bits |= LINUX_POLLERR; 86926795a03SDmitry Chagin if (bev & POLLHUP) 87026795a03SDmitry Chagin bits |= LINUX_POLLHUP; 87126795a03SDmitry Chagin if (bev & POLLNVAL) 87226795a03SDmitry Chagin bits |= LINUX_POLLNVAL; 87326795a03SDmitry Chagin if (bev & POLLRDNORM) 87426795a03SDmitry Chagin bits |= LINUX_POLLRDNORM; 87526795a03SDmitry Chagin if (bev & POLLRDBAND) 87626795a03SDmitry Chagin bits |= LINUX_POLLRDBAND; 87726795a03SDmitry Chagin if (bev & POLLWRBAND) 87826795a03SDmitry Chagin bits |= LINUX_POLLWRBAND; 87926795a03SDmitry Chagin if (bev & POLLRDHUP) 88026795a03SDmitry Chagin bits |= LINUX_POLLRDHUP; 88126795a03SDmitry Chagin 88226795a03SDmitry Chagin *lev = bits; 88326795a03SDmitry Chagin } 88432fdc75fSDmitry Chagin 88532fdc75fSDmitry Chagin bool 88632fdc75fSDmitry Chagin linux_use_real_ifname(const struct ifnet *ifp) 88732fdc75fSDmitry Chagin { 88832fdc75fSDmitry Chagin 8893ab3c9c2SDmitry Chagin return (use_real_ifnames); 89032fdc75fSDmitry Chagin } 891