/*
- * Copyright (c) 1999-2007 Apple Inc. All rights reserved.
+ * Copyright (c) 1999-2010 Apple Inc. All rights reserved.
*
* @APPLE_OSREFERENCE_LICENSE_HEADER_START@
*
* 11-Jan-92 Peter King (king@next.com)
* Created from M68K sources
*/
-
-#include "SYS.h"
-
-#if defined(__ppc__) || defined(__ppc64__)
-
-/* We use mode-independent "g" opcodes such as "srgi". These expand
- * into word operations when targeting __ppc__, and into doubleword
- * operations when targeting __ppc64__.
+
+/*
+ * All of the asm stubs in this file have been adjusted so the pre/post
+ * fork handlers and dyld fixup are done in C inside Libc. As such, Libc
+ * expects the __fork asm to fix up the return code to be -1, 0 or pid
+ * and errno if needed.
*/
-#include <architecture/ppc/mode_independent_asm.h>
-
-MI_ENTRY_POINT(___fork)
- MI_PUSH_STACK_FRAME
-
- MI_CALL_EXTERNAL(__cthread_fork_prepare)
-
- li r0,SYS_fork
- sc // do the fork
- b Lbotch // error return
-
- cmpwi r4,0 // parent (r4==0) or child (r4==1) ?
- beq Lparent // parent, since r4==0
-
-
-/* Here if we are the child. */
-
-#if defined(__DYNAMIC__)
- .cstring
-LC3:
- .ascii "__dyld_fork_child\0"
- .text
- .align 2
- mflr r0
- bcl 20,31,1f
-1: mflr r3
- mtlr r0
- addis r3,r3,ha16(LC3-1b)
- addi r3,r3,lo16(LC3-1b)
- addi r4,r1,SF_LOCAL1
- bl __dyld_func_lookup
- lg r3,SF_LOCAL1(r1)
- mtspr ctr,r3
- bctrl
-#endif
-
- li r9,0
- MI_GET_ADDRESS(r8,__current_pid)
- stw r9,0(r8) // clear cached pid in child
-
- MI_CALL_EXTERNAL(__cthread_fork_child)
-
- li r3,0 // flag for "we are the child"
- b Lreturn
+#include "SYS.h"
-/* Here if we are the parent, with:
- * r3 = child's pid
- */
-Lparent:
- stg r3,SF_LOCAL2(r1) // save child pid in stack
-
- b Lparent_return // clean up and return child's pid
-
-
-/* Here if the fork() syscall failed. We're still the parent. */
-
-Lbotch:
-
- MI_CALL_EXTERNAL(cerror)
- li r3,-1 // get an error return code
- stg r3,SF_LOCAL2(r1) // save return code in stack
-
- /*
- * We use cthread_fork_parent() to clean up after a fork error
- * (unlock cthreads and mailloc packages) so the parent
- * process can Malloc() after fork() errors without
- * deadlocking.
- */
-
-Lparent_return:
- MI_CALL_EXTERNAL(__cthread_fork_parent)
- lg r3,SF_LOCAL2(r1) // return -1 on error, child's pid on success
-
-Lreturn:
- MI_POP_STACK_FRAME_AND_RETURN
-
-#elif defined(__i386__)
+#if defined(__i386__)
LEAF(___fork, 0)
subl $28, %esp // Align the stack, with 16 bytes of extra padding that we'll need
- CALL_EXTERN(__cthread_fork_prepare)
movl $ SYS_fork,%eax; // code for fork -> eax
UNIX_SYSCALL_TRAP // do the system call
jnc L1 // jump if CF==0
- CALL_EXTERN(cerror)
- CALL_EXTERN(__cthread_fork_parent)
+ CALL_EXTERN(tramp_cerror)
movl $-1,%eax
addl $28, %esp // restore the stack
ret
jz L2 // parent, since r1 == 0 in parent, 1 in child
//child here...
-#if defined(__DYNAMIC__)
-// Here on the child side of the fork we need to tell the dynamic linker that
-// we have forked. To do this we call __dyld_fork_child in the dyanmic
-// linker. But since we can't dynamically bind anything until this is done we
-// do this by using the private extern __dyld_func_lookup() function to get the
-// address of __dyld_fork_child (the 'C' code equivlent):
-//
-// _dyld_func_lookup("__dyld_fork_child", &address);
-// address();
-//
-.cstring
-LC0:
- .ascii "__dyld_fork_child\0"
-
-.text
- leal 0x8(%esp),%eax // get the address where we're going to store the pointer
- movl %eax, 0x4(%esp) // copy the address of the pointer
- call 1f
-1: popl %eax
- leal LC0-1b(%eax),%eax
- movl %eax, 0x0(%esp) // copy the name of the function to look up
- call __dyld_func_lookup
- movl 0x8(%esp),%eax // move the value returned in address parameter
- call *%eax // call __dyld_fork_child indirectly
-#endif
- xorl %eax, %eax
- REG_TO_EXTERN(%eax, __current_pid)
- CALL_EXTERN(__cthread_fork_child)
-
xorl %eax,%eax // zero eax
- addl $28, %esp // restore the stack
- ret
-
- //parent here...
+ REG_TO_EXTERN(%eax, __current_pid);
L2:
- movl %eax, 0xc(%esp) // save pid
-
- CALL_EXTERN_AGAIN(__cthread_fork_parent)
- movl 0xc(%esp), %eax // return pid
addl $28, %esp // restore the stack
- ret
+ // parent ends up here skipping child portion
+ ret
#elif defined(__x86_64__)
LEAF(___fork, 0)
subq $24, %rsp // Align the stack, plus room for local storage
- CALL_EXTERN(__cthread_fork_prepare)
movl $ SYSCALL_CONSTRUCT_UNIX(SYS_fork),%eax; // code for fork -> rax
UNIX_SYSCALL_TRAP // do the system call
jnc L1 // jump if CF==0
- CALL_EXTERN(cerror)
- CALL_EXTERN(__cthread_fork_parent)
+ movq %rax, %rdi
+ CALL_EXTERN(_cerror)
movq $-1, %rax
addq $24, %rsp // restore the stack
ret
jz L2 // parent, since r1 == 0 in parent, 1 in child
//child here...
-#if defined(__DYNAMIC__)
-// Here on the child side of the fork we need to tell the dynamic linker that
-// we have forked. To do this we call __dyld_fork_child in the dyanmic
-// linker. But since we can't dynamically bind anything until this is done we
-// do this by using the private extern __dyld_func_lookup() function to get the
-// address of __dyld_fork_child (the 'C' code equivlent):
-//
-// _dyld_func_lookup("__dyld_fork_child", &address);
-// address();
-//
-.cstring
-LC0:
- .ascii "__dyld_fork_child\0"
-
-.text
- leaq 8(%rsp),%rsi // get the address where we're going to store the pointer
- leaq LC0(%rip), %rdi // copy the name of the function to look up
- call __dyld_func_lookup
- call *8(%rsp) // call __dyld_fork_child indirectly
-#endif
xorq %rax, %rax
- REG_TO_EXTERN(%rax, __current_pid)
- CALL_EXTERN(__cthread_fork_child)
-
- xorq %rax,%rax // zero rax
+ PICIFY(__current_pid)
+ movl %eax,(%r11)
+L2:
+ // parent ends up here skipping child portion
addq $24, %rsp // restore the stack
ret
- //parent here...
-L2:
- movl %eax, 16(%rsp) // save pid
+#elif defined(__arm__)
+
+MI_ENTRY_POINT(___fork)
+ stmfd sp!, {r4, r7, lr}
+ add r7, sp, #4
- CALL_EXTERN_AGAIN(__cthread_fork_parent)
- movl 16(%rsp), %eax // return pid
- addq $24, %rsp // restore the stack
- ret
+ mov r1, #1 // prime results
+ mov r12, #SYS_fork
+ swi #SWI_SYSCALL // make the syscall
+ bcs Lbotch // error?
+
+ cmp r1, #0 // parent (r1=0) or child(r1=1)
+ beq Lparent
+
+ //child here...
+ MI_GET_ADDRESS(r3, __current_pid)
+ mov r0, #0
+ str r0, [r3] // clear cached pid in child
+ ldmfd sp!, {r4, r7, pc}
+
+Lbotch:
+ MI_CALL_EXTERNAL(_cerror) // jump here on error
+ mov r0,#-1 // set the error
+ // fall thru
+Lparent:
+ ldmfd sp!, {r4, r7, pc} // pop and return
+
+#elif defined(__arm64__)
+
+#include <mach/arm64/asm.h>
+
+MI_ENTRY_POINT(___fork)
+ ARM64_STACK_PROLOG
+ PUSH_FRAME
+ // ARM moves a 1 in to r1 here, but I can't see why.
+ mov x16, #SYS_fork // Syscall code
+ svc #SWI_SYSCALL // Trap to kernel
+ b.cs Lbotch // Carry bit indicates failure
+ cbz x1, Lparent // x1 == 0 indicates that we are the parent
+
+ // Child
+ MI_GET_ADDRESS(x9, __current_pid) // Get address of cached "current pid"
+ mov w0, #0
+ str w0, [x9] // Clear cached current pid
+ POP_FRAME // And done
+ ARM64_STACK_EPILOG
+
+Lbotch:
+ MI_CALL_EXTERNAL(_cerror) // Handle error
+ mov w0, #-1 // Return value is -1
+Lparent:
+ POP_FRAME // Return
+ ARM64_STACK_EPILOG
#else
#error Unsupported architecture