xref: /freebsd/sys/compat/linux/linux.c (revision 044ab55e41712a07fcec3e803eda03cf9cb36c60)
14ab7403bSDmitry Chagin /*-
24ab7403bSDmitry Chagin  * Copyright (c) 2015 Dmitry Chagin
34ab7403bSDmitry Chagin  * All rights reserved.
44ab7403bSDmitry Chagin  *
54ab7403bSDmitry Chagin  * Redistribution and use in source and binary forms, with or without
64ab7403bSDmitry Chagin  * modification, are permitted provided that the following conditions
74ab7403bSDmitry Chagin  * are met:
84ab7403bSDmitry Chagin  * 1. Redistributions of source code must retain the above copyright
94ab7403bSDmitry Chagin  *    notice, this list of conditions and the following disclaimer.
104ab7403bSDmitry Chagin  * 2. Redistributions in binary form must reproduce the above copyright
114ab7403bSDmitry Chagin  *    notice, this list of conditions and the following disclaimer in the
124ab7403bSDmitry Chagin  *    documentation and/or other materials provided with the distribution.
134ab7403bSDmitry Chagin  *
144ab7403bSDmitry Chagin  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
154ab7403bSDmitry Chagin  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
164ab7403bSDmitry Chagin  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
174ab7403bSDmitry Chagin  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
184ab7403bSDmitry Chagin  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
194ab7403bSDmitry Chagin  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
204ab7403bSDmitry Chagin  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
214ab7403bSDmitry Chagin  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
224ab7403bSDmitry Chagin  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
234ab7403bSDmitry Chagin  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
244ab7403bSDmitry Chagin  * SUCH DAMAGE.
254ab7403bSDmitry Chagin  */
264ab7403bSDmitry Chagin 
274ab7403bSDmitry Chagin #include <sys/cdefs.h>
284ab7403bSDmitry Chagin __FBSDID("$FreeBSD$");
294ab7403bSDmitry Chagin 
30d5368bf3SDmitry Chagin #include <opt_inet6.h>
31d5368bf3SDmitry Chagin 
324ab7403bSDmitry Chagin #include <sys/param.h>
334ab7403bSDmitry Chagin #include <sys/systm.h>
34*044ab55eSEdward Tomasz Napierala #include <sys/conf.h>
35d151344dSDmitry Chagin #include <sys/ctype.h>
36d151344dSDmitry Chagin #include <sys/jail.h>
37d151344dSDmitry Chagin #include <sys/lock.h>
38d5368bf3SDmitry Chagin #include <sys/malloc.h>
394ab7403bSDmitry Chagin #include <sys/signalvar.h>
40d151344dSDmitry Chagin #include <sys/socket.h>
41d5368bf3SDmitry Chagin #include <sys/socketvar.h>
42d151344dSDmitry Chagin 
43d151344dSDmitry Chagin #include <net/if.h>
44d151344dSDmitry Chagin #include <net/if_var.h>
45d151344dSDmitry Chagin #include <net/if_dl.h>
46d151344dSDmitry Chagin #include <net/if_types.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>
53d5368bf3SDmitry Chagin #include <compat/linux/linux_util.h>
544ab7403bSDmitry Chagin 
55b9d3556aSYuri Pankov struct futex_list futex_list;
56a161fba9SYuri Pankov struct mtx futex_mtx;			/* protects the futex list */
57b9d3556aSYuri Pankov 
58d151344dSDmitry Chagin CTASSERT(LINUX_IFNAMSIZ == IFNAMSIZ);
594ab7403bSDmitry Chagin 
604ab7403bSDmitry Chagin static int bsd_to_linux_sigtbl[LINUX_SIGTBLSZ] = {
614ab7403bSDmitry Chagin 	LINUX_SIGHUP,	/* SIGHUP */
624ab7403bSDmitry Chagin 	LINUX_SIGINT,	/* SIGINT */
634ab7403bSDmitry Chagin 	LINUX_SIGQUIT,	/* SIGQUIT */
644ab7403bSDmitry Chagin 	LINUX_SIGILL,	/* SIGILL */
654ab7403bSDmitry Chagin 	LINUX_SIGTRAP,	/* SIGTRAP */
664ab7403bSDmitry Chagin 	LINUX_SIGABRT,	/* SIGABRT */
674ab7403bSDmitry Chagin 	0,		/* SIGEMT */
684ab7403bSDmitry Chagin 	LINUX_SIGFPE,	/* SIGFPE */
694ab7403bSDmitry Chagin 	LINUX_SIGKILL,	/* SIGKILL */
704ab7403bSDmitry Chagin 	LINUX_SIGBUS,	/* SIGBUS */
714ab7403bSDmitry Chagin 	LINUX_SIGSEGV,	/* SIGSEGV */
724ab7403bSDmitry Chagin 	LINUX_SIGSYS,	/* SIGSYS */
734ab7403bSDmitry Chagin 	LINUX_SIGPIPE,	/* SIGPIPE */
744ab7403bSDmitry Chagin 	LINUX_SIGALRM,	/* SIGALRM */
754ab7403bSDmitry Chagin 	LINUX_SIGTERM,	/* SIGTERM */
764ab7403bSDmitry Chagin 	LINUX_SIGURG,	/* SIGURG */
774ab7403bSDmitry Chagin 	LINUX_SIGSTOP,	/* SIGSTOP */
784ab7403bSDmitry Chagin 	LINUX_SIGTSTP,	/* SIGTSTP */
794ab7403bSDmitry Chagin 	LINUX_SIGCONT,	/* SIGCONT */
804ab7403bSDmitry Chagin 	LINUX_SIGCHLD,	/* SIGCHLD */
814ab7403bSDmitry Chagin 	LINUX_SIGTTIN,	/* SIGTTIN */
824ab7403bSDmitry Chagin 	LINUX_SIGTTOU,	/* SIGTTOU */
834ab7403bSDmitry Chagin 	LINUX_SIGIO,	/* SIGIO */
844ab7403bSDmitry Chagin 	LINUX_SIGXCPU,	/* SIGXCPU */
854ab7403bSDmitry Chagin 	LINUX_SIGXFSZ,	/* SIGXFSZ */
864ab7403bSDmitry Chagin 	LINUX_SIGVTALRM,/* SIGVTALRM */
874ab7403bSDmitry Chagin 	LINUX_SIGPROF,	/* SIGPROF */
884ab7403bSDmitry Chagin 	LINUX_SIGWINCH,	/* SIGWINCH */
894ab7403bSDmitry Chagin 	0,		/* SIGINFO */
904ab7403bSDmitry Chagin 	LINUX_SIGUSR1,	/* SIGUSR1 */
914ab7403bSDmitry Chagin 	LINUX_SIGUSR2	/* SIGUSR2 */
924ab7403bSDmitry Chagin };
934ab7403bSDmitry Chagin 
944ab7403bSDmitry Chagin static int linux_to_bsd_sigtbl[LINUX_SIGTBLSZ] = {
954ab7403bSDmitry Chagin 	SIGHUP,		/* LINUX_SIGHUP */
964ab7403bSDmitry Chagin 	SIGINT,		/* LINUX_SIGINT */
974ab7403bSDmitry Chagin 	SIGQUIT,	/* LINUX_SIGQUIT */
984ab7403bSDmitry Chagin 	SIGILL,		/* LINUX_SIGILL */
994ab7403bSDmitry Chagin 	SIGTRAP,	/* LINUX_SIGTRAP */
1004ab7403bSDmitry Chagin 	SIGABRT,	/* LINUX_SIGABRT */
1014ab7403bSDmitry Chagin 	SIGBUS,		/* LINUX_SIGBUS */
1024ab7403bSDmitry Chagin 	SIGFPE,		/* LINUX_SIGFPE */
1034ab7403bSDmitry Chagin 	SIGKILL,	/* LINUX_SIGKILL */
1044ab7403bSDmitry Chagin 	SIGUSR1,	/* LINUX_SIGUSR1 */
1054ab7403bSDmitry Chagin 	SIGSEGV,	/* LINUX_SIGSEGV */
1064ab7403bSDmitry Chagin 	SIGUSR2,	/* LINUX_SIGUSR2 */
1074ab7403bSDmitry Chagin 	SIGPIPE,	/* LINUX_SIGPIPE */
1084ab7403bSDmitry Chagin 	SIGALRM,	/* LINUX_SIGALRM */
1094ab7403bSDmitry Chagin 	SIGTERM,	/* LINUX_SIGTERM */
1104ab7403bSDmitry Chagin 	SIGBUS,		/* LINUX_SIGSTKFLT */
1114ab7403bSDmitry Chagin 	SIGCHLD,	/* LINUX_SIGCHLD */
1124ab7403bSDmitry Chagin 	SIGCONT,	/* LINUX_SIGCONT */
1134ab7403bSDmitry Chagin 	SIGSTOP,	/* LINUX_SIGSTOP */
1144ab7403bSDmitry Chagin 	SIGTSTP,	/* LINUX_SIGTSTP */
1154ab7403bSDmitry Chagin 	SIGTTIN,	/* LINUX_SIGTTIN */
1164ab7403bSDmitry Chagin 	SIGTTOU,	/* LINUX_SIGTTOU */
1174ab7403bSDmitry Chagin 	SIGURG,		/* LINUX_SIGURG */
1184ab7403bSDmitry Chagin 	SIGXCPU,	/* LINUX_SIGXCPU */
1194ab7403bSDmitry Chagin 	SIGXFSZ,	/* LINUX_SIGXFSZ */
1204ab7403bSDmitry Chagin 	SIGVTALRM,	/* LINUX_SIGVTALARM */
1214ab7403bSDmitry Chagin 	SIGPROF,	/* LINUX_SIGPROF */
1224ab7403bSDmitry Chagin 	SIGWINCH,	/* LINUX_SIGWINCH */
1234ab7403bSDmitry Chagin 	SIGIO,		/* LINUX_SIGIO */
1244ab7403bSDmitry Chagin 	/*
1254ab7403bSDmitry Chagin 	 * FreeBSD does not have SIGPWR signal, map Linux SIGPWR signal
1264ab7403bSDmitry Chagin 	 * to the first unused FreeBSD signal number. Since Linux supports
1274ab7403bSDmitry Chagin 	 * signals from 1 to 64 we are ok here as our SIGRTMIN = 65.
1284ab7403bSDmitry Chagin 	 */
1294ab7403bSDmitry Chagin 	SIGRTMIN,	/* LINUX_SIGPWR */
1304ab7403bSDmitry Chagin 	SIGSYS		/* LINUX_SIGSYS */
1314ab7403bSDmitry Chagin };
1324ab7403bSDmitry Chagin 
133*044ab55eSEdward Tomasz Napierala static struct cdev *dev_shm_cdev;
134*044ab55eSEdward Tomasz Napierala static struct cdevsw dev_shm_cdevsw = {
135*044ab55eSEdward Tomasz Napierala      .d_version = D_VERSION,
136*044ab55eSEdward Tomasz Napierala      .d_name    = "dev_shm",
137*044ab55eSEdward Tomasz Napierala };
138*044ab55eSEdward Tomasz Napierala 
1394ab7403bSDmitry Chagin /*
1404ab7403bSDmitry Chagin  * Map Linux RT signals to the FreeBSD RT signals.
1414ab7403bSDmitry Chagin  */
1424ab7403bSDmitry Chagin static inline int
1434ab7403bSDmitry Chagin linux_to_bsd_rt_signal(int sig)
1444ab7403bSDmitry Chagin {
1454ab7403bSDmitry Chagin 
1464ab7403bSDmitry Chagin 	return (SIGRTMIN + 1 + sig - LINUX_SIGRTMIN);
1474ab7403bSDmitry Chagin }
1484ab7403bSDmitry Chagin 
1494ab7403bSDmitry Chagin static inline int
1504ab7403bSDmitry Chagin bsd_to_linux_rt_signal(int sig)
1514ab7403bSDmitry Chagin {
1524ab7403bSDmitry Chagin 
1534ab7403bSDmitry Chagin 	return (sig - SIGRTMIN - 1 + LINUX_SIGRTMIN);
1544ab7403bSDmitry Chagin }
1554ab7403bSDmitry Chagin 
1564ab7403bSDmitry Chagin int
1574ab7403bSDmitry Chagin linux_to_bsd_signal(int sig)
1584ab7403bSDmitry Chagin {
1594ab7403bSDmitry Chagin 
1608081c6ceSEdward Tomasz Napierala 	KASSERT(sig > 0 && sig <= LINUX_SIGRTMAX, ("invalid Linux signal %d\n", sig));
1614ab7403bSDmitry Chagin 
1624ab7403bSDmitry Chagin 	if (sig < LINUX_SIGRTMIN)
1634ab7403bSDmitry Chagin 		return (linux_to_bsd_sigtbl[_SIG_IDX(sig)]);
1644ab7403bSDmitry Chagin 
1654ab7403bSDmitry Chagin 	return (linux_to_bsd_rt_signal(sig));
1664ab7403bSDmitry Chagin }
1674ab7403bSDmitry Chagin 
1684ab7403bSDmitry Chagin int
1694ab7403bSDmitry Chagin bsd_to_linux_signal(int sig)
1704ab7403bSDmitry Chagin {
1714ab7403bSDmitry Chagin 
1724ab7403bSDmitry Chagin 	if (sig <= LINUX_SIGTBLSZ)
1734ab7403bSDmitry Chagin 		return (bsd_to_linux_sigtbl[_SIG_IDX(sig)]);
1744ab7403bSDmitry Chagin 	if (sig == SIGRTMIN)
1754ab7403bSDmitry Chagin 		return (LINUX_SIGPWR);
1764ab7403bSDmitry Chagin 
1774ab7403bSDmitry Chagin 	return (bsd_to_linux_rt_signal(sig));
1784ab7403bSDmitry Chagin }
1794ab7403bSDmitry Chagin 
1804ab7403bSDmitry Chagin int
1814ab7403bSDmitry Chagin linux_to_bsd_sigaltstack(int lsa)
1824ab7403bSDmitry Chagin {
1834ab7403bSDmitry Chagin 	int bsa = 0;
1844ab7403bSDmitry Chagin 
1854ab7403bSDmitry Chagin 	if (lsa & LINUX_SS_DISABLE)
1864ab7403bSDmitry Chagin 		bsa |= SS_DISABLE;
1874ab7403bSDmitry Chagin 	/*
1884ab7403bSDmitry Chagin 	 * Linux ignores SS_ONSTACK flag for ss
1894ab7403bSDmitry Chagin 	 * parameter while FreeBSD prohibits it.
1904ab7403bSDmitry Chagin 	 */
1914ab7403bSDmitry Chagin 	return (bsa);
1924ab7403bSDmitry Chagin }
1934ab7403bSDmitry Chagin 
1944ab7403bSDmitry Chagin int
1954ab7403bSDmitry Chagin bsd_to_linux_sigaltstack(int bsa)
1964ab7403bSDmitry Chagin {
1974ab7403bSDmitry Chagin 	int lsa = 0;
1984ab7403bSDmitry Chagin 
1994ab7403bSDmitry Chagin 	if (bsa & SS_DISABLE)
2004ab7403bSDmitry Chagin 		lsa |= LINUX_SS_DISABLE;
2014ab7403bSDmitry Chagin 	if (bsa & SS_ONSTACK)
2024ab7403bSDmitry Chagin 		lsa |= LINUX_SS_ONSTACK;
2034ab7403bSDmitry Chagin 	return (lsa);
2044ab7403bSDmitry Chagin }
2054ab7403bSDmitry Chagin 
2064ab7403bSDmitry Chagin void
2074ab7403bSDmitry Chagin linux_to_bsd_sigset(l_sigset_t *lss, sigset_t *bss)
2084ab7403bSDmitry Chagin {
2094ab7403bSDmitry Chagin 	int b, l;
2104ab7403bSDmitry Chagin 
2114ab7403bSDmitry Chagin 	SIGEMPTYSET(*bss);
2124ab7403bSDmitry Chagin 	for (l = 1; l <= LINUX_SIGRTMAX; l++) {
2134ab7403bSDmitry Chagin 		if (LINUX_SIGISMEMBER(*lss, l)) {
2144ab7403bSDmitry Chagin 			b = linux_to_bsd_signal(l);
2154ab7403bSDmitry Chagin 			if (b)
2164ab7403bSDmitry Chagin 				SIGADDSET(*bss, b);
2174ab7403bSDmitry Chagin 		}
2184ab7403bSDmitry Chagin 	}
2194ab7403bSDmitry Chagin }
2204ab7403bSDmitry Chagin 
2214ab7403bSDmitry Chagin void
2224ab7403bSDmitry Chagin bsd_to_linux_sigset(sigset_t *bss, l_sigset_t *lss)
2234ab7403bSDmitry Chagin {
2244ab7403bSDmitry Chagin 	int b, l;
2254ab7403bSDmitry Chagin 
2264ab7403bSDmitry Chagin 	LINUX_SIGEMPTYSET(*lss);
2274ab7403bSDmitry Chagin 	for (b = 1; b <= SIGRTMAX; b++) {
2284ab7403bSDmitry Chagin 		if (SIGISMEMBER(*bss, b)) {
2294ab7403bSDmitry Chagin 			l = bsd_to_linux_signal(b);
2304ab7403bSDmitry Chagin 			if (l)
2314ab7403bSDmitry Chagin 				LINUX_SIGADDSET(*lss, l);
2324ab7403bSDmitry Chagin 		}
2334ab7403bSDmitry Chagin 	}
2344ab7403bSDmitry Chagin }
235d151344dSDmitry Chagin 
236d151344dSDmitry Chagin /*
237d151344dSDmitry Chagin  * Translate a Linux interface name to a FreeBSD interface name,
238d151344dSDmitry Chagin  * and return the associated ifnet structure
239d151344dSDmitry Chagin  * bsdname and lxname need to be least IFNAMSIZ bytes long, but
240d151344dSDmitry Chagin  * can point to the same buffer.
241d151344dSDmitry Chagin  */
242d151344dSDmitry Chagin struct ifnet *
243d151344dSDmitry Chagin ifname_linux_to_bsd(struct thread *td, const char *lxname, char *bsdname)
244d151344dSDmitry Chagin {
245d151344dSDmitry Chagin 	struct ifnet *ifp;
246d151344dSDmitry Chagin 	int len, unit;
247d151344dSDmitry Chagin 	char *ep;
2489c1437aeSDmitry Chagin 	int index;
2499c1437aeSDmitry Chagin 	bool is_eth, is_lo;
250d151344dSDmitry Chagin 
251d151344dSDmitry Chagin 	for (len = 0; len < LINUX_IFNAMSIZ; ++len)
2529c1437aeSDmitry Chagin 		if (!isalpha(lxname[len]) || lxname[len] == '\0')
253d151344dSDmitry Chagin 			break;
254d151344dSDmitry Chagin 	if (len == 0 || len == LINUX_IFNAMSIZ)
255d151344dSDmitry Chagin 		return (NULL);
256d151344dSDmitry Chagin 	/* Linux loopback interface name is lo (not lo0) */
2579c1437aeSDmitry Chagin 	is_lo = (len == 2 && strncmp(lxname, "lo", len) == 0);
258d151344dSDmitry Chagin 	unit = (int)strtoul(lxname + len, &ep, 10);
259d151344dSDmitry Chagin 	if ((ep == NULL || ep == lxname + len || ep >= lxname + LINUX_IFNAMSIZ) &&
260d151344dSDmitry Chagin 	    is_lo == 0)
261d151344dSDmitry Chagin 		return (NULL);
262d151344dSDmitry Chagin 	index = 0;
2639c1437aeSDmitry Chagin 	is_eth = (len == 3 && strncmp(lxname, "eth", len) == 0);
264d151344dSDmitry Chagin 
265d151344dSDmitry Chagin 	CURVNET_SET(TD_TO_VNET(td));
266d151344dSDmitry Chagin 	IFNET_RLOCK();
267d151344dSDmitry Chagin 	CK_STAILQ_FOREACH(ifp, &V_ifnet, if_link) {
268d151344dSDmitry Chagin 		/*
269d151344dSDmitry Chagin 		 * Allow Linux programs to use FreeBSD names. Don't presume
270d151344dSDmitry Chagin 		 * we never have an interface named "eth", so don't make
271d151344dSDmitry Chagin 		 * the test optional based on is_eth.
272d151344dSDmitry Chagin 		 */
273d151344dSDmitry Chagin 		if (strncmp(ifp->if_xname, lxname, LINUX_IFNAMSIZ) == 0)
274d151344dSDmitry Chagin 			break;
275d151344dSDmitry Chagin 		if (is_eth && IFP_IS_ETH(ifp) && unit == index++)
276d151344dSDmitry Chagin 			break;
277d151344dSDmitry Chagin 		if (is_lo && IFP_IS_LOOP(ifp))
278d151344dSDmitry Chagin 			break;
279d151344dSDmitry Chagin 	}
280d151344dSDmitry Chagin 	IFNET_RUNLOCK();
281d151344dSDmitry Chagin 	CURVNET_RESTORE();
282d151344dSDmitry Chagin 	if (ifp != NULL && bsdname != NULL)
283d151344dSDmitry Chagin 		strlcpy(bsdname, ifp->if_xname, IFNAMSIZ);
284d151344dSDmitry Chagin 	return (ifp);
285d151344dSDmitry Chagin }
286d151344dSDmitry Chagin 
287d151344dSDmitry Chagin void
288d151344dSDmitry Chagin linux_ifflags(struct ifnet *ifp, short *flags)
289d151344dSDmitry Chagin {
290bbac65c7SDmitry Chagin 	unsigned short fl;
291d151344dSDmitry Chagin 
292bbac65c7SDmitry Chagin 	fl = (ifp->if_flags | ifp->if_drv_flags) & 0xffff;
293bbac65c7SDmitry Chagin 	*flags = 0;
294bbac65c7SDmitry Chagin 	if (fl & IFF_UP)
295bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_UP;
296bbac65c7SDmitry Chagin 	if (fl & IFF_BROADCAST)
297bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_BROADCAST;
298bbac65c7SDmitry Chagin 	if (fl & IFF_DEBUG)
299bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_DEBUG;
300bbac65c7SDmitry Chagin 	if (fl & IFF_LOOPBACK)
301bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_LOOPBACK;
302bbac65c7SDmitry Chagin 	if (fl & IFF_POINTOPOINT)
303bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_POINTOPOINT;
304bbac65c7SDmitry Chagin 	if (fl & IFF_DRV_RUNNING)
305bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_RUNNING;
306bbac65c7SDmitry Chagin 	if (fl & IFF_NOARP)
307bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_NOARP;
308bbac65c7SDmitry Chagin 	if (fl & IFF_PROMISC)
309bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_PROMISC;
310bbac65c7SDmitry Chagin 	if (fl & IFF_ALLMULTI)
311bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_ALLMULTI;
312bbac65c7SDmitry Chagin 	if (fl & IFF_MULTICAST)
313bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_MULTICAST;
314d151344dSDmitry Chagin }
315d151344dSDmitry Chagin 
316d151344dSDmitry Chagin int
317d151344dSDmitry Chagin linux_ifhwaddr(struct ifnet *ifp, struct l_sockaddr *lsa)
318d151344dSDmitry Chagin {
319d151344dSDmitry Chagin 	struct ifaddr *ifa;
320d151344dSDmitry Chagin 	struct sockaddr_dl *sdl;
321d151344dSDmitry Chagin 
322d151344dSDmitry Chagin 	if (IFP_IS_LOOP(ifp)) {
323d151344dSDmitry Chagin 		bzero(lsa, sizeof(*lsa));
324d151344dSDmitry Chagin 		lsa->sa_family = LINUX_ARPHRD_LOOPBACK;
325d151344dSDmitry Chagin 		return (0);
326d151344dSDmitry Chagin 	}
327d151344dSDmitry Chagin 
328d151344dSDmitry Chagin 	if (!IFP_IS_ETH(ifp))
329d151344dSDmitry Chagin 		return (ENOENT);
330d151344dSDmitry Chagin 
331d151344dSDmitry Chagin 	CK_STAILQ_FOREACH(ifa, &ifp->if_addrhead, ifa_link) {
332d151344dSDmitry Chagin 		sdl = (struct sockaddr_dl*)ifa->ifa_addr;
333d151344dSDmitry Chagin 		if (sdl != NULL && (sdl->sdl_family == AF_LINK) &&
334d151344dSDmitry Chagin 		    (sdl->sdl_type == IFT_ETHER)) {
335d151344dSDmitry Chagin 			bzero(lsa, sizeof(*lsa));
336d151344dSDmitry Chagin 			lsa->sa_family = LINUX_ARPHRD_ETHER;
337d151344dSDmitry Chagin 			bcopy(LLADDR(sdl), lsa->sa_data, LINUX_IFHWADDRLEN);
338d151344dSDmitry Chagin 			return (0);
339d151344dSDmitry Chagin 		}
340d151344dSDmitry Chagin 	}
341d151344dSDmitry Chagin 
342d151344dSDmitry Chagin 	return (ENOENT);
343d151344dSDmitry Chagin }
344d5368bf3SDmitry Chagin 
345d5368bf3SDmitry Chagin int
346d5368bf3SDmitry Chagin linux_to_bsd_domain(int domain)
347d5368bf3SDmitry Chagin {
348d5368bf3SDmitry Chagin 
349d5368bf3SDmitry Chagin 	switch (domain) {
350d5368bf3SDmitry Chagin 	case LINUX_AF_UNSPEC:
351d5368bf3SDmitry Chagin 		return (AF_UNSPEC);
352d5368bf3SDmitry Chagin 	case LINUX_AF_UNIX:
353d5368bf3SDmitry Chagin 		return (AF_LOCAL);
354d5368bf3SDmitry Chagin 	case LINUX_AF_INET:
355d5368bf3SDmitry Chagin 		return (AF_INET);
356d5368bf3SDmitry Chagin 	case LINUX_AF_INET6:
357d5368bf3SDmitry Chagin 		return (AF_INET6);
358d5368bf3SDmitry Chagin 	case LINUX_AF_AX25:
359d5368bf3SDmitry Chagin 		return (AF_CCITT);
360d5368bf3SDmitry Chagin 	case LINUX_AF_IPX:
361d5368bf3SDmitry Chagin 		return (AF_IPX);
362d5368bf3SDmitry Chagin 	case LINUX_AF_APPLETALK:
363d5368bf3SDmitry Chagin 		return (AF_APPLETALK);
364d5368bf3SDmitry Chagin 	}
365d5368bf3SDmitry Chagin 	return (-1);
366d5368bf3SDmitry Chagin }
367d5368bf3SDmitry Chagin 
368d5368bf3SDmitry Chagin int
369d5368bf3SDmitry Chagin bsd_to_linux_domain(int domain)
370d5368bf3SDmitry Chagin {
371d5368bf3SDmitry Chagin 
372d5368bf3SDmitry Chagin 	switch (domain) {
373d5368bf3SDmitry Chagin 	case AF_UNSPEC:
374d5368bf3SDmitry Chagin 		return (LINUX_AF_UNSPEC);
375d5368bf3SDmitry Chagin 	case AF_LOCAL:
376d5368bf3SDmitry Chagin 		return (LINUX_AF_UNIX);
377d5368bf3SDmitry Chagin 	case AF_INET:
378d5368bf3SDmitry Chagin 		return (LINUX_AF_INET);
379d5368bf3SDmitry Chagin 	case AF_INET6:
380d5368bf3SDmitry Chagin 		return (LINUX_AF_INET6);
381d5368bf3SDmitry Chagin 	case AF_CCITT:
382d5368bf3SDmitry Chagin 		return (LINUX_AF_AX25);
383d5368bf3SDmitry Chagin 	case AF_IPX:
384d5368bf3SDmitry Chagin 		return (LINUX_AF_IPX);
385d5368bf3SDmitry Chagin 	case AF_APPLETALK:
386d5368bf3SDmitry Chagin 		return (LINUX_AF_APPLETALK);
387d5368bf3SDmitry Chagin 	}
388d5368bf3SDmitry Chagin 	return (-1);
389d5368bf3SDmitry Chagin }
390d5368bf3SDmitry Chagin 
391d5368bf3SDmitry Chagin /*
392d5368bf3SDmitry Chagin  * Based on the fact that:
393d5368bf3SDmitry Chagin  * 1. Native and Linux storage of struct sockaddr
394d5368bf3SDmitry Chagin  * and struct sockaddr_in6 are equal.
395d5368bf3SDmitry Chagin  * 2. On Linux sa_family is the first member of all struct sockaddr.
396d5368bf3SDmitry Chagin  */
397d5368bf3SDmitry Chagin int
398d5368bf3SDmitry Chagin bsd_to_linux_sockaddr(const struct sockaddr *sa, struct l_sockaddr **lsa,
399d5368bf3SDmitry Chagin     socklen_t len)
400d5368bf3SDmitry Chagin {
401d5368bf3SDmitry Chagin 	struct l_sockaddr *kosa;
402d5368bf3SDmitry Chagin 	int error, bdom;
403d5368bf3SDmitry Chagin 
404d5368bf3SDmitry Chagin 	*lsa = NULL;
405d5368bf3SDmitry Chagin 	if (len < 2 || len > UCHAR_MAX)
406d5368bf3SDmitry Chagin 		return (EINVAL);
407d5368bf3SDmitry Chagin 
408d5368bf3SDmitry Chagin 	kosa = malloc(len, M_SONAME, M_WAITOK);
409d5368bf3SDmitry Chagin 	bcopy(sa, kosa, len);
410d5368bf3SDmitry Chagin 
411d5368bf3SDmitry Chagin 	bdom = bsd_to_linux_domain(sa->sa_family);
412d5368bf3SDmitry Chagin 	if (bdom == -1) {
413d5368bf3SDmitry Chagin 		error = EAFNOSUPPORT;
414d5368bf3SDmitry Chagin 		goto out;
415d5368bf3SDmitry Chagin 	}
416d5368bf3SDmitry Chagin 
417d5368bf3SDmitry Chagin 	kosa->sa_family = bdom;
418d5368bf3SDmitry Chagin 	*lsa = kosa;
419d5368bf3SDmitry Chagin 	return (0);
420d5368bf3SDmitry Chagin 
421d5368bf3SDmitry Chagin out:
422d5368bf3SDmitry Chagin 	free(kosa, M_SONAME);
423d5368bf3SDmitry Chagin 	return (error);
424d5368bf3SDmitry Chagin }
425d5368bf3SDmitry Chagin 
426d5368bf3SDmitry Chagin int
427d5368bf3SDmitry Chagin linux_to_bsd_sockaddr(const struct l_sockaddr *osa, struct sockaddr **sap,
428d5368bf3SDmitry Chagin     socklen_t *len)
429d5368bf3SDmitry Chagin {
430d5368bf3SDmitry Chagin 	struct sockaddr *sa;
431d5368bf3SDmitry Chagin 	struct l_sockaddr *kosa;
432d5368bf3SDmitry Chagin #ifdef INET6
433d5368bf3SDmitry Chagin 	struct sockaddr_in6 *sin6;
434d5368bf3SDmitry Chagin 	bool  oldv6size;
435d5368bf3SDmitry Chagin #endif
436d5368bf3SDmitry Chagin 	char *name;
437d5368bf3SDmitry Chagin 	int salen, bdom, error, hdrlen, namelen;
438d5368bf3SDmitry Chagin 
439d5368bf3SDmitry Chagin 	if (*len < 2 || *len > UCHAR_MAX)
440d5368bf3SDmitry Chagin 		return (EINVAL);
441d5368bf3SDmitry Chagin 
442d5368bf3SDmitry Chagin 	salen = *len;
443d5368bf3SDmitry Chagin 
444d5368bf3SDmitry Chagin #ifdef INET6
445d5368bf3SDmitry Chagin 	oldv6size = false;
446d5368bf3SDmitry Chagin 	/*
447d5368bf3SDmitry Chagin 	 * Check for old (pre-RFC2553) sockaddr_in6. We may accept it
448d5368bf3SDmitry Chagin 	 * if it's a v4-mapped address, so reserve the proper space
449d5368bf3SDmitry Chagin 	 * for it.
450d5368bf3SDmitry Chagin 	 */
451d5368bf3SDmitry Chagin 	if (salen == sizeof(struct sockaddr_in6) - sizeof(uint32_t)) {
452d5368bf3SDmitry Chagin 		salen += sizeof(uint32_t);
453d5368bf3SDmitry Chagin 		oldv6size = true;
454d5368bf3SDmitry Chagin 	}
455d5368bf3SDmitry Chagin #endif
456d5368bf3SDmitry Chagin 
457d5368bf3SDmitry Chagin 	kosa = malloc(salen, M_SONAME, M_WAITOK);
458d5368bf3SDmitry Chagin 
459d5368bf3SDmitry Chagin 	if ((error = copyin(osa, kosa, *len)))
460d5368bf3SDmitry Chagin 		goto out;
461d5368bf3SDmitry Chagin 
462d5368bf3SDmitry Chagin 	bdom = linux_to_bsd_domain(kosa->sa_family);
463d5368bf3SDmitry Chagin 	if (bdom == -1) {
464d5368bf3SDmitry Chagin 		error = EAFNOSUPPORT;
465d5368bf3SDmitry Chagin 		goto out;
466d5368bf3SDmitry Chagin 	}
467d5368bf3SDmitry Chagin 
468d5368bf3SDmitry Chagin #ifdef INET6
469d5368bf3SDmitry Chagin 	/*
470d5368bf3SDmitry Chagin 	 * Older Linux IPv6 code uses obsolete RFC2133 struct sockaddr_in6,
471d5368bf3SDmitry Chagin 	 * which lacks the scope id compared with RFC2553 one. If we detect
472d5368bf3SDmitry Chagin 	 * the situation, reject the address and write a message to system log.
473d5368bf3SDmitry Chagin 	 *
474d5368bf3SDmitry Chagin 	 * Still accept addresses for which the scope id is not used.
475d5368bf3SDmitry Chagin 	 */
476d5368bf3SDmitry Chagin 	if (oldv6size) {
477d5368bf3SDmitry Chagin 		if (bdom == AF_INET6) {
478d5368bf3SDmitry Chagin 			sin6 = (struct sockaddr_in6 *)kosa;
479d5368bf3SDmitry Chagin 			if (IN6_IS_ADDR_V4MAPPED(&sin6->sin6_addr) ||
480d5368bf3SDmitry Chagin 			    (!IN6_IS_ADDR_LINKLOCAL(&sin6->sin6_addr) &&
481d5368bf3SDmitry Chagin 			     !IN6_IS_ADDR_SITELOCAL(&sin6->sin6_addr) &&
482d5368bf3SDmitry Chagin 			     !IN6_IS_ADDR_V4COMPAT(&sin6->sin6_addr) &&
483d5368bf3SDmitry Chagin 			     !IN6_IS_ADDR_UNSPECIFIED(&sin6->sin6_addr) &&
484d5368bf3SDmitry Chagin 			     !IN6_IS_ADDR_MULTICAST(&sin6->sin6_addr))) {
485d5368bf3SDmitry Chagin 				sin6->sin6_scope_id = 0;
486d5368bf3SDmitry Chagin 			} else {
487d5368bf3SDmitry Chagin 				linux_msg(curthread,
488d5368bf3SDmitry Chagin 				    "obsolete pre-RFC2553 sockaddr_in6 rejected\n");
489d5368bf3SDmitry Chagin 				error = EINVAL;
490d5368bf3SDmitry Chagin 				goto out;
491d5368bf3SDmitry Chagin 			}
492d5368bf3SDmitry Chagin 		} else
493d5368bf3SDmitry Chagin 			salen -= sizeof(uint32_t);
494d5368bf3SDmitry Chagin 	}
495d5368bf3SDmitry Chagin #endif
496d5368bf3SDmitry Chagin 	if (bdom == AF_INET) {
497d5368bf3SDmitry Chagin 		if (salen < sizeof(struct sockaddr_in)) {
498d5368bf3SDmitry Chagin 			error = EINVAL;
499d5368bf3SDmitry Chagin 			goto out;
500d5368bf3SDmitry Chagin 		}
501d5368bf3SDmitry Chagin 		salen = sizeof(struct sockaddr_in);
502d5368bf3SDmitry Chagin 	}
503d5368bf3SDmitry Chagin 
504d5368bf3SDmitry Chagin 	if (bdom == AF_LOCAL && salen > sizeof(struct sockaddr_un)) {
505d5368bf3SDmitry Chagin 		hdrlen = offsetof(struct sockaddr_un, sun_path);
506d5368bf3SDmitry Chagin 		name = ((struct sockaddr_un *)kosa)->sun_path;
507d5368bf3SDmitry Chagin 		if (*name == '\0') {
508d5368bf3SDmitry Chagin 			/*
509d5368bf3SDmitry Chagin 			 * Linux abstract namespace starts with a NULL byte.
510d5368bf3SDmitry Chagin 			 * XXX We do not support abstract namespace yet.
511d5368bf3SDmitry Chagin 			 */
512d5368bf3SDmitry Chagin 			namelen = strnlen(name + 1, salen - hdrlen - 1) + 1;
513d5368bf3SDmitry Chagin 		} else
514d5368bf3SDmitry Chagin 			namelen = strnlen(name, salen - hdrlen);
515d5368bf3SDmitry Chagin 		salen = hdrlen + namelen;
516d5368bf3SDmitry Chagin 		if (salen > sizeof(struct sockaddr_un)) {
517d5368bf3SDmitry Chagin 			error = ENAMETOOLONG;
518d5368bf3SDmitry Chagin 			goto out;
519d5368bf3SDmitry Chagin 		}
520d5368bf3SDmitry Chagin 	}
521d5368bf3SDmitry Chagin 
522d5368bf3SDmitry Chagin 	sa = (struct sockaddr *)kosa;
523d5368bf3SDmitry Chagin 	sa->sa_family = bdom;
524d5368bf3SDmitry Chagin 	sa->sa_len = salen;
525d5368bf3SDmitry Chagin 
526d5368bf3SDmitry Chagin 	*sap = sa;
527d5368bf3SDmitry Chagin 	*len = salen;
528d5368bf3SDmitry Chagin 	return (0);
529d5368bf3SDmitry Chagin 
530d5368bf3SDmitry Chagin out:
531d5368bf3SDmitry Chagin 	free(kosa, M_SONAME);
532d5368bf3SDmitry Chagin 	return (error);
533d5368bf3SDmitry Chagin }
534*044ab55eSEdward Tomasz Napierala 
535*044ab55eSEdward Tomasz Napierala void
536*044ab55eSEdward Tomasz Napierala linux_dev_shm_create(void)
537*044ab55eSEdward Tomasz Napierala {
538*044ab55eSEdward Tomasz Napierala 	int error;
539*044ab55eSEdward Tomasz Napierala 
540*044ab55eSEdward Tomasz Napierala 	error = make_dev_p(MAKEDEV_CHECKNAME | MAKEDEV_WAITOK, &dev_shm_cdev,
541*044ab55eSEdward Tomasz Napierala 	    &dev_shm_cdevsw, NULL, UID_ROOT, GID_WHEEL, 0, "shm/.mountpoint");
542*044ab55eSEdward Tomasz Napierala 	if (error != 0) {
543*044ab55eSEdward Tomasz Napierala 		printf("%s: failed to create device node, error %d\n",
544*044ab55eSEdward Tomasz Napierala 		    __func__, error);
545*044ab55eSEdward Tomasz Napierala 	}
546*044ab55eSEdward Tomasz Napierala }
547*044ab55eSEdward Tomasz Napierala 
548*044ab55eSEdward Tomasz Napierala void
549*044ab55eSEdward Tomasz Napierala linux_dev_shm_destroy(void)
550*044ab55eSEdward Tomasz Napierala {
551*044ab55eSEdward Tomasz Napierala 
552*044ab55eSEdward Tomasz Napierala 	destroy_dev(dev_shm_cdev);
553*044ab55eSEdward Tomasz Napierala }
554