Update copyright dates with scripts/update-copyrights.
[glibc.git] / sysdeps / unix / sysv / linux / tile / clone.S
blob982ff80675f4d719f9a0900e612ab0131366cd53
1 /* Copyright (C) 2011-2015 Free Software Foundation, Inc.
2    This file is part of the GNU C Library.
3    Contributed by Chris Metcalf <cmetcalf@tilera.com>, 2011.
5    The GNU C Library is free software; you can redistribute it and/or
6    modify it under the terms of the GNU Lesser General Public
7    License as published by the Free Software Foundation; either
8    version 2.1 of the License, or (at your option) any later version.
10    The GNU C Library is distributed in the hope that it will be useful,
11    but WITHOUT ANY WARRANTY; without even the implied warranty of
12    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13    Lesser General Public License for more details.
15    You should have received a copy of the GNU Lesser General Public
16    License along with the GNU C Library.  If not, see
17    <http://www.gnu.org/licenses/>.  */
19 /* clone() is even more special than fork() as it mucks with stacks
20    and invokes a function in the right context after it's all over.  */
22 #include <sysdep.h>
23 #define _ERRNO_H 1
24 #include <bits/errno.h>
26 #include <asm/unistd.h>
27 #include <arch/abi.h>
28 #include <tls.h>
29 #include <linux/sched.h>
31 /* What we save where in the stack frame; must include all callee-saves. */
32 #define FRAME_NEXT_LR   (0 * REGSIZE)  /* reserved by ABI; not used here */
33 #define FRAME_SP        (1 * REGSIZE)
34 #define FRAME_R30       (2 * REGSIZE)
35 #define FRAME_R31       (3 * REGSIZE)
36 #define FRAME_R32       (4 * REGSIZE)
37 #define FRAME_SIZE      (5 * REGSIZE)
39 /* int clone(int (*fn)(void *arg), void *child_stack, int flags, void *arg,
40              pid_t *ptid, struct user_desc *tls, pid_t *ctid); */
42         .text
43 ENTRY (__clone)
44         /* Create a stack frame so we can pass callee-saves to new task. */
45         {
46          move r10, sp
47          ST sp, lr
48          ADDI_PTR sp, sp, -FRAME_SIZE
49         }
50         cfi_offset (lr, 0)
51         cfi_def_cfa_offset (FRAME_SIZE)
52         ADDI_PTR r11, sp, FRAME_SP
53         {
54          ST r11, r10
55          ADDI_PTR r11, sp, FRAME_R30
56         }
57         {
58          ST r11, r30
59          ADDI_PTR r11, sp, FRAME_R31
60         }
61         cfi_offset (r30, FRAME_R30 - FRAME_SIZE)
62         {
63          ST r11, r31
64          ADDI_PTR r11, sp, FRAME_R32
65         }
66         cfi_offset (r31, FRAME_R31 - FRAME_SIZE)
67         ST r11, r32
68         cfi_offset (r32, FRAME_R32 - FRAME_SIZE)
70         /* sanity check arguments */
71         BEQZ r0, .Linvalid
72         BEQZ r1, .Linvalid
74         /* Make sure child stack is properly aligned, and set up the
75            top frame so that we can call out of it immediately in the
76            child.  Setting it up here means we fault in the parent if
77            it's bogus, which is probably cleaner than faulting first
78            thing in the child. */
79         ADDI_PTR r1, r1, -C_ABI_SAVE_AREA_SIZE
80         andi r1, r1, -C_ABI_SAVE_AREA_SIZE
81         ADDI_PTR r9, r1, REGSIZE /* sp of this frame on entry, i.e. zero */
82         ST r9, zero
84         /* We need to switch the argument convention around from
85            libc to kernel:
87            libc:
88             r0 fn
89             r1 child_stack
90             r2 flags
91             r3 arg
92             r4 ptid
93             r5 tls
94             r6 ctid
96            kernel:
97             r0 flags
98             r1 child_stack [same as libc]
99             r2 ptid
100             r3 ctid
101             r4 tls
103            Plus the callee-saves as described at .Lthread_start, below.  */
104         {
105          move r32, r0
106          move r0, r2
107         }
108         {
109          move r31, r3
110          move r3, r6
111         }
112         {
113          move r30, r2
114          move r2, r4
115         }
116         {
117          move r4, r5
118          moveli TREG_SYSCALL_NR_NAME, __NR_clone
119         }
120         swint1
121         BEQZ r0, .Lthread_start  /* If in child task.  */
123 .Ldone:
124         /* Restore the callee-saved registers and return. */
125         ADDLI_PTR lr, sp, FRAME_SIZE
126         {
127          LD lr, lr
128          ADDLI_PTR r30, sp, FRAME_R30
129         }
130         {
131          LD r30, r30
132          ADDLI_PTR r31, sp, FRAME_R31
133         }
134         {
135          LD r31, r31
136          ADDLI_PTR r32, sp, FRAME_R32
137         }
138         {
139          LD r32, r32
140          ADDI_PTR sp, sp, FRAME_SIZE
141         }
142         cfi_def_cfa_offset (0)
144         BNEZ r1, .Lerror
145         jrp lr
147 .Lerror:
148         j SYSCALL_ERROR_NAME
150 .Linvalid:
151         {
152          movei r1, EINVAL
153          j .Ldone
154         }
156 /* This function expects to receive:
158    sp: the top of a valid stack area
159    r30: clone() flags
160    r31: the argument to pass to the user function
161    r32: the user function pointer  */
163 .Lthread_start:
164         cfi_def_cfa_offset (FRAME_SIZE)
165         cfi_undefined (lr)
166         /* Check and see if we need to reset the PID, which we do if
167            CLONE_THREAD isn't set, i.e. we're not staying in the thread group.
168            If CLONE_VM is set, we're doing some kind of thread-like clone,
169            so we set the tid/pid to -1 to disable using the cached values
170            in getpid().  Otherwise (if CLONE_VM isn't set), it's a
171            fork-like clone, and we go ahead and write the cached values
172            from the true system pid (retrieved via __NR_getpid syscall).  */
173 #ifdef __tilegx__
174         {
175          moveli r0, hw1_last(CLONE_VM)
176          moveli r1, hw1_last(CLONE_THREAD)
177         }
178         {
179          shl16insli r0, r0, hw0(CLONE_VM)
180          shl16insli r1, r1, hw0(CLONE_THREAD)
181         }
182 #else
183         {
184          moveli r0, lo16(CLONE_VM)
185          moveli r1, lo16(CLONE_THREAD)
186         }
187         {
188          auli r0, r0, ha16(CLONE_VM)
189          auli r1, r1, ha16(CLONE_THREAD)
190         }
191 #endif
192         {
193          and r0, r30, r0
194          and r1, r30, r1
195         }
196         BNEZ r1, .Lno_reset_pid   /* CLONE_THREAD is set */
197         {
198          movei r0, -1
199          BNEZ r0, .Lgotpid         /* CLONE_VM is set */
200         }
201         moveli TREG_SYSCALL_NR_NAME, __NR_getpid
202         swint1
203 .Lgotpid:
204         ADDLI_PTR r2, tp, PID_OFFSET
205         {
206          ST4 r2, r0
207          ADDLI_PTR r2, tp, TID_OFFSET
208         }
209         ST4 r2, r0
210 .Lno_reset_pid:
211         {
212          /* Invoke user function with specified argument. */
213          move r0, r31
214          jalr r32
215         }
216         {
217          j HIDDEN_JUMPTARGET(_exit)
218          info INFO_OP_CANNOT_BACKTRACE   /* Notify backtracer to stop. */
219         }
220 PSEUDO_END (__clone)
222 weak_alias (__clone, clone)