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