|  |  | 
|  | /* | 
|  | * Copyright (C) Igor Sysoev | 
|  | */ | 
|  |  | 
|  |  | 
|  | #include <sys/syscall.h> | 
|  | #include <machine/asm.h> | 
|  |  | 
|  | /* | 
|  | * rfork_thread(3) - rfork_thread(flags, stack, func, arg); | 
|  | */ | 
|  |  | 
|  | #define	KERNCALL	int $0x80 | 
|  |  | 
|  | ENTRY(rfork_thread) | 
|  | push	%ebp | 
|  | mov	%esp, %ebp | 
|  | push	%esi | 
|  |  | 
|  | mov	12(%ebp), %esi	# the thread stack address | 
|  |  | 
|  | sub	$4, %esi | 
|  | mov	20(%ebp), %eax	# the thread argument | 
|  | mov	%eax, (%esi) | 
|  |  | 
|  | sub	$4, %esi | 
|  | mov	16(%ebp), %eax	# the thread start address | 
|  | mov	%eax, (%esi) | 
|  |  | 
|  | push	8(%ebp)		# rfork(2) flags | 
|  | push	$0 | 
|  | mov	$SYS_rfork, %eax | 
|  | KERNCALL | 
|  | jc	error | 
|  |  | 
|  | cmp	$0, %edx | 
|  | jne	child | 
|  |  | 
|  | parent: | 
|  | add	$8, %esp | 
|  | pop	%esi | 
|  | leave | 
|  | ret | 
|  |  | 
|  | child: | 
|  | mov	%esi, %esp | 
|  | pop	%eax | 
|  | call	*%eax		# call a thread start address ... | 
|  | add	$4, %esp | 
|  |  | 
|  | push	%eax | 
|  | push	$0 | 
|  | mov	$SYS_exit, %eax	# ... and exit(2) after a thread would return | 
|  | KERNCALL | 
|  |  | 
|  | error: | 
|  | add	$8, %esp | 
|  | pop	%esi | 
|  | leave | 
|  | PIC_PROLOGUE | 
|  |  | 
|  | /* libc's cerror: jmp  PIC_PLT(HIDENAME(cerror)) */ | 
|  |  | 
|  | push	%eax | 
|  | call	PIC_PLT(CNAME(__error)) | 
|  | pop	%ecx | 
|  | PIC_EPILOGUE | 
|  | mov	%ecx, (%eax) | 
|  | mov	$-1, %eax | 
|  | mov	$-1, %edx | 
|  | ret |