1
0
mirror of https://github.com/cc65/cc65.git synced 2025-01-12 02:30:44 +00:00
cc65/libsrc/atari/shadow_ram_prepare.s

380 lines
8.6 KiB
ArmAsm
Raw Normal View History

;
; Atari XL shadow RAM preparation routines
;
; Tasks:
; - move screen memory below load address
; - copy ROM chargen to its new place
; - copy shadow RAM contents to their destination
;
; Christian Groessler, chris@groessler.org, 2013
;
2013-06-25 00:48:43 +02:00
DEBUG = 1
2013-06-13 01:07:40 +02:00
.if .defined(__ATARIXL__)
2013-06-25 00:48:43 +02:00
.export sramprep
.import __SRPREP_LOAD__, __SRPREP_SIZE__
2013-06-25 00:48:43 +02:00
.import __SHADOW_RAM_LOAD__, __SHADOW_RAM_SIZE__
.import __SHADOW_RAM_RUN__
.import __CHARGEN_START__, __CHARGEN_SIZE__
.import __SAVEAREA_LOAD__
.include "zeropage.inc"
.include "atari.inc"
2013-06-25 00:48:43 +02:00
.include "save_area.inc"
.macro print_string text
2013-06-25 00:48:43 +02:00
.local start, cont
jmp cont
start: .byte text, ATEOL
cont: ldx #0 ; channel 0
lda #<start
sta ICBAL,x ; address
lda #>start
sta ICBAH,x
lda #<(cont - start)
sta ICBLL,x ; length
lda #>(cont - start)
sta ICBLH,x
lda #PUTCHR
sta ICCOM,x
jsr CIOV_org
.endmacro
; ------------------------------------------------------------------------
; EXE load chunk header
.segment "SRPREPHDR"
.word __SRPREP_LOAD__
2013-05-31 14:49:25 +02:00
.word __SRPREP_LOAD__ + __SRPREP_SIZE__ + __SHADOW_RAM_SIZE__ - 1
; ------------------------------------------------------------------------
; Actual code
.segment "SRPREP"
sramprep:
.ifdef DEBUG
2013-06-25 00:48:43 +02:00
print_string "entering stage #2"
.endif
; save values of modified system variables and ports
2013-06-25 00:48:43 +02:00
lda RAMTOP
sta RAMTOP_save
lda MEMTOP
sta MEMTOP_save
lda MEMTOP+1
sta MEMTOP_save+1
lda APPMHI
sta APPMHI_save
lda APPMHI+1
sta APPMHI_save+1
lda PORTB
sta PORTB_save
; disable BASIC
2013-06-25 00:48:43 +02:00
lda PORTB
ora #2
sta PORTB
2013-06-25 00:48:43 +02:00
.include "xlmemchk.inc" ; calculate lowest address used and new value for RAMTOP
2013-06-25 00:48:43 +02:00
ldx lowadr
stx MEMTOP
lda lowadr+1
sta MEMTOP+1
lda lodadr+1
sta RAMTOP
; set APPMHI to MEMLO (+ 1 for sanity)
lda MEMLO
clc
adc #1
sta APPMHI
lda MEMLO+1
adc #0
sta APPMHI+1
; issue a GRAPHICS 0 call (copied'n'pasted from TGI drivers) to move screen memory down
jsr findfreeiocb
.ifdef DEBUG ; only check in debug version, this shouldn't really happen(tm)
beq iocbok
print_string "Internal error, no free IOCB!"
jsr delay
jsr delay
jsr delay
jsr restore ; restore stuff we've changed
jmp (DOSVEC) ; abort loading
iocbok:
.endif
; Reopen it in Graphics 0
lda #OPEN
sta ICCOM,x
lda #OPNIN | OPNOT
sta ICAX1,x
lda #0
sta ICAX2,x
lda #<screen_device
sta ICBAL,x
lda #>screen_device
sta ICBAH,x
lda #<screen_device_length
sta ICBLL,x
lda #>screen_device_length
sta ICBLH,x
jsr CIOV_org
bpl scrok
; shouldn't happen(tm)
2013-06-25 00:48:43 +02:00
print_string "Internal error, aborting..."
jsr delay
jsr delay
jsr delay
jsr restore ; restore stuff we've changed
2013-06-25 00:48:43 +02:00
jmp (DOSVEC) ; abort loading
scrok: ; now close it again -- we don't need it anymore
lda #CLOSE
sta ICCOM,x
jsr CIOV_org
; copy chargen to low memory
2013-06-25 00:48:26 +02:00
.ifdef DEBUG
2013-06-25 00:48:43 +02:00
print_string "copy chargen to low memory"
2013-06-25 00:48:26 +02:00
.endif
2013-06-25 00:48:43 +02:00
lda #>(__SRPREP_LOAD__ + __SRPREP_SIZE__ + __SHADOW_RAM_SIZE__)
sta ptr3+1
lda #<(__SRPREP_LOAD__ + __SRPREP_SIZE__ + __SHADOW_RAM_SIZE__)
sta ptr3
beq cg_addr_ok
2013-06-25 00:48:43 +02:00
; page align the new chargen address
inc ptr3+1
lda #0
sta ptr3
cg_addr_ok:
lda ptr3+1
and #3
beq cg_addr_ok2
; align to next 1K boundary
lda ptr3+1
and #$fc
clc
adc #4
sta ptr3+1
cg_addr_ok2:
2013-06-25 00:48:43 +02:00
lda #<DCSORG
sta ptr1
lda #>DCSORG
sta ptr1+1
lda ptr3
sta ptr2
lda ptr3+1
pha ; needed later to set CHBAS/CHBASE
2013-06-25 00:48:43 +02:00
sta ptr2+1
lda #>__CHARGEN_SIZE__
sta tmp2
lda #<__CHARGEN_SIZE__
sta tmp1
2013-06-25 00:48:43 +02:00
jsr memcopy
.ifdef DEBUG
print_string "now setting up high memory"
.endif
; disable ROM
2013-06-25 00:48:43 +02:00
sei
ldx #0
stx NMIEN ; disable NMI
2013-06-25 00:48:43 +02:00
lda PORTB
and #$fe
tax
pla ; get temp. chargen address
sta WSYNC ; wait for horiz. retrace
stx PORTB ; now ROM is mapped out
; switch to temporary chargen
sta CHBASE
sta CHBAS
; copy shadow RAM contents to their destination
2013-06-25 00:48:43 +02:00
lda #<__SHADOW_RAM_SIZE__
bne do_copy
lda #>__SHADOW_RAM_SIZE__
beq no_copy ; we have no shadow RAM contents
2013-06-25 00:48:43 +02:00
; ptr1 - src; ptr2 - dest; tmp1, tmp2 - len
do_copy:lda #<__SHADOW_RAM_LOAD__
sta ptr1
lda #>__SHADOW_RAM_LOAD__
sta ptr1+1
lda #<__SHADOW_RAM_RUN__
sta ptr2
lda #>__SHADOW_RAM_RUN__
sta ptr2+1
lda #<__SHADOW_RAM_SIZE__
sta tmp1
lda #>__SHADOW_RAM_SIZE__
sta tmp2
jsr memcopy
no_copy:
2013-06-25 00:48:26 +02:00
; copy chargen to its new (final) location
2013-06-25 00:48:43 +02:00
lda ptr3
sta ptr1
lda ptr3+1
sta ptr1+1
lda #<__CHARGEN_START__
sta ptr2
lda #>__CHARGEN_START__
sta ptr2+1
lda #>__CHARGEN_SIZE__
sta tmp2
lda #<__CHARGEN_SIZE__
sta tmp1
jsr memcopy
; re-enable ROM
2013-06-25 00:48:43 +02:00
lda PORTB
ora #1
ldx #>DCSORG
sta WSYNC ; wait for horiz. retrace
2013-06-25 00:48:43 +02:00
sta PORTB
stx CHBASE
stx CHBAS
2013-06-25 00:48:43 +02:00
lda #$40
sta NMIEN ; enable VB again
cli ; and enable IRQs
2013-06-25 00:48:26 +02:00
.ifdef DEBUG
2013-06-25 00:48:43 +02:00
print_string "Stage #2 OK"
print_string "loading main chunk"
2013-06-25 00:48:43 +02:00
jsr delay
2013-06-25 00:48:26 +02:00
.endif
rts
.include "findfreeiocb.inc"
; routine taken from http://www.obelisk.demon.co.uk/6502/algorithms.html
;
; copy memory
; ptr1 - source
; ptr2 - destination
; tmp2:tmp1 - len
2013-06-25 00:48:43 +02:00
.proc memcopy
ldy #0
ldx tmp2
beq last
pagecp: lda (ptr1),y
sta (ptr2),y
iny
bne pagecp
inc ptr1+1
inc ptr2+1
dex
bne pagecp
last: cpy tmp1
beq done
lda (ptr1),y
sta (ptr2),y
iny
bne last
done: rts
.endproc
; clean up after a fatal error
restore:lda RAMTOP_save
sta RAMTOP
lda MEMTOP_save
sta MEMTOP
lda MEMTOP_save+1
sta MEMTOP+1
lda APPMHI_save
sta APPMHI
lda APPMHI_save+1
sta APPMHI+1
rts
.ifdef DEBUG
.byte "HERE ****************** HERE ***************>>>>>>"
sramsize:
2013-06-25 00:48:43 +02:00
.word __SHADOW_RAM_SIZE__
; short delay
2013-06-25 00:48:43 +02:00
.proc delay
lda #10
l: jsr delay1
clc
sbc #0
bne l
rts
delay1: ldx #0
ldy #0
loop: dey
bne loop
dex
bne loop
rts
.endproc
.endif ; .ifdef DEBUG
2013-06-25 00:48:43 +02:00
screen_device: .byte "S:",0
screen_device_length = * - screen_device
.ifdef DEBUG
2013-06-25 00:48:43 +02:00
.byte " ** srprep ** end-->"
.endif
; ------------------------------------------------------------------------
; Provide an empty SHADOW_RAM segment in order that the linker is happy
; if the user program doesn't have a SHADOW_RAM segment.
.segment "SHADOW_RAM"
; ------------------------------------------------------------------------
; EXE load chunk "trailer" - sets INITAD
.segment "SRPREPTRL"
.word INITAD
.word INITAD+1
.word __SRPREP_LOAD__
2013-06-25 00:48:43 +02:00
.endif ; .if .defined(__ATARIXL__)