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
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
|
; Copyright (c) 2023 Marvin Borner
%include "config.asm"
global main
extern printf
extern array
extern size
bits 64
;;; main
main:
push rbp
mov ecx, size
loop:
push rcx
call thread
pop rcx
dec ecx
jnz loop
end:
call thread_wait_all
pop rbp
mov rax, 0x3c
mov rdi, 0
syscall ; exit
%if USE_SYSCALL
; uses nanosleep syscall
sleep:
mov rax, TIMEOUT
mov rcx, rsi
mul rcx
push qword rax ; ns
push qword 0 ; s
mov rax, 0x23
mov rdi, rsp
xor rsi, rsi
syscall
add rsp, 16
ret
%else
; uses busy loop
sleep:
mov rax, TIMEOUT
mov rcx, rsi
mul rcx
.loop:
dec rax
jnz .loop
ret
%endif
print:
push rsi
call sleep
pop rsi
mov rdi, format
xor rax, rax
call printf
ret
thread:
lea rax, print
mov esi, [array + (ecx - 1) * 4]
call thread_run
ret
;;; utils
thread_run:
push rax
mov rax, 0x39
syscall ; fork
cmp rax, 0
mov r10, rax
pop rax
je child
mov rax, r10
ret
thread_wait_all:
sub rsp, 4
mov rdi, -1
mov rax, 0x3d
mov rsi, rsp
xor rdx, rdx
xor r10, r10
syscall ; wait4
add rsp, 4
cmp rax, -10 ; idk why
jne thread_wait_all
ret
child:
call rax
mov rax, 0x3c
mov rdi, 0
syscall ; exit
;;; data
section .data
format: db "%d", 10, 0
|