.
[glibc-ports.git] / sysdeps / unix / sysv / linux / alpha / wait4.S
blob8d89e3d46ece8e9c1ac930883c31511ded68311c
1 /* Copyright (C) 1998, 2003 Free Software Foundation, Inc.
2    This file is part of the GNU C Library.
4    The GNU C Library is free software; you can redistribute it and/or
5    modify it under the terms of the GNU Lesser General Public
6    License as published by the Free Software Foundation; either
7    version 2.1 of the License, or (at your option) any later version.
9    The GNU C Library is distributed in the hope that it will be useful,
10    but WITHOUT ANY WARRANTY; without even the implied warranty of
11    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12    Lesser General Public License for more details.
14    You should have received a copy of the GNU Lesser General Public
15    License along with the GNU C Library; if not, write to the Free
16    Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
17    02111-1307 USA.  */
19 #include <sysdep.h>
20 #define _ERRNO_H        1
21 #include <bits/errno.h>
22 #include "kernel-features.h"
24 .text
26 #if defined HAVE_ELF && defined PIC && defined DO_VERSIONING
27 #define WAIT4   __wait4_tv64
28 #else
29 #define WAIT4   __wait4
30 #endif
32 #if defined __ASSUME_TIMEVAL64
33 PSEUDO(WAIT4, wait4, 4)
34         ret
35 PSEUDO_END(WAIT4)
36 #else
37 /* The problem here is that initially we made struct timeval compatible with
38    OSF/1, using int32.  But we defined time_t with uint64, and later found
39    that POSIX requires tv_sec to be time_t.
41    So now we have to do compatibility stuff.  */
43 /* The variable is shared between all wrappers around signal handling
44    functions which have RT equivalents.  */
45 .comm __libc_missing_axp_tv64, 4
47 LEAF(WAIT4, 32)
48         ldgp    gp, 0(pv)
49         subq    sp, 32, sp
50 #ifdef PROF
51         .set noat
52         lda     AT, _mcount
53         jsr     AT, (AT), _mcount
54         .set at
55 #endif
56         .prologue 1
58         ldl     t0, __libc_missing_axp_tv64
60         /* Save arguments in case we do need to fall back.  */
61         stq     a0, 0(sp)
62         stq     a1, 8(sp)
63         stq     a2, 16(sp)
64         stq     a3, 24(sp)
66         bne     t0, $do32
68         ldi     v0, SYS_ify(wait4)
69         callsys
70         bne     a3, $err64
72         /* Everything ok.  */
73         addq    sp, 32, sp
74         ret
76         /* If we didn't get ENOSYS, it is a real error.  */
77         .align 3
78 $err64: cmpeq   v0, ENOSYS, t0
79         beq     t0, $error
80         stl     t0, __libc_missing_axp_tv64
82         /* Recover the saved arguments.  */
83         ldq     a3, 24(sp)
84         ldq     a2, 16(sp)
85         ldq     a1, 8(sp)
86         ldq     a0, 0(sp)
88         .align 3
89 $do32:  ldi     v0, SYS_ify(osf_wait4)
90         callsys
91         bne     a3, $error
93         /* Copy back to proper format.  */
94         ldq     a3, 24(sp)
95         beq     a3, 2f
96         ldl     t0, 0(a3)               # ru_utime.tv_sec
97         ldl     t1, 4(a3)               # ru_utime.tv_usec
98         ldl     t2, 8(a3)               # ru_stime.tv_sec
99         ldl     t3, 12(a3)              # ru_stime.tv_usec
100         ldt     $f15, 16(a3)            # ru_maxrss
101         ldt     $f16, 24(a3)            # ru_ixrss
102         ldt     $f17, 32(a3)            # ru_idrss
103         ldt     $f18, 40(a3)            # ru_isrss
104         ldt     $f19, 48(a3)            # ru_minflt
105         ldt     $f20, 56(a3)            # ru_majflt
106         ldt     $f21, 64(a3)            # ru_nswap
107         ldt     $f22, 72(a3)            # ru_inblock
108         ldt     $f23, 80(a3)            # ru_oublock
109         ldt     $f24, 88(a3)            # ru_msgsend
110         ldt     $f25, 96(a3)            # ru_msgrcv
111         ldt     $f26, 104(a3)           # ru_nsignals
112         ldt     $f27, 112(a3)           # ru_nvcsw
113         .set noat
114         ldt     $f28, 120(a3)           # ru_nivcsw
115         stq     t0, 0(a3)
116         stq     t1, 8(a3)
117         stq     t2, 16(a3)
118         stq     t3, 24(a3)
119         stt     $f15, 32(a3)
120         stt     $f16, 40(a3)
121         stt     $f17, 48(a3)
122         stt     $f18, 56(a3)
123         stt     $f19, 64(a3)
124         stt     $f20, 72(a3)
125         stt     $f21, 80(a3)
126         stt     $f22, 88(a3)
127         stt     $f23, 96(a3)
128         stt     $f24, 104(a3)
129         stt     $f25, 112(a3)
130         stt     $f26, 120(a3)
131         stt     $f27, 128(a3)
132         stt     $f28, 136(a3)
133         .set at
135 2:      addq    sp, 32, sp
136         ret
138         .align 3
139 $error:
140         addq    sp, 32, sp
141         SYSCALL_ERROR_HANDLER
143 END(WAIT4)
144 #endif /* __ASSUME_TIMEVAL64 */
146 #if defined HAVE_ELF && defined PIC && defined DO_VERSIONING
147 default_symbol_version (__wait4_tv64, __wait4, GLIBC_2.1)
149 /* It seems to me to be a misfeature of the assembler that we can only
150    have one version-alias per symbol.  So create an alias ourselves.
151    The 'p' is for 'public'.  *Shrug*  */
152 strong_alias (__wait4_tv64, __wait4_tv64p)
153 default_symbol_version (__wait4_tv64p, wait4, GLIBC_2.1)
154 #else
155 weak_alias (__wait4, wait4)
156 #endif