8322 nl: misleading-indentation
[unleashed/tickless.git] / usr / src / lib / libc / port / threads / scalls.c
blob4a62ea76b974f6cb64e37b745a58d87ee7969c5f
1 /*
2 * CDDL HEADER START
4 * The contents of this file are subject to the terms of the
5 * Common Development and Distribution License (the "License").
6 * You may not use this file except in compliance with the License.
8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9 * or http://www.opensolaris.org/os/licensing.
10 * See the License for the specific language governing permissions
11 * and limitations under the License.
13 * When distributing Covered Code, include this CDDL HEADER in each
14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15 * If applicable, add the following below this CDDL HEADER, with the
16 * fields enclosed by brackets "[]" replaced with your own identifying
17 * information: Portions Copyright [yyyy] [name of copyright owner]
19 * CDDL HEADER END
23 * Copyright 2010 Sun Microsystems, Inc. All rights reserved.
24 * Use is subject to license terms.
25 * Copyright (c) 2015, Joyent, Inc. All rights reserved.
28 /* Copyright (c) 2013, OmniTI Computer Consulting, Inc. All rights reserved. */
30 #include "lint.h"
31 #include "thr_uberdata.h"
32 #include <stdarg.h>
33 #include <poll.h>
34 #include <stropts.h>
35 #include <dlfcn.h>
36 #include <wait.h>
37 #include <sys/socket.h>
38 #include <sys/uio.h>
39 #include <sys/file.h>
40 #include <sys/door.h>
43 * These leading-underbar symbols exist because mistakes were made
44 * in the past that put them into non-SUNWprivate versions of
45 * the libc mapfiles. They should be eliminated, but oh well...
47 #pragma weak _fork = fork
48 #pragma weak _read = read
49 #pragma weak _write = write
50 #pragma weak _getmsg = getmsg
51 #pragma weak _getpmsg = getpmsg
52 #pragma weak _putmsg = putmsg
53 #pragma weak _putpmsg = putpmsg
54 #pragma weak _sleep = sleep
55 #pragma weak _close = close
56 #pragma weak _creat = creat
57 #pragma weak _fcntl = fcntl
58 #pragma weak _fsync = fsync
59 #pragma weak _lockf = lockf
60 #pragma weak _msgrcv = msgrcv
61 #pragma weak _msgsnd = msgsnd
62 #pragma weak _msync = msync
63 #pragma weak _open = open
64 #pragma weak _openat = openat
65 #pragma weak _pause = pause
66 #pragma weak _readv = readv
67 #pragma weak _sigpause = sigpause
68 #pragma weak _sigsuspend = sigsuspend
69 #pragma weak _tcdrain = tcdrain
70 #pragma weak _waitid = waitid
71 #pragma weak _writev = writev
73 #if !defined(_LP64)
74 #pragma weak _creat64 = creat64
75 #pragma weak _lockf64 = lockf64
76 #pragma weak _open64 = open64
77 #pragma weak _openat64 = openat64
78 #pragma weak _pread64 = pread64
79 #pragma weak _pwrite64 = pwrite64
80 #endif
83 * These are SUNWprivate, but they are being used by Sun Studio libcollector.
85 #pragma weak _fork1 = fork1
86 #pragma weak _forkall = forkall
89 * atfork_lock protects the pthread_atfork() data structures.
91 * fork_lock does double-duty. Not only does it (and atfork_lock)
92 * serialize calls to fork() and forkall(), but it also serializes calls
93 * to thr_suspend() and thr_continue() (because fork() and forkall() also
94 * suspend and continue other threads and they want no competition).
96 * Functions called in dlopen()ed L10N objects can do anything, including
97 * call malloc() and free(). Such calls are not fork-safe when protected
98 * by an ordinary mutex that is acquired in libc's prefork processing
99 * because, with an interposed malloc library present, there would be a
100 * lock ordering violation due to the pthread_atfork() prefork function
101 * in the interposition library acquiring its malloc lock(s) before the
102 * ordinary mutex in libc being acquired by libc's prefork functions.
104 * Within libc, calls to malloc() and free() are fork-safe if the calls
105 * are made while holding no other libc locks. This covers almost all
106 * of libc's malloc() and free() calls. For those libc code paths, such
107 * as the above-mentioned L10N calls, that require serialization and that
108 * may call malloc() or free(), libc uses callout_lock_enter() to perform
109 * the serialization. This works because callout_lock is not acquired as
110 * part of running the pthread_atfork() prefork handlers (to avoid the
111 * lock ordering violation described above). Rather, it is simply
112 * reinitialized in postfork1_child() to cover the case that some
113 * now-defunct thread might have been suspended while holding it.
116 void
117 fork_lock_enter(void)
119 ASSERT(curthread->ul_critical == 0);
120 (void) mutex_lock(&curthread->ul_uberdata->fork_lock);
123 void
124 fork_lock_exit(void)
126 ASSERT(curthread->ul_critical == 0);
127 (void) mutex_unlock(&curthread->ul_uberdata->fork_lock);
131 * Use cancel_safe_mutex_lock() to protect against being cancelled while
132 * holding callout_lock and calling outside of libc (via L10N plugins).
133 * We will honor a pending cancellation request when callout_lock_exit()
134 * is called, by calling cancel_safe_mutex_unlock().
136 void
137 callout_lock_enter(void)
139 ASSERT(curthread->ul_critical == 0);
140 cancel_safe_mutex_lock(&curthread->ul_uberdata->callout_lock);
143 void
144 callout_lock_exit(void)
146 ASSERT(curthread->ul_critical == 0);
147 cancel_safe_mutex_unlock(&curthread->ul_uberdata->callout_lock);
150 pid_t
151 forkx(int flags)
153 ulwp_t *self = curthread;
154 uberdata_t *udp = self->ul_uberdata;
155 pid_t pid;
157 if (self->ul_vfork) {
159 * We are a child of vfork(); omit all of the fork
160 * logic and go straight to the system call trap.
161 * A vfork() child of a multithreaded parent
162 * must never call fork().
164 if (udp->uberflags.uf_mt) {
165 errno = ENOTSUP;
166 return (-1);
168 pid = __forkx(flags);
169 if (pid == 0) { /* child */
170 udp->pid = getpid();
171 self->ul_vfork = 0;
173 return (pid);
176 sigoff(self);
177 if (self->ul_fork) {
179 * Cannot call fork() from a fork handler.
181 sigon(self);
182 errno = EDEADLK;
183 return (-1);
185 self->ul_fork = 1;
188 * The functions registered by pthread_atfork() are defined by
189 * the application and its libraries and we must not hold any
190 * internal lmutex_lock()-acquired locks while invoking them.
191 * We hold only udp->atfork_lock to protect the atfork linkages.
192 * If one of these pthread_atfork() functions attempts to fork
193 * or to call pthread_atfork(), libc will detect the error and
194 * fail the call with EDEADLK. Otherwise, the pthread_atfork()
195 * functions are free to do anything they please (except they
196 * will not receive any signals).
198 (void) mutex_lock(&udp->atfork_lock);
201 * Posix (SUSv3) requires fork() to be async-signal-safe.
202 * This cannot be made to happen with fork handlers in place
203 * (they grab locks). To be in nominal compliance, don't run
204 * any fork handlers if we are called within a signal context.
205 * This leaves the child process in a questionable state with
206 * respect to its locks, but at least the parent process does
207 * not become deadlocked due to the calling thread attempting
208 * to acquire a lock that it already owns.
210 if (self->ul_siglink == NULL)
211 _prefork_handler();
214 * Block every other thread attempting thr_suspend() or thr_continue().
216 (void) mutex_lock(&udp->fork_lock);
219 * Block all signals.
220 * Just deferring them via sigoff() is not enough.
221 * We have to avoid taking a deferred signal in the child
222 * that was actually sent to the parent before __forkx().
224 block_all_signals(self);
227 * This suspends all threads but this one, leaving them
228 * suspended outside of any critical regions in the library.
229 * Thus, we are assured that no lmutex_lock()-acquired library
230 * locks are held while we invoke fork() from the current thread.
232 suspend_fork();
234 pid = __forkx(flags);
236 if (pid == 0) { /* child */
238 * Clear our schedctl pointer.
239 * Discard any deferred signal that was sent to the parent.
240 * Because we blocked all signals before __forkx(), a
241 * deferred signal cannot have been taken by the child.
243 self->ul_schedctl_called = NULL;
244 self->ul_schedctl = NULL;
245 self->ul_cursig = 0;
246 self->ul_siginfo.si_signo = 0;
247 udp->pid = getpid();
248 /* reset the library's data structures to reflect one thread */
249 unregister_locks();
250 postfork1_child();
251 restore_signals(self);
252 (void) mutex_unlock(&udp->fork_lock);
253 if (self->ul_siglink == NULL)
254 _postfork_child_handler();
255 } else {
256 /* restart all threads that were suspended for fork() */
257 continue_fork(0);
258 restore_signals(self);
259 (void) mutex_unlock(&udp->fork_lock);
260 if (self->ul_siglink == NULL)
261 _postfork_parent_handler();
264 (void) mutex_unlock(&udp->atfork_lock);
265 self->ul_fork = 0;
266 sigon(self);
268 return (pid);
272 * fork() is fork1() for both Posix threads and Solaris threads.
273 * The forkall() interface exists for applications that require
274 * the semantics of replicating all threads.
276 #pragma weak fork1 = fork
277 pid_t
278 fork(void)
280 return (forkx(0));
284 * Much of the logic here is the same as in forkx().
285 * See the comments in forkx(), above.
287 pid_t
288 forkallx(int flags)
290 ulwp_t *self = curthread;
291 uberdata_t *udp = self->ul_uberdata;
292 pid_t pid;
294 if (self->ul_vfork) {
295 if (udp->uberflags.uf_mt) {
296 errno = ENOTSUP;
297 return (-1);
299 pid = __forkallx(flags);
300 if (pid == 0) { /* child */
301 udp->pid = getpid();
302 self->ul_vfork = 0;
304 return (pid);
307 sigoff(self);
308 if (self->ul_fork) {
309 sigon(self);
310 errno = EDEADLK;
311 return (-1);
313 self->ul_fork = 1;
314 (void) mutex_lock(&udp->atfork_lock);
315 (void) mutex_lock(&udp->fork_lock);
316 block_all_signals(self);
317 suspend_fork();
319 pid = __forkallx(flags);
321 if (pid == 0) {
322 self->ul_schedctl_called = NULL;
323 self->ul_schedctl = NULL;
324 self->ul_cursig = 0;
325 self->ul_siginfo.si_signo = 0;
326 udp->pid = getpid();
327 unregister_locks();
328 continue_fork(1);
329 } else {
330 continue_fork(0);
332 restore_signals(self);
333 (void) mutex_unlock(&udp->fork_lock);
334 (void) mutex_unlock(&udp->atfork_lock);
335 self->ul_fork = 0;
336 sigon(self);
338 return (pid);
341 pid_t
342 forkall(void)
344 return (forkallx(0));
348 * For the implementation of cancellation at cancellation points.
350 #define PROLOGUE \
352 ulwp_t *self = curthread; \
353 int nocancel = \
354 (self->ul_vfork | self->ul_nocancel | self->ul_libc_locks | \
355 self->ul_critical | self->ul_sigdefer); \
356 int abort = 0; \
357 if (nocancel == 0) { \
358 self->ul_save_async = self->ul_cancel_async; \
359 if (!self->ul_cancel_disabled) { \
360 self->ul_cancel_async = 1; \
361 if (self->ul_cancel_pending) \
362 pthread_exit(PTHREAD_CANCELED); \
364 self->ul_sp = stkptr(); \
365 } else if (self->ul_cancel_pending && \
366 !self->ul_cancel_disabled) { \
367 set_cancel_eintr_flag(self); \
368 abort = 1; \
371 #define EPILOGUE \
372 if (nocancel == 0) { \
373 self->ul_sp = 0; \
374 self->ul_cancel_async = self->ul_save_async; \
379 * Perform the body of the action required by most of the cancelable
380 * function calls. The return(function_call) part is to allow the
381 * compiler to make the call be executed with tail recursion, which
382 * saves a register window on sparc and slightly (not much) improves
383 * the code for x86/x64 compilations.
385 #define PERFORM(function_call) \
386 PROLOGUE \
387 if (abort) { \
388 *self->ul_errnop = EINTR; \
389 return (-1); \
391 if (nocancel) \
392 return (function_call); \
393 rv = function_call; \
394 EPILOGUE \
395 return (rv);
398 * Specialized prologue for sigsuspend() and pollsys().
399 * These system calls pass a signal mask to the kernel.
400 * The kernel replaces the thread's signal mask with the
401 * temporary mask before the thread goes to sleep. If
402 * a signal is received, the signal handler will execute
403 * with the temporary mask, as modified by the sigaction
404 * for the particular signal.
406 * We block all signals until we reach the kernel with the
407 * temporary mask. This eliminates race conditions with
408 * setting the signal mask while signals are being posted.
410 #define PROLOGUE_MASK(sigmask) \
412 ulwp_t *self = curthread; \
413 int nocancel = \
414 (self->ul_vfork | self->ul_nocancel | self->ul_libc_locks | \
415 self->ul_critical | self->ul_sigdefer); \
416 if (!self->ul_vfork) { \
417 if (sigmask) { \
418 block_all_signals(self); \
419 self->ul_tmpmask = *sigmask; \
420 delete_reserved_signals(&self->ul_tmpmask); \
421 self->ul_sigsuspend = 1; \
423 if (nocancel == 0) { \
424 self->ul_save_async = self->ul_cancel_async; \
425 if (!self->ul_cancel_disabled) { \
426 self->ul_cancel_async = 1; \
427 if (self->ul_cancel_pending) { \
428 if (self->ul_sigsuspend) { \
429 self->ul_sigsuspend = 0;\
430 restore_signals(self); \
432 pthread_exit(PTHREAD_CANCELED); \
435 self->ul_sp = stkptr(); \
440 * If a signal is taken, we return from the system call wrapper with
441 * our original signal mask restored (see code in call_user_handler()).
442 * If not (self->ul_sigsuspend is still non-zero), we must restore our
443 * original signal mask ourself.
445 #define EPILOGUE_MASK \
446 if (nocancel == 0) { \
447 self->ul_sp = 0; \
448 self->ul_cancel_async = self->ul_save_async; \
450 if (self->ul_sigsuspend) { \
451 self->ul_sigsuspend = 0; \
452 restore_signals(self); \
457 * Cancellation prologue and epilogue functions,
458 * for cancellation points too complex to include here.
460 void
461 _cancel_prologue(void)
463 ulwp_t *self = curthread;
465 self->ul_cancel_prologue =
466 (self->ul_vfork | self->ul_nocancel | self->ul_libc_locks |
467 self->ul_critical | self->ul_sigdefer) != 0;
468 if (self->ul_cancel_prologue == 0) {
469 self->ul_save_async = self->ul_cancel_async;
470 if (!self->ul_cancel_disabled) {
471 self->ul_cancel_async = 1;
472 if (self->ul_cancel_pending)
473 pthread_exit(PTHREAD_CANCELED);
475 self->ul_sp = stkptr();
476 } else if (self->ul_cancel_pending &&
477 !self->ul_cancel_disabled) {
478 set_cancel_eintr_flag(self);
482 void
483 _cancel_epilogue(void)
485 ulwp_t *self = curthread;
487 if (self->ul_cancel_prologue == 0) {
488 self->ul_sp = 0;
489 self->ul_cancel_async = self->ul_save_async;
494 * Called from _thrp_join() (thr_join() is a cancellation point)
497 lwp_wait(thread_t tid, thread_t *found)
499 int error;
501 PROLOGUE
502 if (abort)
503 return (EINTR);
504 while ((error = __lwp_wait(tid, found)) == EINTR && !cancel_active())
505 continue;
506 EPILOGUE
507 return (error);
510 ssize_t
511 read(int fd, void *buf, size_t size)
513 extern ssize_t __read(int, void *, size_t);
514 ssize_t rv;
516 PERFORM(__read(fd, buf, size))
519 ssize_t
520 write(int fd, const void *buf, size_t size)
522 extern ssize_t __write(int, const void *, size_t);
523 ssize_t rv;
525 PERFORM(__write(fd, buf, size))
529 getmsg(int fd, struct strbuf *ctlptr, struct strbuf *dataptr,
530 int *flagsp)
532 extern int __getmsg(int, struct strbuf *, struct strbuf *, int *);
533 int rv;
535 PERFORM(__getmsg(fd, ctlptr, dataptr, flagsp))
539 getpmsg(int fd, struct strbuf *ctlptr, struct strbuf *dataptr,
540 int *bandp, int *flagsp)
542 extern int __getpmsg(int, struct strbuf *, struct strbuf *,
543 int *, int *);
544 int rv;
546 PERFORM(__getpmsg(fd, ctlptr, dataptr, bandp, flagsp))
550 putmsg(int fd, const struct strbuf *ctlptr,
551 const struct strbuf *dataptr, int flags)
553 extern int __putmsg(int, const struct strbuf *,
554 const struct strbuf *, int);
555 int rv;
557 PERFORM(__putmsg(fd, ctlptr, dataptr, flags))
561 __xpg4_putmsg(int fd, const struct strbuf *ctlptr,
562 const struct strbuf *dataptr, int flags)
564 extern int __putmsg(int, const struct strbuf *,
565 const struct strbuf *, int);
566 int rv;
568 PERFORM(__putmsg(fd, ctlptr, dataptr, flags|MSG_XPG4))
572 putpmsg(int fd, const struct strbuf *ctlptr,
573 const struct strbuf *dataptr, int band, int flags)
575 extern int __putpmsg(int, const struct strbuf *,
576 const struct strbuf *, int, int);
577 int rv;
579 PERFORM(__putpmsg(fd, ctlptr, dataptr, band, flags))
583 __xpg4_putpmsg(int fd, const struct strbuf *ctlptr,
584 const struct strbuf *dataptr, int band, int flags)
586 extern int __putpmsg(int, const struct strbuf *,
587 const struct strbuf *, int, int);
588 int rv;
590 PERFORM(__putpmsg(fd, ctlptr, dataptr, band, flags|MSG_XPG4))
594 nanosleep(const timespec_t *rqtp, timespec_t *rmtp)
596 int error;
598 PROLOGUE
599 error = abort? EINTR : __nanosleep(rqtp, rmtp);
600 EPILOGUE
601 if (error) {
602 errno = error;
603 return (-1);
605 return (0);
609 clock_nanosleep(clockid_t clock_id, int flags,
610 const timespec_t *rqtp, timespec_t *rmtp)
612 timespec_t reltime;
613 hrtime_t start;
614 hrtime_t rqlapse;
615 hrtime_t lapse;
616 int error;
618 switch (clock_id) {
619 case CLOCK_VIRTUAL:
620 case CLOCK_PROCESS_CPUTIME_ID:
621 case CLOCK_THREAD_CPUTIME_ID:
622 return (ENOTSUP);
623 case CLOCK_REALTIME:
624 case CLOCK_HIGHRES:
625 break;
626 default:
627 return (EINVAL);
629 if (flags & TIMER_ABSTIME) {
630 abstime_to_reltime(clock_id, rqtp, &reltime);
631 rmtp = NULL;
632 } else {
633 reltime = *rqtp;
634 if (clock_id == CLOCK_HIGHRES)
635 start = gethrtime();
637 restart:
638 PROLOGUE
639 error = abort? EINTR : __nanosleep(&reltime, rmtp);
640 EPILOGUE
641 if (error == 0 && clock_id == CLOCK_HIGHRES) {
643 * Don't return yet if we didn't really get a timeout.
644 * This can happen if we return because someone resets
645 * the system clock.
647 if (flags & TIMER_ABSTIME) {
648 if ((hrtime_t)(uint32_t)rqtp->tv_sec * NANOSEC +
649 rqtp->tv_nsec > gethrtime()) {
650 abstime_to_reltime(clock_id, rqtp, &reltime);
651 goto restart;
653 } else {
654 rqlapse = (hrtime_t)(uint32_t)rqtp->tv_sec * NANOSEC +
655 rqtp->tv_nsec;
656 lapse = gethrtime() - start;
657 if (rqlapse > lapse) {
658 hrt2ts(rqlapse - lapse, &reltime);
659 goto restart;
663 if (error == 0 && clock_id == CLOCK_REALTIME &&
664 (flags & TIMER_ABSTIME)) {
666 * Don't return yet just because someone reset the
667 * system clock. Recompute the new relative time
668 * and reissue the nanosleep() call if necessary.
670 * Resetting the system clock causes all sorts of
671 * problems and the SUSV3 standards body should
672 * have made the behavior of clock_nanosleep() be
673 * implementation-defined in such a case rather than
674 * being specific about honoring the new system time.
675 * Standards bodies are filled with fools and idiots.
677 abstime_to_reltime(clock_id, rqtp, &reltime);
678 if (reltime.tv_sec != 0 || reltime.tv_nsec != 0)
679 goto restart;
681 return (error);
684 unsigned int
685 sleep(unsigned int sec)
687 unsigned int rem = 0;
688 timespec_t ts;
689 timespec_t tsr;
691 ts.tv_sec = (time_t)sec;
692 ts.tv_nsec = 0;
693 if (nanosleep(&ts, &tsr) == -1 && errno == EINTR) {
694 rem = (unsigned int)tsr.tv_sec;
695 if (tsr.tv_nsec >= NANOSEC / 2)
696 rem++;
698 return (rem);
702 usleep(useconds_t usec)
704 timespec_t ts;
706 ts.tv_sec = usec / MICROSEC;
707 ts.tv_nsec = (long)(usec % MICROSEC) * 1000;
708 (void) nanosleep(&ts, NULL);
709 return (0);
713 close(int fildes)
715 extern void _aio_close(int);
716 extern int __close(int);
717 int rv;
720 * If we call _aio_close() while in a critical region,
721 * we will draw an ASSERT() failure, so don't do it.
722 * No calls to close() from within libc need _aio_close();
723 * only the application's calls to close() need this,
724 * and such calls are never from a libc critical region.
726 if (curthread->ul_critical == 0)
727 _aio_close(fildes);
728 PERFORM(__close(fildes))
732 door_call(int d, door_arg_t *params)
734 extern int __door_call(int, door_arg_t *);
735 int rv;
737 PERFORM(__door_call(d, params))
741 fcntl(int fildes, int cmd, ...)
743 extern int __fcntl(int, int, ...);
744 intptr_t arg;
745 int rv;
746 va_list ap;
748 va_start(ap, cmd);
749 arg = va_arg(ap, intptr_t);
750 va_end(ap);
751 if (cmd != F_SETLKW)
752 return (__fcntl(fildes, cmd, arg));
753 PERFORM(__fcntl(fildes, cmd, arg))
757 fdatasync(int fildes)
759 extern int __fdsync(int, int);
760 int rv;
762 PERFORM(__fdsync(fildes, FDSYNC))
766 fsync(int fildes)
768 extern int __fdsync(int, int);
769 int rv;
771 PERFORM(__fdsync(fildes, FSYNC))
775 lockf(int fildes, int function, off_t size)
777 extern int __lockf(int, int, off_t);
778 int rv;
780 PERFORM(__lockf(fildes, function, size))
783 #if !defined(_LP64)
785 lockf64(int fildes, int function, off64_t size)
787 extern int __lockf64(int, int, off64_t);
788 int rv;
790 PERFORM(__lockf64(fildes, function, size))
792 #endif /* !_LP64 */
794 ssize_t
795 msgrcv(int msqid, void *msgp, size_t msgsz, long msgtyp, int msgflg)
797 extern ssize_t __msgrcv(int, void *, size_t, long, int);
798 ssize_t rv;
800 PERFORM(__msgrcv(msqid, msgp, msgsz, msgtyp, msgflg))
804 msgsnd(int msqid, const void *msgp, size_t msgsz, int msgflg)
806 extern int __msgsnd(int, const void *, size_t, int);
807 int rv;
809 PERFORM(__msgsnd(msqid, msgp, msgsz, msgflg))
813 msync(caddr_t addr, size_t len, int flags)
815 extern int __msync(caddr_t, size_t, int);
816 int rv;
818 PERFORM(__msync(addr, len, flags))
822 openat(int fd, const char *path, int oflag, ...)
824 mode_t mode;
825 int rv;
826 va_list ap;
828 va_start(ap, oflag);
829 mode = va_arg(ap, mode_t);
830 va_end(ap);
831 PERFORM(__openat(fd, path, oflag, mode))
835 open(const char *path, int oflag, ...)
837 mode_t mode;
838 int rv;
839 va_list ap;
841 va_start(ap, oflag);
842 mode = va_arg(ap, mode_t);
843 va_end(ap);
844 PERFORM(__open(path, oflag, mode))
848 creat(const char *path, mode_t mode)
850 return (open(path, O_WRONLY | O_CREAT | O_TRUNC, mode));
853 #if !defined(_LP64)
855 openat64(int fd, const char *path, int oflag, ...)
857 mode_t mode;
858 int rv;
859 va_list ap;
861 va_start(ap, oflag);
862 mode = va_arg(ap, mode_t);
863 va_end(ap);
864 PERFORM(__openat64(fd, path, oflag, mode))
868 open64(const char *path, int oflag, ...)
870 mode_t mode;
871 int rv;
872 va_list ap;
874 va_start(ap, oflag);
875 mode = va_arg(ap, mode_t);
876 va_end(ap);
877 PERFORM(__open64(path, oflag, mode))
881 creat64(const char *path, mode_t mode)
883 return (open64(path, O_WRONLY | O_CREAT | O_TRUNC, mode));
885 #endif /* !_LP64 */
888 pause(void)
890 extern int __pause(void);
891 int rv;
893 PERFORM(__pause())
896 ssize_t
897 pread(int fildes, void *buf, size_t nbyte, off_t offset)
899 extern ssize_t __pread(int, void *, size_t, off_t);
900 ssize_t rv;
902 PERFORM(__pread(fildes, buf, nbyte, offset))
905 #if !defined(_LP64)
906 ssize_t
907 pread64(int fildes, void *buf, size_t nbyte, off64_t offset)
909 extern ssize_t __pread64(int, void *, size_t, off64_t);
910 ssize_t rv;
912 PERFORM(__pread64(fildes, buf, nbyte, offset))
915 ssize_t
916 preadv64(int fildes, const struct iovec *iov, int iovcnt, off64_t offset)
919 extern ssize_t __preadv64(int, const struct iovec *, int, off_t, off_t);
920 ssize_t rv;
922 PERFORM(__preadv64(fildes, iov, iovcnt, offset & 0xffffffffULL,
923 offset>>32))
925 #endif /* !_LP64 */
927 ssize_t
928 preadv(int fildes, const struct iovec *iov, int iovcnt, off_t offset)
931 extern ssize_t __preadv(int, const struct iovec *, int, off_t, off_t);
932 ssize_t rv;
934 PERFORM(__preadv(fildes, iov, iovcnt, offset, 0))
936 ssize_t
937 pwrite(int fildes, const void *buf, size_t nbyte, off_t offset)
939 extern ssize_t __pwrite(int, const void *, size_t, off_t);
940 ssize_t rv;
942 PERFORM(__pwrite(fildes, buf, nbyte, offset))
945 #if !defined(_LP64)
946 ssize_t
947 pwrite64(int fildes, const void *buf, size_t nbyte, off64_t offset)
949 extern ssize_t __pwrite64(int, const void *, size_t, off64_t);
950 ssize_t rv;
952 PERFORM(__pwrite64(fildes, buf, nbyte, offset))
955 ssize_t
956 pwritev64(int fildes, const struct iovec *iov, int iovcnt, off64_t offset)
959 extern ssize_t __pwritev64(int,
960 const struct iovec *, int, off_t, off_t);
961 ssize_t rv;
963 PERFORM(__pwritev64(fildes, iov, iovcnt, offset &
964 0xffffffffULL, offset>>32))
967 #endif /* !_LP64 */
969 ssize_t
970 pwritev(int fildes, const struct iovec *iov, int iovcnt, off_t offset)
972 extern ssize_t __pwritev(int, const struct iovec *, int, off_t, off_t);
973 ssize_t rv;
975 PERFORM(__pwritev(fildes, iov, iovcnt, offset, 0))
978 ssize_t
979 readv(int fildes, const struct iovec *iov, int iovcnt)
981 extern ssize_t __readv(int, const struct iovec *, int);
982 ssize_t rv;
984 PERFORM(__readv(fildes, iov, iovcnt))
988 sigpause(int sig)
990 extern int __sigpause(int);
991 int rv;
993 PERFORM(__sigpause(sig))
997 sigsuspend(const sigset_t *set)
999 extern int __sigsuspend(const sigset_t *);
1000 int rv;
1002 PROLOGUE_MASK(set)
1003 rv = __sigsuspend(set);
1004 EPILOGUE_MASK
1005 return (rv);
1009 _pollsys(struct pollfd *fds, nfds_t nfd, const timespec_t *timeout,
1010 const sigset_t *sigmask)
1012 extern int __pollsys(struct pollfd *, nfds_t, const timespec_t *,
1013 const sigset_t *);
1014 int rv;
1016 PROLOGUE_MASK(sigmask)
1017 rv = __pollsys(fds, nfd, timeout, sigmask);
1018 EPILOGUE_MASK
1019 return (rv);
1023 sigtimedwait(const sigset_t *set, siginfo_t *infop, const timespec_t *timeout)
1025 extern int __sigtimedwait(const sigset_t *, siginfo_t *,
1026 const timespec_t *);
1027 siginfo_t info;
1028 int sig;
1030 PROLOGUE
1031 if (abort) {
1032 *self->ul_errnop = EINTR;
1033 sig = -1;
1034 } else {
1035 sig = __sigtimedwait(set, &info, timeout);
1036 if (sig == SIGCANCEL &&
1037 (SI_FROMKERNEL(&info) || info.si_code == SI_LWP)) {
1038 do_sigcancel();
1039 *self->ul_errnop = EINTR;
1040 sig = -1;
1043 EPILOGUE
1044 if (sig != -1 && infop)
1045 (void) memcpy(infop, &info, sizeof (*infop));
1046 return (sig);
1050 sigwait(sigset_t *set)
1052 return (sigtimedwait(set, NULL, NULL));
1056 sigwaitinfo(const sigset_t *set, siginfo_t *info)
1058 return (sigtimedwait(set, info, NULL));
1062 sigqueue(pid_t pid, int signo, const union sigval value)
1064 extern int __sigqueue(pid_t pid, int signo,
1065 /* const union sigval */ void *value, int si_code, int block);
1066 return (__sigqueue(pid, signo, value.sival_ptr, SI_QUEUE, 0));
1070 _so_accept(int sock, struct sockaddr *addr, uint_t *addrlen, int version,
1071 int flags)
1073 extern int __so_accept(int, struct sockaddr *, uint_t *, int, int);
1074 int rv;
1076 PERFORM(__so_accept(sock, addr, addrlen, version, flags))
1080 _so_connect(int sock, struct sockaddr *addr, uint_t addrlen, int version)
1082 extern int __so_connect(int, struct sockaddr *, uint_t, int);
1083 int rv;
1085 PERFORM(__so_connect(sock, addr, addrlen, version))
1089 _so_recv(int sock, void *buf, size_t len, int flags)
1091 extern int __so_recv(int, void *, size_t, int);
1092 int rv;
1094 PERFORM(__so_recv(sock, buf, len, flags))
1098 _so_recvfrom(int sock, void *buf, size_t len, int flags,
1099 struct sockaddr *addr, int *addrlen)
1101 extern int __so_recvfrom(int, void *, size_t, int,
1102 struct sockaddr *, int *);
1103 int rv;
1105 PERFORM(__so_recvfrom(sock, buf, len, flags, addr, addrlen))
1109 _so_recvmsg(int sock, struct msghdr *msg, int flags)
1111 extern int __so_recvmsg(int, struct msghdr *, int);
1112 int rv;
1114 PERFORM(__so_recvmsg(sock, msg, flags))
1118 _so_send(int sock, const void *buf, size_t len, int flags)
1120 extern int __so_send(int, const void *, size_t, int);
1121 int rv;
1123 PERFORM(__so_send(sock, buf, len, flags))
1127 _so_sendmsg(int sock, const struct msghdr *msg, int flags)
1129 extern int __so_sendmsg(int, const struct msghdr *, int);
1130 int rv;
1132 PERFORM(__so_sendmsg(sock, msg, flags))
1136 _so_sendto(int sock, const void *buf, size_t len, int flags,
1137 const struct sockaddr *addr, int *addrlen)
1139 extern int __so_sendto(int, const void *, size_t, int,
1140 const struct sockaddr *, int *);
1141 int rv;
1143 PERFORM(__so_sendto(sock, buf, len, flags, addr, addrlen))
1147 tcdrain(int fildes)
1149 extern int __tcdrain(int);
1150 int rv;
1152 PERFORM(__tcdrain(fildes))
1156 waitid(idtype_t idtype, id_t id, siginfo_t *infop, int options)
1158 extern int __waitid(idtype_t, id_t, siginfo_t *, int);
1159 int rv;
1161 if (options & WNOHANG)
1162 return (__waitid(idtype, id, infop, options));
1163 PERFORM(__waitid(idtype, id, infop, options))
1166 ssize_t
1167 writev(int fildes, const struct iovec *iov, int iovcnt)
1169 extern ssize_t __writev(int, const struct iovec *, int);
1170 ssize_t rv;
1172 PERFORM(__writev(fildes, iov, iovcnt))