-
Notifications
You must be signed in to change notification settings - Fork 13
/
Copy pathtrs80m2diag.asm
832 lines (701 loc) · 20.2 KB
/
trs80m2diag.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
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
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
; code: language=asm-collection tabSize=8
; configuration defines:
; Whether to continue to DRAM testing after finding VRAM error. This can be used
; to work on DRAM if you know the VRAM is bad, but you don't have replacement chips
; on hand and want to test DRAM chips too. By default the released builds stop
; if there is a VRAM error and try to display a character test pattern on the screen.
CONTINUE_ON_VRAM_ERROR = 0
; For debugging in an emulator, you can choose a page number to simulate a bit error
; while testing.
SIMULATE_ERROR = 0
; SIMULATE_ERROR = $80
; SIMULATE_ERROR = $3C
VBASE equ $F800
VSIZE equ $0800
VLINE equ 80
.include "inc/z80.mac"
.include "inc/spt.mac"
.include "inc/m2.inc"
; Notes on global register allocation:
;
; This ROM doesn't work like typical Z80 code, which assumes the presence of a stack.
; There may in fact be no working memory in the machine for holding the stack.
;
; An overall goal for this version of the code is to run in the absence of any working
; ram at all. There is no stack and no RAM variables. The only storage of variables
; is in registers, so the registers must be carefully preserved.
;
; Without a stack, that means either saving them to other registers and restoring before
; jumping to other code (remembering there can be no CALLs when there is no stack)
; or avoiding their use altogether. These are extremely confining restrictions.
;
; Assembly purists will shudder at the extensive use of macros, but for sanity it
; cannot be avoided.
;
; Globally, the contents of these registers must be preserved
; e = bit errors in the region of memory currently being tested
; ix = current location in VRAM for printing messages
; iy = current table entry for test parameters
;
.org 0000h ; z80 boot code starts at location 0
reset:
di ; mask INT
im 1
init_crtc:
ld bc,$0FFC ; count $0F, port $FC crtc address reg
ld hl,crtc_setup_table+crtc_setup_len-1
.loop: ld a,(hl) ; fetch bytes from setup table and send top to bottom
out (c),b ; CRTC address register
out ($FD),a ; CRTC data register
dec hl
dec b
jp p,.loop
test_vram:
SPTHREAD_BEGIN ; set up to begin running threaded code
dw vram_map ; map in VRAM so we can print results
;;;; test the VRAM while lighting/clicking the floppy drive
dw spt_call, sptc_fdc_reset_head
dw spt_select_test, tp_vram
dw memtestmarch ; test the VRAM
dw fdc_select_none
dw spt_jp_nc, .vram_ok
;;;; the VRAM tests bad. Report and loop
dw chartest_fullscreen
.vram_bad:
dw spt_call, sptc_blink_biterrs
dw spt_jp, .vram_bad
.vram_ok:
;;;; clear the screen, print charset at bottom, report the results of the VRAM test
dw con_clear
dw spt_con_index,10
dw spt_con_print, msg_banner ; print the banner
dw spt_call, sptc_print_charset
dw spt_select_test, tp_vram
dw spt_call, sptc_announcetest ; print results of VRAM tst
dw spt_con_print, msg_testok
dw con_clear_kbd
.test_4000:
;;;; Test DRAM. Start by testint the bank to which we are going to relocate some code
dw spt_select_test, tp_rdest ; load the test parameters
dw spt_call, sptc_announcetest ; announce what test we are about to run
dw tp_map_bank ; map in the bank to test (this unmaps VRAM)
dw spt_call, sptc_bank_test_and_report
dw spt_jp_nc, .test_0000
;;;; The bank at $4000 is bad, so report this and skip the relocation test
dw spt_select_test, tp_low ; announce that we're skipping the relocating test
dw spt_call, sptc_announcetest ;
dw spt_con_print, msg_skipped ; we can't run the low test
dw spt_jp, .test_banked
.test_0000:
;;;; The bank at $4000 is ok, so run the relocation test
;;;; relocate the memory test into upper half of low RAM, unmap ROM, run test on lowest RAM
dw relocate_memtest
dw spt_select_test, tp_low_reloc ; select the low mem test next
dw spt_call, sptc_announcetest
dw spt_call, sptc_relocated_test
dw spt_call, sptc_bank_report
.test_banked:
;;;; Start loopin through the rest of the banks, testing each one
dw spt_select_test, tp_high ; load the test table
.bank_loop:
;;;; top of the loop for each bank
dw check_boot_keys
;;;; check to see if auto-booting has been cancelled errors or ESC
dw spt_jp_boot_cancelled,.bank_test ; don't try to boot if there are errors
dw spt_jp_fdc_ready,spt_boot_fd ; boot from the floppy if it is ready
.bank_test:
;;;; run one bank test
dw spt_call, sptc_announcetest ; announce what test we are about to run
dw tp_map_bank
dw spt_jp_bank_dup, .bank_dup
dw spt_call, sptc_bank_test_and_report
.bank_cont:
;;;; Finished testing this bank. Determine what to do next.
dw spt_tp_next, .table_done ; start over if we've reached the end
; dw spt_jp_fdc_ready,spt_boot_fd ; boot from the floppy if it is ready
dw spt_jp, .bank_loop ; else test the next bank
.bank_dup:
;;;; we've detected a duplicate bank (bank mapped in more than one place)
dw vram_map ; in this case, VRAM was not remapped, do it now
dw spt_con_print, msg_dup
dw spt_call, sptc_printbank
dw spt_jp, .bank_cont
.table_done:
;;;; reached the end of the table
dw spt_jp_boot_cancelled,.test_4000 ; don't try to boot if there are errors
dw spt_jp_hd_present,spt_boot_hd ; boot from the HD if it's present
dw spt_jp, .test_4000
;; -------------------------------------------------------------------------------------------------
;; end of main program.
check_boot_keys:
in a,(nmi_status_reg) ; see if there is a key available
bit nmi_status_bit_kbd_int,a
ret z ; just return if no key pressed
in a,(kbd_data_reg)
cp $1B ; press ESC for floppy
jp z,boot_fd0
cp $03 ; press BREAK for floppy
jp z,boot_fd0
cp '1' ; press 1 for hard disk
jp z,boot_hd0
cp ' ' ; space to cancel booting and loop tests
ret nz ; just continue if no match
ld a,(VBASE)
cp ' '
jr z,.cancel
ld a,' '
ld (VBASE),a
ret
.cancel:
ld a,'!' ; mark to skip booting
ld (VBASE),a
ret
spt_boot_hd:
; dw spt_call,spt_print_boot_msg
; ; MAC_SPT_CON_GOTO 1,35
; ; dw spt_con_print, msg_boot
; dw spt_con_print, msg_hd
dw boot_hd0
spt_boot_fd:
; dw spt_call,spt_print_boot_msg
; ; MAC_SPT_CON_GOTO 1,35
; ; dw spt_con_print, msg_boot
; dw spt_con_print, msg_fd
dw boot_fd0
sptc_bank_test_and_report:
dw memtestmarch ; test the current bank
sptc_bank_report:
dw mark_bank_map_vram
dw spt_jp_nc, .bank_ok
dw spt_call, sptc_print_errsmsg
dw print_biterrs
dw spt_exit
.bank_ok:
dw spt_con_print, msg_testok
dw spt_exit
spt_jp_bank_dup:
; determine if this bank is a different, already tested bank
ex af,af' ; save old flags
ld h,(iy+TP_BASE) ; get the base address of ram
ld l,15 ; plus 15 bytes
ld bc,15 ; compare bytes at offset 15...1 (not 0)
ld a,$55 ; test that the bank is filled with $55
.test55:
cpd ; compare a,(hl) ; hl-- ; bc--
jr nz,.done ; if mem != $55, this isn't a tested bank, so quit
jp pe,.test55 ; if we have more to test, loop
ld a,(hl) ; HL should now be the base address
cp (iy+TP_BANK) ; compare to current bank number
jr c,.dup ; mem < bank; found a duplicate bank number
.done:
pop hl ; discard the error address
ex af,af'
ret
.dup: ; found a match. Jump to the specified SPT location
ld e,a ; report the error as the bank we found
ex af,af'
pop hl
ld sp,hl
ret
spt_jp_boot_cancelled:
pop hl ; get the address to jump to if there are errors
ld a,(VBASE)
cp ' ' ; if there is no exclamation point
ret z ; continue
ld sp,hl ; else go to new thread location
ret
mark_bank_map_vram:
ex af,af' ; save flags
ld h,(iy+TP_BASE) ; get this bank's base in HL
ld l,0
ld b,(iy+TP_BANK) ; get the bank number in b
ld (hl),b ; mark the base address with the bank number
ld a,b ; get the bank number into A
or $80 ; enable VRAM
out ($FF),a ; map in VRAM and the current bank
ex af,af' ; restore the saved flags
ret nc ; return if the test had no error
; ex af,af' ; save flags
ld a,$ff ; see if this is a an absent bank
cp e
jr z,.absent ; if absent, don't record it as an errored bank
ld a,'!' ; not absent, but bad; note this with the exclamation
ld (VBASE),a ; put an exclamation in the corner if this bank is bad
.absent:
; ex af,af' ; restore the saved flags
scf ; set the carry flag again
ret
ld_a_e:
ld a,e
ret
sptc_printbank:
dw ld_a_e
dw con_printh
dw spt_con_print,msg_space
dw spt_exit
ld_d_0:
ld d,0
ret
spt_dec_d_jp_nz:
pop hl
dec d
ret z
ld sp,hl
ret
spt_jp_fdc_ready:
pop hl
ld a,fdc_sel_side_0+fdc_sel_dr_0
out (fdc_select_reg),a
in a,(fdc_status_reg)
bit 7,a
ret nz ; return if not ready
ld sp,hl
ret
spt_jp_hd_present:
; reset HDC and check for drive presence
xor a
out (hdc_control_reg),a
ld a,hdc_control_soft_reset
out (hdc_control_reg),a
; Earlier versions of the boot ROM didn't enable interrupt and DMA. Not sure
; why this is a good idea.
ld a,hdc_control_deven+hdc_control_wait_enable+hdc_control_intrq_enable+hdc_control_dma_enable
out (hdc_control_reg),a
in a,(hdc_drive_id_45) ; check ID of drive 4
and $0f
pop hl
ret z ; return if not present
ld sp,hl
ret
sptc_fdc_terminate_ready_timeout:
dw ld_d_0 ; wait up to 256 times, then fail out
.loop: dw fdc_terminate_cmd
dw spt_jp_fdc_ready,.done
dw spt_pause,$0500
dw spt_dec_d_jp_nz,.loop
.done: dw spt_exit
; terminate_fdc_cmd:
fdc_terminate_cmd:
.start: ld a,fdc_cmd_force_int+fdc_cmd_force_int_immediate
out (fdc_cmd_reg),a
ld a,fdc_cmd_force_int
out (fdc_cmd_reg),a
ld b,11
.delay: djnz .delay ; delay by 138 T-states (34.5ms@4MHz)
in a,(fdc_data_reg) ; to reset DRQ, presumably
in a,(fdc_status_reg)
ret
fdc_select_none:
ld a,fdc_sel_side_0|fdc_sel_dr_none
jr fdc_select
fdc_select_d0s0:
ld a,fdc_sel_side_0|fdc_sel_dr_0 ; select d0s0
fdc_select:
out (fdc_select_reg),a
ld b,0 ; short delay for FDC to respond
.dly1: djnz .dly1
ret
; fdc_deselect:
; ld a,fdc_sel_dr_none ; deselect the drive
; out (fdc_select_reg),a
; ret
fdc_step_in_5:
ld c,5
fdc_step_in:
.silp: ld a,fdc_cmd_step_in|fdc_cmd_update_track|fdc_cmd_head_load|fdc_cmd_step_rate_15ms
out (fdc_cmd_reg),a
ld b,0 ; short delay for FDC to respond
.dly1: djnz .dly1
.wrdy1: in a,(fdc_status_reg) ; wait for ready indication
bit 0,a
; and $81
jr nz,.wrdy1
dec c
jr nz,.silp ; repeate the stepping
ret
fdc_head_restore:
ld a,fdc_cmd_restore|fdc_cmd_head_load|fdc_cmd_step_rate_15ms
out (fdc_cmd_reg),a ; restore head to track zero
ret
sptc_fdc_reset_head:
dw fdc_select_d0s0
; dw fdc_terminate_cmd ; reset FDC
dw spt_call, sptc_fdc_terminate_ready_timeout
dw fdc_step_in_5
dw fdc_head_restore
; dw spt_pause,1599
dw spt_pause,2000
dw spt_exit
; fdc_head_unload:
; ld a,fdc_cmd_restore|fdc_cmd_head_load|fdc_cmd_step_rate_15ms
; out (fdc_cmd_reg),a ; restore head to track zero
; ret
; sptc_fdc_release_head:
; dw
; test if the e register matches 7-bit vram and jump to spt address if match
spt_jp_e_zero:
pop hl ; get the address for jumping if match
ld a,0 ; test clean
cp e ;
ret nz ; return without jump if there is NOT a match
ld sp,hl ; else jump to the requested location
ret
spt_jp_e_ff:
pop hl ; get the address for jumping if match
ld a,$FF ; test all bits err
cp e ;
ret nz ; return without jump if there is NOT a match
ld sp,hl ; else jump to the requested location
ret
; load the label string address from the current test parameter table entry into hl
ld_hl_tp_label:
ld c,(iy+TP_LABEL)
ld b,0
ld hl,labels_start
add hl,bc
ret
ld_hl_tp_base:
ld h,(iy+TP_BASE)
ld l,0
ret
ld_bc_tp_size:
ld b,(iy+TP_SIZE)
ld c,0
ret
ld_a_tp_bank:
ld a,(iy+TP_BANK)
ret
tp_map_bank:
ld a,(iy+TP_BANK)
cp 0 ; special case: when we say bank 0, we really mean 1
jr nz,.send
ld a,1 ; substitute a 1 when actually banking 0
.send:
out ($FF),a
ret
spt_tp_goto: ld a,(iy+TP_POS)
ld ixl,a
ld a,(iy+TP_POS+1)
ld ixh,a
ret
; move to the next test parameter table entry
spt_tp_next: pop hl ; get the address to jump to if we are starting over
ld bc,tp_entrysize ; find the next entry
add iy,bc
ld a,(iy+TP_SIZE) ; is the length zero?
or a
ret nz ; no, use it
ld c,(iy+TP_GOTO) ; yes, get the address of the first entry
ld b,(iy+TP_GOTO+1)
ld iy,0
add iy,bc
ld sp,hl ; jump to the next location
ret
sptc_announcetest:
dw spt_tp_goto
dw ld_hl_tp_label
dw con_print
dw ld_a_tp_bank
dw con_printh
dw spt_con_print, msg_space
dw spt_call, sptc_tp_print_range
dw spt_con_print, msg_testing
dw spt_con_index, -status_backup
dw spt_exit
ld_a_h: ld a,h
ret
ld_a_l: ld a,l
ret
add_hl_bc: add hl,bc
ret
dec_hl: dec hl
ret
sptc_tp_print_range:
dw ld_hl_tp_base
dw ld_a_h
dw con_printx
dw ld_a_l
dw con_printx
dw spt_con_print, msg_dash
dw ld_hl_tp_base
dw ld_bc_tp_size
dw add_hl_bc
dw dec_hl
dw ld_a_h
dw con_printx
dw ld_a_l
dw con_printx
; dw spt_con_print, msg_space
dw spt_exit
spt_pause: ; pause by an amount specified in BC
pop bc
pause_bc: ; pause by BC*50-5+14 t-states
ex af,af'
.loop:
nop12
nop12
dec bc
ld a,b
or c
jr nz,.loop
ex af,af'
ret
sptc_print_errsmsg:
dw spt_jp_e_ff,.absent
dw spt_con_print,msg_biterrs
dw spt_exit
.absent:
dw spt_con_print,msg_absent
dw spt_exit
print_biterrs:
ld a,'7'
ld b,8
.showbit:
rlc e
jr nc,.zero
ld (ix+0),a
jr .cont
.zero:
ld (ix+0),'.'
.cont:
inc ix
dec a
djnz .showbit
ret
rlc_e:
rlc e
ret
spt_blink_bit_rl:
; rlc e
SPTHREAD_ENTER
dw rlc_e
; dw m2_drivelight_on
dw fdc_select_d0s0
dw spt_jp_c, .long
dw spt_pause, $2000
dw spt_jp, .off
.long:
dw spt_pause, $FFFF
.off:
; dw m2_drivelight_off
dw fdc_select_none
dw spt_pause, $8000
dw spt_exit
sptc_blink_biterrs:
dw spt_blink_bit_rl
dw spt_blink_bit_rl
dw spt_blink_bit_rl
dw spt_blink_bit_rl
dw spt_blink_bit_rl
dw spt_blink_bit_rl
dw spt_blink_bit_rl
dw spt_blink_bit_rl
dw spt_pause, $00
; dw spt_pause, $00
; dw spt_pause, $00
dw pause_bc
dw pause_bc
dw spt_exit
; ld_a_0:
; ld a,0
; ret
sptc_print_charset:
; MAC_SPT_CON_GOTO 20,-8
; dw ld_a_0
dw spt_charset_64_p16_start
dw spt_charset_64_p16
dw spt_charset_64_p16
dw spt_charset_64_p16
dw spt_exit
spt_charset_64_p16_start:
ld hl,VBASE+(VLINE*20)-8
xor a
spt_charset_64_p16:
ld bc,$10
add hl,bc
ld bc,$40
; ld bc,$10
; add ix,bc
; ld bc,$40
jr charset_here
chartest_fullscreen:
ld hl,VBASE
; ld ix,VBASE
ld bc,VSIZE
charset_here:
.charloop:
ld (hl),a
; ld (ix+0),a ; copy A to byte pointed by HL
inc a ; increments A
; inc ix
cpi ; increments HL, decrements BC (and does a CP)
jp pe, .charloop
ret
; V_END = (VBASE+VSIZE-1)
; msg_boot: dbz "Booting "
; msg_hd: dbz "HD"
; msg_fd: dbz "FD"
labels_start:
label_vram: dbz " 2K VRAM "
label_dram16: dbz "16K DRAM "
label_bank16: dbz "16K page "
msg_dash: dbz "-"
msg_space: dbz " "
msg_banner: dbiz "TRS-80 M2 Test ROM - Frank IZ8DWF / Dave KI3V / Adrian Black"
msg_testok: dbz " --OK-- "
; msg_reloc: dbz "(reloc) "
msg_skipped: dbz " *skip* "
msg_biterrs: dbz " errors:"
msg_absent: dbz " absent:"
msg_dup: dbz " DUP "
msg_testing: db " "
dbi " TEST "
dbz " "
; msg_testing: db " ", " "+$80, "t"+$80, "e"+$80, "s"+$80, "t"+$80, " "+$80, " ", 0
status_backup equ $-msg_testing-1
; test parameter table. 2-byte entries:
; 1. size of test in bytes
; 2. starting address
; 3. bank to map before test
; 4. location in screen memory to start printing test data
; 5. address of string for announcing test
; tp_entrysize equ 10
tp_entrysize equ tp_low-tp_vram
COL1 = 2
COL2 = (COL1+40)
TP_SIZE equ 0
TP_BASE equ 1
TP_BANK equ 2
TP_LABEL equ 3
TP_POS equ 4
TP_GOTO equ 1
memtest_ld_bc_size .macro
ld b,(iy+TP_SIZE)
ld c,a
.endm
memtest_ld_hl_base .macro
ld h,(iy+TP_BASE)
ld l,a
.endm
memtest_loadregs .macro
xor a
memtest_ld_bc_size
memtest_ld_hl_base
.endm
tp_vram: db high VSIZE, high VBASE, $0, label_vram-labels_start
dw VBASE+( 2*VLINE)+COL1
tp_low: db $40, $00, $0, label_dram16-labels_start
dw VBASE+( 3*VLINE)+COL1
tp_rdest: db $40, $40, $0, label_dram16-labels_start
dw VBASE+( 4*VLINE)+COL1
tp_high: db $40, $80, $1, label_bank16-labels_start
dw VBASE+( 5*VLINE)+COL1
db $40, $C0, $1, label_bank16-labels_start
dw VBASE+( 6*VLINE)+COL1
db $40, $80, $2, label_bank16-labels_start
dw VBASE+( 7*VLINE)+COL1
db $40, $C0, $2, label_bank16-labels_start
dw VBASE+( 8*VLINE)+COL1
db $40, $80, $3, label_bank16-labels_start
dw VBASE+( 9*VLINE)+COL1
db $40, $C0, $3, label_bank16-labels_start
dw VBASE+(10*VLINE)+COL1
db $40, $80, $4, label_bank16-labels_start
dw VBASE+(11*VLINE)+COL1
db $40, $C0, $4, label_bank16-labels_start
dw VBASE+(12*VLINE)+COL1
db $40, $80, $5, label_bank16-labels_start
dw VBASE+(13*VLINE)+COL1
db $40, $C0, $5, label_bank16-labels_start
dw VBASE+(14*VLINE)+COL1
db $40, $80, $6, label_bank16-labels_start
dw VBASE+(15*VLINE)+COL1
db $40, $C0, $6, label_bank16-labels_start
dw VBASE+(16*VLINE)+COL1
db $40, $80, $7, label_bank16-labels_start
dw VBASE+(17*VLINE)+COL1
db $40, $C0, $7, label_bank16-labels_start
dw VBASE+(18*VLINE)+COL1
db $40, $80, $8, label_bank16-labels_start
dw VBASE+( 3*VLINE)+COL2
db $40, $C0, $8, label_bank16-labels_start
dw VBASE+( 4*VLINE)+COL2
db $40, $80, $9, label_bank16-labels_start
dw VBASE+( 5*VLINE)+COL2
db $40, $C0, $9, label_bank16-labels_start
dw VBASE+( 6*VLINE)+COL2
db $40, $80, $A, label_bank16-labels_start
dw VBASE+( 7*VLINE)+COL2
db $40, $C0, $A, label_bank16-labels_start
dw VBASE+( 8*VLINE)+COL2
db $40, $80, $B, label_bank16-labels_start
dw VBASE+( 9*VLINE)+COL2
db $40, $C0, $B, label_bank16-labels_start
dw VBASE+(10*VLINE)+COL2
db $40, $80, $C, label_bank16-labels_start
dw VBASE+(11*VLINE)+COL2
db $40, $C0, $C, label_bank16-labels_start
dw VBASE+(12*VLINE)+COL2
db $40, $80, $D, label_bank16-labels_start
dw VBASE+(13*VLINE)+COL2
db $40, $C0, $D, label_bank16-labels_start
dw VBASE+(14*VLINE)+COL2
db $40, $80, $E, label_bank16-labels_start
dw VBASE+(15*VLINE)+COL2
db $40, $C0, $E, label_bank16-labels_start
dw VBASE+(16*VLINE)+COL2
db $40, $80, $F, label_bank16-labels_start
dw VBASE+(17*VLINE)+COL2
db $40, $C0, $F, label_bank16-labels_start
dw VBASE+(18*VLINE)+COL2
db $00
dw tp_high
relocate_memtest:
ld de,reloc_dst_begin
ld hl,reloc_src_begin
ld bc,reloc_size
ldir
ret
; ----------------------------------------------------------------------------
; Relocated tests:
; The code below is assembled to be relocated to $4100. It contains threaded
; code which cannot be written in a position-independent manner (consisting
; mostly of absolute subroutine addresses). So we use the assembler's
; .phase directive to tell it to assemble for operation when moved to $4000.
; This code should unmap the ROM from the region starting at $0000, perform
; the memory test, then remap the ROM and return execution there via the
; normal mechanism. The rest of ROM will not be visible during this, so no
; calls back into it can happen until the ROM is remapped, and no data
; from it can be seen either (hence an extra copy of the label).
reloc_src_begin:
.phase $4100
reloc_dst_begin:
sptc_relocated_test:
dw rom_unmap
dw relocated_memtest
dw rom_map
dw spt_exit
rom_unmap:
xor a
out ($F9),a
ret
rom_map:
ld a,1
out ($F9),a
ret
tp_low_reloc: db $40, $00, $0, label_dram16-labels_start
dw VBASE+( 3*VLINE)+COL1
relocated_memtest:
.dephase
.include "inc/memtestmarch.asm"
reloc_size equ relocated_memtest - reloc_dst_begin + $ - memtestmarch
; End of relocated section.
; (But note, the code will place a copy of the memtestmarch routine right
; after the above code. That routine is written in position-independent code
; aka relocatable code in Z80 terminology. We need this RAM copy while
; testing the RAM that is mapped behind the ROM at location $0000.)
; ----------------------------------------------------------------------------
.include "inc/trs80m2fdcboot.asm"
.include "inc/spt.asm"
.include "inc/trs80m2con.asm"
; include "inc/trs80m2music.asm"