xref: /freebsd/sys/compat/linux/linux.c (revision bbac65c772a37b0191445f55bdcdfb6bd0ac5fdb)
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 
304ab7403bSDmitry Chagin #include <sys/param.h>
314ab7403bSDmitry Chagin #include <sys/systm.h>
32d151344dSDmitry Chagin #include <sys/ctype.h>
33d151344dSDmitry Chagin #include <sys/jail.h>
34d151344dSDmitry Chagin #include <sys/lock.h>
354ab7403bSDmitry Chagin #include <sys/signalvar.h>
36d151344dSDmitry Chagin #include <sys/socket.h>
37d151344dSDmitry Chagin 
38d151344dSDmitry Chagin #include <net/if.h>
39d151344dSDmitry Chagin #include <net/if_var.h>
40d151344dSDmitry Chagin #include <net/if_dl.h>
41d151344dSDmitry Chagin #include <net/if_types.h>
424ab7403bSDmitry Chagin 
434ab7403bSDmitry Chagin #include <compat/linux/linux.h>
44d151344dSDmitry Chagin #include <compat/linux/linux_common.h>
454ab7403bSDmitry Chagin 
46d151344dSDmitry Chagin CTASSERT(LINUX_IFNAMSIZ == IFNAMSIZ);
474ab7403bSDmitry Chagin 
484ab7403bSDmitry Chagin static int bsd_to_linux_sigtbl[LINUX_SIGTBLSZ] = {
494ab7403bSDmitry Chagin 	LINUX_SIGHUP,	/* SIGHUP */
504ab7403bSDmitry Chagin 	LINUX_SIGINT,	/* SIGINT */
514ab7403bSDmitry Chagin 	LINUX_SIGQUIT,	/* SIGQUIT */
524ab7403bSDmitry Chagin 	LINUX_SIGILL,	/* SIGILL */
534ab7403bSDmitry Chagin 	LINUX_SIGTRAP,	/* SIGTRAP */
544ab7403bSDmitry Chagin 	LINUX_SIGABRT,	/* SIGABRT */
554ab7403bSDmitry Chagin 	0,		/* SIGEMT */
564ab7403bSDmitry Chagin 	LINUX_SIGFPE,	/* SIGFPE */
574ab7403bSDmitry Chagin 	LINUX_SIGKILL,	/* SIGKILL */
584ab7403bSDmitry Chagin 	LINUX_SIGBUS,	/* SIGBUS */
594ab7403bSDmitry Chagin 	LINUX_SIGSEGV,	/* SIGSEGV */
604ab7403bSDmitry Chagin 	LINUX_SIGSYS,	/* SIGSYS */
614ab7403bSDmitry Chagin 	LINUX_SIGPIPE,	/* SIGPIPE */
624ab7403bSDmitry Chagin 	LINUX_SIGALRM,	/* SIGALRM */
634ab7403bSDmitry Chagin 	LINUX_SIGTERM,	/* SIGTERM */
644ab7403bSDmitry Chagin 	LINUX_SIGURG,	/* SIGURG */
654ab7403bSDmitry Chagin 	LINUX_SIGSTOP,	/* SIGSTOP */
664ab7403bSDmitry Chagin 	LINUX_SIGTSTP,	/* SIGTSTP */
674ab7403bSDmitry Chagin 	LINUX_SIGCONT,	/* SIGCONT */
684ab7403bSDmitry Chagin 	LINUX_SIGCHLD,	/* SIGCHLD */
694ab7403bSDmitry Chagin 	LINUX_SIGTTIN,	/* SIGTTIN */
704ab7403bSDmitry Chagin 	LINUX_SIGTTOU,	/* SIGTTOU */
714ab7403bSDmitry Chagin 	LINUX_SIGIO,	/* SIGIO */
724ab7403bSDmitry Chagin 	LINUX_SIGXCPU,	/* SIGXCPU */
734ab7403bSDmitry Chagin 	LINUX_SIGXFSZ,	/* SIGXFSZ */
744ab7403bSDmitry Chagin 	LINUX_SIGVTALRM,/* SIGVTALRM */
754ab7403bSDmitry Chagin 	LINUX_SIGPROF,	/* SIGPROF */
764ab7403bSDmitry Chagin 	LINUX_SIGWINCH,	/* SIGWINCH */
774ab7403bSDmitry Chagin 	0,		/* SIGINFO */
784ab7403bSDmitry Chagin 	LINUX_SIGUSR1,	/* SIGUSR1 */
794ab7403bSDmitry Chagin 	LINUX_SIGUSR2	/* SIGUSR2 */
804ab7403bSDmitry Chagin };
814ab7403bSDmitry Chagin 
824ab7403bSDmitry Chagin static int linux_to_bsd_sigtbl[LINUX_SIGTBLSZ] = {
834ab7403bSDmitry Chagin 	SIGHUP,		/* LINUX_SIGHUP */
844ab7403bSDmitry Chagin 	SIGINT,		/* LINUX_SIGINT */
854ab7403bSDmitry Chagin 	SIGQUIT,	/* LINUX_SIGQUIT */
864ab7403bSDmitry Chagin 	SIGILL,		/* LINUX_SIGILL */
874ab7403bSDmitry Chagin 	SIGTRAP,	/* LINUX_SIGTRAP */
884ab7403bSDmitry Chagin 	SIGABRT,	/* LINUX_SIGABRT */
894ab7403bSDmitry Chagin 	SIGBUS,		/* LINUX_SIGBUS */
904ab7403bSDmitry Chagin 	SIGFPE,		/* LINUX_SIGFPE */
914ab7403bSDmitry Chagin 	SIGKILL,	/* LINUX_SIGKILL */
924ab7403bSDmitry Chagin 	SIGUSR1,	/* LINUX_SIGUSR1 */
934ab7403bSDmitry Chagin 	SIGSEGV,	/* LINUX_SIGSEGV */
944ab7403bSDmitry Chagin 	SIGUSR2,	/* LINUX_SIGUSR2 */
954ab7403bSDmitry Chagin 	SIGPIPE,	/* LINUX_SIGPIPE */
964ab7403bSDmitry Chagin 	SIGALRM,	/* LINUX_SIGALRM */
974ab7403bSDmitry Chagin 	SIGTERM,	/* LINUX_SIGTERM */
984ab7403bSDmitry Chagin 	SIGBUS,		/* LINUX_SIGSTKFLT */
994ab7403bSDmitry Chagin 	SIGCHLD,	/* LINUX_SIGCHLD */
1004ab7403bSDmitry Chagin 	SIGCONT,	/* LINUX_SIGCONT */
1014ab7403bSDmitry Chagin 	SIGSTOP,	/* LINUX_SIGSTOP */
1024ab7403bSDmitry Chagin 	SIGTSTP,	/* LINUX_SIGTSTP */
1034ab7403bSDmitry Chagin 	SIGTTIN,	/* LINUX_SIGTTIN */
1044ab7403bSDmitry Chagin 	SIGTTOU,	/* LINUX_SIGTTOU */
1054ab7403bSDmitry Chagin 	SIGURG,		/* LINUX_SIGURG */
1064ab7403bSDmitry Chagin 	SIGXCPU,	/* LINUX_SIGXCPU */
1074ab7403bSDmitry Chagin 	SIGXFSZ,	/* LINUX_SIGXFSZ */
1084ab7403bSDmitry Chagin 	SIGVTALRM,	/* LINUX_SIGVTALARM */
1094ab7403bSDmitry Chagin 	SIGPROF,	/* LINUX_SIGPROF */
1104ab7403bSDmitry Chagin 	SIGWINCH,	/* LINUX_SIGWINCH */
1114ab7403bSDmitry Chagin 	SIGIO,		/* LINUX_SIGIO */
1124ab7403bSDmitry Chagin 	/*
1134ab7403bSDmitry Chagin 	 * FreeBSD does not have SIGPWR signal, map Linux SIGPWR signal
1144ab7403bSDmitry Chagin 	 * to the first unused FreeBSD signal number. Since Linux supports
1154ab7403bSDmitry Chagin 	 * signals from 1 to 64 we are ok here as our SIGRTMIN = 65.
1164ab7403bSDmitry Chagin 	 */
1174ab7403bSDmitry Chagin 	SIGRTMIN,	/* LINUX_SIGPWR */
1184ab7403bSDmitry Chagin 	SIGSYS		/* LINUX_SIGSYS */
1194ab7403bSDmitry Chagin };
1204ab7403bSDmitry Chagin 
1214ab7403bSDmitry Chagin /*
1224ab7403bSDmitry Chagin  * Map Linux RT signals to the FreeBSD RT signals.
1234ab7403bSDmitry Chagin  */
1244ab7403bSDmitry Chagin static inline int
1254ab7403bSDmitry Chagin linux_to_bsd_rt_signal(int sig)
1264ab7403bSDmitry Chagin {
1274ab7403bSDmitry Chagin 
1284ab7403bSDmitry Chagin 	return (SIGRTMIN + 1 + sig - LINUX_SIGRTMIN);
1294ab7403bSDmitry Chagin }
1304ab7403bSDmitry Chagin 
1314ab7403bSDmitry Chagin static inline int
1324ab7403bSDmitry Chagin bsd_to_linux_rt_signal(int sig)
1334ab7403bSDmitry Chagin {
1344ab7403bSDmitry Chagin 
1354ab7403bSDmitry Chagin 	return (sig - SIGRTMIN - 1 + LINUX_SIGRTMIN);
1364ab7403bSDmitry Chagin }
1374ab7403bSDmitry Chagin 
1384ab7403bSDmitry Chagin int
1394ab7403bSDmitry Chagin linux_to_bsd_signal(int sig)
1404ab7403bSDmitry Chagin {
1414ab7403bSDmitry Chagin 
1428081c6ceSEdward Tomasz Napierala 	KASSERT(sig > 0 && sig <= LINUX_SIGRTMAX, ("invalid Linux signal %d\n", sig));
1434ab7403bSDmitry Chagin 
1444ab7403bSDmitry Chagin 	if (sig < LINUX_SIGRTMIN)
1454ab7403bSDmitry Chagin 		return (linux_to_bsd_sigtbl[_SIG_IDX(sig)]);
1464ab7403bSDmitry Chagin 
1474ab7403bSDmitry Chagin 	return (linux_to_bsd_rt_signal(sig));
1484ab7403bSDmitry Chagin }
1494ab7403bSDmitry Chagin 
1504ab7403bSDmitry Chagin int
1514ab7403bSDmitry Chagin bsd_to_linux_signal(int sig)
1524ab7403bSDmitry Chagin {
1534ab7403bSDmitry Chagin 
1544ab7403bSDmitry Chagin 	if (sig <= LINUX_SIGTBLSZ)
1554ab7403bSDmitry Chagin 		return (bsd_to_linux_sigtbl[_SIG_IDX(sig)]);
1564ab7403bSDmitry Chagin 	if (sig == SIGRTMIN)
1574ab7403bSDmitry Chagin 		return (LINUX_SIGPWR);
1584ab7403bSDmitry Chagin 
1594ab7403bSDmitry Chagin 	return (bsd_to_linux_rt_signal(sig));
1604ab7403bSDmitry Chagin }
1614ab7403bSDmitry Chagin 
1624ab7403bSDmitry Chagin int
1634ab7403bSDmitry Chagin linux_to_bsd_sigaltstack(int lsa)
1644ab7403bSDmitry Chagin {
1654ab7403bSDmitry Chagin 	int bsa = 0;
1664ab7403bSDmitry Chagin 
1674ab7403bSDmitry Chagin 	if (lsa & LINUX_SS_DISABLE)
1684ab7403bSDmitry Chagin 		bsa |= SS_DISABLE;
1694ab7403bSDmitry Chagin 	/*
1704ab7403bSDmitry Chagin 	 * Linux ignores SS_ONSTACK flag for ss
1714ab7403bSDmitry Chagin 	 * parameter while FreeBSD prohibits it.
1724ab7403bSDmitry Chagin 	 */
1734ab7403bSDmitry Chagin 	return (bsa);
1744ab7403bSDmitry Chagin }
1754ab7403bSDmitry Chagin 
1764ab7403bSDmitry Chagin int
1774ab7403bSDmitry Chagin bsd_to_linux_sigaltstack(int bsa)
1784ab7403bSDmitry Chagin {
1794ab7403bSDmitry Chagin 	int lsa = 0;
1804ab7403bSDmitry Chagin 
1814ab7403bSDmitry Chagin 	if (bsa & SS_DISABLE)
1824ab7403bSDmitry Chagin 		lsa |= LINUX_SS_DISABLE;
1834ab7403bSDmitry Chagin 	if (bsa & SS_ONSTACK)
1844ab7403bSDmitry Chagin 		lsa |= LINUX_SS_ONSTACK;
1854ab7403bSDmitry Chagin 	return (lsa);
1864ab7403bSDmitry Chagin }
1874ab7403bSDmitry Chagin 
1884ab7403bSDmitry Chagin void
1894ab7403bSDmitry Chagin linux_to_bsd_sigset(l_sigset_t *lss, sigset_t *bss)
1904ab7403bSDmitry Chagin {
1914ab7403bSDmitry Chagin 	int b, l;
1924ab7403bSDmitry Chagin 
1934ab7403bSDmitry Chagin 	SIGEMPTYSET(*bss);
1944ab7403bSDmitry Chagin 	for (l = 1; l <= LINUX_SIGRTMAX; l++) {
1954ab7403bSDmitry Chagin 		if (LINUX_SIGISMEMBER(*lss, l)) {
1964ab7403bSDmitry Chagin 			b = linux_to_bsd_signal(l);
1974ab7403bSDmitry Chagin 			if (b)
1984ab7403bSDmitry Chagin 				SIGADDSET(*bss, b);
1994ab7403bSDmitry Chagin 		}
2004ab7403bSDmitry Chagin 	}
2014ab7403bSDmitry Chagin }
2024ab7403bSDmitry Chagin 
2034ab7403bSDmitry Chagin void
2044ab7403bSDmitry Chagin bsd_to_linux_sigset(sigset_t *bss, l_sigset_t *lss)
2054ab7403bSDmitry Chagin {
2064ab7403bSDmitry Chagin 	int b, l;
2074ab7403bSDmitry Chagin 
2084ab7403bSDmitry Chagin 	LINUX_SIGEMPTYSET(*lss);
2094ab7403bSDmitry Chagin 	for (b = 1; b <= SIGRTMAX; b++) {
2104ab7403bSDmitry Chagin 		if (SIGISMEMBER(*bss, b)) {
2114ab7403bSDmitry Chagin 			l = bsd_to_linux_signal(b);
2124ab7403bSDmitry Chagin 			if (l)
2134ab7403bSDmitry Chagin 				LINUX_SIGADDSET(*lss, l);
2144ab7403bSDmitry Chagin 		}
2154ab7403bSDmitry Chagin 	}
2164ab7403bSDmitry Chagin }
217d151344dSDmitry Chagin 
218d151344dSDmitry Chagin /*
219d151344dSDmitry Chagin  * Translate a Linux interface name to a FreeBSD interface name,
220d151344dSDmitry Chagin  * and return the associated ifnet structure
221d151344dSDmitry Chagin  * bsdname and lxname need to be least IFNAMSIZ bytes long, but
222d151344dSDmitry Chagin  * can point to the same buffer.
223d151344dSDmitry Chagin  */
224d151344dSDmitry Chagin struct ifnet *
225d151344dSDmitry Chagin ifname_linux_to_bsd(struct thread *td, const char *lxname, char *bsdname)
226d151344dSDmitry Chagin {
227d151344dSDmitry Chagin 	struct ifnet *ifp;
228d151344dSDmitry Chagin 	int len, unit;
229d151344dSDmitry Chagin 	char *ep;
2309c1437aeSDmitry Chagin 	int index;
2319c1437aeSDmitry Chagin 	bool is_eth, is_lo;
232d151344dSDmitry Chagin 
233d151344dSDmitry Chagin 	for (len = 0; len < LINUX_IFNAMSIZ; ++len)
2349c1437aeSDmitry Chagin 		if (!isalpha(lxname[len]) || lxname[len] == '\0')
235d151344dSDmitry Chagin 			break;
236d151344dSDmitry Chagin 	if (len == 0 || len == LINUX_IFNAMSIZ)
237d151344dSDmitry Chagin 		return (NULL);
238d151344dSDmitry Chagin 	/* Linux loopback interface name is lo (not lo0) */
2399c1437aeSDmitry Chagin 	is_lo = (len == 2 && strncmp(lxname, "lo", len) == 0);
240d151344dSDmitry Chagin 	unit = (int)strtoul(lxname + len, &ep, 10);
241d151344dSDmitry Chagin 	if ((ep == NULL || ep == lxname + len || ep >= lxname + LINUX_IFNAMSIZ) &&
242d151344dSDmitry Chagin 	    is_lo == 0)
243d151344dSDmitry Chagin 		return (NULL);
244d151344dSDmitry Chagin 	index = 0;
2459c1437aeSDmitry Chagin 	is_eth = (len == 3 && strncmp(lxname, "eth", len) == 0);
246d151344dSDmitry Chagin 
247d151344dSDmitry Chagin 	CURVNET_SET(TD_TO_VNET(td));
248d151344dSDmitry Chagin 	IFNET_RLOCK();
249d151344dSDmitry Chagin 	CK_STAILQ_FOREACH(ifp, &V_ifnet, if_link) {
250d151344dSDmitry Chagin 		/*
251d151344dSDmitry Chagin 		 * Allow Linux programs to use FreeBSD names. Don't presume
252d151344dSDmitry Chagin 		 * we never have an interface named "eth", so don't make
253d151344dSDmitry Chagin 		 * the test optional based on is_eth.
254d151344dSDmitry Chagin 		 */
255d151344dSDmitry Chagin 		if (strncmp(ifp->if_xname, lxname, LINUX_IFNAMSIZ) == 0)
256d151344dSDmitry Chagin 			break;
257d151344dSDmitry Chagin 		if (is_eth && IFP_IS_ETH(ifp) && unit == index++)
258d151344dSDmitry Chagin 			break;
259d151344dSDmitry Chagin 		if (is_lo && IFP_IS_LOOP(ifp))
260d151344dSDmitry Chagin 			break;
261d151344dSDmitry Chagin 	}
262d151344dSDmitry Chagin 	IFNET_RUNLOCK();
263d151344dSDmitry Chagin 	CURVNET_RESTORE();
264d151344dSDmitry Chagin 	if (ifp != NULL && bsdname != NULL)
265d151344dSDmitry Chagin 		strlcpy(bsdname, ifp->if_xname, IFNAMSIZ);
266d151344dSDmitry Chagin 	return (ifp);
267d151344dSDmitry Chagin }
268d151344dSDmitry Chagin 
269d151344dSDmitry Chagin void
270d151344dSDmitry Chagin linux_ifflags(struct ifnet *ifp, short *flags)
271d151344dSDmitry Chagin {
272*bbac65c7SDmitry Chagin 	unsigned short fl;
273d151344dSDmitry Chagin 
274*bbac65c7SDmitry Chagin 	fl = (ifp->if_flags | ifp->if_drv_flags) & 0xffff;
275*bbac65c7SDmitry Chagin 	*flags = 0;
276*bbac65c7SDmitry Chagin 	if (fl & IFF_UP)
277*bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_UP;
278*bbac65c7SDmitry Chagin 	if (fl & IFF_BROADCAST)
279*bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_BROADCAST;
280*bbac65c7SDmitry Chagin 	if (fl & IFF_DEBUG)
281*bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_DEBUG;
282*bbac65c7SDmitry Chagin 	if (fl & IFF_LOOPBACK)
283*bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_LOOPBACK;
284*bbac65c7SDmitry Chagin 	if (fl & IFF_POINTOPOINT)
285*bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_POINTOPOINT;
286*bbac65c7SDmitry Chagin 	if (fl & IFF_DRV_RUNNING)
287*bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_RUNNING;
288*bbac65c7SDmitry Chagin 	if (fl & IFF_NOARP)
289*bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_NOARP;
290*bbac65c7SDmitry Chagin 	if (fl & IFF_PROMISC)
291*bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_PROMISC;
292*bbac65c7SDmitry Chagin 	if (fl & IFF_ALLMULTI)
293*bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_ALLMULTI;
294*bbac65c7SDmitry Chagin 	if (fl & IFF_MULTICAST)
295*bbac65c7SDmitry Chagin 		*flags |= LINUX_IFF_MULTICAST;
296d151344dSDmitry Chagin }
297d151344dSDmitry Chagin 
298d151344dSDmitry Chagin int
299d151344dSDmitry Chagin linux_ifhwaddr(struct ifnet *ifp, struct l_sockaddr *lsa)
300d151344dSDmitry Chagin {
301d151344dSDmitry Chagin 	struct ifaddr *ifa;
302d151344dSDmitry Chagin 	struct sockaddr_dl *sdl;
303d151344dSDmitry Chagin 
304d151344dSDmitry Chagin 	if (IFP_IS_LOOP(ifp)) {
305d151344dSDmitry Chagin 		bzero(lsa, sizeof(*lsa));
306d151344dSDmitry Chagin 		lsa->sa_family = LINUX_ARPHRD_LOOPBACK;
307d151344dSDmitry Chagin 		return (0);
308d151344dSDmitry Chagin 	}
309d151344dSDmitry Chagin 
310d151344dSDmitry Chagin 	if (!IFP_IS_ETH(ifp))
311d151344dSDmitry Chagin 		return (ENOENT);
312d151344dSDmitry Chagin 
313d151344dSDmitry Chagin 	CK_STAILQ_FOREACH(ifa, &ifp->if_addrhead, ifa_link) {
314d151344dSDmitry Chagin 		sdl = (struct sockaddr_dl*)ifa->ifa_addr;
315d151344dSDmitry Chagin 		if (sdl != NULL && (sdl->sdl_family == AF_LINK) &&
316d151344dSDmitry Chagin 		    (sdl->sdl_type == IFT_ETHER)) {
317d151344dSDmitry Chagin 			bzero(lsa, sizeof(*lsa));
318d151344dSDmitry Chagin 			lsa->sa_family = LINUX_ARPHRD_ETHER;
319d151344dSDmitry Chagin 			bcopy(LLADDR(sdl), lsa->sa_data, LINUX_IFHWADDRLEN);
320d151344dSDmitry Chagin 			return (0);
321d151344dSDmitry Chagin 		}
322d151344dSDmitry Chagin 	}
323d151344dSDmitry Chagin 
324d151344dSDmitry Chagin 	return (ENOENT);
325d151344dSDmitry Chagin }
326