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
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
|
# Must be included at the top of the file
asm "extern malloc"
asm "extern realloc"
asm "extern free"
asm "extern printf"
asm "extern CreateFileA"
asm "extern ReadFile"
asm "extern WriteFile"
asm "extern CloseHandle"
{}uint8 _alert = "Alert!\n\0"
{}uint8 _dec = "%d\n\0"
/; _alloc (uint size) [~void]
~void out
# Mov size into proper register, and set all extras to zero
asm "mov rcx, r8"
asm "mov rdx, 0"
asm "mov r8, 0"
asm "mov r9, 0"
asm "call malloc"
# Set out to the returned value
# (The compiler assignes spaces sequentially, and we have a uint in r8)
asm "mov r9, rax"
return out
;/
/; _realloc (~void ptr, uint new_size) [~void]
~void out
# Mov ptr and new size into proper registers, and set all extras to zero
asm "mov rcx, r8"
asm "mov rdx, r9"
asm "mov r8, 0"
asm "mov r9, 0"
# Do call
asm "call realloc"
# Set out to the returned value
# (The compiler assignes spaces sequentially. We have a ptr in r8, and a uint in r9)
asm "mov r10, rax"
return out
;/
/; _delete (~void ptr)
# setup call by clearing most values
asm "mov rcx, rax"
asm "mov rdx, 0"
asm "mov r8, 0"
asm "mov r9, 0"
# do call
asm "call free"
# there's no more to do 'cause free returns nothing
;/
/; _printf (~void str)
# setup call by clearing most values
asm "mov rcx, rax"
asm "mov rdx, 0"
asm "mov r8, 0"
asm "mov r9, 0"
# do call
asm "call printf"
# there's no more to do 'cause printf returns nothing
;/
/; _print_num (~void str, int num)
# setup call by clearing most values
asm "mov rcx, rax"
asm "mov rdx, rbx"
asm "mov rdi, 0"
asm "mov rsi, 0"
asm "mov r8, 0"
asm "mov r9, 0"
# do call
asm "call printf"
# there's no more to do 'cause printf returns nothing
;/
# Create file for writing (overwrite)
/; _create_file (~void name) [~void]
~void out
asm "mov rcx, r8" # lpFileName
asm "mov rdx, 3" # dwDesiredAccess
asm "shl rdx, 30" # (GENERIC_READ 1 << 31 | GENERIC_WRITE 1 << 30)
asm "mov r8, 0" # dwShareMode
asm "mov r9, 0" # lpSecurityAttributes
asm "push qword 0" # hTemplateFile
asm "push qword 128" # dwFlagsAndAttributes (NORMAL_FILE = 128)
asm "push qword 2" # dwCreationDisposition (CREATE_ALWAYS = 2)
# Shadow space
asm "push r9"
asm "push r8"
asm "push rdx"
asm "push rcx"
# Do call
asm "call CreateFileA"
# Set out to the returned value
# (The compiler assignes spaces sequentially. We have a ptr in r8)
asm "mov r9, rax"
return out
;/
# Open file for reading or writing (no overwrite)
/; _open_file (~void name) [~void]
~void out
asm "mov rcx, r8" # lpFileName
asm "mov rdx, 3" # dwDesiredAccess
asm "shl rdx, 30" # (GENERIC_READ 1 << 31 | GENERIC_WRITE 1 << 30)
asm "mov r8, 0" # dwShareMode
asm "mov r9, 0" # lpSecurityAttributes
asm "push qword 0" # hTemplateFile
asm "push qword 128" # dwFlagsAndAttributes (NORMAL_FILE = 128)
asm "push qword 2" # dwCreationDisposition (CREATE_ALWAYS = 2)
# Shadow space
asm "push r9"
asm "push r8"
asm "push rdx"
asm "push rcx"
# Do call
asm "call CreateFileA"
# Set out to the returned value
# (The compiler assignes spaces sequentially. We have a ptr in r8, and a uint in r9)
asm "mov r10, rax"
return out
;/
/; _close_file (~void handle)
asm "mov rcx, r8" # handle
asm "mov rdx, 0"
asm "mov r8, 0"
asm "mov r9, 0"
# Shadow space
asm "push r9"
asm "push r8"
asm "push rdx"
asm "push rcx"
asm "call CloseHandle"
asm "add rsp, 32"
;/
/; _read_byte (~void handle, ~uint8 byte) [bool]
bool out
asm "mov rcx, r8" # handle
asm "mov rdx, r9" # buffer
asm "mov r8, 1" # one byte
asm "mov r9, 0"
asm "push qword 0"
# Shadow space
asm "push r9"
asm "push r8"
asm "push rdx"
asm "push rcx"
asm "call ReadFile"
asm "mov r10, rax"
return out
;/
/; _write_byte (~void handle, ~uint8 byte) [bool]
bool out
asm "mov rcx, r8" # handle
asm "mov rdx, r9" # buffer
asm "mov r8, 1" # one byte
asm "mov r9, 0"
asm "push qword 0"
# Shadow space
asm "push r9"
asm "push r8"
asm "push rdx"
asm "push rcx"
asm "call WriteFile"
asm "mov r10, rax"
return out
;/
/; print_alert
_printf(~_alert{0})
;/
/; cstr_len (~uint8 cstr) [int]
int i = 0
/; loop (cstr{i} !== 0) [i++] ;/
return i
;/
|