summaryrefslogtreecommitdiff
path: root/src/user/lib/thread.S
blob: 1a27c300fc43c6d24ecfc3cc27175768031c1950 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
#define ASM_FILE 1
#include <camellia/syscalls.h>
#include <camellia/flags.h>

.section .text
.global thread_creates
.type thread_creates, @function
// void thread_creates(int flags, void (*fn)(void*), void *arg, void *stack);
thread_creates:
	push %r12
	push %r13
	push %r14

	/* save fn, arg, stack */
	mov %rsi, %r12
	mov %rdx, %r13
	mov %rcx, %r14

	mov %rdi, %rsi
	or $(FORK_SHAREMEM | FORK_SHAREHANDLE), %rsi
	mov $_SYSCALL_FORK, %rdi
	xor %rdx, %rdx
	syscall

	test %rax, %rax
	jz 1f
	/* in parent, return normally */
	pop %r14
	pop %r13
	pop %r12
	ret
1:	/* in child */
	mov %r14, %rsp
	mov %r13, %rdi
	call *%r12

	mov $_SYSCALL_EXIT, %rdi
	xor %rsi, %rsi
	syscall
	hlt /* if all else fails... */