xref: /freebsd/sys/compat/linux/linux.c (revision 3ab3c9c29cf0e5df8dbbaaf2003456445534bad8)
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 /*
245*3ab3c9c2SDmitry Chagin  * Translate a FreeBSD interface name to a Linux interface name
246*3ab3c9c2SDmitry Chagin  * by interface name, and return the number of bytes copied to lxname.
247*3ab3c9c2SDmitry Chagin  */
248*3ab3c9c2SDmitry Chagin int
249*3ab3c9c2SDmitry Chagin ifname_bsd_to_linux_name(const char *bsdname, char *lxname, size_t len)
250*3ab3c9c2SDmitry Chagin {
251*3ab3c9c2SDmitry Chagin 	struct epoch_tracker et;
252*3ab3c9c2SDmitry Chagin 	struct ifnet *ifp;
253*3ab3c9c2SDmitry Chagin 	int ret;
254*3ab3c9c2SDmitry Chagin 
255*3ab3c9c2SDmitry Chagin 	ret = 0;
256*3ab3c9c2SDmitry Chagin 	CURVNET_SET(TD_TO_VNET(curthread));
257*3ab3c9c2SDmitry Chagin 	NET_EPOCH_ENTER(et);
258*3ab3c9c2SDmitry Chagin 	ifp = ifunit(bsdname);
259*3ab3c9c2SDmitry Chagin 	if (ifp != NULL)
260*3ab3c9c2SDmitry Chagin 		ret = ifname_bsd_to_linux_ifp(ifp, lxname, len);
261*3ab3c9c2SDmitry Chagin 	NET_EPOCH_EXIT(et);
262*3ab3c9c2SDmitry Chagin 	CURVNET_RESTORE();
263*3ab3c9c2SDmitry Chagin 	return (ret);
264*3ab3c9c2SDmitry Chagin }
265*3ab3c9c2SDmitry Chagin 
266*3ab3c9c2SDmitry Chagin /*
267*3ab3c9c2SDmitry Chagin  * Translate a FreeBSD interface name to a Linux interface name
268*3ab3c9c2SDmitry Chagin  * by interface index, and return the number of bytes copied to lxname.
269*3ab3c9c2SDmitry Chagin  */
270*3ab3c9c2SDmitry Chagin int
271*3ab3c9c2SDmitry Chagin ifname_bsd_to_linux_idx(u_int idx, char *lxname, size_t len)
272*3ab3c9c2SDmitry Chagin {
273*3ab3c9c2SDmitry Chagin 	struct epoch_tracker et;
274*3ab3c9c2SDmitry Chagin 	struct ifnet *ifp;
275*3ab3c9c2SDmitry Chagin 	int ret;
276*3ab3c9c2SDmitry Chagin 
277*3ab3c9c2SDmitry Chagin 	ret = 0;
278*3ab3c9c2SDmitry Chagin 	CURVNET_SET(TD_TO_VNET(curthread));
279*3ab3c9c2SDmitry Chagin 	NET_EPOCH_ENTER(et);
280*3ab3c9c2SDmitry Chagin 	ifp = ifnet_byindex(idx);
281*3ab3c9c2SDmitry Chagin 	if (ifp != NULL)
282*3ab3c9c2SDmitry Chagin 		ret = ifname_bsd_to_linux_ifp(ifp, lxname, len);
283*3ab3c9c2SDmitry Chagin 	NET_EPOCH_EXIT(et);
284*3ab3c9c2SDmitry Chagin 	CURVNET_RESTORE();
285*3ab3c9c2SDmitry Chagin 	return (ret);
286*3ab3c9c2SDmitry Chagin }
287*3ab3c9c2SDmitry Chagin 
288*3ab3c9c2SDmitry Chagin /*
289*3ab3c9c2SDmitry Chagin  * Translate a FreeBSD interface name to a Linux interface name,
290*3ab3c9c2SDmitry Chagin  * and return the number of bytes copied to lxname.
291*3ab3c9c2SDmitry Chagin  */
292*3ab3c9c2SDmitry Chagin int
293*3ab3c9c2SDmitry Chagin ifname_bsd_to_linux_ifp(struct ifnet *ifp, char *lxname, size_t len)
294*3ab3c9c2SDmitry Chagin {
295*3ab3c9c2SDmitry Chagin 	struct ifnet *ifscan;
296*3ab3c9c2SDmitry Chagin 	int unit;
297*3ab3c9c2SDmitry Chagin 
298*3ab3c9c2SDmitry Chagin 	NET_EPOCH_ASSERT();
299*3ab3c9c2SDmitry Chagin 
300*3ab3c9c2SDmitry Chagin 	/*
301*3ab3c9c2SDmitry Chagin 	 * Linux loopback interface name is lo (not lo0),
302*3ab3c9c2SDmitry Chagin 	 * we translate lo to lo0, loX to loX.
303*3ab3c9c2SDmitry Chagin 	 */
304*3ab3c9c2SDmitry Chagin 	if (IFP_IS_LOOP(ifp) && strncmp(ifp->if_xname, "lo0", IFNAMSIZ) == 0)
305*3ab3c9c2SDmitry Chagin 		return (strlcpy(lxname, "lo", len));
306*3ab3c9c2SDmitry Chagin 
307*3ab3c9c2SDmitry Chagin 	/* Short-circuit non ethernet interfaces. */
308*3ab3c9c2SDmitry Chagin 	if (!IFP_IS_ETH(ifp) || linux_use_real_ifname(ifp))
309*3ab3c9c2SDmitry Chagin 		return (strlcpy(lxname, ifp->if_xname, len));
310*3ab3c9c2SDmitry Chagin 
311*3ab3c9c2SDmitry Chagin 	/* Determine the (relative) unit number for ethernet interfaces. */
312*3ab3c9c2SDmitry Chagin 	unit = 0;
313*3ab3c9c2SDmitry Chagin 	CK_STAILQ_FOREACH(ifscan, &V_ifnet, if_link) {
314*3ab3c9c2SDmitry Chagin 		if (ifscan == ifp)
315*3ab3c9c2SDmitry Chagin 			return (snprintf(lxname, len, "eth%d", unit));
316*3ab3c9c2SDmitry Chagin 		if (IFP_IS_ETH(ifscan))
317*3ab3c9c2SDmitry Chagin 			unit++;
318*3ab3c9c2SDmitry Chagin 	}
319*3ab3c9c2SDmitry Chagin 	return (0);
320*3ab3c9c2SDmitry Chagin }
321*3ab3c9c2SDmitry Chagin 
322*3ab3c9c2SDmitry Chagin /*
323d151344dSDmitry Chagin  * Translate a Linux interface name to a FreeBSD interface name,
324d151344dSDmitry Chagin  * and return the associated ifnet structure
325d151344dSDmitry Chagin  * bsdname and lxname need to be least IFNAMSIZ bytes long, but
326d151344dSDmitry Chagin  * can point to the same buffer.
327d151344dSDmitry Chagin  */
328d151344dSDmitry Chagin struct ifnet *
329d151344dSDmitry Chagin ifname_linux_to_bsd(struct thread *td, const char *lxname, char *bsdname)
330d151344dSDmitry Chagin {
331e9e637bfSJustin Hibbits 	struct ifnet *ifp;
332e9e637bfSJustin Hibbits 	int len, unit;
333d151344dSDmitry Chagin 	char *ep;
334e9e637bfSJustin Hibbits 	int index;
335e9e637bfSJustin Hibbits 	bool is_eth, is_lo;
336d151344dSDmitry Chagin 
337d151344dSDmitry Chagin 	for (len = 0; len < LINUX_IFNAMSIZ; ++len)
3389c1437aeSDmitry Chagin 		if (!isalpha(lxname[len]) || lxname[len] == '\0')
339d151344dSDmitry Chagin 			break;
340d151344dSDmitry Chagin 	if (len == 0 || len == LINUX_IFNAMSIZ)
341d151344dSDmitry Chagin 		return (NULL);
342*3ab3c9c2SDmitry Chagin 	/*
343*3ab3c9c2SDmitry Chagin 	 * Linux loopback interface name is lo (not lo0),
344*3ab3c9c2SDmitry Chagin 	 * we translate lo to lo0, loX to loX.
345*3ab3c9c2SDmitry Chagin 	 */
346*3ab3c9c2SDmitry Chagin 	is_lo = (len == 2 && strncmp(lxname, "lo", LINUX_IFNAMSIZ) == 0);
347e9e637bfSJustin Hibbits 	unit = (int)strtoul(lxname + len, &ep, 10);
348d151344dSDmitry Chagin 	if ((ep == NULL || ep == lxname + len || ep >= lxname + LINUX_IFNAMSIZ) &&
349e9e637bfSJustin Hibbits 	    is_lo == 0)
350d151344dSDmitry Chagin 		return (NULL);
351e9e637bfSJustin Hibbits 	index = 0;
352e9e637bfSJustin Hibbits 	is_eth = (len == 3 && strncmp(lxname, "eth", len) == 0);
353d151344dSDmitry Chagin 
354d151344dSDmitry Chagin 	CURVNET_SET(TD_TO_VNET(td));
355d151344dSDmitry Chagin 	IFNET_RLOCK();
356e9e637bfSJustin Hibbits 	CK_STAILQ_FOREACH(ifp, &V_ifnet, if_link) {
357e9e637bfSJustin Hibbits 		/*
358e9e637bfSJustin Hibbits 		 * Allow Linux programs to use FreeBSD names. Don't presume
359e9e637bfSJustin Hibbits 		 * we never have an interface named "eth", so don't make
360e9e637bfSJustin Hibbits 		 * the test optional based on is_eth.
361e9e637bfSJustin Hibbits 		 */
362e9e637bfSJustin Hibbits 		if (strncmp(ifp->if_xname, lxname, LINUX_IFNAMSIZ) == 0)
363e9e637bfSJustin Hibbits 			break;
364e9e637bfSJustin Hibbits 		if (is_eth && IFP_IS_ETH(ifp) && unit == index++)
365e9e637bfSJustin Hibbits 			break;
366e9e637bfSJustin Hibbits 		if (is_lo && IFP_IS_LOOP(ifp))
367e9e637bfSJustin Hibbits 			break;
368e9e637bfSJustin Hibbits 	}
369d151344dSDmitry Chagin 	IFNET_RUNLOCK();
370d151344dSDmitry Chagin 	CURVNET_RESTORE();
371e9e637bfSJustin Hibbits 	if (ifp != NULL && bsdname != NULL)
372e9e637bfSJustin Hibbits 		strlcpy(bsdname, ifp->if_xname, IFNAMSIZ);
373e9e637bfSJustin Hibbits 	return (ifp);
374d151344dSDmitry Chagin }
375d151344dSDmitry Chagin 
376d151344dSDmitry Chagin void
377d151344dSDmitry Chagin linux_ifflags(struct ifnet *ifp, short *flags)
378d151344dSDmitry Chagin {
379bbac65c7SDmitry Chagin 	unsigned short fl;
380d151344dSDmitry Chagin 
381e9e637bfSJustin Hibbits 	fl = (ifp->if_flags | ifp->if_drv_flags) & 0xffff;
382bbac65c7SDmitry Chagin 	*flags = 0;
383bbac65c7SDmitry Chagin 	if (fl & IFF_UP)
384bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_UP;
385bbac65c7SDmitry Chagin 	if (fl & IFF_BROADCAST)
386bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_BROADCAST;
387bbac65c7SDmitry Chagin 	if (fl & IFF_DEBUG)
388bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_DEBUG;
389bbac65c7SDmitry Chagin 	if (fl & IFF_LOOPBACK)
390bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_LOOPBACK;
391bbac65c7SDmitry Chagin 	if (fl & IFF_POINTOPOINT)
392bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_POINTOPOINT;
393bbac65c7SDmitry Chagin 	if (fl & IFF_DRV_RUNNING)
394bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_RUNNING;
395bbac65c7SDmitry Chagin 	if (fl & IFF_NOARP)
396bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_NOARP;
397bbac65c7SDmitry Chagin 	if (fl & IFF_PROMISC)
398bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_PROMISC;
399bbac65c7SDmitry Chagin 	if (fl & IFF_ALLMULTI)
400bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_ALLMULTI;
401bbac65c7SDmitry Chagin 	if (fl & IFF_MULTICAST)
402bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_MULTICAST;
403d151344dSDmitry Chagin }
404d151344dSDmitry Chagin 
405d151344dSDmitry Chagin int
406d151344dSDmitry Chagin linux_ifhwaddr(struct ifnet *ifp, struct l_sockaddr *lsa)
407d151344dSDmitry Chagin {
408e9e637bfSJustin Hibbits 	struct ifaddr *ifa;
409e9e637bfSJustin Hibbits 	struct sockaddr_dl *sdl;
410d151344dSDmitry Chagin 
411d151344dSDmitry Chagin 	if (IFP_IS_LOOP(ifp)) {
412d151344dSDmitry Chagin 		bzero(lsa, sizeof(*lsa));
413d151344dSDmitry Chagin 		lsa->sa_family = LINUX_ARPHRD_LOOPBACK;
414d151344dSDmitry Chagin 		return (0);
415d151344dSDmitry Chagin 	}
416d151344dSDmitry Chagin 
417d151344dSDmitry Chagin 	if (!IFP_IS_ETH(ifp))
418d151344dSDmitry Chagin 		return (ENOENT);
419d151344dSDmitry Chagin 
420e9e637bfSJustin Hibbits 	CK_STAILQ_FOREACH(ifa, &ifp->if_addrhead, ifa_link) {
421e9e637bfSJustin Hibbits 		sdl = (struct sockaddr_dl*)ifa->ifa_addr;
422e9e637bfSJustin Hibbits 		if (sdl != NULL && (sdl->sdl_family == AF_LINK) &&
423e9e637bfSJustin Hibbits 		    (sdl->sdl_type == IFT_ETHER)) {
424e9e637bfSJustin Hibbits 			bzero(lsa, sizeof(*lsa));
425e9e637bfSJustin Hibbits 			lsa->sa_family = LINUX_ARPHRD_ETHER;
426e9e637bfSJustin Hibbits 			bcopy(LLADDR(sdl), lsa->sa_data, LINUX_IFHWADDRLEN);
427d151344dSDmitry Chagin 			return (0);
428e9e637bfSJustin Hibbits 		}
429e9e637bfSJustin Hibbits 	}
430d151344dSDmitry Chagin 
431d151344dSDmitry Chagin 	return (ENOENT);
432d151344dSDmitry Chagin }
433d5368bf3SDmitry Chagin 
434d5368bf3SDmitry Chagin int
435d5368bf3SDmitry Chagin linux_to_bsd_domain(int domain)
436d5368bf3SDmitry Chagin {
437d5368bf3SDmitry Chagin 
438d5368bf3SDmitry Chagin 	switch (domain) {
439d5368bf3SDmitry Chagin 	case LINUX_AF_UNSPEC:
440d5368bf3SDmitry Chagin 		return (AF_UNSPEC);
441d5368bf3SDmitry Chagin 	case LINUX_AF_UNIX:
442d5368bf3SDmitry Chagin 		return (AF_LOCAL);
443d5368bf3SDmitry Chagin 	case LINUX_AF_INET:
444d5368bf3SDmitry Chagin 		return (AF_INET);
445d5368bf3SDmitry Chagin 	case LINUX_AF_INET6:
446d5368bf3SDmitry Chagin 		return (AF_INET6);
447d5368bf3SDmitry Chagin 	case LINUX_AF_AX25:
448d5368bf3SDmitry Chagin 		return (AF_CCITT);
449d5368bf3SDmitry Chagin 	case LINUX_AF_IPX:
450d5368bf3SDmitry Chagin 		return (AF_IPX);
451d5368bf3SDmitry Chagin 	case LINUX_AF_APPLETALK:
452d5368bf3SDmitry Chagin 		return (AF_APPLETALK);
4537c40e2d5SAlexander V. Chernikov 	case LINUX_AF_NETLINK:
4547c40e2d5SAlexander V. Chernikov 		return (AF_NETLINK);
455d5368bf3SDmitry Chagin 	}
456d5368bf3SDmitry Chagin 	return (-1);
457d5368bf3SDmitry Chagin }
458d5368bf3SDmitry Chagin 
459d5368bf3SDmitry Chagin int
460d5368bf3SDmitry Chagin bsd_to_linux_domain(int domain)
461d5368bf3SDmitry Chagin {
462d5368bf3SDmitry Chagin 
463d5368bf3SDmitry Chagin 	switch (domain) {
464d5368bf3SDmitry Chagin 	case AF_UNSPEC:
465d5368bf3SDmitry Chagin 		return (LINUX_AF_UNSPEC);
466d5368bf3SDmitry Chagin 	case AF_LOCAL:
467d5368bf3SDmitry Chagin 		return (LINUX_AF_UNIX);
468d5368bf3SDmitry Chagin 	case AF_INET:
469d5368bf3SDmitry Chagin 		return (LINUX_AF_INET);
470d5368bf3SDmitry Chagin 	case AF_INET6:
471d5368bf3SDmitry Chagin 		return (LINUX_AF_INET6);
472d5368bf3SDmitry Chagin 	case AF_CCITT:
473d5368bf3SDmitry Chagin 		return (LINUX_AF_AX25);
474d5368bf3SDmitry Chagin 	case AF_IPX:
475d5368bf3SDmitry Chagin 		return (LINUX_AF_IPX);
476d5368bf3SDmitry Chagin 	case AF_APPLETALK:
477d5368bf3SDmitry Chagin 		return (LINUX_AF_APPLETALK);
4787c40e2d5SAlexander V. Chernikov 	case AF_NETLINK:
4797c40e2d5SAlexander V. Chernikov 		return (LINUX_AF_NETLINK);
480d5368bf3SDmitry Chagin 	}
481d5368bf3SDmitry Chagin 	return (-1);
482d5368bf3SDmitry Chagin }
483d5368bf3SDmitry Chagin 
484d5368bf3SDmitry Chagin /*
485d5368bf3SDmitry Chagin  * Based on the fact that:
486d5368bf3SDmitry Chagin  * 1. Native and Linux storage of struct sockaddr
487d5368bf3SDmitry Chagin  * and struct sockaddr_in6 are equal.
488d5368bf3SDmitry Chagin  * 2. On Linux sa_family is the first member of all struct sockaddr.
489d5368bf3SDmitry Chagin  */
490d5368bf3SDmitry Chagin int
491d5368bf3SDmitry Chagin bsd_to_linux_sockaddr(const struct sockaddr *sa, struct l_sockaddr **lsa,
492d5368bf3SDmitry Chagin     socklen_t len)
493d5368bf3SDmitry Chagin {
494d5368bf3SDmitry Chagin 	struct l_sockaddr *kosa;
495bbddd588SDmitry Chagin 	int bdom;
496d5368bf3SDmitry Chagin 
497d5368bf3SDmitry Chagin 	*lsa = NULL;
498d5368bf3SDmitry Chagin 	if (len < 2 || len > UCHAR_MAX)
499d5368bf3SDmitry Chagin 		return (EINVAL);
500bbddd588SDmitry Chagin 	bdom = bsd_to_linux_domain(sa->sa_family);
501bbddd588SDmitry Chagin 	if (bdom == -1)
502bbddd588SDmitry Chagin 		return (EAFNOSUPPORT);
503d5368bf3SDmitry Chagin 
50463355839SDmitry Chagin 	kosa = malloc(len, M_LINUX, M_WAITOK);
505d5368bf3SDmitry Chagin 	bcopy(sa, kosa, len);
506d5368bf3SDmitry Chagin 	kosa->sa_family = bdom;
507d5368bf3SDmitry Chagin 	*lsa = kosa;
508d5368bf3SDmitry Chagin 	return (0);
509d5368bf3SDmitry Chagin }
510d5368bf3SDmitry Chagin 
511d5368bf3SDmitry Chagin int
512d5368bf3SDmitry Chagin linux_to_bsd_sockaddr(const struct l_sockaddr *osa, struct sockaddr **sap,
513d5368bf3SDmitry Chagin     socklen_t *len)
514d5368bf3SDmitry Chagin {
515d5368bf3SDmitry Chagin 	struct sockaddr *sa;
516d5368bf3SDmitry Chagin 	struct l_sockaddr *kosa;
517d5368bf3SDmitry Chagin #ifdef INET6
518d5368bf3SDmitry Chagin 	struct sockaddr_in6 *sin6;
519d5368bf3SDmitry Chagin 	bool  oldv6size;
520d5368bf3SDmitry Chagin #endif
521d5368bf3SDmitry Chagin 	char *name;
522d5368bf3SDmitry Chagin 	int salen, bdom, error, hdrlen, namelen;
523d5368bf3SDmitry Chagin 
524d5368bf3SDmitry Chagin 	if (*len < 2 || *len > UCHAR_MAX)
525d5368bf3SDmitry Chagin 		return (EINVAL);
526d5368bf3SDmitry Chagin 
527d5368bf3SDmitry Chagin 	salen = *len;
528d5368bf3SDmitry Chagin 
529d5368bf3SDmitry Chagin #ifdef INET6
530d5368bf3SDmitry Chagin 	oldv6size = false;
531d5368bf3SDmitry Chagin 	/*
532d5368bf3SDmitry Chagin 	 * Check for old (pre-RFC2553) sockaddr_in6. We may accept it
533d5368bf3SDmitry Chagin 	 * if it's a v4-mapped address, so reserve the proper space
534d5368bf3SDmitry Chagin 	 * for it.
535d5368bf3SDmitry Chagin 	 */
536d5368bf3SDmitry Chagin 	if (salen == sizeof(struct sockaddr_in6) - sizeof(uint32_t)) {
537d5368bf3SDmitry Chagin 		salen += sizeof(uint32_t);
538d5368bf3SDmitry Chagin 		oldv6size = true;
539d5368bf3SDmitry Chagin 	}
540d5368bf3SDmitry Chagin #endif
541d5368bf3SDmitry Chagin 
542d5368bf3SDmitry Chagin 	kosa = malloc(salen, M_SONAME, M_WAITOK);
543d5368bf3SDmitry Chagin 
544d5368bf3SDmitry Chagin 	if ((error = copyin(osa, kosa, *len)))
545d5368bf3SDmitry Chagin 		goto out;
546d5368bf3SDmitry Chagin 
547d5368bf3SDmitry Chagin 	bdom = linux_to_bsd_domain(kosa->sa_family);
548d5368bf3SDmitry Chagin 	if (bdom == -1) {
549d5368bf3SDmitry Chagin 		error = EAFNOSUPPORT;
550d5368bf3SDmitry Chagin 		goto out;
551d5368bf3SDmitry Chagin 	}
552d5368bf3SDmitry Chagin 
553d5368bf3SDmitry Chagin #ifdef INET6
554d5368bf3SDmitry Chagin 	/*
555d5368bf3SDmitry Chagin 	 * Older Linux IPv6 code uses obsolete RFC2133 struct sockaddr_in6,
556d5368bf3SDmitry Chagin 	 * which lacks the scope id compared with RFC2553 one. If we detect
557d5368bf3SDmitry Chagin 	 * the situation, reject the address and write a message to system log.
558d5368bf3SDmitry Chagin 	 *
559d5368bf3SDmitry Chagin 	 * Still accept addresses for which the scope id is not used.
560d5368bf3SDmitry Chagin 	 */
561d5368bf3SDmitry Chagin 	if (oldv6size) {
562d5368bf3SDmitry Chagin 		if (bdom == AF_INET6) {
563d5368bf3SDmitry Chagin 			sin6 = (struct sockaddr_in6 *)kosa;
564d5368bf3SDmitry Chagin 			if (IN6_IS_ADDR_V4MAPPED(&sin6->sin6_addr) ||
565d5368bf3SDmitry Chagin 			    (!IN6_IS_ADDR_LINKLOCAL(&sin6->sin6_addr) &&
566d5368bf3SDmitry Chagin 			     !IN6_IS_ADDR_SITELOCAL(&sin6->sin6_addr) &&
567d5368bf3SDmitry Chagin 			     !IN6_IS_ADDR_V4COMPAT(&sin6->sin6_addr) &&
568d5368bf3SDmitry Chagin 			     !IN6_IS_ADDR_UNSPECIFIED(&sin6->sin6_addr) &&
569d5368bf3SDmitry Chagin 			     !IN6_IS_ADDR_MULTICAST(&sin6->sin6_addr))) {
570d5368bf3SDmitry Chagin 				sin6->sin6_scope_id = 0;
571d5368bf3SDmitry Chagin 			} else {
572d5368bf3SDmitry Chagin 				linux_msg(curthread,
57386e794ebSEdward Tomasz Napierala 				    "obsolete pre-RFC2553 sockaddr_in6 rejected");
574d5368bf3SDmitry Chagin 				error = EINVAL;
575d5368bf3SDmitry Chagin 				goto out;
576d5368bf3SDmitry Chagin 			}
577d5368bf3SDmitry Chagin 		} else
578d5368bf3SDmitry Chagin 			salen -= sizeof(uint32_t);
579d5368bf3SDmitry Chagin 	}
580d5368bf3SDmitry Chagin #endif
581d5368bf3SDmitry Chagin 	if (bdom == AF_INET) {
582d5368bf3SDmitry Chagin 		if (salen < sizeof(struct sockaddr_in)) {
583d5368bf3SDmitry Chagin 			error = EINVAL;
584d5368bf3SDmitry Chagin 			goto out;
585d5368bf3SDmitry Chagin 		}
586d5368bf3SDmitry Chagin 		salen = sizeof(struct sockaddr_in);
587d5368bf3SDmitry Chagin 	}
588d5368bf3SDmitry Chagin 
589d5368bf3SDmitry Chagin 	if (bdom == AF_LOCAL && salen > sizeof(struct sockaddr_un)) {
590d5368bf3SDmitry Chagin 		hdrlen = offsetof(struct sockaddr_un, sun_path);
591d5368bf3SDmitry Chagin 		name = ((struct sockaddr_un *)kosa)->sun_path;
592d5368bf3SDmitry Chagin 		if (*name == '\0') {
593d5368bf3SDmitry Chagin 			/*
594d5368bf3SDmitry Chagin 			 * Linux abstract namespace starts with a NULL byte.
595d5368bf3SDmitry Chagin 			 * XXX We do not support abstract namespace yet.
596d5368bf3SDmitry Chagin 			 */
597d5368bf3SDmitry Chagin 			namelen = strnlen(name + 1, salen - hdrlen - 1) + 1;
598d5368bf3SDmitry Chagin 		} else
599d5368bf3SDmitry Chagin 			namelen = strnlen(name, salen - hdrlen);
600d5368bf3SDmitry Chagin 		salen = hdrlen + namelen;
601d5368bf3SDmitry Chagin 		if (salen > sizeof(struct sockaddr_un)) {
602d5368bf3SDmitry Chagin 			error = ENAMETOOLONG;
603d5368bf3SDmitry Chagin 			goto out;
604d5368bf3SDmitry Chagin 		}
605d5368bf3SDmitry Chagin 	}
606d5368bf3SDmitry Chagin 
6077c40e2d5SAlexander V. Chernikov 	if (bdom == AF_NETLINK) {
6087c40e2d5SAlexander V. Chernikov 		if (salen < sizeof(struct sockaddr_nl)) {
6097c40e2d5SAlexander V. Chernikov 			error = EINVAL;
6107c40e2d5SAlexander V. Chernikov 			goto out;
6117c40e2d5SAlexander V. Chernikov 		}
6127c40e2d5SAlexander V. Chernikov 		salen = sizeof(struct sockaddr_nl);
6137c40e2d5SAlexander V. Chernikov 	}
6147c40e2d5SAlexander V. Chernikov 
615d5368bf3SDmitry Chagin 	sa = (struct sockaddr *)kosa;
616d5368bf3SDmitry Chagin 	sa->sa_family = bdom;
617d5368bf3SDmitry Chagin 	sa->sa_len = salen;
618d5368bf3SDmitry Chagin 
619d5368bf3SDmitry Chagin 	*sap = sa;
620d5368bf3SDmitry Chagin 	*len = salen;
621d5368bf3SDmitry Chagin 	return (0);
622d5368bf3SDmitry Chagin 
623d5368bf3SDmitry Chagin out:
624d5368bf3SDmitry Chagin 	free(kosa, M_SONAME);
625d5368bf3SDmitry Chagin 	return (error);
626d5368bf3SDmitry Chagin }
627044ab55eSEdward Tomasz Napierala 
628044ab55eSEdward Tomasz Napierala void
629044ab55eSEdward Tomasz Napierala linux_dev_shm_create(void)
630044ab55eSEdward Tomasz Napierala {
631044ab55eSEdward Tomasz Napierala 	int error;
632044ab55eSEdward Tomasz Napierala 
633044ab55eSEdward Tomasz Napierala 	error = make_dev_p(MAKEDEV_CHECKNAME | MAKEDEV_WAITOK, &dev_shm_cdev,
634044ab55eSEdward Tomasz Napierala 	    &dev_shm_cdevsw, NULL, UID_ROOT, GID_WHEEL, 0, "shm/.mountpoint");
635044ab55eSEdward Tomasz Napierala 	if (error != 0) {
636044ab55eSEdward Tomasz Napierala 		printf("%s: failed to create device node, error %d\n",
637044ab55eSEdward Tomasz Napierala 		    __func__, error);
638044ab55eSEdward Tomasz Napierala 	}
639044ab55eSEdward Tomasz Napierala }
640044ab55eSEdward Tomasz Napierala 
641044ab55eSEdward Tomasz Napierala void
642044ab55eSEdward Tomasz Napierala linux_dev_shm_destroy(void)
643044ab55eSEdward Tomasz Napierala {
644044ab55eSEdward Tomasz Napierala 
645044ab55eSEdward Tomasz Napierala 	destroy_dev(dev_shm_cdev);
646044ab55eSEdward Tomasz Napierala }
6475403f186SKyle Evans 
6485403f186SKyle Evans int
6495403f186SKyle Evans bsd_to_linux_bits_(int value, struct bsd_to_linux_bitmap *bitmap,
6505403f186SKyle Evans     size_t mapcnt, int no_value)
6515403f186SKyle Evans {
6525403f186SKyle Evans 	int bsd_mask, bsd_value, linux_mask, linux_value;
6535403f186SKyle Evans 	int linux_ret;
6545403f186SKyle Evans 	size_t i;
6555403f186SKyle Evans 	bool applied;
6565403f186SKyle Evans 
6575403f186SKyle Evans 	applied = false;
6585403f186SKyle Evans 	linux_ret = 0;
6595403f186SKyle Evans 	for (i = 0; i < mapcnt; ++i) {
6605403f186SKyle Evans 		bsd_mask = bitmap[i].bsd_mask;
6615403f186SKyle Evans 		bsd_value = bitmap[i].bsd_value;
6625403f186SKyle Evans 		if (bsd_mask == 0)
6635403f186SKyle Evans 			bsd_mask = bsd_value;
6645403f186SKyle Evans 
6655403f186SKyle Evans 		linux_mask = bitmap[i].linux_mask;
6665403f186SKyle Evans 		linux_value = bitmap[i].linux_value;
6675403f186SKyle Evans 		if (linux_mask == 0)
6685403f186SKyle Evans 			linux_mask = linux_value;
6695403f186SKyle Evans 
6705403f186SKyle Evans 		/*
6715403f186SKyle Evans 		 * If a mask larger than just the value is set, we explicitly
6725403f186SKyle Evans 		 * want to make sure that only this bit we mapped within that
6735403f186SKyle Evans 		 * mask is set.
6745403f186SKyle Evans 		 */
6755403f186SKyle Evans 		if ((value & bsd_mask) == bsd_value) {
6765403f186SKyle Evans 			linux_ret = (linux_ret & ~linux_mask) | linux_value;
6775403f186SKyle Evans 			applied = true;
6785403f186SKyle Evans 		}
6795403f186SKyle Evans 	}
6805403f186SKyle Evans 
6815403f186SKyle Evans 	if (!applied)
6825403f186SKyle Evans 		return (no_value);
6835403f186SKyle Evans 	return (linux_ret);
6845403f186SKyle Evans }
6855403f186SKyle Evans 
6865403f186SKyle Evans int
6875403f186SKyle Evans linux_to_bsd_bits_(int value, struct bsd_to_linux_bitmap *bitmap,
6885403f186SKyle Evans     size_t mapcnt, int no_value)
6895403f186SKyle Evans {
6905403f186SKyle Evans 	int bsd_mask, bsd_value, linux_mask, linux_value;
6915403f186SKyle Evans 	int bsd_ret;
6925403f186SKyle Evans 	size_t i;
6935403f186SKyle Evans 	bool applied;
6945403f186SKyle Evans 
6955403f186SKyle Evans 	applied = false;
6965403f186SKyle Evans 	bsd_ret = 0;
6975403f186SKyle Evans 	for (i = 0; i < mapcnt; ++i) {
6985403f186SKyle Evans 		bsd_mask = bitmap[i].bsd_mask;
6995403f186SKyle Evans 		bsd_value = bitmap[i].bsd_value;
7005403f186SKyle Evans 		if (bsd_mask == 0)
7015403f186SKyle Evans 			bsd_mask = bsd_value;
7025403f186SKyle Evans 
7035403f186SKyle Evans 		linux_mask = bitmap[i].linux_mask;
7045403f186SKyle Evans 		linux_value = bitmap[i].linux_value;
7055403f186SKyle Evans 		if (linux_mask == 0)
7065403f186SKyle Evans 			linux_mask = linux_value;
7075403f186SKyle Evans 
7085403f186SKyle Evans 		/*
7095403f186SKyle Evans 		 * If a mask larger than just the value is set, we explicitly
7105403f186SKyle Evans 		 * want to make sure that only this bit we mapped within that
7115403f186SKyle Evans 		 * mask is set.
7125403f186SKyle Evans 		 */
7135403f186SKyle Evans 		if ((value & linux_mask) == linux_value) {
7145403f186SKyle Evans 			bsd_ret = (bsd_ret & ~bsd_mask) | bsd_value;
7155403f186SKyle Evans 			applied = true;
7165403f186SKyle Evans 		}
7175403f186SKyle Evans 	}
7185403f186SKyle Evans 
7195403f186SKyle Evans 	if (!applied)
7205403f186SKyle Evans 		return (no_value);
7215403f186SKyle Evans 	return (bsd_ret);
7225403f186SKyle Evans }
72326795a03SDmitry Chagin 
72426795a03SDmitry Chagin void
72526795a03SDmitry Chagin linux_to_bsd_poll_events(struct thread *td, int fd, short lev,
72626795a03SDmitry Chagin     short *bev)
72726795a03SDmitry Chagin {
72826795a03SDmitry Chagin 	struct file *fp;
72926795a03SDmitry Chagin 	int error;
73026795a03SDmitry Chagin 	short bits = 0;
73126795a03SDmitry Chagin 
73226795a03SDmitry Chagin 	if (lev & LINUX_POLLIN)
73326795a03SDmitry Chagin 		bits |= POLLIN;
73426795a03SDmitry Chagin 	if (lev & LINUX_POLLPRI)
73526795a03SDmitry Chagin 		bits |=	POLLPRI;
73626795a03SDmitry Chagin 	if (lev & LINUX_POLLOUT)
73726795a03SDmitry Chagin 		bits |= POLLOUT;
73826795a03SDmitry Chagin 	if (lev & LINUX_POLLERR)
73926795a03SDmitry Chagin 		bits |= POLLERR;
74026795a03SDmitry Chagin 	if (lev & LINUX_POLLHUP)
74126795a03SDmitry Chagin 		bits |= POLLHUP;
74226795a03SDmitry Chagin 	if (lev & LINUX_POLLNVAL)
74326795a03SDmitry Chagin 		bits |= POLLNVAL;
74426795a03SDmitry Chagin 	if (lev & LINUX_POLLRDNORM)
74526795a03SDmitry Chagin 		bits |= POLLRDNORM;
74626795a03SDmitry Chagin 	if (lev & LINUX_POLLRDBAND)
74726795a03SDmitry Chagin 		bits |= POLLRDBAND;
74826795a03SDmitry Chagin 	if (lev & LINUX_POLLWRBAND)
74926795a03SDmitry Chagin 		bits |= POLLWRBAND;
75026795a03SDmitry Chagin 	if (lev & LINUX_POLLWRNORM)
75126795a03SDmitry Chagin 		bits |= POLLWRNORM;
75226795a03SDmitry Chagin 
75326795a03SDmitry Chagin 	if (lev & LINUX_POLLRDHUP) {
75426795a03SDmitry Chagin 		/*
75526795a03SDmitry Chagin 		 * It seems that the Linux silencly ignores POLLRDHUP
75626795a03SDmitry Chagin 		 * on non-socket file descriptors unlike FreeBSD, where
75726795a03SDmitry Chagin 		 * events bits is more strictly checked (POLLSTANDARD).
75826795a03SDmitry Chagin 		 */
759513c7a6eSMateusz Guzik 		error = fget_unlocked(td, fd, &cap_no_rights, &fp);
76026795a03SDmitry Chagin 		if (error == 0) {
76126795a03SDmitry Chagin 			/*
76226795a03SDmitry Chagin 			 * XXX. On FreeBSD POLLRDHUP applies only to
76326795a03SDmitry Chagin 			 * stream sockets.
76426795a03SDmitry Chagin 			 */
76526795a03SDmitry Chagin 			if (fp->f_type == DTYPE_SOCKET)
76626795a03SDmitry Chagin 				bits |= POLLRDHUP;
76726795a03SDmitry Chagin 			fdrop(fp, td);
76826795a03SDmitry Chagin 		}
76926795a03SDmitry Chagin 	}
77026795a03SDmitry Chagin 
77126795a03SDmitry Chagin 	if (lev & LINUX_POLLMSG)
77226795a03SDmitry Chagin 		LINUX_RATELIMIT_MSG_OPT1("unsupported POLLMSG, events(%d)", lev);
77326795a03SDmitry Chagin 	if (lev & LINUX_POLLREMOVE)
77426795a03SDmitry Chagin 		LINUX_RATELIMIT_MSG_OPT1("unsupported POLLREMOVE, events(%d)", lev);
77526795a03SDmitry Chagin 
77626795a03SDmitry Chagin 	*bev = bits;
77726795a03SDmitry Chagin }
77826795a03SDmitry Chagin 
77926795a03SDmitry Chagin void
78026795a03SDmitry Chagin bsd_to_linux_poll_events(short bev, short *lev)
78126795a03SDmitry Chagin {
78226795a03SDmitry Chagin 	short bits = 0;
78326795a03SDmitry Chagin 
78426795a03SDmitry Chagin 	if (bev & POLLIN)
78526795a03SDmitry Chagin 		bits |= LINUX_POLLIN;
78626795a03SDmitry Chagin 	if (bev & POLLPRI)
78726795a03SDmitry Chagin 		bits |=	LINUX_POLLPRI;
78826795a03SDmitry Chagin 	if (bev & (POLLOUT | POLLWRNORM))
78926795a03SDmitry Chagin 		/*
79026795a03SDmitry Chagin 		 * POLLWRNORM is equal to POLLOUT on FreeBSD,
79126795a03SDmitry Chagin 		 * but not on Linux
79226795a03SDmitry Chagin 		 */
79326795a03SDmitry Chagin 		bits |= LINUX_POLLOUT;
79426795a03SDmitry Chagin 	if (bev & POLLERR)
79526795a03SDmitry Chagin 		bits |= LINUX_POLLERR;
79626795a03SDmitry Chagin 	if (bev & POLLHUP)
79726795a03SDmitry Chagin 		bits |= LINUX_POLLHUP;
79826795a03SDmitry Chagin 	if (bev & POLLNVAL)
79926795a03SDmitry Chagin 		bits |= LINUX_POLLNVAL;
80026795a03SDmitry Chagin 	if (bev & POLLRDNORM)
80126795a03SDmitry Chagin 		bits |= LINUX_POLLRDNORM;
80226795a03SDmitry Chagin 	if (bev & POLLRDBAND)
80326795a03SDmitry Chagin 		bits |= LINUX_POLLRDBAND;
80426795a03SDmitry Chagin 	if (bev & POLLWRBAND)
80526795a03SDmitry Chagin 		bits |= LINUX_POLLWRBAND;
80626795a03SDmitry Chagin 	if (bev & POLLRDHUP)
80726795a03SDmitry Chagin 		bits |= LINUX_POLLRDHUP;
80826795a03SDmitry Chagin 
80926795a03SDmitry Chagin 	*lev = bits;
81026795a03SDmitry Chagin }
81132fdc75fSDmitry Chagin 
81232fdc75fSDmitry Chagin bool
81332fdc75fSDmitry Chagin linux_use_real_ifname(const struct ifnet *ifp)
81432fdc75fSDmitry Chagin {
81532fdc75fSDmitry Chagin 
816*3ab3c9c2SDmitry Chagin 	return (use_real_ifnames);
81732fdc75fSDmitry Chagin }
818