xref: /freebsd/sys/compat/linux/linux.c (revision 9d4a08d162d87ba120f418a1a71facd2c631b549)
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