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> 34044ab55eSEdward Tomasz Napierala #include <sys/conf.h> 35d151344dSDmitry Chagin #include <sys/ctype.h> 36*26795a03SDmitry Chagin #include <sys/file.h> 37*26795a03SDmitry Chagin #include <sys/filedesc.h> 38d151344dSDmitry Chagin #include <sys/jail.h> 39d151344dSDmitry Chagin #include <sys/lock.h> 40d5368bf3SDmitry Chagin #include <sys/malloc.h> 41*26795a03SDmitry Chagin #include <sys/poll.h> 42*26795a03SDmitry Chagin #include <sys/proc.h> 434ab7403bSDmitry Chagin #include <sys/signalvar.h> 44d151344dSDmitry Chagin #include <sys/socket.h> 45d5368bf3SDmitry Chagin #include <sys/socketvar.h> 46d151344dSDmitry Chagin 47d151344dSDmitry Chagin #include <net/if.h> 48d151344dSDmitry Chagin #include <net/if_var.h> 49d151344dSDmitry Chagin #include <net/if_dl.h> 50d151344dSDmitry Chagin #include <net/if_types.h> 514ab7403bSDmitry Chagin 52d5368bf3SDmitry Chagin #include <sys/un.h> 53d5368bf3SDmitry Chagin #include <netinet/in.h> 54d5368bf3SDmitry Chagin 554ab7403bSDmitry Chagin #include <compat/linux/linux.h> 56d151344dSDmitry Chagin #include <compat/linux/linux_common.h> 57*26795a03SDmitry Chagin #include <compat/linux/linux_mib.h> 58d5368bf3SDmitry Chagin #include <compat/linux/linux_util.h> 594ab7403bSDmitry Chagin 60b9d3556aSYuri Pankov struct futex_list futex_list; 61a161fba9SYuri Pankov struct mtx futex_mtx; /* protects the futex list */ 62b9d3556aSYuri Pankov 63d151344dSDmitry Chagin CTASSERT(LINUX_IFNAMSIZ == IFNAMSIZ); 644ab7403bSDmitry Chagin 654ab7403bSDmitry Chagin static int bsd_to_linux_sigtbl[LINUX_SIGTBLSZ] = { 664ab7403bSDmitry Chagin LINUX_SIGHUP, /* SIGHUP */ 674ab7403bSDmitry Chagin LINUX_SIGINT, /* SIGINT */ 684ab7403bSDmitry Chagin LINUX_SIGQUIT, /* SIGQUIT */ 694ab7403bSDmitry Chagin LINUX_SIGILL, /* SIGILL */ 704ab7403bSDmitry Chagin LINUX_SIGTRAP, /* SIGTRAP */ 714ab7403bSDmitry Chagin LINUX_SIGABRT, /* SIGABRT */ 724ab7403bSDmitry Chagin 0, /* SIGEMT */ 734ab7403bSDmitry Chagin LINUX_SIGFPE, /* SIGFPE */ 744ab7403bSDmitry Chagin LINUX_SIGKILL, /* SIGKILL */ 754ab7403bSDmitry Chagin LINUX_SIGBUS, /* SIGBUS */ 764ab7403bSDmitry Chagin LINUX_SIGSEGV, /* SIGSEGV */ 774ab7403bSDmitry Chagin LINUX_SIGSYS, /* SIGSYS */ 784ab7403bSDmitry Chagin LINUX_SIGPIPE, /* SIGPIPE */ 794ab7403bSDmitry Chagin LINUX_SIGALRM, /* SIGALRM */ 804ab7403bSDmitry Chagin LINUX_SIGTERM, /* SIGTERM */ 814ab7403bSDmitry Chagin LINUX_SIGURG, /* SIGURG */ 824ab7403bSDmitry Chagin LINUX_SIGSTOP, /* SIGSTOP */ 834ab7403bSDmitry Chagin LINUX_SIGTSTP, /* SIGTSTP */ 844ab7403bSDmitry Chagin LINUX_SIGCONT, /* SIGCONT */ 854ab7403bSDmitry Chagin LINUX_SIGCHLD, /* SIGCHLD */ 864ab7403bSDmitry Chagin LINUX_SIGTTIN, /* SIGTTIN */ 874ab7403bSDmitry Chagin LINUX_SIGTTOU, /* SIGTTOU */ 884ab7403bSDmitry Chagin LINUX_SIGIO, /* SIGIO */ 894ab7403bSDmitry Chagin LINUX_SIGXCPU, /* SIGXCPU */ 904ab7403bSDmitry Chagin LINUX_SIGXFSZ, /* SIGXFSZ */ 914ab7403bSDmitry Chagin LINUX_SIGVTALRM,/* SIGVTALRM */ 924ab7403bSDmitry Chagin LINUX_SIGPROF, /* SIGPROF */ 934ab7403bSDmitry Chagin LINUX_SIGWINCH, /* SIGWINCH */ 944ab7403bSDmitry Chagin 0, /* SIGINFO */ 954ab7403bSDmitry Chagin LINUX_SIGUSR1, /* SIGUSR1 */ 964ab7403bSDmitry Chagin LINUX_SIGUSR2 /* SIGUSR2 */ 974ab7403bSDmitry Chagin }; 984ab7403bSDmitry Chagin 994ab7403bSDmitry Chagin static int linux_to_bsd_sigtbl[LINUX_SIGTBLSZ] = { 1004ab7403bSDmitry Chagin SIGHUP, /* LINUX_SIGHUP */ 1014ab7403bSDmitry Chagin SIGINT, /* LINUX_SIGINT */ 1024ab7403bSDmitry Chagin SIGQUIT, /* LINUX_SIGQUIT */ 1034ab7403bSDmitry Chagin SIGILL, /* LINUX_SIGILL */ 1044ab7403bSDmitry Chagin SIGTRAP, /* LINUX_SIGTRAP */ 1054ab7403bSDmitry Chagin SIGABRT, /* LINUX_SIGABRT */ 1064ab7403bSDmitry Chagin SIGBUS, /* LINUX_SIGBUS */ 1074ab7403bSDmitry Chagin SIGFPE, /* LINUX_SIGFPE */ 1084ab7403bSDmitry Chagin SIGKILL, /* LINUX_SIGKILL */ 1094ab7403bSDmitry Chagin SIGUSR1, /* LINUX_SIGUSR1 */ 1104ab7403bSDmitry Chagin SIGSEGV, /* LINUX_SIGSEGV */ 1114ab7403bSDmitry Chagin SIGUSR2, /* LINUX_SIGUSR2 */ 1124ab7403bSDmitry Chagin SIGPIPE, /* LINUX_SIGPIPE */ 1134ab7403bSDmitry Chagin SIGALRM, /* LINUX_SIGALRM */ 1144ab7403bSDmitry Chagin SIGTERM, /* LINUX_SIGTERM */ 1154ab7403bSDmitry Chagin SIGBUS, /* LINUX_SIGSTKFLT */ 1164ab7403bSDmitry Chagin SIGCHLD, /* LINUX_SIGCHLD */ 1174ab7403bSDmitry Chagin SIGCONT, /* LINUX_SIGCONT */ 1184ab7403bSDmitry Chagin SIGSTOP, /* LINUX_SIGSTOP */ 1194ab7403bSDmitry Chagin SIGTSTP, /* LINUX_SIGTSTP */ 1204ab7403bSDmitry Chagin SIGTTIN, /* LINUX_SIGTTIN */ 1214ab7403bSDmitry Chagin SIGTTOU, /* LINUX_SIGTTOU */ 1224ab7403bSDmitry Chagin SIGURG, /* LINUX_SIGURG */ 1234ab7403bSDmitry Chagin SIGXCPU, /* LINUX_SIGXCPU */ 1244ab7403bSDmitry Chagin SIGXFSZ, /* LINUX_SIGXFSZ */ 1254ab7403bSDmitry Chagin SIGVTALRM, /* LINUX_SIGVTALARM */ 1264ab7403bSDmitry Chagin SIGPROF, /* LINUX_SIGPROF */ 1274ab7403bSDmitry Chagin SIGWINCH, /* LINUX_SIGWINCH */ 1284ab7403bSDmitry Chagin SIGIO, /* LINUX_SIGIO */ 1294ab7403bSDmitry Chagin /* 1304ab7403bSDmitry Chagin * FreeBSD does not have SIGPWR signal, map Linux SIGPWR signal 1314ab7403bSDmitry Chagin * to the first unused FreeBSD signal number. Since Linux supports 1324ab7403bSDmitry Chagin * signals from 1 to 64 we are ok here as our SIGRTMIN = 65. 1334ab7403bSDmitry Chagin */ 1344ab7403bSDmitry Chagin SIGRTMIN, /* LINUX_SIGPWR */ 1354ab7403bSDmitry Chagin SIGSYS /* LINUX_SIGSYS */ 1364ab7403bSDmitry Chagin }; 1374ab7403bSDmitry Chagin 138044ab55eSEdward Tomasz Napierala static struct cdev *dev_shm_cdev; 139044ab55eSEdward Tomasz Napierala static struct cdevsw dev_shm_cdevsw = { 140044ab55eSEdward Tomasz Napierala .d_version = D_VERSION, 141044ab55eSEdward Tomasz Napierala .d_name = "dev_shm", 142044ab55eSEdward Tomasz Napierala }; 143044ab55eSEdward Tomasz Napierala 1444ab7403bSDmitry Chagin /* 1454ab7403bSDmitry Chagin * Map Linux RT signals to the FreeBSD RT signals. 1464ab7403bSDmitry Chagin */ 1474ab7403bSDmitry Chagin static inline int 1484ab7403bSDmitry Chagin linux_to_bsd_rt_signal(int sig) 1494ab7403bSDmitry Chagin { 1504ab7403bSDmitry Chagin 1514ab7403bSDmitry Chagin return (SIGRTMIN + 1 + sig - LINUX_SIGRTMIN); 1524ab7403bSDmitry Chagin } 1534ab7403bSDmitry Chagin 1544ab7403bSDmitry Chagin static inline int 1554ab7403bSDmitry Chagin bsd_to_linux_rt_signal(int sig) 1564ab7403bSDmitry Chagin { 1574ab7403bSDmitry Chagin 1584ab7403bSDmitry Chagin return (sig - SIGRTMIN - 1 + LINUX_SIGRTMIN); 1594ab7403bSDmitry Chagin } 1604ab7403bSDmitry Chagin 1614ab7403bSDmitry Chagin int 1624ab7403bSDmitry Chagin linux_to_bsd_signal(int sig) 1634ab7403bSDmitry Chagin { 1644ab7403bSDmitry Chagin 1658081c6ceSEdward Tomasz Napierala KASSERT(sig > 0 && sig <= LINUX_SIGRTMAX, ("invalid Linux signal %d\n", sig)); 1664ab7403bSDmitry Chagin 1674ab7403bSDmitry Chagin if (sig < LINUX_SIGRTMIN) 1684ab7403bSDmitry Chagin return (linux_to_bsd_sigtbl[_SIG_IDX(sig)]); 1694ab7403bSDmitry Chagin 1704ab7403bSDmitry Chagin return (linux_to_bsd_rt_signal(sig)); 1714ab7403bSDmitry Chagin } 1724ab7403bSDmitry Chagin 1734ab7403bSDmitry Chagin int 1744ab7403bSDmitry Chagin bsd_to_linux_signal(int sig) 1754ab7403bSDmitry Chagin { 1764ab7403bSDmitry Chagin 1774ab7403bSDmitry Chagin if (sig <= LINUX_SIGTBLSZ) 1784ab7403bSDmitry Chagin return (bsd_to_linux_sigtbl[_SIG_IDX(sig)]); 1794ab7403bSDmitry Chagin if (sig == SIGRTMIN) 1804ab7403bSDmitry Chagin return (LINUX_SIGPWR); 1814ab7403bSDmitry Chagin 1824ab7403bSDmitry Chagin return (bsd_to_linux_rt_signal(sig)); 1834ab7403bSDmitry Chagin } 1844ab7403bSDmitry Chagin 1854ab7403bSDmitry Chagin int 1864ab7403bSDmitry Chagin linux_to_bsd_sigaltstack(int lsa) 1874ab7403bSDmitry Chagin { 1884ab7403bSDmitry Chagin int bsa = 0; 1894ab7403bSDmitry Chagin 1904ab7403bSDmitry Chagin if (lsa & LINUX_SS_DISABLE) 1914ab7403bSDmitry Chagin bsa |= SS_DISABLE; 1924ab7403bSDmitry Chagin /* 1934ab7403bSDmitry Chagin * Linux ignores SS_ONSTACK flag for ss 1944ab7403bSDmitry Chagin * parameter while FreeBSD prohibits it. 1954ab7403bSDmitry Chagin */ 1964ab7403bSDmitry Chagin return (bsa); 1974ab7403bSDmitry Chagin } 1984ab7403bSDmitry Chagin 1994ab7403bSDmitry Chagin int 2004ab7403bSDmitry Chagin bsd_to_linux_sigaltstack(int bsa) 2014ab7403bSDmitry Chagin { 2024ab7403bSDmitry Chagin int lsa = 0; 2034ab7403bSDmitry Chagin 2044ab7403bSDmitry Chagin if (bsa & SS_DISABLE) 2054ab7403bSDmitry Chagin lsa |= LINUX_SS_DISABLE; 2064ab7403bSDmitry Chagin if (bsa & SS_ONSTACK) 2074ab7403bSDmitry Chagin lsa |= LINUX_SS_ONSTACK; 2084ab7403bSDmitry Chagin return (lsa); 2094ab7403bSDmitry Chagin } 2104ab7403bSDmitry Chagin 2114ab7403bSDmitry Chagin void 2124ab7403bSDmitry Chagin linux_to_bsd_sigset(l_sigset_t *lss, sigset_t *bss) 2134ab7403bSDmitry Chagin { 2144ab7403bSDmitry Chagin int b, l; 2154ab7403bSDmitry Chagin 2164ab7403bSDmitry Chagin SIGEMPTYSET(*bss); 2174ab7403bSDmitry Chagin for (l = 1; l <= LINUX_SIGRTMAX; l++) { 2184ab7403bSDmitry Chagin if (LINUX_SIGISMEMBER(*lss, l)) { 2194ab7403bSDmitry Chagin b = linux_to_bsd_signal(l); 2204ab7403bSDmitry Chagin if (b) 2214ab7403bSDmitry Chagin SIGADDSET(*bss, b); 2224ab7403bSDmitry Chagin } 2234ab7403bSDmitry Chagin } 2244ab7403bSDmitry Chagin } 2254ab7403bSDmitry Chagin 2264ab7403bSDmitry Chagin void 2274ab7403bSDmitry Chagin bsd_to_linux_sigset(sigset_t *bss, l_sigset_t *lss) 2284ab7403bSDmitry Chagin { 2294ab7403bSDmitry Chagin int b, l; 2304ab7403bSDmitry Chagin 2314ab7403bSDmitry Chagin LINUX_SIGEMPTYSET(*lss); 2324ab7403bSDmitry Chagin for (b = 1; b <= SIGRTMAX; b++) { 2334ab7403bSDmitry Chagin if (SIGISMEMBER(*bss, b)) { 2344ab7403bSDmitry Chagin l = bsd_to_linux_signal(b); 2354ab7403bSDmitry Chagin if (l) 2364ab7403bSDmitry Chagin LINUX_SIGADDSET(*lss, l); 2374ab7403bSDmitry Chagin } 2384ab7403bSDmitry Chagin } 2394ab7403bSDmitry Chagin } 240d151344dSDmitry Chagin 241d151344dSDmitry Chagin /* 242d151344dSDmitry Chagin * Translate a Linux interface name to a FreeBSD interface name, 243d151344dSDmitry Chagin * and return the associated ifnet structure 244d151344dSDmitry Chagin * bsdname and lxname need to be least IFNAMSIZ bytes long, but 245d151344dSDmitry Chagin * can point to the same buffer. 246d151344dSDmitry Chagin */ 247d151344dSDmitry Chagin struct ifnet * 248d151344dSDmitry Chagin ifname_linux_to_bsd(struct thread *td, const char *lxname, char *bsdname) 249d151344dSDmitry Chagin { 250d151344dSDmitry Chagin struct ifnet *ifp; 251d151344dSDmitry Chagin int len, unit; 252d151344dSDmitry Chagin char *ep; 2539c1437aeSDmitry Chagin int index; 2549c1437aeSDmitry Chagin bool is_eth, is_lo; 255d151344dSDmitry Chagin 256d151344dSDmitry Chagin for (len = 0; len < LINUX_IFNAMSIZ; ++len) 2579c1437aeSDmitry Chagin if (!isalpha(lxname[len]) || lxname[len] == '\0') 258d151344dSDmitry Chagin break; 259d151344dSDmitry Chagin if (len == 0 || len == LINUX_IFNAMSIZ) 260d151344dSDmitry Chagin return (NULL); 261d151344dSDmitry Chagin /* Linux loopback interface name is lo (not lo0) */ 2629c1437aeSDmitry Chagin is_lo = (len == 2 && strncmp(lxname, "lo", len) == 0); 263d151344dSDmitry Chagin unit = (int)strtoul(lxname + len, &ep, 10); 264d151344dSDmitry Chagin if ((ep == NULL || ep == lxname + len || ep >= lxname + LINUX_IFNAMSIZ) && 265d151344dSDmitry Chagin is_lo == 0) 266d151344dSDmitry Chagin return (NULL); 267d151344dSDmitry Chagin index = 0; 2689c1437aeSDmitry Chagin is_eth = (len == 3 && strncmp(lxname, "eth", len) == 0); 269d151344dSDmitry Chagin 270d151344dSDmitry Chagin CURVNET_SET(TD_TO_VNET(td)); 271d151344dSDmitry Chagin IFNET_RLOCK(); 272d151344dSDmitry Chagin CK_STAILQ_FOREACH(ifp, &V_ifnet, if_link) { 273d151344dSDmitry Chagin /* 274d151344dSDmitry Chagin * Allow Linux programs to use FreeBSD names. Don't presume 275d151344dSDmitry Chagin * we never have an interface named "eth", so don't make 276d151344dSDmitry Chagin * the test optional based on is_eth. 277d151344dSDmitry Chagin */ 278d151344dSDmitry Chagin if (strncmp(ifp->if_xname, lxname, LINUX_IFNAMSIZ) == 0) 279d151344dSDmitry Chagin break; 280d151344dSDmitry Chagin if (is_eth && IFP_IS_ETH(ifp) && unit == index++) 281d151344dSDmitry Chagin break; 282d151344dSDmitry Chagin if (is_lo && IFP_IS_LOOP(ifp)) 283d151344dSDmitry Chagin break; 284d151344dSDmitry Chagin } 285d151344dSDmitry Chagin IFNET_RUNLOCK(); 286d151344dSDmitry Chagin CURVNET_RESTORE(); 287d151344dSDmitry Chagin if (ifp != NULL && bsdname != NULL) 288d151344dSDmitry Chagin strlcpy(bsdname, ifp->if_xname, IFNAMSIZ); 289d151344dSDmitry Chagin return (ifp); 290d151344dSDmitry Chagin } 291d151344dSDmitry Chagin 292d151344dSDmitry Chagin void 293d151344dSDmitry Chagin linux_ifflags(struct ifnet *ifp, short *flags) 294d151344dSDmitry Chagin { 295bbac65c7SDmitry Chagin unsigned short fl; 296d151344dSDmitry Chagin 297bbac65c7SDmitry Chagin fl = (ifp->if_flags | ifp->if_drv_flags) & 0xffff; 298bbac65c7SDmitry Chagin *flags = 0; 299bbac65c7SDmitry Chagin if (fl & IFF_UP) 300bbac65c7SDmitry Chagin *flags |= LINUX_IFF_UP; 301bbac65c7SDmitry Chagin if (fl & IFF_BROADCAST) 302bbac65c7SDmitry Chagin *flags |= LINUX_IFF_BROADCAST; 303bbac65c7SDmitry Chagin if (fl & IFF_DEBUG) 304bbac65c7SDmitry Chagin *flags |= LINUX_IFF_DEBUG; 305bbac65c7SDmitry Chagin if (fl & IFF_LOOPBACK) 306bbac65c7SDmitry Chagin *flags |= LINUX_IFF_LOOPBACK; 307bbac65c7SDmitry Chagin if (fl & IFF_POINTOPOINT) 308bbac65c7SDmitry Chagin *flags |= LINUX_IFF_POINTOPOINT; 309bbac65c7SDmitry Chagin if (fl & IFF_DRV_RUNNING) 310bbac65c7SDmitry Chagin *flags |= LINUX_IFF_RUNNING; 311bbac65c7SDmitry Chagin if (fl & IFF_NOARP) 312bbac65c7SDmitry Chagin *flags |= LINUX_IFF_NOARP; 313bbac65c7SDmitry Chagin if (fl & IFF_PROMISC) 314bbac65c7SDmitry Chagin *flags |= LINUX_IFF_PROMISC; 315bbac65c7SDmitry Chagin if (fl & IFF_ALLMULTI) 316bbac65c7SDmitry Chagin *flags |= LINUX_IFF_ALLMULTI; 317bbac65c7SDmitry Chagin if (fl & IFF_MULTICAST) 318bbac65c7SDmitry Chagin *flags |= LINUX_IFF_MULTICAST; 319d151344dSDmitry Chagin } 320d151344dSDmitry Chagin 321d151344dSDmitry Chagin int 322d151344dSDmitry Chagin linux_ifhwaddr(struct ifnet *ifp, struct l_sockaddr *lsa) 323d151344dSDmitry Chagin { 324d151344dSDmitry Chagin struct ifaddr *ifa; 325d151344dSDmitry Chagin struct sockaddr_dl *sdl; 326d151344dSDmitry Chagin 327d151344dSDmitry Chagin if (IFP_IS_LOOP(ifp)) { 328d151344dSDmitry Chagin bzero(lsa, sizeof(*lsa)); 329d151344dSDmitry Chagin lsa->sa_family = LINUX_ARPHRD_LOOPBACK; 330d151344dSDmitry Chagin return (0); 331d151344dSDmitry Chagin } 332d151344dSDmitry Chagin 333d151344dSDmitry Chagin if (!IFP_IS_ETH(ifp)) 334d151344dSDmitry Chagin return (ENOENT); 335d151344dSDmitry Chagin 336d151344dSDmitry Chagin CK_STAILQ_FOREACH(ifa, &ifp->if_addrhead, ifa_link) { 337d151344dSDmitry Chagin sdl = (struct sockaddr_dl*)ifa->ifa_addr; 338d151344dSDmitry Chagin if (sdl != NULL && (sdl->sdl_family == AF_LINK) && 339d151344dSDmitry Chagin (sdl->sdl_type == IFT_ETHER)) { 340d151344dSDmitry Chagin bzero(lsa, sizeof(*lsa)); 341d151344dSDmitry Chagin lsa->sa_family = LINUX_ARPHRD_ETHER; 342d151344dSDmitry Chagin bcopy(LLADDR(sdl), lsa->sa_data, LINUX_IFHWADDRLEN); 343d151344dSDmitry Chagin return (0); 344d151344dSDmitry Chagin } 345d151344dSDmitry Chagin } 346d151344dSDmitry Chagin 347d151344dSDmitry Chagin return (ENOENT); 348d151344dSDmitry Chagin } 349d5368bf3SDmitry Chagin 350d5368bf3SDmitry Chagin int 351d5368bf3SDmitry Chagin linux_to_bsd_domain(int domain) 352d5368bf3SDmitry Chagin { 353d5368bf3SDmitry Chagin 354d5368bf3SDmitry Chagin switch (domain) { 355d5368bf3SDmitry Chagin case LINUX_AF_UNSPEC: 356d5368bf3SDmitry Chagin return (AF_UNSPEC); 357d5368bf3SDmitry Chagin case LINUX_AF_UNIX: 358d5368bf3SDmitry Chagin return (AF_LOCAL); 359d5368bf3SDmitry Chagin case LINUX_AF_INET: 360d5368bf3SDmitry Chagin return (AF_INET); 361d5368bf3SDmitry Chagin case LINUX_AF_INET6: 362d5368bf3SDmitry Chagin return (AF_INET6); 363d5368bf3SDmitry Chagin case LINUX_AF_AX25: 364d5368bf3SDmitry Chagin return (AF_CCITT); 365d5368bf3SDmitry Chagin case LINUX_AF_IPX: 366d5368bf3SDmitry Chagin return (AF_IPX); 367d5368bf3SDmitry Chagin case LINUX_AF_APPLETALK: 368d5368bf3SDmitry Chagin return (AF_APPLETALK); 369d5368bf3SDmitry Chagin } 370d5368bf3SDmitry Chagin return (-1); 371d5368bf3SDmitry Chagin } 372d5368bf3SDmitry Chagin 373d5368bf3SDmitry Chagin int 374d5368bf3SDmitry Chagin bsd_to_linux_domain(int domain) 375d5368bf3SDmitry Chagin { 376d5368bf3SDmitry Chagin 377d5368bf3SDmitry Chagin switch (domain) { 378d5368bf3SDmitry Chagin case AF_UNSPEC: 379d5368bf3SDmitry Chagin return (LINUX_AF_UNSPEC); 380d5368bf3SDmitry Chagin case AF_LOCAL: 381d5368bf3SDmitry Chagin return (LINUX_AF_UNIX); 382d5368bf3SDmitry Chagin case AF_INET: 383d5368bf3SDmitry Chagin return (LINUX_AF_INET); 384d5368bf3SDmitry Chagin case AF_INET6: 385d5368bf3SDmitry Chagin return (LINUX_AF_INET6); 386d5368bf3SDmitry Chagin case AF_CCITT: 387d5368bf3SDmitry Chagin return (LINUX_AF_AX25); 388d5368bf3SDmitry Chagin case AF_IPX: 389d5368bf3SDmitry Chagin return (LINUX_AF_IPX); 390d5368bf3SDmitry Chagin case AF_APPLETALK: 391d5368bf3SDmitry Chagin return (LINUX_AF_APPLETALK); 392d5368bf3SDmitry Chagin } 393d5368bf3SDmitry Chagin return (-1); 394d5368bf3SDmitry Chagin } 395d5368bf3SDmitry Chagin 396d5368bf3SDmitry Chagin /* 397d5368bf3SDmitry Chagin * Based on the fact that: 398d5368bf3SDmitry Chagin * 1. Native and Linux storage of struct sockaddr 399d5368bf3SDmitry Chagin * and struct sockaddr_in6 are equal. 400d5368bf3SDmitry Chagin * 2. On Linux sa_family is the first member of all struct sockaddr. 401d5368bf3SDmitry Chagin */ 402d5368bf3SDmitry Chagin int 403d5368bf3SDmitry Chagin bsd_to_linux_sockaddr(const struct sockaddr *sa, struct l_sockaddr **lsa, 404d5368bf3SDmitry Chagin socklen_t len) 405d5368bf3SDmitry Chagin { 406d5368bf3SDmitry Chagin struct l_sockaddr *kosa; 407d5368bf3SDmitry Chagin int error, bdom; 408d5368bf3SDmitry Chagin 409d5368bf3SDmitry Chagin *lsa = NULL; 410d5368bf3SDmitry Chagin if (len < 2 || len > UCHAR_MAX) 411d5368bf3SDmitry Chagin return (EINVAL); 412d5368bf3SDmitry Chagin 413d5368bf3SDmitry Chagin kosa = malloc(len, M_SONAME, M_WAITOK); 414d5368bf3SDmitry Chagin bcopy(sa, kosa, len); 415d5368bf3SDmitry Chagin 416d5368bf3SDmitry Chagin bdom = bsd_to_linux_domain(sa->sa_family); 417d5368bf3SDmitry Chagin if (bdom == -1) { 418d5368bf3SDmitry Chagin error = EAFNOSUPPORT; 419d5368bf3SDmitry Chagin goto out; 420d5368bf3SDmitry Chagin } 421d5368bf3SDmitry Chagin 422d5368bf3SDmitry Chagin kosa->sa_family = bdom; 423d5368bf3SDmitry Chagin *lsa = kosa; 424d5368bf3SDmitry Chagin return (0); 425d5368bf3SDmitry Chagin 426d5368bf3SDmitry Chagin out: 427d5368bf3SDmitry Chagin free(kosa, M_SONAME); 428d5368bf3SDmitry Chagin return (error); 429d5368bf3SDmitry Chagin } 430d5368bf3SDmitry Chagin 431d5368bf3SDmitry Chagin int 432d5368bf3SDmitry Chagin linux_to_bsd_sockaddr(const struct l_sockaddr *osa, struct sockaddr **sap, 433d5368bf3SDmitry Chagin socklen_t *len) 434d5368bf3SDmitry Chagin { 435d5368bf3SDmitry Chagin struct sockaddr *sa; 436d5368bf3SDmitry Chagin struct l_sockaddr *kosa; 437d5368bf3SDmitry Chagin #ifdef INET6 438d5368bf3SDmitry Chagin struct sockaddr_in6 *sin6; 439d5368bf3SDmitry Chagin bool oldv6size; 440d5368bf3SDmitry Chagin #endif 441d5368bf3SDmitry Chagin char *name; 442d5368bf3SDmitry Chagin int salen, bdom, error, hdrlen, namelen; 443d5368bf3SDmitry Chagin 444d5368bf3SDmitry Chagin if (*len < 2 || *len > UCHAR_MAX) 445d5368bf3SDmitry Chagin return (EINVAL); 446d5368bf3SDmitry Chagin 447d5368bf3SDmitry Chagin salen = *len; 448d5368bf3SDmitry Chagin 449d5368bf3SDmitry Chagin #ifdef INET6 450d5368bf3SDmitry Chagin oldv6size = false; 451d5368bf3SDmitry Chagin /* 452d5368bf3SDmitry Chagin * Check for old (pre-RFC2553) sockaddr_in6. We may accept it 453d5368bf3SDmitry Chagin * if it's a v4-mapped address, so reserve the proper space 454d5368bf3SDmitry Chagin * for it. 455d5368bf3SDmitry Chagin */ 456d5368bf3SDmitry Chagin if (salen == sizeof(struct sockaddr_in6) - sizeof(uint32_t)) { 457d5368bf3SDmitry Chagin salen += sizeof(uint32_t); 458d5368bf3SDmitry Chagin oldv6size = true; 459d5368bf3SDmitry Chagin } 460d5368bf3SDmitry Chagin #endif 461d5368bf3SDmitry Chagin 462d5368bf3SDmitry Chagin kosa = malloc(salen, M_SONAME, M_WAITOK); 463d5368bf3SDmitry Chagin 464d5368bf3SDmitry Chagin if ((error = copyin(osa, kosa, *len))) 465d5368bf3SDmitry Chagin goto out; 466d5368bf3SDmitry Chagin 467d5368bf3SDmitry Chagin bdom = linux_to_bsd_domain(kosa->sa_family); 468d5368bf3SDmitry Chagin if (bdom == -1) { 469d5368bf3SDmitry Chagin error = EAFNOSUPPORT; 470d5368bf3SDmitry Chagin goto out; 471d5368bf3SDmitry Chagin } 472d5368bf3SDmitry Chagin 473d5368bf3SDmitry Chagin #ifdef INET6 474d5368bf3SDmitry Chagin /* 475d5368bf3SDmitry Chagin * Older Linux IPv6 code uses obsolete RFC2133 struct sockaddr_in6, 476d5368bf3SDmitry Chagin * which lacks the scope id compared with RFC2553 one. If we detect 477d5368bf3SDmitry Chagin * the situation, reject the address and write a message to system log. 478d5368bf3SDmitry Chagin * 479d5368bf3SDmitry Chagin * Still accept addresses for which the scope id is not used. 480d5368bf3SDmitry Chagin */ 481d5368bf3SDmitry Chagin if (oldv6size) { 482d5368bf3SDmitry Chagin if (bdom == AF_INET6) { 483d5368bf3SDmitry Chagin sin6 = (struct sockaddr_in6 *)kosa; 484d5368bf3SDmitry Chagin if (IN6_IS_ADDR_V4MAPPED(&sin6->sin6_addr) || 485d5368bf3SDmitry Chagin (!IN6_IS_ADDR_LINKLOCAL(&sin6->sin6_addr) && 486d5368bf3SDmitry Chagin !IN6_IS_ADDR_SITELOCAL(&sin6->sin6_addr) && 487d5368bf3SDmitry Chagin !IN6_IS_ADDR_V4COMPAT(&sin6->sin6_addr) && 488d5368bf3SDmitry Chagin !IN6_IS_ADDR_UNSPECIFIED(&sin6->sin6_addr) && 489d5368bf3SDmitry Chagin !IN6_IS_ADDR_MULTICAST(&sin6->sin6_addr))) { 490d5368bf3SDmitry Chagin sin6->sin6_scope_id = 0; 491d5368bf3SDmitry Chagin } else { 492d5368bf3SDmitry Chagin linux_msg(curthread, 49386e794ebSEdward Tomasz Napierala "obsolete pre-RFC2553 sockaddr_in6 rejected"); 494d5368bf3SDmitry Chagin error = EINVAL; 495d5368bf3SDmitry Chagin goto out; 496d5368bf3SDmitry Chagin } 497d5368bf3SDmitry Chagin } else 498d5368bf3SDmitry Chagin salen -= sizeof(uint32_t); 499d5368bf3SDmitry Chagin } 500d5368bf3SDmitry Chagin #endif 501d5368bf3SDmitry Chagin if (bdom == AF_INET) { 502d5368bf3SDmitry Chagin if (salen < sizeof(struct sockaddr_in)) { 503d5368bf3SDmitry Chagin error = EINVAL; 504d5368bf3SDmitry Chagin goto out; 505d5368bf3SDmitry Chagin } 506d5368bf3SDmitry Chagin salen = sizeof(struct sockaddr_in); 507d5368bf3SDmitry Chagin } 508d5368bf3SDmitry Chagin 509d5368bf3SDmitry Chagin if (bdom == AF_LOCAL && salen > sizeof(struct sockaddr_un)) { 510d5368bf3SDmitry Chagin hdrlen = offsetof(struct sockaddr_un, sun_path); 511d5368bf3SDmitry Chagin name = ((struct sockaddr_un *)kosa)->sun_path; 512d5368bf3SDmitry Chagin if (*name == '\0') { 513d5368bf3SDmitry Chagin /* 514d5368bf3SDmitry Chagin * Linux abstract namespace starts with a NULL byte. 515d5368bf3SDmitry Chagin * XXX We do not support abstract namespace yet. 516d5368bf3SDmitry Chagin */ 517d5368bf3SDmitry Chagin namelen = strnlen(name + 1, salen - hdrlen - 1) + 1; 518d5368bf3SDmitry Chagin } else 519d5368bf3SDmitry Chagin namelen = strnlen(name, salen - hdrlen); 520d5368bf3SDmitry Chagin salen = hdrlen + namelen; 521d5368bf3SDmitry Chagin if (salen > sizeof(struct sockaddr_un)) { 522d5368bf3SDmitry Chagin error = ENAMETOOLONG; 523d5368bf3SDmitry Chagin goto out; 524d5368bf3SDmitry Chagin } 525d5368bf3SDmitry Chagin } 526d5368bf3SDmitry Chagin 527d5368bf3SDmitry Chagin sa = (struct sockaddr *)kosa; 528d5368bf3SDmitry Chagin sa->sa_family = bdom; 529d5368bf3SDmitry Chagin sa->sa_len = salen; 530d5368bf3SDmitry Chagin 531d5368bf3SDmitry Chagin *sap = sa; 532d5368bf3SDmitry Chagin *len = salen; 533d5368bf3SDmitry Chagin return (0); 534d5368bf3SDmitry Chagin 535d5368bf3SDmitry Chagin out: 536d5368bf3SDmitry Chagin free(kosa, M_SONAME); 537d5368bf3SDmitry Chagin return (error); 538d5368bf3SDmitry Chagin } 539044ab55eSEdward Tomasz Napierala 540044ab55eSEdward Tomasz Napierala void 541044ab55eSEdward Tomasz Napierala linux_dev_shm_create(void) 542044ab55eSEdward Tomasz Napierala { 543044ab55eSEdward Tomasz Napierala int error; 544044ab55eSEdward Tomasz Napierala 545044ab55eSEdward Tomasz Napierala error = make_dev_p(MAKEDEV_CHECKNAME | MAKEDEV_WAITOK, &dev_shm_cdev, 546044ab55eSEdward Tomasz Napierala &dev_shm_cdevsw, NULL, UID_ROOT, GID_WHEEL, 0, "shm/.mountpoint"); 547044ab55eSEdward Tomasz Napierala if (error != 0) { 548044ab55eSEdward Tomasz Napierala printf("%s: failed to create device node, error %d\n", 549044ab55eSEdward Tomasz Napierala __func__, error); 550044ab55eSEdward Tomasz Napierala } 551044ab55eSEdward Tomasz Napierala } 552044ab55eSEdward Tomasz Napierala 553044ab55eSEdward Tomasz Napierala void 554044ab55eSEdward Tomasz Napierala linux_dev_shm_destroy(void) 555044ab55eSEdward Tomasz Napierala { 556044ab55eSEdward Tomasz Napierala 557044ab55eSEdward Tomasz Napierala destroy_dev(dev_shm_cdev); 558044ab55eSEdward Tomasz Napierala } 5595403f186SKyle Evans 5605403f186SKyle Evans int 5615403f186SKyle Evans bsd_to_linux_bits_(int value, struct bsd_to_linux_bitmap *bitmap, 5625403f186SKyle Evans size_t mapcnt, int no_value) 5635403f186SKyle Evans { 5645403f186SKyle Evans int bsd_mask, bsd_value, linux_mask, linux_value; 5655403f186SKyle Evans int linux_ret; 5665403f186SKyle Evans size_t i; 5675403f186SKyle Evans bool applied; 5685403f186SKyle Evans 5695403f186SKyle Evans applied = false; 5705403f186SKyle Evans linux_ret = 0; 5715403f186SKyle Evans for (i = 0; i < mapcnt; ++i) { 5725403f186SKyle Evans bsd_mask = bitmap[i].bsd_mask; 5735403f186SKyle Evans bsd_value = bitmap[i].bsd_value; 5745403f186SKyle Evans if (bsd_mask == 0) 5755403f186SKyle Evans bsd_mask = bsd_value; 5765403f186SKyle Evans 5775403f186SKyle Evans linux_mask = bitmap[i].linux_mask; 5785403f186SKyle Evans linux_value = bitmap[i].linux_value; 5795403f186SKyle Evans if (linux_mask == 0) 5805403f186SKyle Evans linux_mask = linux_value; 5815403f186SKyle Evans 5825403f186SKyle Evans /* 5835403f186SKyle Evans * If a mask larger than just the value is set, we explicitly 5845403f186SKyle Evans * want to make sure that only this bit we mapped within that 5855403f186SKyle Evans * mask is set. 5865403f186SKyle Evans */ 5875403f186SKyle Evans if ((value & bsd_mask) == bsd_value) { 5885403f186SKyle Evans linux_ret = (linux_ret & ~linux_mask) | linux_value; 5895403f186SKyle Evans applied = true; 5905403f186SKyle Evans } 5915403f186SKyle Evans } 5925403f186SKyle Evans 5935403f186SKyle Evans if (!applied) 5945403f186SKyle Evans return (no_value); 5955403f186SKyle Evans return (linux_ret); 5965403f186SKyle Evans } 5975403f186SKyle Evans 5985403f186SKyle Evans int 5995403f186SKyle Evans linux_to_bsd_bits_(int value, struct bsd_to_linux_bitmap *bitmap, 6005403f186SKyle Evans size_t mapcnt, int no_value) 6015403f186SKyle Evans { 6025403f186SKyle Evans int bsd_mask, bsd_value, linux_mask, linux_value; 6035403f186SKyle Evans int bsd_ret; 6045403f186SKyle Evans size_t i; 6055403f186SKyle Evans bool applied; 6065403f186SKyle Evans 6075403f186SKyle Evans applied = false; 6085403f186SKyle Evans bsd_ret = 0; 6095403f186SKyle Evans for (i = 0; i < mapcnt; ++i) { 6105403f186SKyle Evans bsd_mask = bitmap[i].bsd_mask; 6115403f186SKyle Evans bsd_value = bitmap[i].bsd_value; 6125403f186SKyle Evans if (bsd_mask == 0) 6135403f186SKyle Evans bsd_mask = bsd_value; 6145403f186SKyle Evans 6155403f186SKyle Evans linux_mask = bitmap[i].linux_mask; 6165403f186SKyle Evans linux_value = bitmap[i].linux_value; 6175403f186SKyle Evans if (linux_mask == 0) 6185403f186SKyle Evans linux_mask = linux_value; 6195403f186SKyle Evans 6205403f186SKyle Evans /* 6215403f186SKyle Evans * If a mask larger than just the value is set, we explicitly 6225403f186SKyle Evans * want to make sure that only this bit we mapped within that 6235403f186SKyle Evans * mask is set. 6245403f186SKyle Evans */ 6255403f186SKyle Evans if ((value & linux_mask) == linux_value) { 6265403f186SKyle Evans bsd_ret = (bsd_ret & ~bsd_mask) | bsd_value; 6275403f186SKyle Evans applied = true; 6285403f186SKyle Evans } 6295403f186SKyle Evans } 6305403f186SKyle Evans 6315403f186SKyle Evans if (!applied) 6325403f186SKyle Evans return (no_value); 6335403f186SKyle Evans return (bsd_ret); 6345403f186SKyle Evans } 635*26795a03SDmitry Chagin 636*26795a03SDmitry Chagin void 637*26795a03SDmitry Chagin linux_to_bsd_poll_events(struct thread *td, int fd, short lev, 638*26795a03SDmitry Chagin short *bev) 639*26795a03SDmitry Chagin { 640*26795a03SDmitry Chagin struct proc *p = td->td_proc; 641*26795a03SDmitry Chagin struct filedesc *fdp; 642*26795a03SDmitry Chagin struct file *fp; 643*26795a03SDmitry Chagin int error; 644*26795a03SDmitry Chagin short bits = 0; 645*26795a03SDmitry Chagin 646*26795a03SDmitry Chagin if (lev & LINUX_POLLIN) 647*26795a03SDmitry Chagin bits |= POLLIN; 648*26795a03SDmitry Chagin if (lev & LINUX_POLLPRI) 649*26795a03SDmitry Chagin bits |= POLLPRI; 650*26795a03SDmitry Chagin if (lev & LINUX_POLLOUT) 651*26795a03SDmitry Chagin bits |= POLLOUT; 652*26795a03SDmitry Chagin if (lev & LINUX_POLLERR) 653*26795a03SDmitry Chagin bits |= POLLERR; 654*26795a03SDmitry Chagin if (lev & LINUX_POLLHUP) 655*26795a03SDmitry Chagin bits |= POLLHUP; 656*26795a03SDmitry Chagin if (lev & LINUX_POLLNVAL) 657*26795a03SDmitry Chagin bits |= POLLNVAL; 658*26795a03SDmitry Chagin if (lev & LINUX_POLLRDNORM) 659*26795a03SDmitry Chagin bits |= POLLRDNORM; 660*26795a03SDmitry Chagin if (lev & LINUX_POLLRDBAND) 661*26795a03SDmitry Chagin bits |= POLLRDBAND; 662*26795a03SDmitry Chagin if (lev & LINUX_POLLWRBAND) 663*26795a03SDmitry Chagin bits |= POLLWRBAND; 664*26795a03SDmitry Chagin if (lev & LINUX_POLLWRNORM) 665*26795a03SDmitry Chagin bits |= POLLWRNORM; 666*26795a03SDmitry Chagin 667*26795a03SDmitry Chagin if (lev & LINUX_POLLRDHUP) { 668*26795a03SDmitry Chagin /* 669*26795a03SDmitry Chagin * It seems that the Linux silencly ignores POLLRDHUP 670*26795a03SDmitry Chagin * on non-socket file descriptors unlike FreeBSD, where 671*26795a03SDmitry Chagin * events bits is more strictly checked (POLLSTANDARD). 672*26795a03SDmitry Chagin */ 673*26795a03SDmitry Chagin fdp = p->p_fd; 674*26795a03SDmitry Chagin error = fget_unlocked(fdp, fd, &cap_no_rights, &fp); 675*26795a03SDmitry Chagin if (error == 0) { 676*26795a03SDmitry Chagin /* 677*26795a03SDmitry Chagin * XXX. On FreeBSD POLLRDHUP applies only to 678*26795a03SDmitry Chagin * stream sockets. 679*26795a03SDmitry Chagin */ 680*26795a03SDmitry Chagin if (fp->f_type == DTYPE_SOCKET) 681*26795a03SDmitry Chagin bits |= POLLRDHUP; 682*26795a03SDmitry Chagin fdrop(fp, td); 683*26795a03SDmitry Chagin } 684*26795a03SDmitry Chagin } 685*26795a03SDmitry Chagin 686*26795a03SDmitry Chagin if (lev & LINUX_POLLMSG) 687*26795a03SDmitry Chagin LINUX_RATELIMIT_MSG_OPT1("unsupported POLLMSG, events(%d)", lev); 688*26795a03SDmitry Chagin if (lev & LINUX_POLLREMOVE) 689*26795a03SDmitry Chagin LINUX_RATELIMIT_MSG_OPT1("unsupported POLLREMOVE, events(%d)", lev); 690*26795a03SDmitry Chagin 691*26795a03SDmitry Chagin *bev = bits; 692*26795a03SDmitry Chagin } 693*26795a03SDmitry Chagin 694*26795a03SDmitry Chagin void 695*26795a03SDmitry Chagin bsd_to_linux_poll_events(short bev, short *lev) 696*26795a03SDmitry Chagin { 697*26795a03SDmitry Chagin short bits = 0; 698*26795a03SDmitry Chagin 699*26795a03SDmitry Chagin if (bev & POLLIN) 700*26795a03SDmitry Chagin bits |= LINUX_POLLIN; 701*26795a03SDmitry Chagin if (bev & POLLPRI) 702*26795a03SDmitry Chagin bits |= LINUX_POLLPRI; 703*26795a03SDmitry Chagin if (bev & (POLLOUT | POLLWRNORM)) 704*26795a03SDmitry Chagin /* 705*26795a03SDmitry Chagin * POLLWRNORM is equal to POLLOUT on FreeBSD, 706*26795a03SDmitry Chagin * but not on Linux 707*26795a03SDmitry Chagin */ 708*26795a03SDmitry Chagin bits |= LINUX_POLLOUT; 709*26795a03SDmitry Chagin if (bev & POLLERR) 710*26795a03SDmitry Chagin bits |= LINUX_POLLERR; 711*26795a03SDmitry Chagin if (bev & POLLHUP) 712*26795a03SDmitry Chagin bits |= LINUX_POLLHUP; 713*26795a03SDmitry Chagin if (bev & POLLNVAL) 714*26795a03SDmitry Chagin bits |= LINUX_POLLNVAL; 715*26795a03SDmitry Chagin if (bev & POLLRDNORM) 716*26795a03SDmitry Chagin bits |= LINUX_POLLRDNORM; 717*26795a03SDmitry Chagin if (bev & POLLRDBAND) 718*26795a03SDmitry Chagin bits |= LINUX_POLLRDBAND; 719*26795a03SDmitry Chagin if (bev & POLLWRBAND) 720*26795a03SDmitry Chagin bits |= LINUX_POLLWRBAND; 721*26795a03SDmitry Chagin if (bev & POLLRDHUP) 722*26795a03SDmitry Chagin bits |= LINUX_POLLRDHUP; 723*26795a03SDmitry Chagin 724*26795a03SDmitry Chagin *lev = bits; 725*26795a03SDmitry Chagin } 726