1 /* $NetBSD: svr4_32_misc.c,v 1.66 2009/11/04 21:23:03 rmind Exp $ */
4 * Copyright (c) 1994, 2008 The NetBSD Foundation, Inc.
7 * This code is derived from software contributed to The NetBSD Foundation
10 * Redistribution and use in source and binary forms, with or without
11 * modification, are permitted provided that the following conditions
13 * 1. Redistributions of source code must retain the above copyright
14 * notice, this list of conditions and the following disclaimer.
15 * 2. Redistributions in binary form must reproduce the above copyright
16 * notice, this list of conditions and the following disclaimer in the
17 * documentation and/or other materials provided with the distribution.
19 * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
20 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
21 * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
22 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
23 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
24 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
25 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
26 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
27 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
28 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29 * POSSIBILITY OF SUCH DAMAGE.
33 * SVR4 compatibility module.
35 * SVR4 system calls that are implemented differently in BSD are
39 #include <sys/cdefs.h>
40 __KERNEL_RCSID(0, "$NetBSD: svr4_32_misc.c,v 1.66 2009/11/04 21:23:03 rmind Exp $");
42 #include <sys/param.h>
43 #include <sys/systm.h>
44 #include <sys/namei.h>
45 #include <sys/dirent.h>
50 #include <sys/filedesc.h>
51 #include <sys/ioctl.h>
52 #include <sys/kernel.h>
53 #include <sys/malloc.h>
57 #include <sys/mount.h>
58 #include <sys/resource.h>
59 #include <sys/resourcevar.h>
60 #include <sys/socket.h>
61 #include <sys/vnode.h>
64 #include <sys/utsname.h>
65 #include <sys/unistd.h>
66 #include <sys/vfs_syscalls.h>
67 #include <sys/times.h>
70 #include <sys/ptrace.h>
71 #include <sys/signalvar.h>
73 #include <netinet/in.h>
74 #include <sys/syscallargs.h>
76 #include <miscfs/specfs/specdev.h>
78 #include <compat/svr4_32/svr4_32_types.h>
79 #include <compat/netbsd32/netbsd32_syscallargs.h>
80 #include <compat/svr4_32/svr4_32_signal.h>
81 #include <compat/svr4_32/svr4_32_lwp.h>
82 #include <compat/svr4_32/svr4_32_ucontext.h>
83 #include <compat/svr4_32/svr4_32_syscallargs.h>
84 #include <compat/svr4_32/svr4_32_util.h>
85 #include <compat/svr4_32/svr4_32_time.h>
86 #include <compat/svr4_32/svr4_32_dirent.h>
87 #include <compat/svr4/svr4_ulimit.h>
88 #include <compat/svr4_32/svr4_32_hrt.h>
89 #include <compat/svr4/svr4_wait.h>
90 #include <compat/svr4_32/svr4_32_statvfs.h>
91 #include <compat/svr4/svr4_sysconfig.h>
92 #include <compat/svr4_32/svr4_32_acl.h>
93 #include <compat/svr4/svr4_mman.h>
97 #include <uvm/uvm_extern.h>
99 static int svr4_to_bsd_mmap_flags(int);
101 static inline clock_t timeval_to_clock_t(struct timeval
*);
102 static int svr4_32_setinfo(int, struct rusage
*, int, svr4_32_siginfo_tp
);
104 #define svr4_32_pfind(pid) p_find((pid), PFIND_UNLOCK | PFIND_ZOMBIE)
106 static int svr4_32_mknod(struct lwp
*, register_t
*, const char *,
107 svr4_32_mode_t
, svr4_32_dev_t
);
110 svr4_32_sys_wait(struct lwp
*l
, const struct svr4_32_sys_wait_args
*uap
,
113 int error
, st
, sig
, pid
= WAIT_ANY
;
115 error
= do_sys_wait(&pid
, &st
, 0, NULL
);
121 if (WIFSIGNALED(st
)) {
123 if (sig
>= 0 && sig
< NSIG
)
124 st
= (st
& ~0177) | native_to_svr4_signo
[sig
];
125 } else if (WIFSTOPPED(st
)) {
127 if (sig
>= 0 && sig
< NSIG
)
128 st
= (st
& ~0xff00) | (native_to_svr4_signo
[sig
] << 8);
132 * It looks like wait(2) on svr4/solaris/2.4 returns
133 * the status in retval[1], and the pid on retval[0].
137 if (SCARG_P32(uap
, status
))
138 error
= copyout(&st
, SCARG_P32(uap
, status
), sizeof(st
));
144 svr4_32_sys_execv(struct lwp
*l
, const struct svr4_32_sys_execv_args
*uap
, register_t
*retval
)
147 syscallarg(char *) path;
148 syscallarg(char **) argv;
150 struct netbsd32_execve_args ap
;
152 SCARG(&ap
, path
) = SCARG(uap
, path
);
153 SCARG(&ap
, argp
) = SCARG(uap
, argp
);
154 NETBSD32PTR32(SCARG(&ap
, envp
), 0);
156 return netbsd32_execve(l
, &ap
, retval
);
161 svr4_32_sys_execve(struct proc
*p
, void *v
, register_t
*retval
)
163 struct svr4_32_sys_execve_args
/* {
164 syscallarg(const char *) path;
165 syscallarg(char **) argv;
166 syscallarg(char **) envp;
168 struct sys_execve_args ap
;
170 SCARG(&ap
, path
) = SCARG_P32(uap
, path
);
171 SCARG(&ap
, argp
) = SCARG_P32(uap
, argp
);
172 SCARG(&ap
, envp
) = SCARG_P32(uap
, envp
);
174 return netbsd32_execve(p
, &ap
, retval
);
179 svr4_32_sys_time(struct lwp
*l
, const struct svr4_32_sys_time_args
*uap
, register_t
*retval
)
183 struct netbsd32_timeval ntv
;
186 ntv
.tv_sec
= tv
.tv_sec
;
187 ntv
.tv_usec
= tv
.tv_usec
;
188 if (SCARG_P32(uap
, t
))
189 error
= copyout(&ntv
.tv_sec
, SCARG_P32(uap
, t
),
191 *retval
= (int) ntv
.tv_sec
;
198 * Read SVR4-style directory entries. We suck them into kernel space so
199 * that they can be massaged before being copied out to user code. Like
200 * SunOS, we squish out `empty' entries.
202 * This is quite ugly, but what do you expect from compatibility code?
205 svr4_32_sys_getdents64(struct lwp
*l
, const struct svr4_32_sys_getdents64_args
*uap
, register_t
*retval
)
209 char *inp
, *sbuf
; /* BSD-format */
210 int len
, reclen
; /* BSD-format */
211 char *outp
; /* SVR4-format */
212 int resid
, svr4_32_reclen
; /* SVR4-format */
216 struct svr4_32_dirent64 idb
;
217 off_t off
; /* true file offset */
218 int buflen
, error
, eofflag
;
219 off_t
*cookiebuf
= NULL
, *cookie
;
222 /* fd_getvnode() will use the descriptor for us */
223 if ((error
= fd_getvnode(SCARG(uap
, fd
), &fp
)) != 0)
226 if ((fp
->f_flag
& FREAD
) == 0) {
232 if (vp
->v_type
!= VDIR
) {
237 buflen
= min(MAXBSIZE
, SCARG(uap
, nbytes
));
238 sbuf
= malloc(buflen
, M_TEMP
, M_WAITOK
);
239 vn_lock(vp
, LK_EXCLUSIVE
| LK_RETRY
);
242 aiov
.iov_base
= sbuf
;
243 aiov
.iov_len
= buflen
;
244 auio
.uio_iov
= &aiov
;
246 auio
.uio_rw
= UIO_READ
;
247 auio
.uio_resid
= buflen
;
248 auio
.uio_offset
= off
;
249 UIO_SETUP_SYSSPACE(&auio
);
251 * First we read into the malloc'ed buffer, then
252 * we massage it into user space, one record at a time.
254 error
= VOP_READDIR(vp
, &auio
, fp
->f_cred
, &eofflag
, &cookiebuf
,
260 outp
= SCARG_P32(uap
, dp
);
261 resid
= SCARG(uap
, nbytes
);
262 if ((len
= buflen
- auio
.uio_resid
) == 0)
265 for (cookie
= cookiebuf
; len
> 0; len
-= reclen
) {
266 bdp
= (struct dirent
*)inp
;
267 reclen
= bdp
->d_reclen
;
269 panic("svr4_32_getdents64: bad reclen");
270 if (bdp
->d_fileno
== 0) {
271 inp
+= reclen
; /* it is a hole; squish it out */
278 svr4_32_reclen
= SVR4_RECLEN(&idb
, bdp
->d_namlen
);
279 if (reclen
> len
|| resid
< svr4_32_reclen
) {
280 /* entry too big for buffer, so just stop */
285 off
= *cookie
++; /* each entry points to the next */
289 * Massage in place to make a SVR4-shaped dirent (otherwise
290 * we have to worry about touching user memory outside of
291 * the copyout() call).
293 idb
.d_ino
= (svr4_32_ino64_t
)bdp
->d_fileno
;
294 idb
.d_off
= (svr4_32_off64_t
)off
;
295 idb
.d_reclen
= (u_short
)svr4_32_reclen
;
296 strlcpy(idb
.d_name
, bdp
->d_name
, sizeof(idb
.d_name
));
297 if ((error
= copyout((void *)&idb
, outp
, svr4_32_reclen
)))
299 /* advance past this real entry */
301 /* advance output past SVR4-shaped entry */
302 outp
+= svr4_32_reclen
;
303 resid
-= svr4_32_reclen
;
306 /* if we squished out the whole block, try again */
307 if (outp
== SCARG_P32(uap
, dp
))
309 fp
->f_offset
= off
; /* update the vnode offset */
312 *retval
= SCARG(uap
, nbytes
) - resid
;
316 free(cookiebuf
, M_TEMP
);
319 fd_putfile(SCARG(uap
, fd
));
325 svr4_32_sys_getdents(struct lwp
*l
, const struct svr4_32_sys_getdents_args
*uap
, register_t
*retval
)
329 char *inp
, *sbuf
; /* BSD-format */
330 int len
, reclen
; /* BSD-format */
331 char *outp
; /* SVR4-format */
332 int resid
, svr4_reclen
; /* SVR4-format */
336 struct svr4_32_dirent idb
;
337 off_t off
; /* true file offset */
338 int buflen
, error
, eofflag
;
339 off_t
*cookiebuf
= NULL
, *cookie
;
342 /* fd_getvnode() will use the descriptor for us */
343 if ((error
= fd_getvnode(SCARG(uap
, fd
), &fp
)) != 0)
346 if ((fp
->f_flag
& FREAD
) == 0) {
352 if (vp
->v_type
!= VDIR
) {
357 buflen
= min(MAXBSIZE
, SCARG(uap
, nbytes
));
358 sbuf
= malloc(buflen
, M_TEMP
, M_WAITOK
);
359 vn_lock(vp
, LK_EXCLUSIVE
| LK_RETRY
);
362 aiov
.iov_base
= sbuf
;
363 aiov
.iov_len
= buflen
;
364 auio
.uio_iov
= &aiov
;
366 auio
.uio_rw
= UIO_READ
;
367 auio
.uio_resid
= buflen
;
368 auio
.uio_offset
= off
;
369 UIO_SETUP_SYSSPACE(&auio
);
371 * First we read into the malloc'ed buffer, then
372 * we massage it into user space, one record at a time.
374 error
= VOP_READDIR(vp
, &auio
, fp
->f_cred
, &eofflag
, &cookiebuf
,
380 outp
= SCARG_P32(uap
, buf
);
381 resid
= SCARG(uap
, nbytes
);
382 if ((len
= buflen
- auio
.uio_resid
) == 0)
385 for (cookie
= cookiebuf
; len
> 0; len
-= reclen
) {
386 bdp
= (struct dirent
*)inp
;
387 reclen
= bdp
->d_reclen
;
389 panic("svr4_32_getdents: bad reclen");
391 off
= *cookie
++; /* each entry points to the next */
394 if ((off
>> 32) != 0) {
395 compat_offseterr(vp
, "svr4_32_getdents");
399 if (bdp
->d_fileno
== 0) {
400 inp
+= reclen
; /* it is a hole; squish it out */
403 svr4_reclen
= SVR4_RECLEN(&idb
, bdp
->d_namlen
);
404 if (reclen
> len
|| resid
< svr4_reclen
) {
405 /* entry too big for buffer, so just stop */
410 * Massage in place to make a SVR4-shaped dirent (otherwise
411 * we have to worry about touching user memory outside of
412 * the copyout() call).
414 idb
.d_ino
= (svr4_32_ino_t
)bdp
->d_fileno
;
415 idb
.d_off
= (svr4_32_off_t
)off
;
416 idb
.d_reclen
= (u_short
)svr4_reclen
;
417 strlcpy(idb
.d_name
, bdp
->d_name
, sizeof(idb
.d_name
));
418 if ((error
= copyout((void *)&idb
, outp
, svr4_reclen
)))
420 /* advance past this real entry */
422 /* advance output past SVR4-shaped entry */
424 resid
-= svr4_reclen
;
427 /* if we squished out the whole block, try again */
428 if (outp
== SCARG_P32(uap
, buf
))
430 fp
->f_offset
= off
; /* update the vnode offset */
433 *retval
= SCARG(uap
, nbytes
) - resid
;
437 free(cookiebuf
, M_TEMP
);
440 fd_putfile(SCARG(uap
, fd
));
446 svr4_to_bsd_mmap_flags(int f
)
448 int type
= f
& SVR4_MAP_TYPE
;
451 if (type
!= MAP_PRIVATE
&& type
!= MAP_SHARED
)
454 nf
= f
& SVR4_MAP_COPYFLAGS
;
455 if (f
& SVR4_MAP_ANON
)
463 svr4_32_sys_mmap(struct lwp
*l
, const struct svr4_32_sys_mmap_args
*uap
, register_t
*retval
)
465 struct sys_mmap_args mm
;
468 * Verify the arguments.
470 if (SCARG(uap
, prot
) & ~(PROT_READ
| PROT_WRITE
| PROT_EXEC
))
471 return EINVAL
; /* XXX still needed? */
473 if (SCARG(uap
, len
) == 0)
476 if ((SCARG(&mm
, flags
) = svr4_to_bsd_mmap_flags(SCARG(uap
, flags
))) == -1)
479 SCARG(&mm
, prot
) = SCARG(uap
, prot
);
480 SCARG(&mm
, len
) = SCARG(uap
, len
);
481 SCARG(&mm
, fd
) = SCARG(uap
, fd
);
482 SCARG(&mm
, addr
) = SCARG_P32(uap
, addr
);
483 SCARG(&mm
, pos
) = SCARG(uap
, pos
);
485 error
= sys_mmap(l
, &mm
, retval
);
486 if ((u_long
)*retval
> (u_long
)UINT_MAX
) {
487 printf("svr4_32_mmap: retval out of range: 0x%lx",
489 /* Should try to recover and return an error here. */
496 svr4_32_sys_mmap64(struct lwp
*l
, const struct svr4_32_sys_mmap64_args
*uap
, register_t
*retval
)
498 struct sys_mmap_args mm
;
501 * Verify the arguments.
503 if (SCARG(uap
, prot
) & ~(PROT_READ
| PROT_WRITE
| PROT_EXEC
))
504 return EINVAL
; /* XXX still needed? */
506 if (SCARG(uap
, len
) == 0)
509 if ((SCARG(&mm
, flags
) = svr4_to_bsd_mmap_flags(SCARG(uap
, flags
))) == -1)
512 SCARG(&mm
, prot
) = SCARG(uap
, prot
);
513 SCARG(&mm
, len
) = SCARG(uap
, len
);
514 SCARG(&mm
, fd
) = SCARG(uap
, fd
);
515 SCARG(&mm
, addr
) = SCARG_P32(uap
, addr
);
516 SCARG(&mm
, pos
) = SCARG(uap
, pos
);
518 error
= sys_mmap(l
, &mm
, retval
);
519 if ((u_long
)*retval
> (u_long
)UINT_MAX
) {
520 printf("svr4_32_mmap64: retval out of range: 0x%lx",
522 /* Should try to recover and return an error here. */
529 svr4_32_mknod(struct lwp
*l
, register_t
*retval
, const char *path
, svr4_32_mode_t mode
, svr4_32_dev_t dev
)
531 if (S_ISFIFO(mode
)) {
532 struct sys_mkfifo_args ap
;
533 SCARG(&ap
, path
) = path
;
534 SCARG(&ap
, mode
) = mode
;
535 return sys_mkfifo(l
, &ap
, retval
);
537 return do_sys_mknod(l
, path
, mode
, dev
, retval
, UIO_USERSPACE
);
543 svr4_32_sys_mknod(struct lwp
*l
, const struct svr4_32_sys_mknod_args
*uap
, register_t
*retval
)
545 return svr4_32_mknod(l
, retval
,
546 SCARG_P32(uap
, path
), SCARG(uap
, mode
),
547 svr4_32_to_bsd_odev_t(SCARG(uap
, dev
)));
552 svr4_32_sys_xmknod(struct lwp
*l
, const struct svr4_32_sys_xmknod_args
*uap
, register_t
*retval
)
554 return svr4_32_mknod(l
, retval
,
555 SCARG_P32(uap
, path
), SCARG(uap
, mode
),
556 svr4_32_to_bsd_dev_t(SCARG(uap
, dev
)));
561 svr4_32_sys_vhangup(struct lwp
*l
, const void *v
, register_t
*retval
)
568 svr4_32_sys_sysconfig(struct lwp
*l
, const struct svr4_32_sys_sysconfig_args
*uap
, register_t
*retval
)
570 extern u_int maxfiles
;
573 switch (SCARG(uap
, name
)) {
574 case SVR4_CONFIG_NGROUPS
:
575 *retval
= NGROUPS_MAX
;
577 case SVR4_CONFIG_CHILD_MAX
:
580 case SVR4_CONFIG_OPEN_FILES
:
583 case SVR4_CONFIG_POSIX_VER
:
586 case SVR4_CONFIG_PAGESIZE
:
589 case SVR4_CONFIG_CLK_TCK
:
590 *retval
= 60; /* should this be `hz', ie. 100? */
592 case SVR4_CONFIG_XOPEN_VER
:
593 *retval
= 2; /* XXX: What should that be? */
595 case SVR4_CONFIG_PROF_TCK
:
596 *retval
= 60; /* XXX: What should that be? */
598 case SVR4_CONFIG_NPROC_CONF
:
599 *retval
= 1; /* Only one processor for now */
601 case SVR4_CONFIG_NPROC_ONLN
:
602 *retval
= 1; /* And it better be online */
604 case SVR4_CONFIG_AIO_LISTIO_MAX
:
605 case SVR4_CONFIG_AIO_MAX
:
606 case SVR4_CONFIG_AIO_PRIO_DELTA_MAX
:
607 *retval
= 0; /* No aio support */
609 case SVR4_CONFIG_DELAYTIMER_MAX
:
610 *retval
= 0; /* No delaytimer support */
612 case SVR4_CONFIG_MQ_OPEN_MAX
:
614 *retval
= msginfo
.msgmni
;
619 case SVR4_CONFIG_MQ_PRIO_MAX
:
620 *retval
= 0; /* XXX: Don't know */
622 case SVR4_CONFIG_RTSIG_MAX
:
625 case SVR4_CONFIG_SEM_NSEMS_MAX
:
627 *retval
= seminfo
.semmni
;
632 case SVR4_CONFIG_SEM_VALUE_MAX
:
634 *retval
= seminfo
.semvmx
;
639 case SVR4_CONFIG_SIGQUEUE_MAX
:
640 *retval
= 0; /* XXX: Don't know */
642 case SVR4_CONFIG_SIGRT_MIN
:
643 case SVR4_CONFIG_SIGRT_MAX
:
644 *retval
= 0; /* No real time signals */
646 case SVR4_CONFIG_TIMER_MAX
:
647 *retval
= 3; /* XXX: real, virtual, profiling */
649 case SVR4_CONFIG_PHYS_PAGES
:
650 *retval
= uvmexp
.free
; /* XXX: free instead of total */
652 case SVR4_CONFIG_AVPHYS_PAGES
:
653 uvm_estimatepageable(&active
, NULL
);
654 *retval
= active
; /* XXX: active instead of avg */
656 case SVR4_CONFIG_COHERENCY
:
657 *retval
= 0; /* XXX */
659 case SVR4_CONFIG_SPLIT_CACHE
:
660 *retval
= 0; /* XXX */
662 case SVR4_CONFIG_ICACHESZ
:
663 *retval
= 256; /* XXX */
665 case SVR4_CONFIG_DCACHESZ
:
666 *retval
= 256; /* XXX */
668 case SVR4_CONFIG_ICACHELINESZ
:
669 *retval
= 64; /* XXX */
671 case SVR4_CONFIG_DCACHELINESZ
:
672 *retval
= 64; /* XXX */
674 case SVR4_CONFIG_ICACHEBLKSZ
:
675 *retval
= 64; /* XXX */
677 case SVR4_CONFIG_DCACHEBLKSZ
:
678 *retval
= 64; /* XXX */
680 case SVR4_CONFIG_DCACHETBLKSZ
:
681 *retval
= 64; /* XXX */
683 case SVR4_CONFIG_ICACHE_ASSOC
:
684 *retval
= 1; /* XXX */
686 case SVR4_CONFIG_DCACHE_ASSOC
:
687 *retval
= 1; /* XXX */
689 case SVR4_CONFIG_MAXPID
:
692 case SVR4_CONFIG_STACK_PROT
:
693 *retval
= PROT_READ
|PROT_WRITE
|PROT_EXEC
;
704 svr4_32_sys_break(struct lwp
*l
, const struct svr4_32_sys_break_args
*uap
, register_t
*retval
)
706 struct proc
*p
= l
->l_proc
;
707 struct vmspace
*vm
= p
->p_vmspace
;
711 old
= (vaddr_t
) vm
->vm_daddr
;
712 new = round_page((vaddr_t
)SCARG_P32(uap
, nsize
));
714 if (new - old
> p
->p_rlimit
[RLIMIT_DATA
].rlim_cur
&& new > old
)
717 old
= round_page(old
+ ctob(vm
->vm_dsize
));
718 DPRINTF(("break(2): dsize = %x ctob %x\n",
719 vm
->vm_dsize
, ctob(vm
->vm_dsize
)));
722 error
= uvm_map(&vm
->vm_map
, &old
, new - old
, NULL
,
723 UVM_UNKNOWN_OFFSET
, 0,
724 UVM_MAPFLAG(UVM_PROT_ALL
, UVM_PROT_ALL
, UVM_INH_COPY
,
726 UVM_FLAG_AMAPPAD
|UVM_FLAG_FIXED
|
727 UVM_FLAG_OVERLAY
|UVM_FLAG_COPYONW
));
729 uprintf("sbrk: grow failed, return = %d\n", error
);
732 vm
->vm_dsize
+= btoc(new - old
);
733 } else if (new < old
) {
734 uvm_deallocate(&vm
->vm_map
, new, old
- new);
735 vm
->vm_dsize
-= btoc(old
- new);
741 static inline clock_t
742 timeval_to_clock_t(struct timeval
*tv
)
744 return tv
->tv_sec
* hz
+ tv
->tv_usec
/ (1000000 / hz
);
748 svr4_32_sys_times(struct lwp
*l
, const struct svr4_32_sys_times_args
*uap
, register_t
*retval
)
752 struct rusage ru
, *rup
;
753 struct proc
*p
= l
->l_proc
;
755 ru
= l
->l_proc
->p_stats
->p_ru
;
756 mutex_enter(p
->p_lock
);
757 calcru(p
, &ru
.ru_utime
, &ru
.ru_stime
, NULL
, NULL
);
759 mutex_exit(p
->p_lock
);
761 tms
.tms_utime
= timeval_to_clock_t(&ru
.ru_utime
);
762 tms
.tms_stime
= timeval_to_clock_t(&ru
.ru_stime
);
764 rup
= &l
->l_proc
->p_stats
->p_cru
;
765 tms
.tms_cutime
= timeval_to_clock_t(&rup
->ru_utime
);
766 tms
.tms_cstime
= timeval_to_clock_t(&rup
->ru_stime
);
769 *retval
= timeval_to_clock_t(&t
);
771 return copyout(&tms
, SCARG_P32(uap
, tp
), sizeof(tms
));
776 svr4_32_sys_ulimit(struct lwp
*l
, const struct svr4_32_sys_ulimit_args
*uap
, register_t
*retval
)
778 struct proc
*p
= l
->l_proc
;
783 switch (SCARG(uap
, cmd
)) {
785 r
= p
->p_rlimit
[RLIMIT_FSIZE
].rlim_cur
/ 512;
789 krl
.rlim_cur
= SCARG(uap
, newlimit
) * 512;
790 krl
.rlim_max
= p
->p_rlimit
[RLIMIT_FSIZE
].rlim_max
;
792 error
= dosetrlimit(l
, l
->l_proc
, RLIMIT_FSIZE
, &krl
);
796 r
= p
->p_rlimit
[RLIMIT_FSIZE
].rlim_cur
;
800 r
= p
->p_rlimit
[RLIMIT_DATA
].rlim_cur
;
803 r
+= (long)p
->p_vmspace
->vm_daddr
;
807 r
= p
->p_rlimit
[RLIMIT_NOFILE
].rlim_cur
;
814 *retval
= r
> 0x7fffffff ? 0x7fffffff : r
;
820 svr4_32_sys_pgrpsys(struct lwp
*l
, const struct svr4_32_sys_pgrpsys_args
*uap
, register_t
*retval
)
822 struct proc
*p
= l
->l_proc
;
824 switch (SCARG(uap
, cmd
)) {
825 case 1: /* setpgrp() */
827 * SVR4 setpgrp() (which takes no arguments) has the
828 * semantics that the session ID is also created anew, so
829 * in almost every sense, setpgrp() is identical to
830 * setsid() for SVR4. (Under BSD, the difference is that
831 * a setpgid(0,0) will not create a new session.)
833 sys_setsid(l
, NULL
, retval
);
836 case 0: /* getpgrp() */
837 *retval
= p
->p_pgrp
->pg_id
;
840 case 2: /* getsid(pid) */
841 if (SCARG(uap
, pid
) != 0 &&
842 (p
= svr4_32_pfind(SCARG(uap
, pid
))) == NULL
)
845 * This has already been initialized to the pid of
846 * the session leader.
848 *retval
= (register_t
) p
->p_session
->s_sid
;
851 case 3: /* setsid() */
852 return sys_setsid(l
, NULL
, retval
);
854 case 4: /* getpgid(pid) */
856 if (SCARG(uap
, pid
) != 0 &&
857 (p
= svr4_32_pfind(SCARG(uap
, pid
))) == NULL
)
860 *retval
= (int) p
->p_pgrp
->pg_id
;
863 case 5: /* setpgid(pid, pgid); */
865 struct sys_setpgid_args sa
;
867 SCARG(&sa
, pid
) = SCARG(uap
, pid
);
868 SCARG(&sa
, pgid
) = SCARG(uap
, pgid
);
869 return sys_setpgid(l
, &sa
, retval
);
877 struct svr4_32_hrtcntl_args
{
881 syscallarg(svr4_32_hrt_interval_tp
) iv
;
882 syscallarg(svr4_32_hrt_time_tp
) ti
;
887 svr4_32_hrtcntl(struct proc
*p
, const struct svr4_32_hrtcntl_args
*uap
, register_t
*retval
)
889 switch (SCARG(uap
, fun
)) {
890 case SVR4_HRT_CNTL_RES
:
891 DPRINTF(("htrcntl(RES)\n"));
892 *retval
= SVR4_HRT_USEC
;
895 case SVR4_HRT_CNTL_TOFD
:
896 DPRINTF(("htrcntl(TOFD)\n"));
900 if (SCARG(uap
, clk
) != SVR4_HRT_CLK_STD
) {
901 DPRINTF(("clk == %d\n", SCARG(uap
, clk
)));
904 if (SCARG_P32(uap
, ti
) == 0) {
905 DPRINTF(("ti NULL\n"));
910 t
.h_rem
= tv
.tv_usec
;
911 t
.h_res
= SVR4_HRT_USEC
;
912 return copyout(&t
, SCARG_P32(uap
, ti
),
916 case SVR4_HRT_CNTL_START
:
917 DPRINTF(("htrcntl(START)\n"));
920 case SVR4_HRT_CNTL_GET
:
921 DPRINTF(("htrcntl(GET)\n"));
924 DPRINTF(("Bad htrcntl command %d\n", SCARG(uap
, fun
)));
931 svr4_32_sys_hrtsys(struct lwp
*l
, const struct svr4_32_sys_hrtsys_args
*uap
, register_t
*retval
)
934 switch (SCARG(uap
, cmd
)) {
936 return svr4_32_hrtcntl(l
->l_proc
, (const struct svr4_32_hrtcntl_args
*) uap
,
940 DPRINTF(("hrtalarm\n"));
944 DPRINTF(("hrtsleep\n"));
948 DPRINTF(("hrtcancel\n"));
952 DPRINTF(("Bad hrtsys command %d\n", SCARG(uap
, cmd
)));
959 svr4_32_setinfo(int pid
, struct rusage
*ru
, int st
, svr4_32_siginfo_tp si
)
961 svr4_32_siginfo_t
*s
= NETBSD32PTR64(si
);
965 memset(&i
, 0, sizeof(i
));
967 i
.si_signo
= SVR4_SIGCHLD
;
968 i
.si_errno
= 0; /* XXX? */
972 i
.si_stime
= ru
->ru_stime
.tv_sec
;
973 i
.si_utime
= ru
->ru_utime
.tv_sec
;
977 i
.si_status
= WEXITSTATUS(st
);
978 i
.si_code
= SVR4_CLD_EXITED
;
979 } else if (WIFSTOPPED(st
)) {
981 if (sig
>= 0 && sig
< NSIG
)
982 i
.si_status
= native_to_svr4_signo
[sig
];
984 if (i
.si_status
== SVR4_SIGCONT
)
985 i
.si_code
= SVR4_CLD_CONTINUED
;
987 i
.si_code
= SVR4_CLD_STOPPED
;
990 if (sig
>= 0 && sig
< NSIG
)
991 i
.si_status
= native_to_svr4_signo
[sig
];
994 i
.si_code
= SVR4_CLD_DUMPED
;
996 i
.si_code
= SVR4_CLD_KILLED
;
999 DPRINTF(("siginfo [pid %ld signo %d code %d errno %d status %d]\n",
1000 i
.si_pid
, i
.si_signo
, i
.si_code
, i
.si_errno
, i
.si_status
));
1002 return copyout(&i
, s
, sizeof(i
));
1007 svr4_32_sys_waitsys(struct lwp
*l
, const struct svr4_32_sys_waitsys_args
*uap
, register_t
*retval
)
1009 int options
, error
, status
;
1011 int id
= SCARG(uap
, id
);
1013 switch (SCARG(uap
, grp
)) {
1018 id
= -l
->l_proc
->p_pgid
;
1029 DPRINTF(("waitsys(%d, %d, %p, %x)\n",
1030 SCARG(uap
, grp
), id
,
1031 SCARG(uap
, info
), SCARG(uap
, options
)));
1033 /* Translate options */
1034 options
= WOPTSCHECKED
;
1035 if (SCARG(uap
, options
) & SVR4_WNOWAIT
)
1037 if (SCARG(uap
, options
) & SVR4_WNOHANG
)
1039 if ((SCARG(uap
, options
) & (SVR4_WEXITED
|SVR4_WTRAPPED
)) == 0)
1040 options
|= WNOZOMBIE
;
1041 if (SCARG(uap
, options
) & (SVR4_WSTOPPED
|SVR4_WCONTINUED
))
1042 options
|= WUNTRACED
;
1044 error
= do_sys_wait(&id
, &status
, options
, &ru
);
1050 return svr4_32_setinfo(id
, &ru
, status
, SCARG(uap
, info
));
1054 svr4_32_copyout_statvfs(const struct statvfs
*bfs
, struct svr4_32_statvfs
*sufs
)
1056 struct svr4_32_statvfs
*sfs
= malloc(sizeof(*sfs
), M_TEMP
, M_WAITOK
);
1059 sfs
->f_bsize
= bfs
->f_iosize
; /* XXX */
1060 sfs
->f_frsize
= bfs
->f_bsize
;
1061 sfs
->f_blocks
= bfs
->f_blocks
;
1062 sfs
->f_bfree
= bfs
->f_bfree
;
1063 sfs
->f_bavail
= bfs
->f_bavail
;
1064 sfs
->f_files
= bfs
->f_files
;
1065 sfs
->f_ffree
= bfs
->f_ffree
;
1066 sfs
->f_favail
= bfs
->f_ffree
;
1067 sfs
->f_fsid
= bfs
->f_fsidx
.__fsid_val
[0];
1068 memcpy(sfs
->f_basetype
, bfs
->f_fstypename
, sizeof(sfs
->f_basetype
));
1070 if (bfs
->f_flag
& MNT_RDONLY
)
1071 sfs
->f_flag
|= SVR4_ST_RDONLY
;
1072 if (bfs
->f_flag
& MNT_NOSUID
)
1073 sfs
->f_flag
|= SVR4_ST_NOSUID
;
1074 sfs
->f_namemax
= MAXNAMLEN
;
1075 memcpy(sfs
->f_fstr
, bfs
->f_fstypename
, sizeof(sfs
->f_fstr
)); /* XXX */
1076 memset(sfs
->f_filler
, 0, sizeof(sfs
->f_filler
));
1078 error
= copyout(sfs
, sufs
, sizeof(*sfs
));
1086 svr4_32_copyout_statvfs64(const struct statvfs
*bfs
, struct svr4_32_statvfs64
*sufs
)
1088 struct svr4_32_statvfs64
*sfs
= malloc(sizeof(*sfs
), M_TEMP
, M_WAITOK
);
1091 sfs
->f_bsize
= bfs
->f_iosize
; /* XXX */
1092 sfs
->f_frsize
= bfs
->f_bsize
;
1093 sfs
->f_blocks
= bfs
->f_blocks
;
1094 sfs
->f_bfree
= bfs
->f_bfree
;
1095 sfs
->f_bavail
= bfs
->f_bavail
;
1096 sfs
->f_files
= bfs
->f_files
;
1097 sfs
->f_ffree
= bfs
->f_ffree
;
1098 sfs
->f_favail
= bfs
->f_ffree
;
1099 sfs
->f_fsid
= bfs
->f_fsidx
.__fsid_val
[0];
1100 memcpy(sfs
->f_basetype
, bfs
->f_fstypename
, sizeof(sfs
->f_basetype
));
1102 if (bfs
->f_flag
& MNT_RDONLY
)
1103 sfs
->f_flag
|= SVR4_ST_RDONLY
;
1104 if (bfs
->f_flag
& MNT_NOSUID
)
1105 sfs
->f_flag
|= SVR4_ST_NOSUID
;
1106 sfs
->f_namemax
= MAXNAMLEN
;
1107 memcpy(sfs
->f_fstr
, bfs
->f_fstypename
, sizeof(sfs
->f_fstr
)); /* XXX */
1108 memset(sfs
->f_filler
, 0, sizeof(sfs
->f_filler
));
1110 error
= copyout(sfs
, sufs
, sizeof(*sfs
));
1118 svr4_32_sys_statvfs(struct lwp
*l
, const struct svr4_32_sys_statvfs_args
*uap
, register_t
*retval
)
1123 sb
= STATVFSBUF_GET();
1124 error
= do_sys_pstatvfs(l
, SCARG_P32(uap
, path
), ST_WAIT
, sb
);
1126 error
= svr4_32_copyout_statvfs(sb
, SCARG_P32(uap
, fs
));
1133 svr4_32_sys_fstatvfs(struct lwp
*l
, const struct svr4_32_sys_fstatvfs_args
*uap
, register_t
*retval
)
1138 sb
= STATVFSBUF_GET();
1139 error
= do_sys_fstatvfs(l
, SCARG(uap
, fd
), ST_WAIT
, sb
);
1141 error
= svr4_32_copyout_statvfs(sb
, SCARG_P32(uap
, fs
));
1148 svr4_32_sys_statvfs64(struct lwp
*l
, const struct svr4_32_sys_statvfs64_args
*uap
, register_t
*retval
)
1153 sb
= STATVFSBUF_GET();
1154 error
= do_sys_pstatvfs(l
, SCARG_P32(uap
, path
), ST_WAIT
, sb
);
1156 error
= svr4_32_copyout_statvfs64(sb
, SCARG_P32(uap
, fs
));
1163 svr4_32_sys_fstatvfs64(struct lwp
*l
, const struct svr4_32_sys_fstatvfs64_args
*uap
, register_t
*retval
)
1168 sb
= STATVFSBUF_GET();
1169 error
= do_sys_fstatvfs(l
, SCARG(uap
, fd
), ST_WAIT
, sb
);
1171 error
= svr4_32_copyout_statvfs64(sb
, SCARG_P32(uap
, fs
));
1179 svr4_32_sys_alarm(struct lwp
*l
, const struct svr4_32_sys_alarm_args
*uap
, register_t
*retval
)
1181 struct itimerval tp
;
1183 dogetitimer(l
->l_proc
, ITIMER_REAL
, &tp
);
1184 if (tp
.it_value
.tv_usec
)
1185 tp
.it_value
.tv_sec
++;
1186 *retval
= (register_t
)tp
.it_value
.tv_sec
;
1188 timerclear(&tp
.it_interval
);
1189 tp
.it_value
.tv_sec
= SCARG(uap
, sec
);
1190 tp
.it_value
.tv_usec
= 0;
1192 return dosetitimer(l
->l_proc
, ITIMER_REAL
, &tp
);
1197 svr4_32_sys_gettimeofday(struct lwp
*l
, const struct svr4_32_sys_gettimeofday_args
*uap
, register_t
*retval
)
1200 if (SCARG_P32(uap
, tp
)) {
1204 return copyout(&atv
, SCARG_P32(uap
, tp
), sizeof (atv
));
1212 svr4_32_sys_facl(struct lwp
*l
, const struct svr4_32_sys_facl_args
*uap
, register_t
*retval
)
1217 switch (SCARG(uap
, cmd
)) {
1218 case SVR4_SYS_SETACL
:
1219 /* We don't support acls on any filesystem */
1222 case SVR4_SYS_GETACL
:
1225 return copyout(retval, &SCARG(uap, num),
1226 sizeof(SCARG(uap, num)));
1229 case SVR4_SYS_GETACLCNT
:
1239 svr4_32_sys_acl(struct lwp
*l
, const struct svr4_32_sys_acl_args
*uap
, register_t
*retval
)
1241 return svr4_32_sys_facl(l
, (const void *)uap
, retval
); /* XXX: for now the same */
1246 svr4_32_sys_auditsys(struct lwp
*l
, const struct svr4_32_sys_auditsys_args
*uap
, register_t
*retval
)
1249 * XXX: Big brother is *not* watching.
1256 svr4_32_sys_memcntl(struct lwp
*l
, const struct svr4_32_sys_memcntl_args
*uap
, register_t
*retval
)
1258 switch (SCARG(uap
, cmd
)) {
1261 struct sys___msync13_args msa
;
1263 SCARG(&msa
, addr
) = SCARG_P32(uap
, addr
);
1264 SCARG(&msa
, len
) = SCARG(uap
, len
);
1265 SCARG(&msa
, flags
) = (uintptr_t)SCARG_P32(uap
, arg
);
1267 return sys___msync13(l
, &msa
, retval
);
1269 case SVR4_MC_ADVISE
:
1271 struct sys_madvise_args maa
;
1273 SCARG(&maa
, addr
) = SCARG_P32(uap
, addr
);
1274 SCARG(&maa
, len
) = SCARG(uap
, len
);
1275 SCARG(&maa
, behav
) = (uintptr_t)SCARG_P32(uap
, arg
);
1277 return sys_madvise(l
, &maa
, retval
);
1280 case SVR4_MC_UNLOCK
:
1281 case SVR4_MC_LOCKAS
:
1282 case SVR4_MC_UNLOCKAS
:
1291 svr4_32_sys_nice(struct lwp
*l
, const struct svr4_32_sys_nice_args
*uap
, register_t
*retval
)
1293 struct sys_setpriority_args ap
;
1296 SCARG(&ap
, which
) = PRIO_PROCESS
;
1297 SCARG(&ap
, who
) = 0;
1298 SCARG(&ap
, prio
) = SCARG(uap
, prio
);
1300 if ((error
= sys_setpriority(l
, &ap
, retval
)) != 0)
1303 if ((error
= sys_getpriority(l
, (const void *)&ap
, retval
)) != 0)
1311 svr4_32_sys_resolvepath(struct lwp
*l
, const struct svr4_32_sys_resolvepath_args
*uap
, register_t
*retval
)
1313 struct nameidata nd
;
1317 NDINIT(&nd
, LOOKUP
, NOFOLLOW
| SAVENAME
| TRYEMULROOT
, UIO_USERSPACE
,
1318 SCARG_P32(uap
, path
));
1320 if ((error
= namei(&nd
)) != 0)
1323 if ((error
= copyoutstr(nd
.ni_cnd
.cn_pnbuf
,
1324 SCARG_P32(uap
, buf
),
1325 SCARG(uap
, bufsiz
), &len
)) != 0)
1331 PNBUF_PUT(nd
.ni_cnd
.cn_pnbuf
);