-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathhelper_functions.asm
185 lines (170 loc) · 4.85 KB
/
helper_functions.asm
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
;
; Title: Helper Functions
; Author: Dean Belfield, Reinhard Schu
; Created: 06/11/2022
; Last Updated: 20/12/2022
;
include "helper_macros.inc"
; Print a zero-terminated string
; IX: pointer to string
; destroys: IX, A
;
PRSTR: LD A,(IX)
OR A
RET Z ; finish if zero
PRT_CHR
INC IX
JR PRSTR ; loop around to next byte
; Print a 24-bit HEX number
; HLU: Number to print
;
Print_Hex24: PUSH.LIL HL
LD.LIL HL, 2
ADD.LIL HL, SP
LD.LIL A, (HL)
POP.LIL HL
CALL Print_Hex8
; Print a 16-bit HEX number
; HL: Number to print
;
Print_Hex16: LD A,H
CALL Print_Hex8
LD A,L
; Print an 8-bit HEX number
; A: Number to print
;
Print_Hex8: LD C,A
RRA
RRA
RRA
RRA
CALL prtnbl
LD A,C
prtnbl: AND 0Fh
ADD A,90h
DAA
ADC A,40h
DAA
PRT_CHR
RET
; Skip whitespaces
; HLU: Pointer in string buffer
;
SKIPSP: LD.LIL A, (HL)
CP ' '
RET NZ
INC.LIL HL
JR SKIPSP
; Read next argument from buffer
; Inputs: HL: Pointer in string buffer
; DE: Pointer to Buffer where to copy argument
; Outputs: HL: Updated pointer in string buffer
; DE: Pointer to Buffer argument was copied to (preserved from input)
; F: Carry set if argument read, otherwise reset
; Destroys: A,H,L,F;
;
READ_ARG: CALL SKIPSP ; Skip whitespace
LD.LIL A,(HL) ; Fetch first character
OR A ; Check for end of string
RET Z ; Return with no carry if not
PUSH DE ; Preserve DE
READ_ARG1: LD.LIL A,(HL) ; Fetch the character
OR A ; Check for end of string
JR Z,READ_ARG4 ; end of argument, finish
CP ' ' ; if whitespace
JR Z,READ_ARG4 ; end of argument, finish
LD (DE),A ; store the character
INC.LIL HL ; next charcater
INC DE
JR READ_ARG1
READ_ARG4: XOR A
LD (DE),A ; terminate output buffer
POP DE ; restore DE
SCF ; We have a valid argument so set carry
RET
; Read a number aqs a string and convert to binary (int)
; If prefixed with $, will read as hex, otherwise decimal
; Inputs: HL: Pointer in string buffer
; Outputs: DE: Value (24-bit)
; F: Zero reset if valid number, otherwise set
; Destroys: A,D,E,H,L,F
;
AtoI: PUSH.LIL IX ; Preserve IX
PUSH.LIL BC ; Preserve BC
LD IX,_AtoI_flags ; start assuming invalid number,
RES 0,(IX) ; so reset "valid number" flag
LD A,(HL) ; Read first character
OR A ; Check for end of string
JR Z, _AtoI_end ; Return if no input
LD.LIL DE,0 ; Initialise DE
CP '$' ; Is it prefixed with '$' (HEX number)?
JR NZ, _AtoI3 ; Jump to decimal parser if not
INC HL ; Otherwise fall through to ASC_TO_HEX
;
_AtoI1: LD A,(HL) ; Fetch the character
OR A ; Check for end of string
JR Z,_AtoI_end
CALL UPPRC ; Convert to uppercase
SUB '0' ; Normalise to 0
JR C, _AtoI_invalid ; Return if < ASCII '0' (out of range)
CP 10 ; Check if >= 10
JR C, _AtoI2 ; No, so skip next bit
SUB 7 ; Adjust ASCII A-F to nibble
CP 16 ; Check for > F
JR NC, _AtoI_invalid ; Return if out of range
;
_AtoI2: SET 0,(IX) ; set "valid number" flag
PUSH HL ; Stack HL
LD HL, DE
ADD.LIL HL, HL ; RXS: times 4? why not shift left?
ADD.LIL HL, HL
ADD.LIL HL, HL
ADD.LIL HL, HL
LD HL, DE
POP HL ; Restore HL
OR E ; OR the new digit in to the least significant nibble
LD E, A
;
INC HL ; Onto the next character
JR _AtoI1 ; And loop
;
_AtoI3: LD A, (HL) ; Fetch the character
OR A ; Check for end of string
JR Z,_AtoI_end
SUB '0' ; Normalise to 0
JR C, _AtoI_invalid ; Return if < ASCII '0'
CP 10 ; Check if >= 10
JR NC, _AtoI_invalid ; Return if >= 10
SET 0,(IX) ; set "valid number" flag
PUSH HL ; Stack HL
LD HL, DE
PUSH.LIL HL ; LD BC, HL
POP.LIL BC
ADD.LIL HL, HL ; x 2
ADD.LIL HL, HL ; x 4
ADD.LIL HL, BC ; x 5
ADD.LIL HL, HL ; x 10
LD.LIL BC, 0
LD C, A ; LD BCU, A
ADD.LIL HL, BC ; Add BCU to HL
LD DE, HL
POP HL ; Restore HL
;
INC HL ; Onto the next character
JR _AtoI3 ; And loop
_AtoI_invalid: RES 0,(IX) ; reset "valid number" flag
_AtoI_end: POP.LIL BC ; recover BC
BIT 0,(IX) ; transfer "valid number" bit to Z flag
POP.LIL IX ; recover IX
RET
_AtoI_flags: DB 0 ; storage for flags
; Convert a character to upper case
; A: Character to convert
;
UPPRC: AND 7Fh
CP 'a' ; check range
RET C
CP '{' ; check range
RET NC
AND 5Fh ; Convert to upper case
RET