mirror of
https://github.com/KarolS/millfork.git
synced 2025-01-10 20:29:35 +00:00
Document the current state of 6809 support
This commit is contained in:
parent
ff46129c4e
commit
166f5c397d
@ -74,7 +74,7 @@ and the most significant word is passed via the DE register pair
|
|||||||
|
|
||||||
* callee may clobber all flags
|
* callee may clobber all flags
|
||||||
|
|
||||||
* callee may clobber all registers except for IX, IY and shadow registers
|
* callee may clobber all registers except for SP, IX, IY and shadow registers
|
||||||
|
|
||||||
## 8086
|
## 8086
|
||||||
|
|
||||||
@ -118,4 +118,32 @@ and the most significant word is passed via the DX register
|
|||||||
|
|
||||||
* callee may clobber all flags
|
* callee may clobber all flags
|
||||||
|
|
||||||
* callee may clobber all registers except for BP
|
* callee may clobber all registers except for SP and BP
|
||||||
|
|
||||||
|
## 6809
|
||||||
|
|
||||||
|
**WARNING!** Motorola 6809 support is not yet complete.
|
||||||
|
|
||||||
|
**TODO: this convention may change**
|
||||||
|
|
||||||
|
#### Parameters:
|
||||||
|
|
||||||
|
* if the function has one parameter of size one byte, it is passed via the B register
|
||||||
|
|
||||||
|
* if the function has one parameter of size two bytes, it is passed via the D register
|
||||||
|
|
||||||
|
* otherwise, all parameters are passed via static locations
|
||||||
|
|
||||||
|
#### Return values:
|
||||||
|
|
||||||
|
* one-byte return values are passed via the B register
|
||||||
|
|
||||||
|
* two-byte return values are passed via the D register
|
||||||
|
|
||||||
|
* otherwise, the return value is passed via a static location
|
||||||
|
|
||||||
|
#### Register preservation:
|
||||||
|
|
||||||
|
* callee may clobber all flags
|
||||||
|
|
||||||
|
* callee may clobber all registers except for S and U
|
||||||
|
@ -44,6 +44,9 @@ if a line ends with a backslash character, the value continues to the next line.
|
|||||||
* `i8086` (Intel 8086; very experimental, very buggy and very, very incomplete –
|
* `i8086` (Intel 8086; very experimental, very buggy and very, very incomplete –
|
||||||
see the [8086 support disclaimer](../lang/x86disclaimer.md))
|
see the [8086 support disclaimer](../lang/x86disclaimer.md))
|
||||||
|
|
||||||
|
* `6809` (Motorola 6809; very experimental, very buggy and very, very incomplete –
|
||||||
|
many language features simply do not work at all for this target)
|
||||||
|
|
||||||
* `encoding` – default encoding for console I/O. Default: `ascii`.
|
* `encoding` – default encoding for console I/O. Default: `ascii`.
|
||||||
See [the list of available encodings](../lang/text.md).
|
See [the list of available encodings](../lang/text.md).
|
||||||
|
|
||||||
|
@ -34,6 +34,8 @@
|
|||||||
|
|
||||||
* [Inline 8080/LR35902/Z80 assembly syntax](lang/assemblyz80.md)
|
* [Inline 8080/LR35902/Z80 assembly syntax](lang/assemblyz80.md)
|
||||||
|
|
||||||
|
* [Inline 6809 assembly syntax](lang/assembly6809.md)
|
||||||
|
|
||||||
* [Important guidelines regarding reentrancy](lang/reentrancy.md)
|
* [Important guidelines regarding reentrancy](lang/reentrancy.md)
|
||||||
|
|
||||||
* [List of keywords](lang/keywords.md)
|
* [List of keywords](lang/keywords.md)
|
||||||
|
134
docs/lang/assembly6809.md
Normal file
134
docs/lang/assembly6809.md
Normal file
@ -0,0 +1,134 @@
|
|||||||
|
[< back to index](../doc_index.md)
|
||||||
|
|
||||||
|
# Using 6809 assembly within Millfork programs
|
||||||
|
|
||||||
|
**WARNING!** Motorola 6809 support is not yet complete.
|
||||||
|
|
||||||
|
There are two ways to include raw assembly code in your Millfork programs:
|
||||||
|
|
||||||
|
* inline assembly code blocks
|
||||||
|
|
||||||
|
* whole assembly functions
|
||||||
|
|
||||||
|
## Assembly syntax
|
||||||
|
|
||||||
|
Millfork inline assembly uses the same three-letter opcodes as most other 6809 assemblers.
|
||||||
|
|
||||||
|
Labels have to be followed by a colon and they can optionally be on a separate line.
|
||||||
|
Indentation is not important:
|
||||||
|
|
||||||
|
first: INC a
|
||||||
|
second:
|
||||||
|
INC b
|
||||||
|
INC c
|
||||||
|
|
||||||
|
|
||||||
|
Label names have to start with a letter and can contain digits, underscores and letters.
|
||||||
|
This means than they cannot start with a period like in many other assemblers.
|
||||||
|
Similarly, anonymous labels designated with `+` or `-` are also not supported.
|
||||||
|
|
||||||
|
Assembly can refer to variables and constants defined in Millfork,
|
||||||
|
but you need to be careful with using absolute vs immediate addressing:
|
||||||
|
|
||||||
|
const byte fiveConstant = 5
|
||||||
|
byte fiveVariable = 5
|
||||||
|
|
||||||
|
byte ten() {
|
||||||
|
byte result
|
||||||
|
asm {
|
||||||
|
LDB fiveVariable // not LDB #fiveVariable
|
||||||
|
ADDB #fiveConstant
|
||||||
|
STB result
|
||||||
|
}
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
|
Any assembly opcode can be prefixed with `?`, which allows the optimizer change it or elide it if needed.
|
||||||
|
Opcodes without that prefix will be always compiled as written.
|
||||||
|
|
||||||
|
The '!' prefix marks the statement as volatile, which means it will be a subject to certain, but not all optimizations,
|
||||||
|
in order to preserve its semantics.
|
||||||
|
|
||||||
|
You can insert macros into assembly, by prefixing them with `+` and using the same syntax as in Millfork:
|
||||||
|
|
||||||
|
macro void run(byte x) {
|
||||||
|
output = x
|
||||||
|
}
|
||||||
|
|
||||||
|
byte output @$c000
|
||||||
|
|
||||||
|
void main () {
|
||||||
|
byte a
|
||||||
|
a = 7
|
||||||
|
asm {
|
||||||
|
+ run(a)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
You can insert raw bytes into your assembly using the array syntax:
|
||||||
|
|
||||||
|
[ $00, $00 ]
|
||||||
|
"this is a string to print" bbc
|
||||||
|
["this is a string to print but this time it's zero-terminated so it will actually work" bbc, 0]
|
||||||
|
[for x,0,until,8 [x]]
|
||||||
|
|
||||||
|
## Assembly functions
|
||||||
|
|
||||||
|
Assembly functions can be declared as `macro` or not.
|
||||||
|
|
||||||
|
A macro assembly function is inserted into the calling function like an inline assembly block,
|
||||||
|
and therefore usually it shouldn't end with `RTS`.
|
||||||
|
|
||||||
|
A non-macro assembly function should end with `RTS`, `JMP`, or `BRA` as appropriate,
|
||||||
|
or it should be an external function.
|
||||||
|
|
||||||
|
For both macro and non-macro assembly functions,
|
||||||
|
the return type can be any valid return type, like for Millfork functions.
|
||||||
|
If the size of the return type is one byte,
|
||||||
|
then the result is passed via the B register.
|
||||||
|
If the size of the return type is two bytes,
|
||||||
|
then the result is passed via the D register.
|
||||||
|
|
||||||
|
### Assembly function parameters
|
||||||
|
|
||||||
|
An assembly function can have parameters.
|
||||||
|
They differ from what is used by Millfork functions.
|
||||||
|
|
||||||
|
Macro assembly functions can have the following parameter types:
|
||||||
|
|
||||||
|
* reference parameters: `byte ref paramname`: every occurrence of the parameter will be replaced with the variable given as an argument
|
||||||
|
|
||||||
|
* constant parameters: `byte const paramname`: every occurrence of the parameter will be replaced with the constant value given as an argument
|
||||||
|
|
||||||
|
For example, if you have:
|
||||||
|
|
||||||
|
macro asm void increase(byte ref v, byte const inc) {
|
||||||
|
LDB v
|
||||||
|
ADDB #inc
|
||||||
|
STB v
|
||||||
|
}
|
||||||
|
|
||||||
|
and call `increase(score, 10)`, the entire call will compile into:
|
||||||
|
|
||||||
|
LDB score
|
||||||
|
ADDB #10
|
||||||
|
STB score
|
||||||
|
|
||||||
|
Non-macro functions can only have their parameters passed via registers:
|
||||||
|
|
||||||
|
* `byte a`, `byte b`: a single byte passed via the given CPU register; any 1-byte type can be used
|
||||||
|
|
||||||
|
* `word d`, `word x`, `word y`: a 2-byte word byte passed via given 16-bit register; any 2-byte type can be used
|
||||||
|
|
||||||
|
Parameters passed via other registers (`U`, `S` etc.) or combinations of registers do not work yet.
|
||||||
|
|
||||||
|
**Work in progress**:
|
||||||
|
Only the following combinations of register parameters work reliably:
|
||||||
|
|
||||||
|
* zero or one register parameters
|
||||||
|
|
||||||
|
Macro assembly functions cannot have any parameter passed via registers.
|
||||||
|
|
||||||
|
## Safe assembly
|
||||||
|
|
||||||
|
**TODO**
|
@ -29,6 +29,7 @@ nav:
|
|||||||
- Functions: lang/functions.md
|
- Functions: lang/functions.md
|
||||||
- Inline 6502 assembly: lang/assembly.md
|
- Inline 6502 assembly: lang/assembly.md
|
||||||
- Inline 8080/LR35902/Z80 assembly: lang/assemblyz80.md
|
- Inline 8080/LR35902/Z80 assembly: lang/assemblyz80.md
|
||||||
|
- Inline 6809 assembly: lang/assembly6809.md
|
||||||
- Reentrancy guidelines: lang/reentrancy.md
|
- Reentrancy guidelines: lang/reentrancy.md
|
||||||
- List of keywords: lang/keywords.md
|
- List of keywords: lang/keywords.md
|
||||||
- Library reference:
|
- Library reference:
|
||||||
|
Loading…
x
Reference in New Issue
Block a user