-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathlibrary.asm
More file actions
501 lines (489 loc) · 14.4 KB
/
library.asm
File metadata and controls
501 lines (489 loc) · 14.4 KB
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
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
; This file is part of Programator.
;
; Programator is free software: you can redistribute it and/or
; modify it under the terms of the GNU General Public License as
; published by the Free Software Foundation, either version 3 of the
; License, or (at your option) any later version.
;
; Programator is distributed in the hope that it will be useful,
; but WITHOUT ANY WARRANTY; without even the implied warranty of
; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
; General Public License for more details.
;
; You should have received a copy of the GNU General Public License
; along with Programator. If not, see <https://www.gnu.org/licenses/>.
;
; Copyright (c) 2022, 2024, 2025 Aleksander Mazur
;
; Funkcje pomocnicze, głównie do obsługi wejścia (poleceń)
;-----------------------------------------------------------
; Wczytuje 2 opcjonalne argumenty z linii poleceń - 2 liczby szesnastkowe.
; Pierwszą do R4:R5, drugą do R2:R3.
; Rejestry te powinny być zainicjowane domyślnymi wartościami na odwrót,
; tj. domyślna wartość pierwszej w R2:R3, a drugiej w R4:R5.
; R1 = adres znaku przed liczbą (spodziewana spacja)
; R0 = adres pierwszego miejsca za linią poleceń
; Nie wraca, jeśli wystąpi błąd.
get_2_hex_numbers:
; wczytujemy argument 1 do R2:R3
acall get_hex_arg
; zamiana R2:R3 z R4:R5
mov A, R3
xch A, R5
mov R3, A
mov A, R2
xch A, R4
mov R2, A
; wczytujemy argument 2 do R2:R3
acall get_hex_arg
; nie powinno być żadnych więcej argumentów
;ajmp ensure_no_args
;-----------------------------------------------------------
; Wraca tylko jeśli polecenie nie ma argumentów.
; W przeciwnym razie wypisuje błąd i wraca do pętli głównej.
; Niszczy A, C
ensure_no_args:
acall get_args_len
jz ret1
error_extarg:
mov DPTR, #s_error_extarg
ajmp print_error_then_prompt
;-----------------------------------------------------------
; Wywołuje get_2_hex_numbers i subtract_address_range
get_address_range:
acall get_2_hex_numbers
;sjmp subtract_address_range
;-----------------------------------------------------------
; Odejmuje adres początkowy (R4:R5) od końcowego (R2:R3);
; jeśli wynik jest ujemny -> wypisuje błąd i nie wraca;
; jeśli wynik jest prawidłowy, nadpisuje adres końcowy liczbą bajtów
; w zakresie od R4:R5 do R2:R3 włącznie, tj. R2:R3 = R2:R3 - R4:R5 + 1.
; Niszczy A, C
subtract_address_range:
clr C
mov A, R3
subb A, R5
mov R3, A
mov A, R2
subb A, R4
mov R2, A
jc error_illopt_fwd
mov A, R3
add A, #1
mov R3, A
mov A, R2
addc A, #0
mov R2, A
ret
error_illopt_fwd:
ajmp error_illopt
;-----------------------------------------------------------
; Oblicza długość argumentów w linii poleceń, tj.
; A = R0 - R1
; Niszczy C
get_args_len:
mov A, R0
clr C
subb A, R1
ret1:
ret
;-----------------------------------------------------------
; Dekoduje cyfrę szesnastkową podaną w ASCII
; A = znak ASCII
; Zwraca C=0 i wartość cyfry w A, lub C=1 jeśli to nie cyfra (wtedy A trzyma starą wartość!)
convert_hex_digit:
cjne A, #'0', convert_hex_digit2
convert_hex_digit2:
jc ret4 ; A < '0' -> błąd
cjne A, #'9' + 1, convert_hex_digit3
convert_hex_digit3:
jc convert_hex_digit_09 ; '0' <= A < '9' + 1 -> 0..9
cjne A, #'A', convert_hex_digit4
convert_hex_digit4:
jc ret4 ; '9' + 1 <= A < 'A' -> błąd
cjne A, #'F' + 1, convert_hex_digit5
convert_hex_digit5:
jc convert_hex_digit_AF ; 'A' <= A < 'F' + 1 -> A..F
; A >= 'F' + 1 -> błąd
setb C
ret4:
ret
convert_hex_digit_AF:
add A, #'0' - 'A' + 10
convert_hex_digit_09:
add A, #0 - '0'
clr C
ret
if USE_SPI
;-----------------------------------------------------------
; Konwertuje liczbę w A (< 100) na BCD
; Niszczy B
convert_to_bcd:
mov B, #10
div AB
swap A
orl A, B
ret
endif
;-----------------------------------------------------------
; Wypisuje rekordy Hex dla obszaru pamięci o długości R2:R3 od R4:R5
; (R2:R3=0000 oznacza pełne 64KB).
; Niszczy A, B, C, R0
; Uaktualnia R4/R5, zeruje R2/R3
; DPTR = callback dostający w R7 potrzebną ilość bajtów od adresu R4:R5
; i zwracający przez R0 adres, gdzie te dane wczytał. C=0 gdy sukces.
; W razie błędu callback może przerwać zrzut zwracając C=1 i liczbę
; pozostałych do zrzucenia (nieudanych) bajtów w R7, a w DPTR komunikat błędu.
; Nie może zniszczyć R2, R3, R4, R5, B, a jeśli zwraca C=0, to też DPTR.
dump_hex_file:
mov A, R2
orl A, R3
jz dump_hex_file_loop_limit ; 0 -> 64KB
dump_hex_file_loop:
; R4:R5 = bieżący adres zrzutu (aktualizowane przez dump_hex)
; R2:R3 = ile bajtów pozostało zrzucić
mov A, R2
jnz dump_hex_file_loop_limit ; zostało > 255 bajtów
mov A, R3
jz dump_hex_file_finish
cjne A, #input_end - input, dump_hex_file_loop2
dump_hex_file_loop2:
jc dump_hex_file_loop_all ; zostało < 20 bajtów
dump_hex_file_loop_limit:
mov A, #input_end - input ; nie więcej niż 20h bajtów naraz w jednej linii
dump_hex_file_loop_all:
mov B, A ; w B przechowujemy długość na potem (R7 zostanie wyzerowany przez dump_hex)
mov R7, A
; wywołujemy callback, który ustawi nam R0
clr A
acall jmp_dptr
jnc dump_hex_file_loop_ok
; procedura z callbacka napotkała błąd - w R7 jest liczba bajtów, których nie udało się pobrać
mov A, B
clr C
subb A, R7 ; A = liczba bajtów, które udało się pobrać
jnz dump_hex_file_part
dump_hex_file_error:
; w DPTR musi już być komunikat błędu
ajmp print_error_then_prompt
dump_hex_file_part:
mov B, A
setb C
dump_hex_file_loop_ok:
push PSW ; zachowujemy flagę C - jeśli jest tam 1, to po wypisaniu rekordu idziemy do print_error_then_prompt
mov R7, B
mov R6, #0 ; rekord typu 0
acall dump_hex_rec
; R2:R3 -= B
mov A, R3
clr C
subb A, B
mov R3, A
jnc dump_hex_file_no_carry1
dec R2
dump_hex_file_no_carry1:
; R4:R5 += B
mov A, R5
add A, B
mov R5, A
jnc dump_hex_file_no_carry2
inc R4
dump_hex_file_no_carry2:
pop PSW
jnc dump_hex_file_loop
; w DPTR musi już być komunikat błędu
ajmp print_error_then_prompt
dump_hex_file_finish:
; koniec zrzutu - jeszcze tylko rekord typu 01 (:00000001FF)
clr A
mov R4, A
mov R5, A
mov R7, A
inc A
mov R6, A
; R0 jest nieważny gdy długość (R7) = 0
;-----------------------------------------------------------
; Wypisuje linię (rekord) Intel Hex.
; R0 = adres danych do zrzucenia w pamięci
; R4:R5 = adres danych źródłowych (do wypisania)
; R6 = typ rekordu
; R7 = długość zrzutu
; Niszczy A, C, R6
; Zeruje R7, aktualizuje R0
dump_hex_rec:
; :llaaaattdddddd...ddxx
acall uart_send_colon
mov A, R7
acall uart_send_hex_byte
mov A, R4
acall uart_send_hex_byte
mov A, R5
acall uart_send_hex_byte
mov A, R6
acall uart_send_hex_byte
; teraz w R6 będziemy liczyć sumę kontrolną
mov A, R7
add A, R4
add A, R5
add A, R6
mov R6, A
mov A, R7
jz dump_hex_checksum
dump_hex_loop:
mov A, @R0
acall uart_send_hex_byte
mov A, @R0
add A, R6
mov R6, A
inc R0
djnz R7, dump_hex_loop
dump_hex_checksum:
mov A, R6
cpl A
inc A
; A = "two’s complement"
acall uart_send_hex_byte
; możnaby tak, ale zniszczylibyśmy DPTR
;mov DPTR, #s_enter
;ajmp uart_send_rom
uart_send_crlf:
mov A, #13
acall uart_send_char
mov A, #10
ajmp uart_send_char
;-----------------------------------------------------------
; Wczytuje rekord Intel Hex do #input
; Jeśli uda się wczytać dane z rekordu, zwraca C=0 i:
; - rozmiar wczytanych danych w R7
; - adres początkowy w R4:R5
; - początek bufora w RAM w R0 (czyli #input)
; W razie błędu zwraca C=1 i niezerowy kod błędu (znak do wypisania) w A.
; Jeśli napotkaliśmy rekord końcowy, to danych nie ma, ale jest sukces
; - wówczas zwraca C=1 i A=0. Należy wtedy wrócić do pętli głównej programu.
; W innych przypadkach można wołać niniejszą funkcję jeszcze raz - ona
; zignoruje wszystko do następnego dwukropka i będzie próbowała
; zinterpretować następny rekord.
; Kody błędów:
; "H–Invalid Intel Hex record format: Intel Hex record contains a nonhex character."
; "L–Invalid Intel Hex record length: Intel Hex record length exceeds
; allowable length [20 bytes (type 0); 0 bytes (type 1 EOF)]."
; "S–Invalid checksum in Intel Hex record: Intel Hex record contains a
; checksum that does not correspond to its hex record. This error is
; caused by manual edits to the Intel Hex file or a compiler error."
; "R–Invalid Intel Hex record type: ROM loader only accepts Intel Hex
; record types 00 and 01 in standard Intel Hex format; make sure the
; assembler/compiler is not configured for Intel Extended Hex or HEX-386 format."
; Niszczy A, C, R0, R2, R3, R4, R5, R6, R7
receive_hex_rec:
; "All characters are discarded before the header character <:> is read."
; "All characters following the record checksum and prior to the next <:> are discarded."
acall uart_receive_char
cjne A, #':', receive_hex_rec
; :llaaaattdddddd...ddxx
acall receive_hex_byte
jc receive_hex_rec_error_H ; nie hex
cjne A, #input_end - input + 1, receive_hex_rec2
receive_hex_rec2:
jnc receive_hex_rec_error_L ; za długi rekord
mov R3, A ; R3 = długość rekordu (na potrzeby pętli poniżej)
mov R7, A ; R7 = długość rekordu (do zwrócenia)
acall receive_hex_byte
jc receive_hex_rec_error_H ; nie hex
mov R4, A ; R4 = starszy bajt adresu początkowego
acall receive_hex_byte
jc receive_hex_rec_error_H ; nie hex
mov R5, A ; R5 = młodszy bajt adresu początkowego
acall receive_hex_byte
jc receive_hex_rec_error_H ; nie hex
; A = typ rekordu (0 - dane, 1 - koniec)
mov R6, A
; w R6 policzymy sobie sumę kontrolną
add A, R7
add A, R4
add A, R5
xch A, R6
jz receive_hex_rec_payload
cjne A, #1, receive_hex_rec_error_R
receive_hex_rec_check_sum:
; wszystkie bajty odebrane, teraz powinna być suma kontrolna
acall receive_hex_byte
jc receive_hex_rec_error_H ; nie hex
add A, R6
setb C
jz ret7 ; suma OK? to C=1 i A=0
mov A, #'S'
ret7:
ret
receive_hex_rec_payload:
mov R0, #input ; R0 = pozycja w buforze
mov A, R7
jz receive_hex_rec_checksum
receive_hex_rec_loop:
acall receive_hex_byte
jc receive_hex_rec_error_H ; nie hex
mov @R0, A
add A, R6
mov R6, A ; aktualizacja sumy kontrolnej
inc R0
djnz R3, receive_hex_rec_loop
receive_hex_rec_checksum:
acall receive_hex_rec_check_sum
jnz ret9
; rekord wczytany poprawnie
mov R0, #input
clr C
ret9:
ret
receive_hex_rec_error_H:
; C=1
mov A, #'H'
ret
receive_hex_rec_error_L:
setb C
mov A, #'L'
ret
receive_hex_rec_error_R:
setb C
mov A, #'R'
ret
;-----------------------------------------------------------
; Wczytuje dwucyfrową liczbę szesnastkową jako bajt
; Zwraca C=0 i liczbę w A, albo C=1, jeśli napotkano nieprawidłowe znaki.
; Niszczy A, C, R2
receive_hex_byte:
acall uart_receive_char
acall convert_hex_digit
jc ret8
swap A
mov R2, A
acall uart_receive_char
acall convert_hex_digit
orl A, R2
ret8:
ret
;-----------------------------------------------------------
; Uogólniona procedura przetwarzania rekordów Intel Hex
; DPTR = callback dostający każdorazowo R7 bajtów (>0) zaczynając od R0
; do wpisania pod adres R4:R5
; Callback musi zwrócić znak ACK/NAK w A.
; Niezależnie od kodu (sukcesu/porażki) o ewentualnej kontynuacji
; decyduje użytkownik po drugiej stronie portu szeregowego.
; Chyba, że callback zwróci A=0 - wtedy funkcja przerywa działanie bez ACK/NAK.
load_hex_file:
acall ensure_no_args
load_hex_file_loop:
acall receive_hex_rec
jc load_hex_rec_check
; pod R0 jest R7 bajtów do zapisu pod adres R4:R5
; (adres z R4:R5 jest w dziedzinie specyficznej dla callbacka)
mov A, R7
jz load_hex_rec_empty ; zabezpieczenie przed rekordem o zerowej długości
clr A
acall jmp_dptr
jz ret8
load_hex_file_error:
; w A jest znak ACK/NAK
acall uart_send_char
sjmp load_hex_file_loop
load_hex_rec_empty:
mov A, #'G'
sjmp load_hex_file_error
load_hex_rec_check:
; C=1 i A=0 -> EOF; wypisujemy G i wychodzimy
jnz load_hex_file_error
mov A, #'G'
ajmp uart_send_char
;-----------------------------------------------------------
; Dekoduje liczbę szesnastkową podaną w ASCII do rejestrów R2:R3.
; R1 = adres znaku przed liczbą (spodziewana spacja)
; R0 = adres pierwszego miejsca za argumentem
; Jeśli nie ma żadnego argumentu (R0==R1), zwraca C=1 nie ruszając R2/R3.
; Jeśli argument jest, ale zły, to funkcja nie wraca.
; W przeciwnym razie zwraca C=0 i liczbę w R2:R3 oraz przesuwa R1
; na pierwszą pozycję za zdekodowaną liczbę.
; Niszczy A, B, C, R7
get_hex_arg:
acall get_args_len
setb C
jz ret3 ; brak argumentu
mov R7, A ; R7 = długość tekstu, której nie możemy przekroczyć
; pierwszy znak powinien być spacją
cjne @R1, #' ', error_notspc
djnz R7, get_hex_arg_ok
; za spacją nic nie ma
error_extspc:
mov DPTR, #s_error_extspc
ajmp print_error_then_prompt
get_hex_arg_ok:
inc R1
; zerujemy R2:R3, bo będziemy or'ować
clr A
mov R2, A
mov R3, A
get_hex_arg_loop:
mov A, @R1
cjne A, #' ', get_hex_arg_loop_nospc
; wychodzimy z R1 wskazującym na spację, C=0 w wyniku cjne
; TODO: jeśli to jest druga spacja zaraz za pierwszą (bez żadnej cyfry), to powinien być błąd E:NOTHEX
ret3:
ret
get_hex_arg_loop_nospc:
acall convert_hex_digit
jc error_nothex
inc R1 ; przechodzimy za właśnie przetworzoną cyfrę
mov B, A ; B = wczytana cyfra Y (0-F)
; R2:R3 = UV:WX -> VW:XY
mov A, R3 ; A = WX
swap A ; A = XW
push ACC
anl A, #0F0h ; A = X0
orl A, B ; A = XY
mov R3, A ; R3 = XY
mov A, R2 ; A = UV
swap A ; A = VU
anl A, #0F0h ; A = V0
mov B, A ; B = V0
pop ACC ; A = XW
anl A, #0Fh ; A = 0W
orl A, B ; A = VW
mov R2, A ; R2 = VW
djnz R7, get_hex_arg_loop
clr C
ret
error_notspc:
mov DPTR, #s_error_notspc
ajmp print_error_then_prompt
error_nothex:
mov DPTR, #s_error_nothex
ajmp print_error_then_prompt
if ICP51_W79EX051 or USE_1WIRE or USE_I2C
;-----------------------------------------------------------
; Dekoduje dwucyfrową liczbę szesnastkową podaną w ASCII i zwraca C=0,
; albo - jeśli na pierwszej pozycji jest inny znak niż cyfra szesnastkowa
; - zwraca kod tego znaku i C=1. Jeśli R1==R0, zwraca C=1 i A=0.
; Jeśli zamiast drugiej cyfry jest nieprawidłowy znak, to nie wraca.
; R1 = adres pierwszego znaku do zdekodowania
; R0 = adres pierwszego miejsca za argumentem
; Niszczy A, C, R7, przesuwa R1 za zdekodowane znaki
get_hex_or_char:
acall get_args_len
setb C
jz ret5 ; koniec argumentów
mov A, @R1
inc R1
acall convert_hex_digit
jc ret5 ; nie-cyfra
swap A
mov R7, A
acall get_args_len
jz error_argreq_fwd ; brak drugiej cyfry
mov A, @R1
inc R1
acall convert_hex_digit
jc error_nothex ; drugi znak to nie cyfra
orl A, R7
ret5:
ret
error_argreq_fwd:
ajmp error_argreq
endif