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
|
# MACRO: start
# All assembler tests should start with a call to "start"
.macro start
.text
# Skip over these inlined funcs.
jmp __start;
.global __pass
.type __pass, function
__pass:
# Note - we cannot just invoke:
#
# write 1, _passmsg, 5
#
# here because _passmsg contains the run-time (VMA) address of
# the pass string (probably 0x500) not the load-time (LMA)
# address (probably 0x804c). Normally using the VMA address
# would be the correct thing to do - *if* there was some start
# up code which copied data from LMA to VMA. But we have no
# start up code, so the data still resides at the LMA
# address. Hence we use __romdatastart instead.
#
# Note - we are cheating because the address that we pass to
# "write" should actually be:
#
# __romdatastart + (_passmsg - __datastart)
#
# but the assembler cannot cope with this expression. So we
# cheat and use the fact that we know that _passmsg is the
# first string in the .data section and so (_passmsg -
# __datastart) evaluates to zero.
write 1, __romdatastart, 5
exit 0
.global __fail
.type __fail, function
__fail:
# Note - see above.
#
# write 1, _failmsg, 5
#
# This time we use the fact that _passmsg is aligned to a
# 16 byte boundary to work out that (_failmsg - __datastart)
# evaluates to 0x10.
write 1, __romdatastart + 0x10, 5
exit 1
.data
_passmsg:
.ascii "pass\n"
.align 4
_failmsg:
.ascii "fail\n"
.align 4
.text
.global __start
.type __start, function
__start:
.endm
# MACRO: system_call
# Make a libgloss/Linux system call
.macro system_call nr:req
call #(0x180|\nr);
.endm
# MACRO: exit
# Quit the current test
.macro exit rc:req
mov #\rc, r12
system_call 1
.endm
# MACRO: pass
# Write 'pass' to stdout via syscalls and quit;
# meant for non-OS operating environments
.macro pass
jmp __pass;
.endm
# MACRO: fail
# Write 'fail' to stdout via syscalls and quit;
# meant for non-OS operating environments
.macro fail
jmp __fail;
.endm
# MACRO: write
# Just like the write() C function; uses system calls
.macro write fd:req, buf:req, count:req
mov #\fd, r12;
mov #\buf, r13;
mov #\count, r14;
system_call 5
.endm
|