xref: /freebsd/sys/compat/linux/linux.c (revision f9b0675b014d06995e5337b71129fc94bd1cedd0)
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 /*
2453ab3c9c2SDmitry Chagin  * Translate a FreeBSD interface name to a Linux interface name
2463ab3c9c2SDmitry Chagin  * by interface name, and return the number of bytes copied to lxname.
2473ab3c9c2SDmitry Chagin  */
2483ab3c9c2SDmitry Chagin int
2493ab3c9c2SDmitry Chagin ifname_bsd_to_linux_name(const char *bsdname, char *lxname, size_t len)
2503ab3c9c2SDmitry Chagin {
2513ab3c9c2SDmitry Chagin 	struct epoch_tracker et;
2523ab3c9c2SDmitry Chagin 	struct ifnet *ifp;
2533ab3c9c2SDmitry Chagin 	int ret;
2543ab3c9c2SDmitry Chagin 
2553ab3c9c2SDmitry Chagin 	ret = 0;
2563ab3c9c2SDmitry Chagin 	CURVNET_SET(TD_TO_VNET(curthread));
2573ab3c9c2SDmitry Chagin 	NET_EPOCH_ENTER(et);
2583ab3c9c2SDmitry Chagin 	ifp = ifunit(bsdname);
2593ab3c9c2SDmitry Chagin 	if (ifp != NULL)
2603ab3c9c2SDmitry Chagin 		ret = ifname_bsd_to_linux_ifp(ifp, lxname, len);
2613ab3c9c2SDmitry Chagin 	NET_EPOCH_EXIT(et);
2623ab3c9c2SDmitry Chagin 	CURVNET_RESTORE();
2633ab3c9c2SDmitry Chagin 	return (ret);
2643ab3c9c2SDmitry Chagin }
2653ab3c9c2SDmitry Chagin 
2663ab3c9c2SDmitry Chagin /*
2673ab3c9c2SDmitry Chagin  * Translate a FreeBSD interface name to a Linux interface name
2683ab3c9c2SDmitry Chagin  * by interface index, and return the number of bytes copied to lxname.
2693ab3c9c2SDmitry Chagin  */
2703ab3c9c2SDmitry Chagin int
2713ab3c9c2SDmitry Chagin ifname_bsd_to_linux_idx(u_int idx, char *lxname, size_t len)
2723ab3c9c2SDmitry Chagin {
2733ab3c9c2SDmitry Chagin 	struct epoch_tracker et;
2743ab3c9c2SDmitry Chagin 	struct ifnet *ifp;
2753ab3c9c2SDmitry Chagin 	int ret;
2763ab3c9c2SDmitry Chagin 
2773ab3c9c2SDmitry Chagin 	ret = 0;
2783ab3c9c2SDmitry Chagin 	CURVNET_SET(TD_TO_VNET(curthread));
2793ab3c9c2SDmitry Chagin 	NET_EPOCH_ENTER(et);
2803ab3c9c2SDmitry Chagin 	ifp = ifnet_byindex(idx);
2813ab3c9c2SDmitry Chagin 	if (ifp != NULL)
2823ab3c9c2SDmitry Chagin 		ret = ifname_bsd_to_linux_ifp(ifp, lxname, len);
2833ab3c9c2SDmitry Chagin 	NET_EPOCH_EXIT(et);
2843ab3c9c2SDmitry Chagin 	CURVNET_RESTORE();
2853ab3c9c2SDmitry Chagin 	return (ret);
2863ab3c9c2SDmitry Chagin }
2873ab3c9c2SDmitry Chagin 
2883ab3c9c2SDmitry Chagin /*
2893ab3c9c2SDmitry Chagin  * Translate a FreeBSD interface name to a Linux interface name,
2906c5786fdSDmitry Chagin  * and return the number of bytes copied to lxname, 0 if interface
2916c5786fdSDmitry Chagin  * not found, -1 on error.
2923ab3c9c2SDmitry Chagin  */
2936c5786fdSDmitry Chagin struct ifname_bsd_to_linux_ifp_cb_s {
2946c5786fdSDmitry Chagin 	struct ifnet	*ifp;
2956c5786fdSDmitry Chagin 	int		ethno;
2966c5786fdSDmitry Chagin 	char		*lxname;
2976c5786fdSDmitry Chagin 	size_t		len;
2986c5786fdSDmitry Chagin };
2996c5786fdSDmitry Chagin 
3006c5786fdSDmitry Chagin static int
3016c5786fdSDmitry Chagin ifname_bsd_to_linux_ifp_cb(if_t ifp, void *arg)
3026c5786fdSDmitry Chagin {
3036c5786fdSDmitry Chagin 	struct ifname_bsd_to_linux_ifp_cb_s *cbs = arg;
3046c5786fdSDmitry Chagin 
3056c5786fdSDmitry Chagin 	if (ifp == cbs->ifp)
3066c5786fdSDmitry Chagin 		return (snprintf(cbs->lxname, cbs->len, "eth%d", cbs->ethno));
3076c5786fdSDmitry Chagin 	if (IFP_IS_ETH(ifp))
3086c5786fdSDmitry Chagin 		cbs->ethno++;
3096c5786fdSDmitry Chagin 	return (0);
3106c5786fdSDmitry Chagin }
3116c5786fdSDmitry Chagin 
3123ab3c9c2SDmitry Chagin int
3133ab3c9c2SDmitry Chagin ifname_bsd_to_linux_ifp(struct ifnet *ifp, char *lxname, size_t len)
3143ab3c9c2SDmitry Chagin {
3156c5786fdSDmitry Chagin 	struct ifname_bsd_to_linux_ifp_cb_s arg = {
3166c5786fdSDmitry Chagin 		.ifp = ifp,
3176c5786fdSDmitry Chagin 		.ethno = 0,
3186c5786fdSDmitry Chagin 		.lxname = lxname,
3196c5786fdSDmitry Chagin 		.len = len,
3206c5786fdSDmitry Chagin 	};
3213ab3c9c2SDmitry Chagin 
3223ab3c9c2SDmitry Chagin 	NET_EPOCH_ASSERT();
3233ab3c9c2SDmitry Chagin 
3243ab3c9c2SDmitry Chagin 	/*
3253ab3c9c2SDmitry Chagin 	 * Linux loopback interface name is lo (not lo0),
3263ab3c9c2SDmitry Chagin 	 * we translate lo to lo0, loX to loX.
3273ab3c9c2SDmitry Chagin 	 */
3283ab3c9c2SDmitry Chagin 	if (IFP_IS_LOOP(ifp) && strncmp(ifp->if_xname, "lo0", IFNAMSIZ) == 0)
3293ab3c9c2SDmitry Chagin 		return (strlcpy(lxname, "lo", len));
3303ab3c9c2SDmitry Chagin 
3313ab3c9c2SDmitry Chagin 	/* Short-circuit non ethernet interfaces. */
3323ab3c9c2SDmitry Chagin 	if (!IFP_IS_ETH(ifp) || linux_use_real_ifname(ifp))
3336c5786fdSDmitry Chagin 		return (strlcpy(lxname, if_name(ifp), len));
3343ab3c9c2SDmitry Chagin 
3353ab3c9c2SDmitry Chagin  	/* Determine the (relative) unit number for ethernet interfaces. */
3366c5786fdSDmitry Chagin 	return (if_foreach(ifname_bsd_to_linux_ifp_cb, &arg));
3373ab3c9c2SDmitry Chagin }
3383ab3c9c2SDmitry Chagin 
3393ab3c9c2SDmitry Chagin /*
340d151344dSDmitry Chagin  * Translate a Linux interface name to a FreeBSD interface name,
341d151344dSDmitry Chagin  * and return the associated ifnet structure
342d151344dSDmitry Chagin  * bsdname and lxname need to be least IFNAMSIZ bytes long, but
343d151344dSDmitry Chagin  * can point to the same buffer.
344d151344dSDmitry Chagin  */
3455d5b633dSDmitry Chagin struct ifname_linux_to_ifp_cb_s {
3466c5786fdSDmitry Chagin 	bool		is_lo;
3476c5786fdSDmitry Chagin 	bool		is_eth;
3486c5786fdSDmitry Chagin 	int		ethno;
3496c5786fdSDmitry Chagin 	int		unit;
3506c5786fdSDmitry Chagin 	const char	*lxname;
3516c5786fdSDmitry Chagin 	if_t		ifp;
3526c5786fdSDmitry Chagin };
3536c5786fdSDmitry Chagin 
3546c5786fdSDmitry Chagin static int
3555d5b633dSDmitry Chagin ifname_linux_to_ifp_cb(if_t ifp, void *arg)
3566c5786fdSDmitry Chagin {
3575d5b633dSDmitry Chagin 	struct ifname_linux_to_ifp_cb_s *cbs = arg;
3586c5786fdSDmitry Chagin 
3596c5786fdSDmitry Chagin 	NET_EPOCH_ASSERT();
3606c5786fdSDmitry Chagin 
3616c5786fdSDmitry Chagin 	/*
3626c5786fdSDmitry Chagin 	 * Allow Linux programs to use FreeBSD names. Don't presume
3636c5786fdSDmitry Chagin 	 * we never have an interface named "eth", so don't make
3646c5786fdSDmitry Chagin 	 * the test optional based on is_eth.
3656c5786fdSDmitry Chagin 	 */
3666c5786fdSDmitry Chagin 	if (strncmp(if_name(ifp), cbs->lxname, LINUX_IFNAMSIZ) == 0)
3676c5786fdSDmitry Chagin 		goto out;
3686c5786fdSDmitry Chagin 	if (cbs->is_eth && IFP_IS_ETH(ifp) && cbs->unit == cbs->ethno)
3696c5786fdSDmitry Chagin 		goto out;
3706c5786fdSDmitry Chagin 	if (cbs->is_lo && IFP_IS_LOOP(ifp))
3716c5786fdSDmitry Chagin 		goto out;
3726c5786fdSDmitry Chagin 	if (IFP_IS_ETH(ifp))
3736c5786fdSDmitry Chagin 		cbs->ethno++;
3746c5786fdSDmitry Chagin 	return (0);
3756c5786fdSDmitry Chagin 
3766c5786fdSDmitry Chagin out:
3776c5786fdSDmitry Chagin 	cbs->ifp = ifp;
3786c5786fdSDmitry Chagin 	return (1);
3796c5786fdSDmitry Chagin }
3806c5786fdSDmitry Chagin 
381d151344dSDmitry Chagin struct ifnet *
3825d5b633dSDmitry Chagin ifname_linux_to_ifp(struct thread *td, const char *lxname)
383d151344dSDmitry Chagin {
3845d5b633dSDmitry Chagin 	struct ifname_linux_to_ifp_cb_s arg = {
3856c5786fdSDmitry Chagin 		.ethno = 0,
3866c5786fdSDmitry Chagin 		.lxname = lxname,
3876c5786fdSDmitry Chagin 		.ifp = NULL,
3886c5786fdSDmitry Chagin 	};
3895d5b633dSDmitry Chagin 	int len;
390d151344dSDmitry Chagin 	char *ep;
391d151344dSDmitry Chagin 
3925d5b633dSDmitry Chagin 	NET_EPOCH_ASSERT();
3935d5b633dSDmitry Chagin 
394d151344dSDmitry Chagin 	for (len = 0; len < LINUX_IFNAMSIZ; ++len)
3959c1437aeSDmitry Chagin 		if (!isalpha(lxname[len]) || lxname[len] == '\0')
396d151344dSDmitry Chagin 			break;
397d151344dSDmitry Chagin 	if (len == 0 || len == LINUX_IFNAMSIZ)
398d151344dSDmitry Chagin 		return (NULL);
3993ab3c9c2SDmitry Chagin 	/*
4003ab3c9c2SDmitry Chagin 	 * Linux loopback interface name is lo (not lo0),
4013ab3c9c2SDmitry Chagin 	 * we translate lo to lo0, loX to loX.
4023ab3c9c2SDmitry Chagin 	 */
4036c5786fdSDmitry Chagin 	arg.is_lo = (len == 2 && strncmp(lxname, "lo", LINUX_IFNAMSIZ) == 0);
4046c5786fdSDmitry Chagin 	arg.unit = (int)strtoul(lxname + len, &ep, 10);
405d151344dSDmitry Chagin 	if ((ep == NULL || ep == lxname + len || ep >= lxname + LINUX_IFNAMSIZ) &&
4066c5786fdSDmitry Chagin 	    arg.is_lo == 0)
407d151344dSDmitry Chagin 		return (NULL);
4086c5786fdSDmitry Chagin 	arg.is_eth = (len == 3 && strncmp(lxname, "eth", len) == 0);
409d151344dSDmitry Chagin 
4105d5b633dSDmitry Chagin 	if_foreach(ifname_linux_to_ifp_cb, &arg);
4115d5b633dSDmitry Chagin 	return (arg.ifp);
4125d5b633dSDmitry Chagin }
4135d5b633dSDmitry Chagin 
414*f9b0675bSDmitry Chagin int
4155d5b633dSDmitry Chagin ifname_linux_to_bsd(struct thread *td, const char *lxname, char *bsdname)
4165d5b633dSDmitry Chagin {
4175d5b633dSDmitry Chagin 	struct epoch_tracker et;
4185d5b633dSDmitry Chagin 	struct ifnet *ifp;
4195d5b633dSDmitry Chagin 
420d151344dSDmitry Chagin 	CURVNET_SET(TD_TO_VNET(td));
4216c5786fdSDmitry Chagin 	NET_EPOCH_ENTER(et);
4225d5b633dSDmitry Chagin 	ifp = ifname_linux_to_ifp(td, lxname);
4235d5b633dSDmitry Chagin 	if (ifp != NULL && bsdname != NULL)
4245d5b633dSDmitry Chagin 		strlcpy(bsdname, if_name(ifp), IFNAMSIZ);
4256c5786fdSDmitry Chagin 	NET_EPOCH_EXIT(et);
426d151344dSDmitry Chagin 	CURVNET_RESTORE();
427*f9b0675bSDmitry Chagin 	return (ifp != NULL ? 0 : EINVAL);
428d151344dSDmitry Chagin }
429d151344dSDmitry Chagin 
430130383f2SDmitry Chagin unsigned short
431130383f2SDmitry Chagin linux_ifflags(struct ifnet *ifp)
432d151344dSDmitry Chagin {
433130383f2SDmitry Chagin 	unsigned short fl, flags;
434d151344dSDmitry Chagin 
4356c5786fdSDmitry Chagin 	fl = (if_getflags(ifp) | if_getdrvflags(ifp)) & 0xffff;
436130383f2SDmitry Chagin 	flags = 0;
437bbac65c7SDmitry Chagin 	if (fl & IFF_UP)
438130383f2SDmitry Chagin 		flags |= LINUX_IFF_UP;
439bbac65c7SDmitry Chagin 	if (fl & IFF_BROADCAST)
440130383f2SDmitry Chagin 		flags |= LINUX_IFF_BROADCAST;
441bbac65c7SDmitry Chagin 	if (fl & IFF_DEBUG)
442130383f2SDmitry Chagin 		flags |= LINUX_IFF_DEBUG;
443bbac65c7SDmitry Chagin 	if (fl & IFF_LOOPBACK)
444130383f2SDmitry Chagin 		flags |= LINUX_IFF_LOOPBACK;
445bbac65c7SDmitry Chagin 	if (fl & IFF_POINTOPOINT)
446130383f2SDmitry Chagin 		flags |= LINUX_IFF_POINTOPOINT;
447bbac65c7SDmitry Chagin 	if (fl & IFF_DRV_RUNNING)
448130383f2SDmitry Chagin 		flags |= LINUX_IFF_RUNNING;
449bbac65c7SDmitry Chagin 	if (fl & IFF_NOARP)
450130383f2SDmitry Chagin 		flags |= LINUX_IFF_NOARP;
451bbac65c7SDmitry Chagin 	if (fl & IFF_PROMISC)
452130383f2SDmitry Chagin 		flags |= LINUX_IFF_PROMISC;
453bbac65c7SDmitry Chagin 	if (fl & IFF_ALLMULTI)
454130383f2SDmitry Chagin 		flags |= LINUX_IFF_ALLMULTI;
455bbac65c7SDmitry Chagin 	if (fl & IFF_MULTICAST)
456130383f2SDmitry Chagin 		flags |= LINUX_IFF_MULTICAST;
457130383f2SDmitry Chagin 	return (flags);
458d151344dSDmitry Chagin }
459d151344dSDmitry Chagin 
4606c5786fdSDmitry Chagin static u_int
4616c5786fdSDmitry Chagin linux_ifhwaddr_cb(void *arg, struct ifaddr *ifa, u_int count)
4626c5786fdSDmitry Chagin {
4636c5786fdSDmitry Chagin 	struct sockaddr_dl *sdl = (struct sockaddr_dl *)ifa->ifa_addr;
4646c5786fdSDmitry Chagin 	struct l_sockaddr *lsa = arg;
4656c5786fdSDmitry Chagin 
4666c5786fdSDmitry Chagin 	if (count > 0)
4676c5786fdSDmitry Chagin 		return (0);
4686c5786fdSDmitry Chagin 	if (sdl->sdl_type != IFT_ETHER)
4696c5786fdSDmitry Chagin 		return (0);
4706c5786fdSDmitry Chagin 	bzero(lsa, sizeof(*lsa));
4716c5786fdSDmitry Chagin 	lsa->sa_family = LINUX_ARPHRD_ETHER;
4726c5786fdSDmitry Chagin 	bcopy(LLADDR(sdl), lsa->sa_data, LINUX_IFHWADDRLEN);
4736c5786fdSDmitry Chagin 	return (1);
4746c5786fdSDmitry Chagin }
4756c5786fdSDmitry Chagin 
476d151344dSDmitry Chagin int
477d151344dSDmitry Chagin linux_ifhwaddr(struct ifnet *ifp, struct l_sockaddr *lsa)
478d151344dSDmitry Chagin {
479d151344dSDmitry Chagin 
480d151344dSDmitry Chagin 	if (IFP_IS_LOOP(ifp)) {
481d151344dSDmitry Chagin 		bzero(lsa, sizeof(*lsa));
482d151344dSDmitry Chagin 		lsa->sa_family = LINUX_ARPHRD_LOOPBACK;
483d151344dSDmitry Chagin 		return (0);
484d151344dSDmitry Chagin 	}
485d151344dSDmitry Chagin 	if (!IFP_IS_ETH(ifp))
486d151344dSDmitry Chagin 		return (ENOENT);
4876c5786fdSDmitry Chagin 	if (if_foreach_addr_type(ifp, AF_LINK, linux_ifhwaddr_cb, lsa) > 0)
488d151344dSDmitry Chagin 		return (0);
489d151344dSDmitry Chagin 	return (ENOENT);
490d151344dSDmitry Chagin }
491d5368bf3SDmitry Chagin 
492d5368bf3SDmitry Chagin int
493d5368bf3SDmitry Chagin linux_to_bsd_domain(int domain)
494d5368bf3SDmitry Chagin {
495d5368bf3SDmitry Chagin 
496d5368bf3SDmitry Chagin 	switch (domain) {
497d5368bf3SDmitry Chagin 	case LINUX_AF_UNSPEC:
498d5368bf3SDmitry Chagin 		return (AF_UNSPEC);
499d5368bf3SDmitry Chagin 	case LINUX_AF_UNIX:
500d5368bf3SDmitry Chagin 		return (AF_LOCAL);
501d5368bf3SDmitry Chagin 	case LINUX_AF_INET:
502d5368bf3SDmitry Chagin 		return (AF_INET);
503d5368bf3SDmitry Chagin 	case LINUX_AF_INET6:
504d5368bf3SDmitry Chagin 		return (AF_INET6);
505d5368bf3SDmitry Chagin 	case LINUX_AF_AX25:
506d5368bf3SDmitry Chagin 		return (AF_CCITT);
507d5368bf3SDmitry Chagin 	case LINUX_AF_IPX:
508d5368bf3SDmitry Chagin 		return (AF_IPX);
509d5368bf3SDmitry Chagin 	case LINUX_AF_APPLETALK:
510d5368bf3SDmitry Chagin 		return (AF_APPLETALK);
5117c40e2d5SAlexander V. Chernikov 	case LINUX_AF_NETLINK:
5127c40e2d5SAlexander V. Chernikov 		return (AF_NETLINK);
513d5368bf3SDmitry Chagin 	}
514d5368bf3SDmitry Chagin 	return (-1);
515d5368bf3SDmitry Chagin }
516d5368bf3SDmitry Chagin 
517d5368bf3SDmitry Chagin int
518d5368bf3SDmitry Chagin bsd_to_linux_domain(int domain)
519d5368bf3SDmitry Chagin {
520d5368bf3SDmitry Chagin 
521d5368bf3SDmitry Chagin 	switch (domain) {
522d5368bf3SDmitry Chagin 	case AF_UNSPEC:
523d5368bf3SDmitry Chagin 		return (LINUX_AF_UNSPEC);
524d5368bf3SDmitry Chagin 	case AF_LOCAL:
525d5368bf3SDmitry Chagin 		return (LINUX_AF_UNIX);
526d5368bf3SDmitry Chagin 	case AF_INET:
527d5368bf3SDmitry Chagin 		return (LINUX_AF_INET);
528d5368bf3SDmitry Chagin 	case AF_INET6:
529d5368bf3SDmitry Chagin 		return (LINUX_AF_INET6);
530d5368bf3SDmitry Chagin 	case AF_CCITT:
531d5368bf3SDmitry Chagin 		return (LINUX_AF_AX25);
532d5368bf3SDmitry Chagin 	case AF_IPX:
533d5368bf3SDmitry Chagin 		return (LINUX_AF_IPX);
534d5368bf3SDmitry Chagin 	case AF_APPLETALK:
535d5368bf3SDmitry Chagin 		return (LINUX_AF_APPLETALK);
5367c40e2d5SAlexander V. Chernikov 	case AF_NETLINK:
5377c40e2d5SAlexander V. Chernikov 		return (LINUX_AF_NETLINK);
538d5368bf3SDmitry Chagin 	}
539d5368bf3SDmitry Chagin 	return (-1);
540d5368bf3SDmitry Chagin }
541d5368bf3SDmitry Chagin 
542d5368bf3SDmitry Chagin /*
543d5368bf3SDmitry Chagin  * Based on the fact that:
544d5368bf3SDmitry Chagin  * 1. Native and Linux storage of struct sockaddr
545d5368bf3SDmitry Chagin  * and struct sockaddr_in6 are equal.
546d5368bf3SDmitry Chagin  * 2. On Linux sa_family is the first member of all struct sockaddr.
547d5368bf3SDmitry Chagin  */
548d5368bf3SDmitry Chagin int
549d5368bf3SDmitry Chagin bsd_to_linux_sockaddr(const struct sockaddr *sa, struct l_sockaddr **lsa,
550d5368bf3SDmitry Chagin     socklen_t len)
551d5368bf3SDmitry Chagin {
552d5368bf3SDmitry Chagin 	struct l_sockaddr *kosa;
553bbddd588SDmitry Chagin 	int bdom;
554d5368bf3SDmitry Chagin 
555d5368bf3SDmitry Chagin 	*lsa = NULL;
556d5368bf3SDmitry Chagin 	if (len < 2 || len > UCHAR_MAX)
557d5368bf3SDmitry Chagin 		return (EINVAL);
558bbddd588SDmitry Chagin 	bdom = bsd_to_linux_domain(sa->sa_family);
559bbddd588SDmitry Chagin 	if (bdom == -1)
560bbddd588SDmitry Chagin 		return (EAFNOSUPPORT);
561d5368bf3SDmitry Chagin 
56263355839SDmitry Chagin 	kosa = malloc(len, M_LINUX, M_WAITOK);
563d5368bf3SDmitry Chagin 	bcopy(sa, kosa, len);
564d5368bf3SDmitry Chagin 	kosa->sa_family = bdom;
565d5368bf3SDmitry Chagin 	*lsa = kosa;
566d5368bf3SDmitry Chagin 	return (0);
567d5368bf3SDmitry Chagin }
568d5368bf3SDmitry Chagin 
569d5368bf3SDmitry Chagin int
570d5368bf3SDmitry Chagin linux_to_bsd_sockaddr(const struct l_sockaddr *osa, struct sockaddr **sap,
571d5368bf3SDmitry Chagin     socklen_t *len)
572d5368bf3SDmitry Chagin {
573d5368bf3SDmitry Chagin 	struct sockaddr *sa;
574d5368bf3SDmitry Chagin 	struct l_sockaddr *kosa;
575d5368bf3SDmitry Chagin #ifdef INET6
576d5368bf3SDmitry Chagin 	struct sockaddr_in6 *sin6;
577d5368bf3SDmitry Chagin 	bool  oldv6size;
578d5368bf3SDmitry Chagin #endif
579d5368bf3SDmitry Chagin 	char *name;
580d5368bf3SDmitry Chagin 	int salen, bdom, error, hdrlen, namelen;
581d5368bf3SDmitry Chagin 
582d5368bf3SDmitry Chagin 	if (*len < 2 || *len > UCHAR_MAX)
583d5368bf3SDmitry Chagin 		return (EINVAL);
584d5368bf3SDmitry Chagin 
585d5368bf3SDmitry Chagin 	salen = *len;
586d5368bf3SDmitry Chagin 
587d5368bf3SDmitry Chagin #ifdef INET6
588d5368bf3SDmitry Chagin 	oldv6size = false;
589d5368bf3SDmitry Chagin 	/*
590d5368bf3SDmitry Chagin 	 * Check for old (pre-RFC2553) sockaddr_in6. We may accept it
591d5368bf3SDmitry Chagin 	 * if it's a v4-mapped address, so reserve the proper space
592d5368bf3SDmitry Chagin 	 * for it.
593d5368bf3SDmitry Chagin 	 */
594d5368bf3SDmitry Chagin 	if (salen == sizeof(struct sockaddr_in6) - sizeof(uint32_t)) {
595d5368bf3SDmitry Chagin 		salen += sizeof(uint32_t);
596d5368bf3SDmitry Chagin 		oldv6size = true;
597d5368bf3SDmitry Chagin 	}
598d5368bf3SDmitry Chagin #endif
599d5368bf3SDmitry Chagin 
600d5368bf3SDmitry Chagin 	kosa = malloc(salen, M_SONAME, M_WAITOK);
601d5368bf3SDmitry Chagin 
602d5368bf3SDmitry Chagin 	if ((error = copyin(osa, kosa, *len)))
603d5368bf3SDmitry Chagin 		goto out;
604d5368bf3SDmitry Chagin 
605d5368bf3SDmitry Chagin 	bdom = linux_to_bsd_domain(kosa->sa_family);
606d5368bf3SDmitry Chagin 	if (bdom == -1) {
607d5368bf3SDmitry Chagin 		error = EAFNOSUPPORT;
608d5368bf3SDmitry Chagin 		goto out;
609d5368bf3SDmitry Chagin 	}
610d5368bf3SDmitry Chagin 
611d5368bf3SDmitry Chagin #ifdef INET6
612d5368bf3SDmitry Chagin 	/*
613d5368bf3SDmitry Chagin 	 * Older Linux IPv6 code uses obsolete RFC2133 struct sockaddr_in6,
614d5368bf3SDmitry Chagin 	 * which lacks the scope id compared with RFC2553 one. If we detect
615d5368bf3SDmitry Chagin 	 * the situation, reject the address and write a message to system log.
616d5368bf3SDmitry Chagin 	 *
617d5368bf3SDmitry Chagin 	 * Still accept addresses for which the scope id is not used.
618d5368bf3SDmitry Chagin 	 */
619d5368bf3SDmitry Chagin 	if (oldv6size) {
620d5368bf3SDmitry Chagin 		if (bdom == AF_INET6) {
621d5368bf3SDmitry Chagin 			sin6 = (struct sockaddr_in6 *)kosa;
622d5368bf3SDmitry Chagin 			if (IN6_IS_ADDR_V4MAPPED(&sin6->sin6_addr) ||
623d5368bf3SDmitry Chagin 			    (!IN6_IS_ADDR_LINKLOCAL(&sin6->sin6_addr) &&
624d5368bf3SDmitry Chagin 			     !IN6_IS_ADDR_SITELOCAL(&sin6->sin6_addr) &&
625d5368bf3SDmitry Chagin 			     !IN6_IS_ADDR_V4COMPAT(&sin6->sin6_addr) &&
626d5368bf3SDmitry Chagin 			     !IN6_IS_ADDR_UNSPECIFIED(&sin6->sin6_addr) &&
627d5368bf3SDmitry Chagin 			     !IN6_IS_ADDR_MULTICAST(&sin6->sin6_addr))) {
628d5368bf3SDmitry Chagin 				sin6->sin6_scope_id = 0;
629d5368bf3SDmitry Chagin 			} else {
630d5368bf3SDmitry Chagin 				linux_msg(curthread,
63186e794ebSEdward Tomasz Napierala 				    "obsolete pre-RFC2553 sockaddr_in6 rejected");
632d5368bf3SDmitry Chagin 				error = EINVAL;
633d5368bf3SDmitry Chagin 				goto out;
634d5368bf3SDmitry Chagin 			}
635d5368bf3SDmitry Chagin 		} else
636d5368bf3SDmitry Chagin 			salen -= sizeof(uint32_t);
637d5368bf3SDmitry Chagin 	}
638d5368bf3SDmitry Chagin #endif
639d5368bf3SDmitry Chagin 	if (bdom == AF_INET) {
640d5368bf3SDmitry Chagin 		if (salen < sizeof(struct sockaddr_in)) {
641d5368bf3SDmitry Chagin 			error = EINVAL;
642d5368bf3SDmitry Chagin 			goto out;
643d5368bf3SDmitry Chagin 		}
644d5368bf3SDmitry Chagin 		salen = sizeof(struct sockaddr_in);
645d5368bf3SDmitry Chagin 	}
646d5368bf3SDmitry Chagin 
647d5368bf3SDmitry Chagin 	if (bdom == AF_LOCAL && salen > sizeof(struct sockaddr_un)) {
648d5368bf3SDmitry Chagin 		hdrlen = offsetof(struct sockaddr_un, sun_path);
649d5368bf3SDmitry Chagin 		name = ((struct sockaddr_un *)kosa)->sun_path;
650d5368bf3SDmitry Chagin 		if (*name == '\0') {
651d5368bf3SDmitry Chagin 			/*
652d5368bf3SDmitry Chagin 			 * Linux abstract namespace starts with a NULL byte.
653d5368bf3SDmitry Chagin 			 * XXX We do not support abstract namespace yet.
654d5368bf3SDmitry Chagin 			 */
655d5368bf3SDmitry Chagin 			namelen = strnlen(name + 1, salen - hdrlen - 1) + 1;
656d5368bf3SDmitry Chagin 		} else
657d5368bf3SDmitry Chagin 			namelen = strnlen(name, salen - hdrlen);
658d5368bf3SDmitry Chagin 		salen = hdrlen + namelen;
659d5368bf3SDmitry Chagin 		if (salen > sizeof(struct sockaddr_un)) {
660d5368bf3SDmitry Chagin 			error = ENAMETOOLONG;
661d5368bf3SDmitry Chagin 			goto out;
662d5368bf3SDmitry Chagin 		}
663d5368bf3SDmitry Chagin 	}
664d5368bf3SDmitry Chagin 
6657c40e2d5SAlexander V. Chernikov 	if (bdom == AF_NETLINK) {
6667c40e2d5SAlexander V. Chernikov 		if (salen < sizeof(struct sockaddr_nl)) {
6677c40e2d5SAlexander V. Chernikov 			error = EINVAL;
6687c40e2d5SAlexander V. Chernikov 			goto out;
6697c40e2d5SAlexander V. Chernikov 		}
6707c40e2d5SAlexander V. Chernikov 		salen = sizeof(struct sockaddr_nl);
6717c40e2d5SAlexander V. Chernikov 	}
6727c40e2d5SAlexander V. Chernikov 
673d5368bf3SDmitry Chagin 	sa = (struct sockaddr *)kosa;
674d5368bf3SDmitry Chagin 	sa->sa_family = bdom;
675d5368bf3SDmitry Chagin 	sa->sa_len = salen;
676d5368bf3SDmitry Chagin 
677d5368bf3SDmitry Chagin 	*sap = sa;
678d5368bf3SDmitry Chagin 	*len = salen;
679d5368bf3SDmitry Chagin 	return (0);
680d5368bf3SDmitry Chagin 
681d5368bf3SDmitry Chagin out:
682d5368bf3SDmitry Chagin 	free(kosa, M_SONAME);
683d5368bf3SDmitry Chagin 	return (error);
684d5368bf3SDmitry Chagin }
685044ab55eSEdward Tomasz Napierala 
686044ab55eSEdward Tomasz Napierala void
687044ab55eSEdward Tomasz Napierala linux_dev_shm_create(void)
688044ab55eSEdward Tomasz Napierala {
689044ab55eSEdward Tomasz Napierala 	int error;
690044ab55eSEdward Tomasz Napierala 
691044ab55eSEdward Tomasz Napierala 	error = make_dev_p(MAKEDEV_CHECKNAME | MAKEDEV_WAITOK, &dev_shm_cdev,
692044ab55eSEdward Tomasz Napierala 	    &dev_shm_cdevsw, NULL, UID_ROOT, GID_WHEEL, 0, "shm/.mountpoint");
693044ab55eSEdward Tomasz Napierala 	if (error != 0) {
694044ab55eSEdward Tomasz Napierala 		printf("%s: failed to create device node, error %d\n",
695044ab55eSEdward Tomasz Napierala 		    __func__, error);
696044ab55eSEdward Tomasz Napierala 	}
697044ab55eSEdward Tomasz Napierala }
698044ab55eSEdward Tomasz Napierala 
699044ab55eSEdward Tomasz Napierala void
700044ab55eSEdward Tomasz Napierala linux_dev_shm_destroy(void)
701044ab55eSEdward Tomasz Napierala {
702044ab55eSEdward Tomasz Napierala 
703044ab55eSEdward Tomasz Napierala 	destroy_dev(dev_shm_cdev);
704044ab55eSEdward Tomasz Napierala }
7055403f186SKyle Evans 
7065403f186SKyle Evans int
7075403f186SKyle Evans bsd_to_linux_bits_(int value, struct bsd_to_linux_bitmap *bitmap,
7085403f186SKyle Evans     size_t mapcnt, int no_value)
7095403f186SKyle Evans {
7105403f186SKyle Evans 	int bsd_mask, bsd_value, linux_mask, linux_value;
7115403f186SKyle Evans 	int linux_ret;
7125403f186SKyle Evans 	size_t i;
7135403f186SKyle Evans 	bool applied;
7145403f186SKyle Evans 
7155403f186SKyle Evans 	applied = false;
7165403f186SKyle Evans 	linux_ret = 0;
7175403f186SKyle Evans 	for (i = 0; i < mapcnt; ++i) {
7185403f186SKyle Evans 		bsd_mask = bitmap[i].bsd_mask;
7195403f186SKyle Evans 		bsd_value = bitmap[i].bsd_value;
7205403f186SKyle Evans 		if (bsd_mask == 0)
7215403f186SKyle Evans 			bsd_mask = bsd_value;
7225403f186SKyle Evans 
7235403f186SKyle Evans 		linux_mask = bitmap[i].linux_mask;
7245403f186SKyle Evans 		linux_value = bitmap[i].linux_value;
7255403f186SKyle Evans 		if (linux_mask == 0)
7265403f186SKyle Evans 			linux_mask = linux_value;
7275403f186SKyle Evans 
7285403f186SKyle Evans 		/*
7295403f186SKyle Evans 		 * If a mask larger than just the value is set, we explicitly
7305403f186SKyle Evans 		 * want to make sure that only this bit we mapped within that
7315403f186SKyle Evans 		 * mask is set.
7325403f186SKyle Evans 		 */
7335403f186SKyle Evans 		if ((value & bsd_mask) == bsd_value) {
7345403f186SKyle Evans 			linux_ret = (linux_ret & ~linux_mask) | linux_value;
7355403f186SKyle Evans 			applied = true;
7365403f186SKyle Evans 		}
7375403f186SKyle Evans 	}
7385403f186SKyle Evans 
7395403f186SKyle Evans 	if (!applied)
7405403f186SKyle Evans 		return (no_value);
7415403f186SKyle Evans 	return (linux_ret);
7425403f186SKyle Evans }
7435403f186SKyle Evans 
7445403f186SKyle Evans int
7455403f186SKyle Evans linux_to_bsd_bits_(int value, struct bsd_to_linux_bitmap *bitmap,
7465403f186SKyle Evans     size_t mapcnt, int no_value)
7475403f186SKyle Evans {
7485403f186SKyle Evans 	int bsd_mask, bsd_value, linux_mask, linux_value;
7495403f186SKyle Evans 	int bsd_ret;
7505403f186SKyle Evans 	size_t i;
7515403f186SKyle Evans 	bool applied;
7525403f186SKyle Evans 
7535403f186SKyle Evans 	applied = false;
7545403f186SKyle Evans 	bsd_ret = 0;
7555403f186SKyle Evans 	for (i = 0; i < mapcnt; ++i) {
7565403f186SKyle Evans 		bsd_mask = bitmap[i].bsd_mask;
7575403f186SKyle Evans 		bsd_value = bitmap[i].bsd_value;
7585403f186SKyle Evans 		if (bsd_mask == 0)
7595403f186SKyle Evans 			bsd_mask = bsd_value;
7605403f186SKyle Evans 
7615403f186SKyle Evans 		linux_mask = bitmap[i].linux_mask;
7625403f186SKyle Evans 		linux_value = bitmap[i].linux_value;
7635403f186SKyle Evans 		if (linux_mask == 0)
7645403f186SKyle Evans 			linux_mask = linux_value;
7655403f186SKyle Evans 
7665403f186SKyle Evans 		/*
7675403f186SKyle Evans 		 * If a mask larger than just the value is set, we explicitly
7685403f186SKyle Evans 		 * want to make sure that only this bit we mapped within that
7695403f186SKyle Evans 		 * mask is set.
7705403f186SKyle Evans 		 */
7715403f186SKyle Evans 		if ((value & linux_mask) == linux_value) {
7725403f186SKyle Evans 			bsd_ret = (bsd_ret & ~bsd_mask) | bsd_value;
7735403f186SKyle Evans 			applied = true;
7745403f186SKyle Evans 		}
7755403f186SKyle Evans 	}
7765403f186SKyle Evans 
7775403f186SKyle Evans 	if (!applied)
7785403f186SKyle Evans 		return (no_value);
7795403f186SKyle Evans 	return (bsd_ret);
7805403f186SKyle Evans }
78126795a03SDmitry Chagin 
78226795a03SDmitry Chagin void
78326795a03SDmitry Chagin linux_to_bsd_poll_events(struct thread *td, int fd, short lev,
78426795a03SDmitry Chagin     short *bev)
78526795a03SDmitry Chagin {
78626795a03SDmitry Chagin 	struct file *fp;
78726795a03SDmitry Chagin 	int error;
78826795a03SDmitry Chagin 	short bits = 0;
78926795a03SDmitry Chagin 
79026795a03SDmitry Chagin 	if (lev & LINUX_POLLIN)
79126795a03SDmitry Chagin 		bits |= POLLIN;
79226795a03SDmitry Chagin 	if (lev & LINUX_POLLPRI)
79326795a03SDmitry Chagin 		bits |=	POLLPRI;
79426795a03SDmitry Chagin 	if (lev & LINUX_POLLOUT)
79526795a03SDmitry Chagin 		bits |= POLLOUT;
79626795a03SDmitry Chagin 	if (lev & LINUX_POLLERR)
79726795a03SDmitry Chagin 		bits |= POLLERR;
79826795a03SDmitry Chagin 	if (lev & LINUX_POLLHUP)
79926795a03SDmitry Chagin 		bits |= POLLHUP;
80026795a03SDmitry Chagin 	if (lev & LINUX_POLLNVAL)
80126795a03SDmitry Chagin 		bits |= POLLNVAL;
80226795a03SDmitry Chagin 	if (lev & LINUX_POLLRDNORM)
80326795a03SDmitry Chagin 		bits |= POLLRDNORM;
80426795a03SDmitry Chagin 	if (lev & LINUX_POLLRDBAND)
80526795a03SDmitry Chagin 		bits |= POLLRDBAND;
80626795a03SDmitry Chagin 	if (lev & LINUX_POLLWRBAND)
80726795a03SDmitry Chagin 		bits |= POLLWRBAND;
80826795a03SDmitry Chagin 	if (lev & LINUX_POLLWRNORM)
80926795a03SDmitry Chagin 		bits |= POLLWRNORM;
81026795a03SDmitry Chagin 
81126795a03SDmitry Chagin 	if (lev & LINUX_POLLRDHUP) {
81226795a03SDmitry Chagin 		/*
81326795a03SDmitry Chagin 		 * It seems that the Linux silencly ignores POLLRDHUP
81426795a03SDmitry Chagin 		 * on non-socket file descriptors unlike FreeBSD, where
81526795a03SDmitry Chagin 		 * events bits is more strictly checked (POLLSTANDARD).
81626795a03SDmitry Chagin 		 */
817513c7a6eSMateusz Guzik 		error = fget_unlocked(td, fd, &cap_no_rights, &fp);
81826795a03SDmitry Chagin 		if (error == 0) {
81926795a03SDmitry Chagin 			/*
82026795a03SDmitry Chagin 			 * XXX. On FreeBSD POLLRDHUP applies only to
82126795a03SDmitry Chagin 			 * stream sockets.
82226795a03SDmitry Chagin 			 */
82326795a03SDmitry Chagin 			if (fp->f_type == DTYPE_SOCKET)
82426795a03SDmitry Chagin 				bits |= POLLRDHUP;
82526795a03SDmitry Chagin 			fdrop(fp, td);
82626795a03SDmitry Chagin 		}
82726795a03SDmitry Chagin 	}
82826795a03SDmitry Chagin 
82926795a03SDmitry Chagin 	if (lev & LINUX_POLLMSG)
83026795a03SDmitry Chagin 		LINUX_RATELIMIT_MSG_OPT1("unsupported POLLMSG, events(%d)", lev);
83126795a03SDmitry Chagin 	if (lev & LINUX_POLLREMOVE)
83226795a03SDmitry Chagin 		LINUX_RATELIMIT_MSG_OPT1("unsupported POLLREMOVE, events(%d)", lev);
83326795a03SDmitry Chagin 
83426795a03SDmitry Chagin 	*bev = bits;
83526795a03SDmitry Chagin }
83626795a03SDmitry Chagin 
83726795a03SDmitry Chagin void
83826795a03SDmitry Chagin bsd_to_linux_poll_events(short bev, short *lev)
83926795a03SDmitry Chagin {
84026795a03SDmitry Chagin 	short bits = 0;
84126795a03SDmitry Chagin 
84226795a03SDmitry Chagin 	if (bev & POLLIN)
84326795a03SDmitry Chagin 		bits |= LINUX_POLLIN;
84426795a03SDmitry Chagin 	if (bev & POLLPRI)
84526795a03SDmitry Chagin 		bits |=	LINUX_POLLPRI;
84626795a03SDmitry Chagin 	if (bev & (POLLOUT | POLLWRNORM))
84726795a03SDmitry Chagin 		/*
84826795a03SDmitry Chagin 		 * POLLWRNORM is equal to POLLOUT on FreeBSD,
84926795a03SDmitry Chagin 		 * but not on Linux
85026795a03SDmitry Chagin 		 */
85126795a03SDmitry Chagin 		bits |= LINUX_POLLOUT;
85226795a03SDmitry Chagin 	if (bev & POLLERR)
85326795a03SDmitry Chagin 		bits |= LINUX_POLLERR;
85426795a03SDmitry Chagin 	if (bev & POLLHUP)
85526795a03SDmitry Chagin 		bits |= LINUX_POLLHUP;
85626795a03SDmitry Chagin 	if (bev & POLLNVAL)
85726795a03SDmitry Chagin 		bits |= LINUX_POLLNVAL;
85826795a03SDmitry Chagin 	if (bev & POLLRDNORM)
85926795a03SDmitry Chagin 		bits |= LINUX_POLLRDNORM;
86026795a03SDmitry Chagin 	if (bev & POLLRDBAND)
86126795a03SDmitry Chagin 		bits |= LINUX_POLLRDBAND;
86226795a03SDmitry Chagin 	if (bev & POLLWRBAND)
86326795a03SDmitry Chagin 		bits |= LINUX_POLLWRBAND;
86426795a03SDmitry Chagin 	if (bev & POLLRDHUP)
86526795a03SDmitry Chagin 		bits |= LINUX_POLLRDHUP;
86626795a03SDmitry Chagin 
86726795a03SDmitry Chagin 	*lev = bits;
86826795a03SDmitry Chagin }
86932fdc75fSDmitry Chagin 
87032fdc75fSDmitry Chagin bool
87132fdc75fSDmitry Chagin linux_use_real_ifname(const struct ifnet *ifp)
87232fdc75fSDmitry Chagin {
87332fdc75fSDmitry Chagin 
8743ab3c9c2SDmitry Chagin 	return (use_real_ifnames);
87532fdc75fSDmitry Chagin }
876