collapseos/apps/zasm/main.asm

250 lines
5.6 KiB
NASM
Raw Normal View History

2019-05-11 10:32:05 +10:00
; zasm
;
; Reads input from specified blkdev ID, assemble the binary in two passes and
; spit the result in another specified blkdev ID.
;
; We don't buffer the whole source in memory, so we need our input blkdev to
; support Seek so we can read the file a second time. So, for input, we need
; GetC and Seek.
;
; For output, we only need PutC. Output doesn't start until the second pass.
;
; The goal of the second pass is to assign values to all symbols so that we
; can have forward references (instructions referencing a label that happens
; later).
;
; Labels and constants are both treated the same way, that is, they can be
; forward-referenced in instructions. ".equ" directives, however, are evaluated
; during the first pass so forward references are not allowed.
;
2019-05-10 04:09:40 +10:00
; *** Requirements ***
; blockdev
; strncmp
; addDE
; addHL
; upcase
; unsetZ
; intoDE
; intoHL
; findchar
; blkSel
; fsFindFN
; fsOpen
; fsGetC
; fsSeek
; fsTell
2019-05-10 11:21:08 +10:00
; RAMSTART (where we put our variables in RAM)
2019-05-17 11:15:00 +10:00
; FS_HANDLE_SIZE
2019-05-01 05:51:39 +10:00
2019-05-11 10:32:05 +10:00
; *** Variables ***
; A bool flag indicating that we're on first pass. When we are, we don't care
; about actual output, but only about the length of each upcode. This means
; that when we parse instructions and directive that error out because of a
; missing symbol, we don't error out and just write down a dummy value.
2019-05-16 10:07:21 +10:00
.equ ZASM_FIRST_PASS RAMSTART
; whether we're in "local pass", that is, in local label scanning mode. During
; this special pass, ZASM_FIRST_PASS will also be set so that the rest of the
; code behaves as is we were in the first pass.
.equ ZASM_LOCAL_PASS ZASM_FIRST_PASS+1
; What IO_PC was when we started our context
2019-05-17 11:15:00 +10:00
.equ ZASM_CTX_PC ZASM_LOCAL_PASS+1
2019-05-16 10:07:21 +10:00
.equ ZASM_RAMEND ZASM_CTX_PC+2
2019-05-11 10:32:05 +10:00
; *** Code ***
jp zasmMain
2019-05-18 00:00:30 +10:00
#include "util_z.asm"
2019-05-11 10:32:05 +10:00
.equ IO_RAMSTART ZASM_RAMEND
#include "io.asm"
2019-05-11 11:19:34 +10:00
#include "tok.asm"
#include "parse_z.asm"
2019-05-15 05:26:29 +10:00
#include "expr.asm"
#include "instr.asm"
2019-05-12 12:11:05 +10:00
.equ DIREC_RAMSTART IO_RAMEND
#include "directive.asm"
2019-05-12 12:11:05 +10:00
.equ SYM_RAMSTART DIREC_RAMEND
#include "symbol.asm"
; Read file through blockdev ID in H and outputs its upcodes through blockdev
; ID in L.
2019-05-11 10:32:05 +10:00
zasmMain:
2019-05-14 10:23:10 +10:00
; Init I/O
ld a, h
ld de, IO_IN_GETC
call blkSel
ld a, l
ld de, IO_OUT_GETC
call blkSel
2019-05-16 10:07:21 +10:00
2019-05-14 10:23:10 +10:00
; Init modules
2019-05-16 10:07:21 +10:00
xor a
ld (ZASM_LOCAL_PASS), a
call ioInit
2019-05-14 10:23:10 +10:00
call symInit
2019-05-11 10:32:05 +10:00
; First pass
ld a, 1
ld (ZASM_FIRST_PASS), a
call zasmParseFile
ret nz
2019-05-11 10:32:05 +10:00
; Second pass
2019-05-17 11:15:00 +10:00
call ioRewind
2019-05-11 10:32:05 +10:00
xor a
ld (ZASM_FIRST_PASS), a
call zasmParseFile
ret
; Sets Z according to whether we're in first pass.
zasmIsFirstPass:
ld a, (ZASM_FIRST_PASS)
cp 1
2019-05-01 05:51:39 +10:00
ret
2019-05-16 10:07:21 +10:00
; Sets Z according to whether we're in local pass.
zasmIsLocalPass:
ld a, (ZASM_LOCAL_PASS)
cp 1
ret
; Repeatedly reads lines from IO, assemble them and spit the binary code in
; IO. Z is set on success, unset on error. DE contains the last line number to
; be read (first line is 1).
2019-05-11 10:32:05 +10:00
zasmParseFile:
call ioResetPC
2019-05-11 10:32:05 +10:00
.loop:
call parseLine
ret nz ; error
ld a, b ; TOK_*
cp TOK_EOF
2019-05-16 10:07:21 +10:00
jr z, .eof
jr .loop
.eof:
call zasmIsLocalPass
jr nz, .end ; EOF and not local pass
; we're in local pass and EOF. Unwind this
call _endLocalPass
2019-05-11 10:32:05 +10:00
jr .loop
2019-05-16 10:07:21 +10:00
.end:
cp a ; ensure Z
ret
2019-05-11 10:32:05 +10:00
; Parse next token and accompanying args (when relevant) in I/O, write the
; resulting opcode(s) through ioPutC and increases (IO_PC) by the number of
; bytes written. BC is set to the result of the call to tokenize.
; Sets Z if parse was successful, unset if there was an error. EOF is not an
; error.
2019-05-01 05:51:39 +10:00
parseLine:
call tokenize
2019-05-01 12:27:11 +10:00
ld a, b ; TOK_*
2019-05-01 11:40:22 +10:00
cp TOK_INSTR
2019-05-16 05:16:35 +10:00
jp z, _parseInstr
2019-05-02 01:26:41 +10:00
cp TOK_DIRECTIVE
2019-05-16 05:16:35 +10:00
jp z, _parseDirec
2019-05-10 11:21:08 +10:00
cp TOK_LABEL
jr z, _parseLabel
cp TOK_EOF
ret ; Z is correct. If EOF, Z is set and not an
2019-05-15 04:32:12 +10:00
; error, otherwise, it means bad token and
; errors out.
_parseInstr:
2019-05-01 12:27:11 +10:00
ld a, c ; I_*
2019-05-01 11:40:22 +10:00
call parseInstruction
2019-05-01 05:51:39 +10:00
or a ; is zero?
jr z, .error
2019-05-11 10:32:05 +10:00
ld b, a ; save output byte count
2019-05-02 01:26:41 +10:00
ld hl, instrUpcode
.loopInstr:
ld a, (hl)
call ioPutC
inc hl
djnz .loopInstr
2019-05-15 04:32:12 +10:00
; continue to success
.success:
xor a ; ensure Z
ret
.error:
call unsetZ
2019-05-15 04:32:12 +10:00
ret
_parseDirec:
2019-05-02 01:26:41 +10:00
ld a, c ; D_*
call parseDirective
2019-05-18 05:14:38 +10:00
cp a ; ensure Z
2019-05-15 04:32:12 +10:00
ret
_parseLabel:
2019-05-10 11:21:08 +10:00
; The string in (scratchpad) is a label with its trailing ':' removed.
ld hl, scratchpad
2019-05-16 10:07:21 +10:00
call zasmIsLocalPass
jr z, .processLocalPass
; Is this a local label? If yes, we don't process it in the context of
; parseLine, whether it's first or second pass. Local labels are only
; parsed during the Local Pass
call symIsLabelLocal
jr z, .success ; local? don't do anything.
2019-05-14 10:23:10 +10:00
call zasmIsFirstPass
jr z, .registerLabel ; When we encounter a label in the first
; pass, we register it in the symbol
; list
2019-05-16 10:07:21 +10:00
; At this point, we're in second pass, we've encountered a global label
; and we'll soon continue processing our file. However, before we do
; that, we should process our local labels.
call _beginLocalPass
jr .success
.processLocalPass:
2019-05-14 10:23:10 +10:00
call symIsLabelLocal
2019-05-16 10:07:21 +10:00
jr z, .registerLabel ; local label? all good, register it
; normally
; not a local label? Then we need to end local pass
call _endLocalPass
jr .success
2019-05-14 10:23:10 +10:00
.registerLabel:
ld de, (IO_PC)
2019-05-10 11:21:08 +10:00
call symRegister
2019-05-14 10:23:10 +10:00
jr nz, .error
; continue to .success
.success:
xor a ; ensure Z
2019-05-15 04:32:12 +10:00
ret
2019-05-01 05:51:39 +10:00
.error:
call unsetZ
2019-05-01 05:51:39 +10:00
ret
2019-05-16 10:07:21 +10:00
_beginLocalPass:
; remember were I/O was
2019-05-17 11:15:00 +10:00
call ioSavePos
2019-05-16 10:07:21 +10:00
; Remember where PC was
ld hl, (IO_PC)
2019-05-16 10:07:21 +10:00
ld (ZASM_CTX_PC), hl
; Fake first pass
ld a, 1
ld (ZASM_FIRST_PASS), a
; Set local pass
ld (ZASM_LOCAL_PASS), a
; Empty local label registry
xor a
ld (SYM_LOC_NAMES), a
call symSelectLocalRegistry
ret
_endLocalPass:
call symSelectGlobalRegistry
; recall I/O pos
2019-05-17 11:15:00 +10:00
call ioRecallPos
2019-05-16 10:07:21 +10:00
; recall PC
ld hl, (ZASM_CTX_PC)
ld (IO_PC), hl
2019-05-16 10:07:21 +10:00
; unfake first pass
xor a
ld (ZASM_FIRST_PASS), a
; Unset local pass
ld (ZASM_LOCAL_PASS), a
cp a ; ensure Z
ret