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
2677f66834SDmitry Chagin #include "opt_inet6.h"
27d5368bf3SDmitry Chagin
284ab7403bSDmitry Chagin #include <sys/param.h>
29044ab55eSEdward Tomasz Napierala #include <sys/conf.h>
30d151344dSDmitry Chagin #include <sys/ctype.h>
3126795a03SDmitry Chagin #include <sys/file.h>
3226795a03SDmitry Chagin #include <sys/filedesc.h>
33d151344dSDmitry Chagin #include <sys/jail.h>
34d151344dSDmitry Chagin #include <sys/lock.h>
35d5368bf3SDmitry Chagin #include <sys/malloc.h>
3626795a03SDmitry Chagin #include <sys/poll.h>
3726795a03SDmitry Chagin #include <sys/proc.h>
384ab7403bSDmitry Chagin #include <sys/signalvar.h>
39d151344dSDmitry Chagin #include <sys/socket.h>
40d5368bf3SDmitry Chagin #include <sys/socketvar.h>
41d151344dSDmitry Chagin
42d151344dSDmitry Chagin #include <net/if.h>
43d151344dSDmitry Chagin #include <net/if_var.h>
44d151344dSDmitry Chagin #include <net/if_dl.h>
45d151344dSDmitry Chagin #include <net/if_types.h>
467c40e2d5SAlexander V. Chernikov #include <netlink/netlink.h>
474ab7403bSDmitry Chagin
48d5368bf3SDmitry Chagin #include <sys/un.h>
49d5368bf3SDmitry Chagin #include <netinet/in.h>
50d5368bf3SDmitry Chagin
514ab7403bSDmitry Chagin #include <compat/linux/linux.h>
52d151344dSDmitry Chagin #include <compat/linux/linux_common.h>
5326795a03SDmitry Chagin #include <compat/linux/linux_mib.h>
54d5368bf3SDmitry Chagin #include <compat/linux/linux_util.h>
554ab7403bSDmitry Chagin
569883961eSDmitry Chagin _Static_assert(LINUX_IFNAMSIZ == IFNAMSIZ, "Linux IFNAMSIZ");
57945a7f0dSDmitry Chagin _Static_assert(sizeof(struct sockaddr) == sizeof(struct l_sockaddr),
58945a7f0dSDmitry Chagin "Linux struct sockaddr size");
59945a7f0dSDmitry Chagin _Static_assert(offsetof(struct sockaddr, sa_data) ==
60945a7f0dSDmitry Chagin offsetof(struct l_sockaddr, sa_data), "Linux struct sockaddr layout");
614ab7403bSDmitry Chagin
6232fdc75fSDmitry Chagin static bool use_real_ifnames = false;
6332fdc75fSDmitry Chagin SYSCTL_BOOL(_compat_linux, OID_AUTO, use_real_ifnames, CTLFLAG_RWTUN,
6432fdc75fSDmitry Chagin &use_real_ifnames, 0,
6532fdc75fSDmitry Chagin "Use FreeBSD interface names instead of generating ethN aliases");
6632fdc75fSDmitry Chagin
674ab7403bSDmitry Chagin static int bsd_to_linux_sigtbl[LINUX_SIGTBLSZ] = {
684ab7403bSDmitry Chagin LINUX_SIGHUP, /* SIGHUP */
694ab7403bSDmitry Chagin LINUX_SIGINT, /* SIGINT */
704ab7403bSDmitry Chagin LINUX_SIGQUIT, /* SIGQUIT */
714ab7403bSDmitry Chagin LINUX_SIGILL, /* SIGILL */
724ab7403bSDmitry Chagin LINUX_SIGTRAP, /* SIGTRAP */
734ab7403bSDmitry Chagin LINUX_SIGABRT, /* SIGABRT */
744ab7403bSDmitry Chagin 0, /* SIGEMT */
754ab7403bSDmitry Chagin LINUX_SIGFPE, /* SIGFPE */
764ab7403bSDmitry Chagin LINUX_SIGKILL, /* SIGKILL */
774ab7403bSDmitry Chagin LINUX_SIGBUS, /* SIGBUS */
784ab7403bSDmitry Chagin LINUX_SIGSEGV, /* SIGSEGV */
794ab7403bSDmitry Chagin LINUX_SIGSYS, /* SIGSYS */
804ab7403bSDmitry Chagin LINUX_SIGPIPE, /* SIGPIPE */
814ab7403bSDmitry Chagin LINUX_SIGALRM, /* SIGALRM */
824ab7403bSDmitry Chagin LINUX_SIGTERM, /* SIGTERM */
834ab7403bSDmitry Chagin LINUX_SIGURG, /* SIGURG */
844ab7403bSDmitry Chagin LINUX_SIGSTOP, /* SIGSTOP */
854ab7403bSDmitry Chagin LINUX_SIGTSTP, /* SIGTSTP */
864ab7403bSDmitry Chagin LINUX_SIGCONT, /* SIGCONT */
874ab7403bSDmitry Chagin LINUX_SIGCHLD, /* SIGCHLD */
884ab7403bSDmitry Chagin LINUX_SIGTTIN, /* SIGTTIN */
894ab7403bSDmitry Chagin LINUX_SIGTTOU, /* SIGTTOU */
904ab7403bSDmitry Chagin LINUX_SIGIO, /* SIGIO */
914ab7403bSDmitry Chagin LINUX_SIGXCPU, /* SIGXCPU */
924ab7403bSDmitry Chagin LINUX_SIGXFSZ, /* SIGXFSZ */
934ab7403bSDmitry Chagin LINUX_SIGVTALRM,/* SIGVTALRM */
944ab7403bSDmitry Chagin LINUX_SIGPROF, /* SIGPROF */
954ab7403bSDmitry Chagin LINUX_SIGWINCH, /* SIGWINCH */
964ab7403bSDmitry Chagin 0, /* SIGINFO */
974ab7403bSDmitry Chagin LINUX_SIGUSR1, /* SIGUSR1 */
984ab7403bSDmitry Chagin LINUX_SIGUSR2 /* SIGUSR2 */
994ab7403bSDmitry Chagin };
1004ab7403bSDmitry Chagin
1017a7cee55SDmitry Chagin #define LINUX_SIGPWREMU (SIGRTMIN + (LINUX_SIGRTMAX - LINUX_SIGRTMIN) + 1)
1027a7cee55SDmitry Chagin
1034ab7403bSDmitry Chagin static int linux_to_bsd_sigtbl[LINUX_SIGTBLSZ] = {
1044ab7403bSDmitry Chagin SIGHUP, /* LINUX_SIGHUP */
1054ab7403bSDmitry Chagin SIGINT, /* LINUX_SIGINT */
1064ab7403bSDmitry Chagin SIGQUIT, /* LINUX_SIGQUIT */
1074ab7403bSDmitry Chagin SIGILL, /* LINUX_SIGILL */
1084ab7403bSDmitry Chagin SIGTRAP, /* LINUX_SIGTRAP */
1094ab7403bSDmitry Chagin SIGABRT, /* LINUX_SIGABRT */
1104ab7403bSDmitry Chagin SIGBUS, /* LINUX_SIGBUS */
1114ab7403bSDmitry Chagin SIGFPE, /* LINUX_SIGFPE */
1124ab7403bSDmitry Chagin SIGKILL, /* LINUX_SIGKILL */
1134ab7403bSDmitry Chagin SIGUSR1, /* LINUX_SIGUSR1 */
1144ab7403bSDmitry Chagin SIGSEGV, /* LINUX_SIGSEGV */
1154ab7403bSDmitry Chagin SIGUSR2, /* LINUX_SIGUSR2 */
1164ab7403bSDmitry Chagin SIGPIPE, /* LINUX_SIGPIPE */
1174ab7403bSDmitry Chagin SIGALRM, /* LINUX_SIGALRM */
1184ab7403bSDmitry Chagin SIGTERM, /* LINUX_SIGTERM */
1194ab7403bSDmitry Chagin SIGBUS, /* LINUX_SIGSTKFLT */
1204ab7403bSDmitry Chagin SIGCHLD, /* LINUX_SIGCHLD */
1214ab7403bSDmitry Chagin SIGCONT, /* LINUX_SIGCONT */
1224ab7403bSDmitry Chagin SIGSTOP, /* LINUX_SIGSTOP */
1234ab7403bSDmitry Chagin SIGTSTP, /* LINUX_SIGTSTP */
1244ab7403bSDmitry Chagin SIGTTIN, /* LINUX_SIGTTIN */
1254ab7403bSDmitry Chagin SIGTTOU, /* LINUX_SIGTTOU */
1264ab7403bSDmitry Chagin SIGURG, /* LINUX_SIGURG */
1274ab7403bSDmitry Chagin SIGXCPU, /* LINUX_SIGXCPU */
1284ab7403bSDmitry Chagin SIGXFSZ, /* LINUX_SIGXFSZ */
1294ab7403bSDmitry Chagin SIGVTALRM, /* LINUX_SIGVTALARM */
1304ab7403bSDmitry Chagin SIGPROF, /* LINUX_SIGPROF */
1314ab7403bSDmitry Chagin SIGWINCH, /* LINUX_SIGWINCH */
1324ab7403bSDmitry Chagin SIGIO, /* LINUX_SIGIO */
1334ab7403bSDmitry Chagin /*
1344ab7403bSDmitry Chagin * FreeBSD does not have SIGPWR signal, map Linux SIGPWR signal
1354ab7403bSDmitry Chagin * to the first unused FreeBSD signal number. Since Linux supports
1364ab7403bSDmitry Chagin * signals from 1 to 64 we are ok here as our SIGRTMIN = 65.
1374ab7403bSDmitry Chagin */
1387a7cee55SDmitry Chagin LINUX_SIGPWREMU,/* LINUX_SIGPWR */
1394ab7403bSDmitry Chagin SIGSYS /* LINUX_SIGSYS */
1404ab7403bSDmitry Chagin };
1414ab7403bSDmitry Chagin
142044ab55eSEdward Tomasz Napierala static struct cdev *dev_shm_cdev;
143044ab55eSEdward Tomasz Napierala static struct cdevsw dev_shm_cdevsw = {
144044ab55eSEdward Tomasz Napierala .d_version = D_VERSION,
145044ab55eSEdward Tomasz Napierala .d_name = "dev_shm",
146044ab55eSEdward Tomasz Napierala };
147044ab55eSEdward Tomasz Napierala
1484ab7403bSDmitry Chagin /*
1494ab7403bSDmitry Chagin * Map Linux RT signals to the FreeBSD RT signals.
1504ab7403bSDmitry Chagin */
1514ab7403bSDmitry Chagin static inline int
linux_to_bsd_rt_signal(int sig)1524ab7403bSDmitry Chagin linux_to_bsd_rt_signal(int sig)
1534ab7403bSDmitry Chagin {
1544ab7403bSDmitry Chagin
1557a7cee55SDmitry Chagin return (SIGRTMIN + sig - LINUX_SIGRTMIN);
1564ab7403bSDmitry Chagin }
1574ab7403bSDmitry Chagin
1584ab7403bSDmitry Chagin static inline int
bsd_to_linux_rt_signal(int sig)1594ab7403bSDmitry Chagin bsd_to_linux_rt_signal(int sig)
1604ab7403bSDmitry Chagin {
1614ab7403bSDmitry Chagin
1627a7cee55SDmitry Chagin return (sig - SIGRTMIN + LINUX_SIGRTMIN);
1634ab7403bSDmitry Chagin }
1644ab7403bSDmitry Chagin
1654ab7403bSDmitry Chagin int
linux_to_bsd_signal(int sig)1664ab7403bSDmitry Chagin linux_to_bsd_signal(int sig)
1674ab7403bSDmitry Chagin {
1684ab7403bSDmitry Chagin
1698081c6ceSEdward Tomasz Napierala KASSERT(sig > 0 && sig <= LINUX_SIGRTMAX, ("invalid Linux signal %d\n", sig));
1704ab7403bSDmitry Chagin
1714ab7403bSDmitry Chagin if (sig < LINUX_SIGRTMIN)
1724ab7403bSDmitry Chagin return (linux_to_bsd_sigtbl[_SIG_IDX(sig)]);
1734ab7403bSDmitry Chagin
1744ab7403bSDmitry Chagin return (linux_to_bsd_rt_signal(sig));
1754ab7403bSDmitry Chagin }
1764ab7403bSDmitry Chagin
1774ab7403bSDmitry Chagin int
bsd_to_linux_signal(int sig)1784ab7403bSDmitry Chagin bsd_to_linux_signal(int sig)
1794ab7403bSDmitry Chagin {
1804ab7403bSDmitry Chagin
1814ab7403bSDmitry Chagin if (sig <= LINUX_SIGTBLSZ)
1824ab7403bSDmitry Chagin return (bsd_to_linux_sigtbl[_SIG_IDX(sig)]);
1837a7cee55SDmitry Chagin if (sig == LINUX_SIGPWREMU)
1844ab7403bSDmitry Chagin return (LINUX_SIGPWR);
1854ab7403bSDmitry Chagin
1864ab7403bSDmitry Chagin return (bsd_to_linux_rt_signal(sig));
1874ab7403bSDmitry Chagin }
1884ab7403bSDmitry Chagin
1894ab7403bSDmitry Chagin int
linux_to_bsd_sigaltstack(int lsa)1904ab7403bSDmitry Chagin linux_to_bsd_sigaltstack(int lsa)
1914ab7403bSDmitry Chagin {
1924ab7403bSDmitry Chagin int bsa = 0;
1934ab7403bSDmitry Chagin
1944ab7403bSDmitry Chagin if (lsa & LINUX_SS_DISABLE)
1954ab7403bSDmitry Chagin bsa |= SS_DISABLE;
1964ab7403bSDmitry Chagin /*
1974ab7403bSDmitry Chagin * Linux ignores SS_ONSTACK flag for ss
1984ab7403bSDmitry Chagin * parameter while FreeBSD prohibits it.
1994ab7403bSDmitry Chagin */
2004ab7403bSDmitry Chagin return (bsa);
2014ab7403bSDmitry Chagin }
2024ab7403bSDmitry Chagin
2034ab7403bSDmitry Chagin int
bsd_to_linux_sigaltstack(int bsa)2044ab7403bSDmitry Chagin bsd_to_linux_sigaltstack(int bsa)
2054ab7403bSDmitry Chagin {
2064ab7403bSDmitry Chagin int lsa = 0;
2074ab7403bSDmitry Chagin
2084ab7403bSDmitry Chagin if (bsa & SS_DISABLE)
2094ab7403bSDmitry Chagin lsa |= LINUX_SS_DISABLE;
2104ab7403bSDmitry Chagin if (bsa & SS_ONSTACK)
2114ab7403bSDmitry Chagin lsa |= LINUX_SS_ONSTACK;
2124ab7403bSDmitry Chagin return (lsa);
2134ab7403bSDmitry Chagin }
2144ab7403bSDmitry Chagin
2154ab7403bSDmitry Chagin void
linux_to_bsd_sigset(l_sigset_t * lss,sigset_t * bss)2164ab7403bSDmitry Chagin linux_to_bsd_sigset(l_sigset_t *lss, sigset_t *bss)
2174ab7403bSDmitry Chagin {
2184ab7403bSDmitry Chagin int b, l;
2194ab7403bSDmitry Chagin
2204ab7403bSDmitry Chagin SIGEMPTYSET(*bss);
2214ab7403bSDmitry Chagin for (l = 1; l <= LINUX_SIGRTMAX; l++) {
2224ab7403bSDmitry Chagin if (LINUX_SIGISMEMBER(*lss, l)) {
2234ab7403bSDmitry Chagin b = linux_to_bsd_signal(l);
2244ab7403bSDmitry Chagin if (b)
2254ab7403bSDmitry Chagin SIGADDSET(*bss, b);
2264ab7403bSDmitry Chagin }
2274ab7403bSDmitry Chagin }
2284ab7403bSDmitry Chagin }
2294ab7403bSDmitry Chagin
2304ab7403bSDmitry Chagin void
bsd_to_linux_sigset(sigset_t * bss,l_sigset_t * lss)2314ab7403bSDmitry Chagin bsd_to_linux_sigset(sigset_t *bss, l_sigset_t *lss)
2324ab7403bSDmitry Chagin {
2334ab7403bSDmitry Chagin int b, l;
2344ab7403bSDmitry Chagin
2354ab7403bSDmitry Chagin LINUX_SIGEMPTYSET(*lss);
2364ab7403bSDmitry Chagin for (b = 1; b <= SIGRTMAX; b++) {
2374ab7403bSDmitry Chagin if (SIGISMEMBER(*bss, b)) {
2384ab7403bSDmitry Chagin l = bsd_to_linux_signal(b);
2394ab7403bSDmitry Chagin if (l)
2404ab7403bSDmitry Chagin LINUX_SIGADDSET(*lss, l);
2414ab7403bSDmitry Chagin }
2424ab7403bSDmitry Chagin }
2434ab7403bSDmitry Chagin }
244d151344dSDmitry Chagin
245d151344dSDmitry Chagin /*
2463ab3c9c2SDmitry Chagin * Translate a FreeBSD interface name to a Linux interface name
2473ab3c9c2SDmitry Chagin * by interface name, and return the number of bytes copied to lxname.
2483ab3c9c2SDmitry Chagin */
2493ab3c9c2SDmitry Chagin int
ifname_bsd_to_linux_name(const char * bsdname,char * lxname,size_t len)2503ab3c9c2SDmitry Chagin ifname_bsd_to_linux_name(const char *bsdname, char *lxname, size_t len)
2513ab3c9c2SDmitry Chagin {
2523ab3c9c2SDmitry Chagin struct epoch_tracker et;
2533ab3c9c2SDmitry Chagin struct ifnet *ifp;
2543ab3c9c2SDmitry Chagin int ret;
2553ab3c9c2SDmitry Chagin
2562f5a315bSGleb Smirnoff CURVNET_ASSERT_SET();
2572f5a315bSGleb Smirnoff
2583ab3c9c2SDmitry Chagin ret = 0;
2593ab3c9c2SDmitry Chagin NET_EPOCH_ENTER(et);
2603ab3c9c2SDmitry Chagin ifp = ifunit(bsdname);
2613ab3c9c2SDmitry Chagin if (ifp != NULL)
2623ab3c9c2SDmitry Chagin ret = ifname_bsd_to_linux_ifp(ifp, lxname, len);
2633ab3c9c2SDmitry Chagin NET_EPOCH_EXIT(et);
2643ab3c9c2SDmitry Chagin return (ret);
2653ab3c9c2SDmitry Chagin }
2663ab3c9c2SDmitry Chagin
2673ab3c9c2SDmitry Chagin /*
2683ab3c9c2SDmitry Chagin * Translate a FreeBSD interface name to a Linux interface name
2693ab3c9c2SDmitry Chagin * by interface index, and return the number of bytes copied to lxname.
2703ab3c9c2SDmitry Chagin */
2713ab3c9c2SDmitry Chagin int
ifname_bsd_to_linux_idx(u_int idx,char * lxname,size_t len)2723ab3c9c2SDmitry Chagin ifname_bsd_to_linux_idx(u_int idx, char *lxname, size_t len)
2733ab3c9c2SDmitry Chagin {
2743ab3c9c2SDmitry Chagin struct epoch_tracker et;
2753ab3c9c2SDmitry Chagin struct ifnet *ifp;
2763ab3c9c2SDmitry Chagin int ret;
2773ab3c9c2SDmitry Chagin
2783ab3c9c2SDmitry Chagin ret = 0;
2793ab3c9c2SDmitry Chagin CURVNET_SET(TD_TO_VNET(curthread));
2803ab3c9c2SDmitry Chagin NET_EPOCH_ENTER(et);
2813ab3c9c2SDmitry Chagin ifp = ifnet_byindex(idx);
2823ab3c9c2SDmitry Chagin if (ifp != NULL)
2833ab3c9c2SDmitry Chagin ret = ifname_bsd_to_linux_ifp(ifp, lxname, len);
2843ab3c9c2SDmitry Chagin NET_EPOCH_EXIT(et);
2853ab3c9c2SDmitry Chagin CURVNET_RESTORE();
2863ab3c9c2SDmitry Chagin return (ret);
2873ab3c9c2SDmitry Chagin }
2883ab3c9c2SDmitry Chagin
2893ab3c9c2SDmitry Chagin /*
2903ab3c9c2SDmitry Chagin * Translate a FreeBSD interface name to a Linux interface name,
2916c5786fdSDmitry Chagin * and return the number of bytes copied to lxname, 0 if interface
2926c5786fdSDmitry Chagin * not found, -1 on error.
2933ab3c9c2SDmitry Chagin */
2946c5786fdSDmitry Chagin struct ifname_bsd_to_linux_ifp_cb_s {
2956c5786fdSDmitry Chagin struct ifnet *ifp;
2966c5786fdSDmitry Chagin int ethno;
2976c5786fdSDmitry Chagin char *lxname;
2986c5786fdSDmitry Chagin size_t len;
2996c5786fdSDmitry Chagin };
3006c5786fdSDmitry Chagin
3016c5786fdSDmitry Chagin static int
ifname_bsd_to_linux_ifp_cb(if_t ifp,void * arg)3026c5786fdSDmitry Chagin ifname_bsd_to_linux_ifp_cb(if_t ifp, void *arg)
3036c5786fdSDmitry Chagin {
3046c5786fdSDmitry Chagin struct ifname_bsd_to_linux_ifp_cb_s *cbs = arg;
3056c5786fdSDmitry Chagin
3066c5786fdSDmitry Chagin if (ifp == cbs->ifp)
3076c5786fdSDmitry Chagin return (snprintf(cbs->lxname, cbs->len, "eth%d", cbs->ethno));
3086c5786fdSDmitry Chagin if (IFP_IS_ETH(ifp))
3096c5786fdSDmitry Chagin cbs->ethno++;
3106c5786fdSDmitry Chagin return (0);
3116c5786fdSDmitry Chagin }
3126c5786fdSDmitry Chagin
3133ab3c9c2SDmitry Chagin int
ifname_bsd_to_linux_ifp(struct ifnet * ifp,char * lxname,size_t len)3143ab3c9c2SDmitry Chagin ifname_bsd_to_linux_ifp(struct ifnet *ifp, char *lxname, size_t len)
3153ab3c9c2SDmitry Chagin {
3166c5786fdSDmitry Chagin struct ifname_bsd_to_linux_ifp_cb_s arg = {
3176c5786fdSDmitry Chagin .ifp = ifp,
3186c5786fdSDmitry Chagin .ethno = 0,
3196c5786fdSDmitry Chagin .lxname = lxname,
3206c5786fdSDmitry Chagin .len = len,
3216c5786fdSDmitry Chagin };
3223ab3c9c2SDmitry Chagin
3233ab3c9c2SDmitry Chagin NET_EPOCH_ASSERT();
3243ab3c9c2SDmitry Chagin
3253ab3c9c2SDmitry Chagin /*
3263ab3c9c2SDmitry Chagin * Linux loopback interface name is lo (not lo0),
3273ab3c9c2SDmitry Chagin * we translate lo to lo0, loX to loX.
3283ab3c9c2SDmitry Chagin */
32956928500SJustin Hibbits if (IFP_IS_LOOP(ifp) && strncmp(if_name(ifp), "lo0", IFNAMSIZ) == 0)
3303ab3c9c2SDmitry Chagin return (strlcpy(lxname, "lo", len));
3313ab3c9c2SDmitry Chagin
3323ab3c9c2SDmitry Chagin /* Short-circuit non ethernet interfaces. */
3333ab3c9c2SDmitry Chagin if (!IFP_IS_ETH(ifp) || linux_use_real_ifname(ifp))
3346c5786fdSDmitry Chagin return (strlcpy(lxname, if_name(ifp), len));
3353ab3c9c2SDmitry Chagin
3363ab3c9c2SDmitry Chagin /* Determine the (relative) unit number for ethernet interfaces. */
3376c5786fdSDmitry Chagin return (if_foreach(ifname_bsd_to_linux_ifp_cb, &arg));
3383ab3c9c2SDmitry Chagin }
3393ab3c9c2SDmitry Chagin
3403ab3c9c2SDmitry Chagin /*
341d151344dSDmitry Chagin * Translate a Linux interface name to a FreeBSD interface name,
342d151344dSDmitry Chagin * and return the associated ifnet structure
343d151344dSDmitry Chagin * bsdname and lxname need to be least IFNAMSIZ bytes long, but
344d151344dSDmitry Chagin * can point to the same buffer.
345d151344dSDmitry Chagin */
3465d5b633dSDmitry Chagin struct ifname_linux_to_ifp_cb_s {
3476c5786fdSDmitry Chagin bool is_lo;
3486c5786fdSDmitry Chagin bool is_eth;
3496c5786fdSDmitry Chagin int ethno;
3506c5786fdSDmitry Chagin int unit;
3516c5786fdSDmitry Chagin const char *lxname;
3526c5786fdSDmitry Chagin if_t ifp;
3536c5786fdSDmitry Chagin };
3546c5786fdSDmitry Chagin
3556c5786fdSDmitry Chagin static int
ifname_linux_to_ifp_cb(if_t ifp,void * arg)3565d5b633dSDmitry Chagin ifname_linux_to_ifp_cb(if_t ifp, void *arg)
3576c5786fdSDmitry Chagin {
3585d5b633dSDmitry Chagin struct ifname_linux_to_ifp_cb_s *cbs = arg;
3596c5786fdSDmitry Chagin
3606c5786fdSDmitry Chagin NET_EPOCH_ASSERT();
3616c5786fdSDmitry Chagin
3626c5786fdSDmitry Chagin /*
3636c5786fdSDmitry Chagin * Allow Linux programs to use FreeBSD names. Don't presume
3646c5786fdSDmitry Chagin * we never have an interface named "eth", so don't make
3656c5786fdSDmitry Chagin * the test optional based on is_eth.
3666c5786fdSDmitry Chagin */
3676c5786fdSDmitry Chagin if (strncmp(if_name(ifp), cbs->lxname, LINUX_IFNAMSIZ) == 0)
3686c5786fdSDmitry Chagin goto out;
3696c5786fdSDmitry Chagin if (cbs->is_eth && IFP_IS_ETH(ifp) && cbs->unit == cbs->ethno)
3706c5786fdSDmitry Chagin goto out;
3716c5786fdSDmitry Chagin if (cbs->is_lo && IFP_IS_LOOP(ifp))
3726c5786fdSDmitry Chagin goto out;
3736c5786fdSDmitry Chagin if (IFP_IS_ETH(ifp))
3746c5786fdSDmitry Chagin cbs->ethno++;
3756c5786fdSDmitry Chagin return (0);
3766c5786fdSDmitry Chagin
3776c5786fdSDmitry Chagin out:
3786c5786fdSDmitry Chagin cbs->ifp = ifp;
3796c5786fdSDmitry Chagin return (1);
3806c5786fdSDmitry Chagin }
3816c5786fdSDmitry Chagin
382d151344dSDmitry Chagin struct ifnet *
ifname_linux_to_ifp(struct thread * td,const char * lxname)3835d5b633dSDmitry Chagin ifname_linux_to_ifp(struct thread *td, const char *lxname)
384d151344dSDmitry Chagin {
3855d5b633dSDmitry Chagin struct ifname_linux_to_ifp_cb_s arg = {
3866c5786fdSDmitry Chagin .ethno = 0,
3876c5786fdSDmitry Chagin .lxname = lxname,
3886c5786fdSDmitry Chagin .ifp = NULL,
3896c5786fdSDmitry Chagin };
3905d5b633dSDmitry Chagin int len;
391d151344dSDmitry Chagin char *ep;
392d151344dSDmitry Chagin
3935d5b633dSDmitry Chagin NET_EPOCH_ASSERT();
3945d5b633dSDmitry Chagin
395d151344dSDmitry Chagin for (len = 0; len < LINUX_IFNAMSIZ; ++len)
3969c1437aeSDmitry Chagin if (!isalpha(lxname[len]) || lxname[len] == '\0')
397d151344dSDmitry Chagin break;
398d151344dSDmitry Chagin if (len == 0 || len == LINUX_IFNAMSIZ)
399d151344dSDmitry Chagin return (NULL);
4003ab3c9c2SDmitry Chagin /*
4013ab3c9c2SDmitry Chagin * Linux loopback interface name is lo (not lo0),
4023ab3c9c2SDmitry Chagin * we translate lo to lo0, loX to loX.
4033ab3c9c2SDmitry Chagin */
4046c5786fdSDmitry Chagin arg.is_lo = (len == 2 && strncmp(lxname, "lo", LINUX_IFNAMSIZ) == 0);
4056c5786fdSDmitry Chagin arg.unit = (int)strtoul(lxname + len, &ep, 10);
406d151344dSDmitry Chagin if ((ep == NULL || ep == lxname + len || ep >= lxname + LINUX_IFNAMSIZ) &&
4076c5786fdSDmitry Chagin arg.is_lo == 0)
408d151344dSDmitry Chagin return (NULL);
4096c5786fdSDmitry Chagin arg.is_eth = (len == 3 && strncmp(lxname, "eth", len) == 0);
410d151344dSDmitry Chagin
4115d5b633dSDmitry Chagin if_foreach(ifname_linux_to_ifp_cb, &arg);
4125d5b633dSDmitry Chagin return (arg.ifp);
4135d5b633dSDmitry Chagin }
4145d5b633dSDmitry Chagin
415f9b0675bSDmitry Chagin int
ifname_linux_to_bsd(struct thread * td,const char * lxname,char * bsdname)4165d5b633dSDmitry Chagin ifname_linux_to_bsd(struct thread *td, const char *lxname, char *bsdname)
4175d5b633dSDmitry Chagin {
4185d5b633dSDmitry Chagin struct epoch_tracker et;
4195d5b633dSDmitry Chagin struct ifnet *ifp;
4205d5b633dSDmitry Chagin
421d151344dSDmitry Chagin CURVNET_SET(TD_TO_VNET(td));
4226c5786fdSDmitry Chagin NET_EPOCH_ENTER(et);
4235d5b633dSDmitry Chagin ifp = ifname_linux_to_ifp(td, lxname);
4245d5b633dSDmitry Chagin if (ifp != NULL && bsdname != NULL)
4255d5b633dSDmitry Chagin strlcpy(bsdname, if_name(ifp), IFNAMSIZ);
4266c5786fdSDmitry Chagin NET_EPOCH_EXIT(et);
427d151344dSDmitry Chagin CURVNET_RESTORE();
428f9b0675bSDmitry Chagin return (ifp != NULL ? 0 : EINVAL);
429d151344dSDmitry Chagin }
430d151344dSDmitry Chagin
431130383f2SDmitry Chagin unsigned short
linux_ifflags(struct ifnet * ifp)432130383f2SDmitry Chagin linux_ifflags(struct ifnet *ifp)
433d151344dSDmitry Chagin {
4345c32488dSDmitry Chagin unsigned short flags;
435d151344dSDmitry Chagin
4368243e174SDmitry Chagin NET_EPOCH_ASSERT();
4378243e174SDmitry Chagin
4385c32488dSDmitry Chagin flags = if_getflags(ifp) | if_getdrvflags(ifp);
4395c32488dSDmitry Chagin return (bsd_to_linux_ifflags(flags));
4405c32488dSDmitry Chagin }
4415c32488dSDmitry Chagin
4425c32488dSDmitry Chagin unsigned short
bsd_to_linux_ifflags(int fl)4435c32488dSDmitry Chagin bsd_to_linux_ifflags(int fl)
4445c32488dSDmitry Chagin {
4455c32488dSDmitry Chagin unsigned short flags = 0;
4465c32488dSDmitry Chagin
447bbac65c7SDmitry Chagin if (fl & IFF_UP)
448130383f2SDmitry Chagin flags |= LINUX_IFF_UP;
449bbac65c7SDmitry Chagin if (fl & IFF_BROADCAST)
450130383f2SDmitry Chagin flags |= LINUX_IFF_BROADCAST;
451bbac65c7SDmitry Chagin if (fl & IFF_DEBUG)
452130383f2SDmitry Chagin flags |= LINUX_IFF_DEBUG;
453bbac65c7SDmitry Chagin if (fl & IFF_LOOPBACK)
454130383f2SDmitry Chagin flags |= LINUX_IFF_LOOPBACK;
455bbac65c7SDmitry Chagin if (fl & IFF_POINTOPOINT)
456130383f2SDmitry Chagin flags |= LINUX_IFF_POINTOPOINT;
457bbac65c7SDmitry Chagin if (fl & IFF_DRV_RUNNING)
458130383f2SDmitry Chagin flags |= LINUX_IFF_RUNNING;
459bbac65c7SDmitry Chagin if (fl & IFF_NOARP)
460130383f2SDmitry Chagin flags |= LINUX_IFF_NOARP;
461bbac65c7SDmitry Chagin if (fl & IFF_PROMISC)
462130383f2SDmitry Chagin flags |= LINUX_IFF_PROMISC;
463bbac65c7SDmitry Chagin if (fl & IFF_ALLMULTI)
464130383f2SDmitry Chagin flags |= LINUX_IFF_ALLMULTI;
465bbac65c7SDmitry Chagin if (fl & IFF_MULTICAST)
466130383f2SDmitry Chagin flags |= LINUX_IFF_MULTICAST;
467130383f2SDmitry Chagin return (flags);
468d151344dSDmitry Chagin }
469d151344dSDmitry Chagin
4706c5786fdSDmitry Chagin static u_int
linux_ifhwaddr_cb(void * arg,struct ifaddr * ifa,u_int count)4716c5786fdSDmitry Chagin linux_ifhwaddr_cb(void *arg, struct ifaddr *ifa, u_int count)
4726c5786fdSDmitry Chagin {
4736c5786fdSDmitry Chagin struct sockaddr_dl *sdl = (struct sockaddr_dl *)ifa->ifa_addr;
4746c5786fdSDmitry Chagin struct l_sockaddr *lsa = arg;
4756c5786fdSDmitry Chagin
4766c5786fdSDmitry Chagin if (count > 0)
4776c5786fdSDmitry Chagin return (0);
4786c5786fdSDmitry Chagin if (sdl->sdl_type != IFT_ETHER)
4796c5786fdSDmitry Chagin return (0);
4806c5786fdSDmitry Chagin bzero(lsa, sizeof(*lsa));
4816c5786fdSDmitry Chagin lsa->sa_family = LINUX_ARPHRD_ETHER;
4826c5786fdSDmitry Chagin bcopy(LLADDR(sdl), lsa->sa_data, LINUX_IFHWADDRLEN);
4836c5786fdSDmitry Chagin return (1);
4846c5786fdSDmitry Chagin }
4856c5786fdSDmitry Chagin
486d151344dSDmitry Chagin int
linux_ifhwaddr(struct ifnet * ifp,struct l_sockaddr * lsa)487d151344dSDmitry Chagin linux_ifhwaddr(struct ifnet *ifp, struct l_sockaddr *lsa)
488d151344dSDmitry Chagin {
489d151344dSDmitry Chagin
4908d6dd96dSDmitry Chagin NET_EPOCH_ASSERT();
4918d6dd96dSDmitry Chagin
492d151344dSDmitry Chagin if (IFP_IS_LOOP(ifp)) {
493d151344dSDmitry Chagin bzero(lsa, sizeof(*lsa));
494d151344dSDmitry Chagin lsa->sa_family = LINUX_ARPHRD_LOOPBACK;
495d151344dSDmitry Chagin return (0);
496d151344dSDmitry Chagin }
497d151344dSDmitry Chagin if (!IFP_IS_ETH(ifp))
498d151344dSDmitry Chagin return (ENOENT);
4996c5786fdSDmitry Chagin if (if_foreach_addr_type(ifp, AF_LINK, linux_ifhwaddr_cb, lsa) > 0)
500d151344dSDmitry Chagin return (0);
501d151344dSDmitry Chagin return (ENOENT);
502d151344dSDmitry Chagin }
503d5368bf3SDmitry Chagin
504*9d4a08d1SGleb Smirnoff sa_family_t
linux_to_bsd_domain(sa_family_t domain)505*9d4a08d1SGleb Smirnoff linux_to_bsd_domain(sa_family_t domain)
506d5368bf3SDmitry Chagin {
507d5368bf3SDmitry Chagin
508d5368bf3SDmitry Chagin switch (domain) {
509d5368bf3SDmitry Chagin case LINUX_AF_UNSPEC:
510d5368bf3SDmitry Chagin return (AF_UNSPEC);
511d5368bf3SDmitry Chagin case LINUX_AF_UNIX:
512d5368bf3SDmitry Chagin return (AF_LOCAL);
513d5368bf3SDmitry Chagin case LINUX_AF_INET:
514d5368bf3SDmitry Chagin return (AF_INET);
515d5368bf3SDmitry Chagin case LINUX_AF_INET6:
516d5368bf3SDmitry Chagin return (AF_INET6);
517d5368bf3SDmitry Chagin case LINUX_AF_AX25:
518d5368bf3SDmitry Chagin return (AF_CCITT);
519d5368bf3SDmitry Chagin case LINUX_AF_IPX:
520d5368bf3SDmitry Chagin return (AF_IPX);
521d5368bf3SDmitry Chagin case LINUX_AF_APPLETALK:
522d5368bf3SDmitry Chagin return (AF_APPLETALK);
5237c40e2d5SAlexander V. Chernikov case LINUX_AF_NETLINK:
5247c40e2d5SAlexander V. Chernikov return (AF_NETLINK);
525d5368bf3SDmitry Chagin }
526*9d4a08d1SGleb Smirnoff return (AF_UNKNOWN);
527d5368bf3SDmitry Chagin }
528d5368bf3SDmitry Chagin
529*9d4a08d1SGleb Smirnoff sa_family_t
bsd_to_linux_domain(sa_family_t domain)530*9d4a08d1SGleb Smirnoff bsd_to_linux_domain(sa_family_t domain)
531d5368bf3SDmitry Chagin {
532d5368bf3SDmitry Chagin
533d5368bf3SDmitry Chagin switch (domain) {
534d5368bf3SDmitry Chagin case AF_UNSPEC:
535d5368bf3SDmitry Chagin return (LINUX_AF_UNSPEC);
536d5368bf3SDmitry Chagin case AF_LOCAL:
537d5368bf3SDmitry Chagin return (LINUX_AF_UNIX);
538d5368bf3SDmitry Chagin case AF_INET:
539d5368bf3SDmitry Chagin return (LINUX_AF_INET);
540d5368bf3SDmitry Chagin case AF_INET6:
541d5368bf3SDmitry Chagin return (LINUX_AF_INET6);
542d5368bf3SDmitry Chagin case AF_CCITT:
543d5368bf3SDmitry Chagin return (LINUX_AF_AX25);
544d5368bf3SDmitry Chagin case AF_IPX:
545d5368bf3SDmitry Chagin return (LINUX_AF_IPX);
546d5368bf3SDmitry Chagin case AF_APPLETALK:
547d5368bf3SDmitry Chagin return (LINUX_AF_APPLETALK);
5487c40e2d5SAlexander V. Chernikov case AF_NETLINK:
5497c40e2d5SAlexander V. Chernikov return (LINUX_AF_NETLINK);
550d5368bf3SDmitry Chagin }
551*9d4a08d1SGleb Smirnoff return (AF_UNKNOWN);
552d5368bf3SDmitry Chagin }
553d5368bf3SDmitry Chagin
554d5368bf3SDmitry Chagin /*
555d5368bf3SDmitry Chagin * Based on the fact that:
556d5368bf3SDmitry Chagin * 1. Native and Linux storage of struct sockaddr
557d5368bf3SDmitry Chagin * and struct sockaddr_in6 are equal.
558d5368bf3SDmitry Chagin * 2. On Linux sa_family is the first member of all struct sockaddr.
559d5368bf3SDmitry Chagin */
560d5368bf3SDmitry Chagin int
bsd_to_linux_sockaddr(const struct sockaddr * sa,struct l_sockaddr ** lsa,socklen_t len)561d5368bf3SDmitry Chagin bsd_to_linux_sockaddr(const struct sockaddr *sa, struct l_sockaddr **lsa,
562d5368bf3SDmitry Chagin socklen_t len)
563d5368bf3SDmitry Chagin {
564d5368bf3SDmitry Chagin struct l_sockaddr *kosa;
565*9d4a08d1SGleb Smirnoff sa_family_t bdom;
566d5368bf3SDmitry Chagin
567d5368bf3SDmitry Chagin *lsa = NULL;
568d5368bf3SDmitry Chagin if (len < 2 || len > UCHAR_MAX)
569d5368bf3SDmitry Chagin return (EINVAL);
570bbddd588SDmitry Chagin bdom = bsd_to_linux_domain(sa->sa_family);
571*9d4a08d1SGleb Smirnoff if (bdom == AF_UNKNOWN)
572bbddd588SDmitry Chagin return (EAFNOSUPPORT);
573d5368bf3SDmitry Chagin
57463355839SDmitry Chagin kosa = malloc(len, M_LINUX, M_WAITOK);
575d5368bf3SDmitry Chagin bcopy(sa, kosa, len);
576d5368bf3SDmitry Chagin kosa->sa_family = bdom;
577d5368bf3SDmitry Chagin *lsa = kosa;
578d5368bf3SDmitry Chagin return (0);
579d5368bf3SDmitry Chagin }
580d5368bf3SDmitry Chagin
581d5368bf3SDmitry Chagin int
linux_to_bsd_sockaddr(const struct l_sockaddr * osa,struct sockaddr ** sap,socklen_t * len)582d5368bf3SDmitry Chagin linux_to_bsd_sockaddr(const struct l_sockaddr *osa, struct sockaddr **sap,
583d5368bf3SDmitry Chagin socklen_t *len)
584d5368bf3SDmitry Chagin {
585d5368bf3SDmitry Chagin struct sockaddr *sa;
586d5368bf3SDmitry Chagin struct l_sockaddr *kosa;
587d5368bf3SDmitry Chagin #ifdef INET6
588d5368bf3SDmitry Chagin struct sockaddr_in6 *sin6;
589d5368bf3SDmitry Chagin bool oldv6size;
590d5368bf3SDmitry Chagin #endif
591d5368bf3SDmitry Chagin char *name;
592d5368bf3SDmitry Chagin int salen, bdom, error, hdrlen, namelen;
593d5368bf3SDmitry Chagin
594d5368bf3SDmitry Chagin if (*len < 2 || *len > UCHAR_MAX)
595d5368bf3SDmitry Chagin return (EINVAL);
596d5368bf3SDmitry Chagin
597d5368bf3SDmitry Chagin salen = *len;
598d5368bf3SDmitry Chagin
599d5368bf3SDmitry Chagin #ifdef INET6
600d5368bf3SDmitry Chagin oldv6size = false;
601d5368bf3SDmitry Chagin /*
602d5368bf3SDmitry Chagin * Check for old (pre-RFC2553) sockaddr_in6. We may accept it
603d5368bf3SDmitry Chagin * if it's a v4-mapped address, so reserve the proper space
604d5368bf3SDmitry Chagin * for it.
605d5368bf3SDmitry Chagin */
606d5368bf3SDmitry Chagin if (salen == sizeof(struct sockaddr_in6) - sizeof(uint32_t)) {
607d5368bf3SDmitry Chagin salen += sizeof(uint32_t);
608d5368bf3SDmitry Chagin oldv6size = true;
609d5368bf3SDmitry Chagin }
610d5368bf3SDmitry Chagin #endif
611d5368bf3SDmitry Chagin
612d5368bf3SDmitry Chagin kosa = malloc(salen, M_SONAME, M_WAITOK);
613d5368bf3SDmitry Chagin
614d5368bf3SDmitry Chagin if ((error = copyin(osa, kosa, *len)))
615d5368bf3SDmitry Chagin goto out;
616d5368bf3SDmitry Chagin
617d5368bf3SDmitry Chagin bdom = linux_to_bsd_domain(kosa->sa_family);
618*9d4a08d1SGleb Smirnoff if (bdom == AF_UNKNOWN) {
619d5368bf3SDmitry Chagin error = EAFNOSUPPORT;
620d5368bf3SDmitry Chagin goto out;
621d5368bf3SDmitry Chagin }
622d5368bf3SDmitry Chagin
623d5368bf3SDmitry Chagin #ifdef INET6
624d5368bf3SDmitry Chagin /*
625d5368bf3SDmitry Chagin * Older Linux IPv6 code uses obsolete RFC2133 struct sockaddr_in6,
626d5368bf3SDmitry Chagin * which lacks the scope id compared with RFC2553 one. If we detect
627d5368bf3SDmitry Chagin * the situation, reject the address and write a message to system log.
628d5368bf3SDmitry Chagin *
629d5368bf3SDmitry Chagin * Still accept addresses for which the scope id is not used.
630d5368bf3SDmitry Chagin */
631d5368bf3SDmitry Chagin if (oldv6size) {
632d5368bf3SDmitry Chagin if (bdom == AF_INET6) {
633d5368bf3SDmitry Chagin sin6 = (struct sockaddr_in6 *)kosa;
634d5368bf3SDmitry Chagin if (IN6_IS_ADDR_V4MAPPED(&sin6->sin6_addr) ||
635d5368bf3SDmitry Chagin (!IN6_IS_ADDR_LINKLOCAL(&sin6->sin6_addr) &&
636d5368bf3SDmitry Chagin !IN6_IS_ADDR_SITELOCAL(&sin6->sin6_addr) &&
637d5368bf3SDmitry Chagin !IN6_IS_ADDR_V4COMPAT(&sin6->sin6_addr) &&
638d5368bf3SDmitry Chagin !IN6_IS_ADDR_UNSPECIFIED(&sin6->sin6_addr) &&
639d5368bf3SDmitry Chagin !IN6_IS_ADDR_MULTICAST(&sin6->sin6_addr))) {
640d5368bf3SDmitry Chagin sin6->sin6_scope_id = 0;
641d5368bf3SDmitry Chagin } else {
642d5368bf3SDmitry Chagin linux_msg(curthread,
64386e794ebSEdward Tomasz Napierala "obsolete pre-RFC2553 sockaddr_in6 rejected");
644d5368bf3SDmitry Chagin error = EINVAL;
645d5368bf3SDmitry Chagin goto out;
646d5368bf3SDmitry Chagin }
647d5368bf3SDmitry Chagin } else
648d5368bf3SDmitry Chagin salen -= sizeof(uint32_t);
649d5368bf3SDmitry Chagin }
650d5368bf3SDmitry Chagin #endif
651d5368bf3SDmitry Chagin if (bdom == AF_INET) {
652d5368bf3SDmitry Chagin if (salen < sizeof(struct sockaddr_in)) {
653d5368bf3SDmitry Chagin error = EINVAL;
654d5368bf3SDmitry Chagin goto out;
655d5368bf3SDmitry Chagin }
656d5368bf3SDmitry Chagin salen = sizeof(struct sockaddr_in);
657d5368bf3SDmitry Chagin }
658d5368bf3SDmitry Chagin
659d5368bf3SDmitry Chagin if (bdom == AF_LOCAL && salen > sizeof(struct sockaddr_un)) {
660d5368bf3SDmitry Chagin hdrlen = offsetof(struct sockaddr_un, sun_path);
661d5368bf3SDmitry Chagin name = ((struct sockaddr_un *)kosa)->sun_path;
662d5368bf3SDmitry Chagin if (*name == '\0') {
663d5368bf3SDmitry Chagin /*
664d5368bf3SDmitry Chagin * Linux abstract namespace starts with a NULL byte.
665d5368bf3SDmitry Chagin * XXX We do not support abstract namespace yet.
666d5368bf3SDmitry Chagin */
667d5368bf3SDmitry Chagin namelen = strnlen(name + 1, salen - hdrlen - 1) + 1;
668d5368bf3SDmitry Chagin } else
669d5368bf3SDmitry Chagin namelen = strnlen(name, salen - hdrlen);
670d5368bf3SDmitry Chagin salen = hdrlen + namelen;
671d5368bf3SDmitry Chagin if (salen > sizeof(struct sockaddr_un)) {
672d5368bf3SDmitry Chagin error = ENAMETOOLONG;
673d5368bf3SDmitry Chagin goto out;
674d5368bf3SDmitry Chagin }
675d5368bf3SDmitry Chagin }
676d5368bf3SDmitry Chagin
6777c40e2d5SAlexander V. Chernikov if (bdom == AF_NETLINK) {
6787c40e2d5SAlexander V. Chernikov if (salen < sizeof(struct sockaddr_nl)) {
6797c40e2d5SAlexander V. Chernikov error = EINVAL;
6807c40e2d5SAlexander V. Chernikov goto out;
6817c40e2d5SAlexander V. Chernikov }
6827c40e2d5SAlexander V. Chernikov salen = sizeof(struct sockaddr_nl);
6837c40e2d5SAlexander V. Chernikov }
6847c40e2d5SAlexander V. Chernikov
685d5368bf3SDmitry Chagin sa = (struct sockaddr *)kosa;
686d5368bf3SDmitry Chagin sa->sa_family = bdom;
687d5368bf3SDmitry Chagin sa->sa_len = salen;
688d5368bf3SDmitry Chagin
689d5368bf3SDmitry Chagin *sap = sa;
690d5368bf3SDmitry Chagin *len = salen;
691d5368bf3SDmitry Chagin return (0);
692d5368bf3SDmitry Chagin
693d5368bf3SDmitry Chagin out:
694d5368bf3SDmitry Chagin free(kosa, M_SONAME);
695d5368bf3SDmitry Chagin return (error);
696d5368bf3SDmitry Chagin }
697044ab55eSEdward Tomasz Napierala
698044ab55eSEdward Tomasz Napierala void
linux_dev_shm_create(void)699044ab55eSEdward Tomasz Napierala linux_dev_shm_create(void)
700044ab55eSEdward Tomasz Napierala {
701044ab55eSEdward Tomasz Napierala int error;
702044ab55eSEdward Tomasz Napierala
703044ab55eSEdward Tomasz Napierala error = make_dev_p(MAKEDEV_CHECKNAME | MAKEDEV_WAITOK, &dev_shm_cdev,
704044ab55eSEdward Tomasz Napierala &dev_shm_cdevsw, NULL, UID_ROOT, GID_WHEEL, 0, "shm/.mountpoint");
705044ab55eSEdward Tomasz Napierala if (error != 0) {
706044ab55eSEdward Tomasz Napierala printf("%s: failed to create device node, error %d\n",
707044ab55eSEdward Tomasz Napierala __func__, error);
708044ab55eSEdward Tomasz Napierala }
709044ab55eSEdward Tomasz Napierala }
710044ab55eSEdward Tomasz Napierala
711044ab55eSEdward Tomasz Napierala void
linux_dev_shm_destroy(void)712044ab55eSEdward Tomasz Napierala linux_dev_shm_destroy(void)
713044ab55eSEdward Tomasz Napierala {
714044ab55eSEdward Tomasz Napierala
715044ab55eSEdward Tomasz Napierala destroy_dev(dev_shm_cdev);
716044ab55eSEdward Tomasz Napierala }
7175403f186SKyle Evans
7185403f186SKyle Evans int
bsd_to_linux_bits_(int value,struct bsd_to_linux_bitmap * bitmap,size_t mapcnt,int no_value)7195403f186SKyle Evans bsd_to_linux_bits_(int value, struct bsd_to_linux_bitmap *bitmap,
7205403f186SKyle Evans size_t mapcnt, int no_value)
7215403f186SKyle Evans {
7225403f186SKyle Evans int bsd_mask, bsd_value, linux_mask, linux_value;
7235403f186SKyle Evans int linux_ret;
7245403f186SKyle Evans size_t i;
7255403f186SKyle Evans bool applied;
7265403f186SKyle Evans
7275403f186SKyle Evans applied = false;
7285403f186SKyle Evans linux_ret = 0;
7295403f186SKyle Evans for (i = 0; i < mapcnt; ++i) {
7305403f186SKyle Evans bsd_mask = bitmap[i].bsd_mask;
7315403f186SKyle Evans bsd_value = bitmap[i].bsd_value;
7325403f186SKyle Evans if (bsd_mask == 0)
7335403f186SKyle Evans bsd_mask = bsd_value;
7345403f186SKyle Evans
7355403f186SKyle Evans linux_mask = bitmap[i].linux_mask;
7365403f186SKyle Evans linux_value = bitmap[i].linux_value;
7375403f186SKyle Evans if (linux_mask == 0)
7385403f186SKyle Evans linux_mask = linux_value;
7395403f186SKyle Evans
7405403f186SKyle Evans /*
7415403f186SKyle Evans * If a mask larger than just the value is set, we explicitly
7425403f186SKyle Evans * want to make sure that only this bit we mapped within that
7435403f186SKyle Evans * mask is set.
7445403f186SKyle Evans */
7455403f186SKyle Evans if ((value & bsd_mask) == bsd_value) {
7465403f186SKyle Evans linux_ret = (linux_ret & ~linux_mask) | linux_value;
7475403f186SKyle Evans applied = true;
7485403f186SKyle Evans }
7495403f186SKyle Evans }
7505403f186SKyle Evans
7515403f186SKyle Evans if (!applied)
7525403f186SKyle Evans return (no_value);
7535403f186SKyle Evans return (linux_ret);
7545403f186SKyle Evans }
7555403f186SKyle Evans
7565403f186SKyle Evans int
linux_to_bsd_bits_(int value,struct bsd_to_linux_bitmap * bitmap,size_t mapcnt,int no_value)7575403f186SKyle Evans linux_to_bsd_bits_(int value, struct bsd_to_linux_bitmap *bitmap,
7585403f186SKyle Evans size_t mapcnt, int no_value)
7595403f186SKyle Evans {
7605403f186SKyle Evans int bsd_mask, bsd_value, linux_mask, linux_value;
7615403f186SKyle Evans int bsd_ret;
7625403f186SKyle Evans size_t i;
7635403f186SKyle Evans bool applied;
7645403f186SKyle Evans
7655403f186SKyle Evans applied = false;
7665403f186SKyle Evans bsd_ret = 0;
7675403f186SKyle Evans for (i = 0; i < mapcnt; ++i) {
7685403f186SKyle Evans bsd_mask = bitmap[i].bsd_mask;
7695403f186SKyle Evans bsd_value = bitmap[i].bsd_value;
7705403f186SKyle Evans if (bsd_mask == 0)
7715403f186SKyle Evans bsd_mask = bsd_value;
7725403f186SKyle Evans
7735403f186SKyle Evans linux_mask = bitmap[i].linux_mask;
7745403f186SKyle Evans linux_value = bitmap[i].linux_value;
7755403f186SKyle Evans if (linux_mask == 0)
7765403f186SKyle Evans linux_mask = linux_value;
7775403f186SKyle Evans
7785403f186SKyle Evans /*
7795403f186SKyle Evans * If a mask larger than just the value is set, we explicitly
7805403f186SKyle Evans * want to make sure that only this bit we mapped within that
7815403f186SKyle Evans * mask is set.
7825403f186SKyle Evans */
7835403f186SKyle Evans if ((value & linux_mask) == linux_value) {
7845403f186SKyle Evans bsd_ret = (bsd_ret & ~bsd_mask) | bsd_value;
7855403f186SKyle Evans applied = true;
7865403f186SKyle Evans }
7875403f186SKyle Evans }
7885403f186SKyle Evans
7895403f186SKyle Evans if (!applied)
7905403f186SKyle Evans return (no_value);
7915403f186SKyle Evans return (bsd_ret);
7925403f186SKyle Evans }
79326795a03SDmitry Chagin
79426795a03SDmitry Chagin void
linux_to_bsd_poll_events(struct thread * td,int fd,short lev,short * bev)79526795a03SDmitry Chagin linux_to_bsd_poll_events(struct thread *td, int fd, short lev,
79626795a03SDmitry Chagin short *bev)
79726795a03SDmitry Chagin {
79826795a03SDmitry Chagin struct file *fp;
79926795a03SDmitry Chagin int error;
80026795a03SDmitry Chagin short bits = 0;
80126795a03SDmitry Chagin
80226795a03SDmitry Chagin if (lev & LINUX_POLLIN)
80326795a03SDmitry Chagin bits |= POLLIN;
80426795a03SDmitry Chagin if (lev & LINUX_POLLPRI)
80526795a03SDmitry Chagin bits |= POLLPRI;
80626795a03SDmitry Chagin if (lev & LINUX_POLLOUT)
80726795a03SDmitry Chagin bits |= POLLOUT;
80826795a03SDmitry Chagin if (lev & LINUX_POLLERR)
80926795a03SDmitry Chagin bits |= POLLERR;
81026795a03SDmitry Chagin if (lev & LINUX_POLLHUP)
81126795a03SDmitry Chagin bits |= POLLHUP;
81226795a03SDmitry Chagin if (lev & LINUX_POLLNVAL)
81326795a03SDmitry Chagin bits |= POLLNVAL;
81426795a03SDmitry Chagin if (lev & LINUX_POLLRDNORM)
81526795a03SDmitry Chagin bits |= POLLRDNORM;
81626795a03SDmitry Chagin if (lev & LINUX_POLLRDBAND)
81726795a03SDmitry Chagin bits |= POLLRDBAND;
81826795a03SDmitry Chagin if (lev & LINUX_POLLWRBAND)
81926795a03SDmitry Chagin bits |= POLLWRBAND;
82026795a03SDmitry Chagin if (lev & LINUX_POLLWRNORM)
82126795a03SDmitry Chagin bits |= POLLWRNORM;
82226795a03SDmitry Chagin
82326795a03SDmitry Chagin if (lev & LINUX_POLLRDHUP) {
82426795a03SDmitry Chagin /*
82526795a03SDmitry Chagin * It seems that the Linux silencly ignores POLLRDHUP
82626795a03SDmitry Chagin * on non-socket file descriptors unlike FreeBSD, where
82726795a03SDmitry Chagin * events bits is more strictly checked (POLLSTANDARD).
82826795a03SDmitry Chagin */
829513c7a6eSMateusz Guzik error = fget_unlocked(td, fd, &cap_no_rights, &fp);
83026795a03SDmitry Chagin if (error == 0) {
83126795a03SDmitry Chagin /*
83226795a03SDmitry Chagin * XXX. On FreeBSD POLLRDHUP applies only to
83326795a03SDmitry Chagin * stream sockets.
83426795a03SDmitry Chagin */
83526795a03SDmitry Chagin if (fp->f_type == DTYPE_SOCKET)
83626795a03SDmitry Chagin bits |= POLLRDHUP;
83726795a03SDmitry Chagin fdrop(fp, td);
83826795a03SDmitry Chagin }
83926795a03SDmitry Chagin }
84026795a03SDmitry Chagin
84126795a03SDmitry Chagin if (lev & LINUX_POLLMSG)
84226795a03SDmitry Chagin LINUX_RATELIMIT_MSG_OPT1("unsupported POLLMSG, events(%d)", lev);
84326795a03SDmitry Chagin if (lev & LINUX_POLLREMOVE)
84426795a03SDmitry Chagin LINUX_RATELIMIT_MSG_OPT1("unsupported POLLREMOVE, events(%d)", lev);
84526795a03SDmitry Chagin
84626795a03SDmitry Chagin *bev = bits;
84726795a03SDmitry Chagin }
84826795a03SDmitry Chagin
84926795a03SDmitry Chagin void
bsd_to_linux_poll_events(short bev,short * lev)85026795a03SDmitry Chagin bsd_to_linux_poll_events(short bev, short *lev)
85126795a03SDmitry Chagin {
85226795a03SDmitry Chagin short bits = 0;
85326795a03SDmitry Chagin
85426795a03SDmitry Chagin if (bev & POLLIN)
85526795a03SDmitry Chagin bits |= LINUX_POLLIN;
85626795a03SDmitry Chagin if (bev & POLLPRI)
85726795a03SDmitry Chagin bits |= LINUX_POLLPRI;
85826795a03SDmitry Chagin if (bev & (POLLOUT | POLLWRNORM))
85926795a03SDmitry Chagin /*
86026795a03SDmitry Chagin * POLLWRNORM is equal to POLLOUT on FreeBSD,
86126795a03SDmitry Chagin * but not on Linux
86226795a03SDmitry Chagin */
86326795a03SDmitry Chagin bits |= LINUX_POLLOUT;
86426795a03SDmitry Chagin if (bev & POLLERR)
86526795a03SDmitry Chagin bits |= LINUX_POLLERR;
86626795a03SDmitry Chagin if (bev & POLLHUP)
86726795a03SDmitry Chagin bits |= LINUX_POLLHUP;
86826795a03SDmitry Chagin if (bev & POLLNVAL)
86926795a03SDmitry Chagin bits |= LINUX_POLLNVAL;
87026795a03SDmitry Chagin if (bev & POLLRDNORM)
87126795a03SDmitry Chagin bits |= LINUX_POLLRDNORM;
87226795a03SDmitry Chagin if (bev & POLLRDBAND)
87326795a03SDmitry Chagin bits |= LINUX_POLLRDBAND;
87426795a03SDmitry Chagin if (bev & POLLWRBAND)
87526795a03SDmitry Chagin bits |= LINUX_POLLWRBAND;
87626795a03SDmitry Chagin if (bev & POLLRDHUP)
87726795a03SDmitry Chagin bits |= LINUX_POLLRDHUP;
87826795a03SDmitry Chagin
87926795a03SDmitry Chagin *lev = bits;
88026795a03SDmitry Chagin }
88132fdc75fSDmitry Chagin
88232fdc75fSDmitry Chagin bool
linux_use_real_ifname(const struct ifnet * ifp)88332fdc75fSDmitry Chagin linux_use_real_ifname(const struct ifnet *ifp)
88432fdc75fSDmitry Chagin {
88532fdc75fSDmitry Chagin
8863ab3c9c2SDmitry Chagin return (use_real_ifnames);
88732fdc75fSDmitry Chagin }
888