mirror of
https://github.com/cc65/cc65.git
synced 2025-01-07 13:29:45 +00:00
76a5a72403
Moving __cwd from BSS into INITBSS does of course ;-) not only impact the CBM targets but all targets with disk I/O support. Note: Code using `__cwd-1` may trigger an ld65 range error because __cwd may end up at the very begining of a segment. As far as I see this is an ld65 bug which I'm not try to fix - at least here.
211 lines
5.9 KiB
ArmAsm
211 lines
5.9 KiB
ArmAsm
;
|
|
; Oliver Schmidt, 2009-09-15
|
|
;
|
|
; Startup code for cc65 (Apple2 version)
|
|
;
|
|
|
|
.export _exit, done, return
|
|
.export __STARTUP__ : absolute = 1 ; Mark as startup
|
|
.import zerobss
|
|
.import initlib, donelib
|
|
.import callmain
|
|
.import __LC_START__, __LC_LAST__ ; Linker generated
|
|
.import __INIT_RUN__, __INIT_SIZE__ ; Linker generated
|
|
.import __INITBSS_RUN__ ; Linker generated
|
|
|
|
.include "zeropage.inc"
|
|
.include "apple2.inc"
|
|
|
|
.segment "STARTUP"
|
|
|
|
; ProDOS TechRefMan, chapter 5.2.1:
|
|
; "For maximum interrupt efficiency, a system program should not
|
|
; use more than the upper 3/4 of the stack."
|
|
ldx #$FF
|
|
txs ; Init stack pointer
|
|
|
|
; Switch in LC bank 2 for W/O.
|
|
bit $C081
|
|
bit $C081
|
|
|
|
; Set the source start address.
|
|
lda #<(__INITBSS_RUN__ + __INIT_SIZE__)
|
|
ldy #>(__INITBSS_RUN__ + __INIT_SIZE__)
|
|
sta $9B
|
|
sty $9C
|
|
|
|
; Set the source last address.
|
|
lda #<(__INITBSS_RUN__ + __INIT_SIZE__ + __LC_LAST__ - __LC_START__)
|
|
ldy #>(__INITBSS_RUN__ + __INIT_SIZE__ + __LC_LAST__ - __LC_START__)
|
|
sta $96
|
|
sty $97
|
|
|
|
; Set the destination last address.
|
|
lda #<__LC_LAST__
|
|
ldy #>__LC_LAST__
|
|
sta $94
|
|
sty $95
|
|
|
|
; Call into Applesoft Block Transfer Up -- which handles zero-
|
|
; sized blocks well -- to move the content of the LC memory area.
|
|
jsr $D39A ; BLTU2
|
|
|
|
; Set the source start address.
|
|
lda #<__INITBSS_RUN__
|
|
ldy #>__INITBSS_RUN__
|
|
sta $9B
|
|
sty $9C
|
|
|
|
; Set the source last address.
|
|
lda #<(__INITBSS_RUN__ + __INIT_SIZE__)
|
|
ldy #>(__INITBSS_RUN__ + __INIT_SIZE__)
|
|
sta $96
|
|
sty $97
|
|
|
|
; Set the destination last address.
|
|
lda #<(__INIT_RUN__ + __INIT_SIZE__)
|
|
ldy #>(__INIT_RUN__ + __INIT_SIZE__)
|
|
sta $94
|
|
sty $95
|
|
|
|
; Call into Applesoft Block Transfer Up -- which handles moving
|
|
; overlapping blocks upwards well -- to move the INIT segment.
|
|
jsr $D39A ; BLTU2
|
|
|
|
; Delegate all further processing, to keep the STARTUP segment small.
|
|
jsr init
|
|
|
|
; Avoid a re-entrance of donelib. This is also the exit() entry.
|
|
_exit: ldx #<exit
|
|
lda #>exit
|
|
jsr reset ; Setup RESET vector
|
|
|
|
; Switch in ROM, in case it wasn't already switched in by a RESET.
|
|
bit $C082
|
|
|
|
; Call the module destructors.
|
|
jsr donelib
|
|
|
|
; Restore the original RESET vector.
|
|
exit: ldx #$02
|
|
: lda rvsave,x
|
|
sta SOFTEV,x
|
|
dex
|
|
bpl :-
|
|
|
|
; Copy back the zero-page stuff.
|
|
ldx #zpspace-1
|
|
: lda zpsave,x
|
|
sta sp,x
|
|
dex
|
|
bpl :-
|
|
|
|
; ProDOS TechRefMan, chapter 5.2.1:
|
|
; "System programs should set the stack pointer to $FF at the
|
|
; warm-start entry point."
|
|
ldx #$FF
|
|
txs ; Re-init stack pointer
|
|
|
|
; We're done
|
|
jmp done
|
|
|
|
.segment "INIT"
|
|
|
|
; Save the zero-page locations that we need.
|
|
init: ldx #zpspace-1
|
|
: lda sp,x
|
|
sta zpsave,x
|
|
dex
|
|
bpl :-
|
|
|
|
; Clear the BSS data.
|
|
jsr zerobss
|
|
|
|
; Save the original RESET vector.
|
|
ldx #$02
|
|
: lda SOFTEV,x
|
|
sta rvsave,x
|
|
dex
|
|
bpl :-
|
|
|
|
; ProDOS TechRefMan, chapter 5.3.5:
|
|
; "Your system program should place in the RESET vector the
|
|
; address of a routine that ... closes the files."
|
|
ldx #<_exit
|
|
lda #>_exit
|
|
jsr reset ; Setup RESET vector
|
|
|
|
; Check for ProDOS.
|
|
ldy $BF00 ; MLI call entry point
|
|
cpy #$4C ; Is MLI present? (JMP opcode)
|
|
bne basic
|
|
|
|
; Check the ProDOS system bit map.
|
|
lda $BF6F ; Protection for pages $B8 - $BF
|
|
cmp #%00000001 ; Exactly system global page is protected
|
|
bne basic
|
|
|
|
; No BASIC.SYSTEM; so, quit to the ProDOS dispatcher instead.
|
|
lda #<quit
|
|
ldx #>quit
|
|
sta done+1
|
|
stx done+2
|
|
|
|
; No BASIC.SYSTEM; so, use the addr of the ProDOS system global page.
|
|
lda #<$BF00
|
|
ldx #>$BF00
|
|
bne :+ ; Branch always
|
|
|
|
; Get the highest available mem addr from the BASIC interpreter.
|
|
basic: lda HIMEM
|
|
ldx HIMEM+1
|
|
|
|
; Set up the C stack.
|
|
: sta sp
|
|
stx sp+1
|
|
|
|
; Call the module constructors.
|
|
jsr initlib
|
|
|
|
; Switch in LC bank 2 for R/O.
|
|
bit $C080
|
|
|
|
; Push the command-line arguments; and, call main().
|
|
jmp callmain
|
|
|
|
.code
|
|
|
|
; Set up the RESET vector.
|
|
reset: stx SOFTEV
|
|
sta SOFTEV+1
|
|
eor #$A5
|
|
sta PWREDUP
|
|
return: rts
|
|
|
|
; Quit to the ProDOS dispatcher.
|
|
quit: jsr $BF00 ; MLI call entry point
|
|
.byte $65 ; Quit
|
|
.word q_param
|
|
|
|
.rodata
|
|
|
|
; MLI parameter list for quit
|
|
q_param:.byte $04 ; param_count
|
|
.byte $00 ; quit_type
|
|
.word $0000 ; reserved
|
|
.byte $00 ; reserved
|
|
.word $0000 ; reserved
|
|
|
|
.data
|
|
|
|
; Final jump when we're done
|
|
done: jmp DOSWARM ; Potentially patched at runtime
|
|
|
|
.segment "INITBSS"
|
|
|
|
zpsave: .res zpspace
|
|
|
|
.bss
|
|
|
|
rvsave: .res 3
|