Assembly Routines
Assembly routines can be linked into a C/C++ program by putting them into the same src directory that your C/C++ sources are in, but with an .asm extension. These can be placed in any subdirectory of src just like C/C++ sources.
Assembly Files
Constants
The top of the file is a good place defining constants or including other files that define constants.
These will be availabe throughout the rest of the file, but not in other files.
You can define a constant by saying my_constant := 42
.
You can include common constants in multiple files by defining them in a file named, say, file.inc and putting include 'file.inc'
in every file that needs them.
You should not generally put any labels, code, or data here.
If you try to reference a label defined here, you will get an Error: variable term used where not expected.
linker error which means you are trying to resolve an address that doesn’t belong to any section.
See Section to fix this problem.
Assume
You should add a assume adl=1
before trying to emit any code, which ensures that you get 24-bit eZ80 instructions.
If you end an assembly file with assume adl=0
(which is the eZ80’s 16-bit Z80 compatibility mode), it will propogate to another random assembly file.
All toolchain and compiler-generated sources make sure to reset the mode at the top of the file and end the file in the same mode, but if one of your sources end in Z80 mode, then any other one of your sources might begin in Z80 mode, so it is safer to put the assume
line in every file.
Section
Now that we are in the correct mode, we need to tell the linker where to put things.
We use section .text
for code, section .data
for variables, and section .rodata
for constant data.
Currently these are all placed in RAM, so which section you choose to switch to is much less important than how often you switch sections, even if you are switching to the same section you are already in.
This is because every time you start a new, or restart the same, section, the linker gets a new opportunity to delete a block of dead code/data.
Because of this, the correct time to switch sections is usually every time you start a new function or variable.
You should not let execution fall off the end of a block because you won’t know if that block will be included or deleted from the output, however you can if you say require _symbol
of some public or private symbol defined in the next section to ensure that if the current block is included, then that will force the next block to also be included.
To define a symbol in a block that can be referenced from other blocks, you should do private _symbol
or public _symbol
right before its definition.
If it is private then it is only able to be referenced from the same file and no extern should be used.
If it is public then it can be referenced within the same file without extern just like private symbols, but public symbols can also be referenced from other files and even C/C++!
The public assembly symbol named _symbol
is accessible in C by the global name symbol
, assuming it is properly declared, with your asm symbol acting as the definition.
Extern
At the end of the file is a good place to list every external symbol that you might depend on like extern _symbol
.
This includes both public symbols defined in another assembly file and global symbols from C, prefixed with an underscore like usual.
Lastly, you should not let execution fall off the end of a file because the next file that gets assembled is unpredictable and you could end up anywhere!
Block ordering can only be relied on within a single file, and only for blocks belonging to the same section.
Linking ASM routines to C/C++
If an assembly function needs to be called from C, a separate header file should define a extern C global prototype.
In C this looks like a normal function or global declaration, and in C++ it’s the same thing but in an extern "C" {}
block.
Below is an example C prototype followed by the assembly implementation:
asm_func.h
void asm_func(int arg);
asm_func.asm
assume adl=1
section .text
public _asm_func
_asm_func:
pop hl
pop de
push de ; de = arg
push hl
push de
call _external_func
pop de
ret
extern _external_func
asm_func.c
int external_func(int arg) {
printf("external_func called with %d\n", arg);
return 4321;
}
void test() {
int arg = 1234;
printf("calling asm_func with %d\n", arg);
int ret = asm_func(arg);
printf("asm_func returned %d\n", ret);
}
Preserve
Assembly routines must preserve the IX
and SP
registers.
All other registers are free for use.
Arguments
Arguments are pushed from last to first corresponding to the C prototype.
In eZ80, 3 bytes are always pushed to the stack regardless of the actual size.
However, the assembly function must be careful to only use the valid bytes that are pushed.
For example, if a short type is used, the upper byte of the value pushed on the stack will contain arbitrary data.
This table lists the locations relative to sp from within the called funciton.
Note that sp + [0,2]
contains the return address.
C/C++ Type |
Size |
Stack Location |
---|---|---|
char |
1 byte |
sp + [3] |
short |
2 bytes |
sp + [3,4] |
int |
3 bytes |
sp + [3,5] |
long |
4 bytes |
sp + [3,6] |
(u)int48_t |
6 bytes |
sp + [3,8] |
long long |
8 bytes |
sp + [3,10] |
float |
4 bytes |
sp + [3,6] |
double |
4 bytes |
sp + [3,6] |
pointer |
3 bytes |
sp + [3,5] |
Returns
This table lists which registers are used for return values from a function.
The type’s sign does not affect the registers used, but may affect the value returned.
The LSB is located in the register on the far right of the expression, e.g. E:HL
indicates register L
stores the LSB.
C/C++ Type |
Return Register |
---|---|
char |
A |
short |
HL |
int |
UHL |
long |
E:UHL |
(u)int48_t |
UDE:UHL |
long long |
BC:UDE:UHL |
float |
E:UHL |
double |
E:UHL |
pointer |
UHL |