; ; Oliver Schmidt, 2009-09-15 ; ; Startup code for cc65 (Apple2 version) ; .export done, return .export zpsave, rvsave, reset .export __STARTUP__ : absolute = 1 ; Mark as startup .import initlib, _exit .import zerobss, callmain .import bltu2 .import __ONCE_LOAD__, __ONCE_SIZE__ ; Linker generated .import __LC_START__, __LC_LAST__ ; 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 ; Save space by putting some of the start-up code in the ONCE segment, ; which can be re-used by the BSS segment, the heap and the C stack. jsr init ; Clear the BSS data. jsr zerobss ; Push the command-line arguments; and, call main(). jmp callmain ; ------------------------------------------------------------------------ .segment "ONCE" ; Save the zero-page locations that we need. init: ldx #zpspace-1 : lda c_sp,x sta zpsave,x dex bpl :- ; Save the original RESET vector. ldx #$02 : lda SOFTEV,x sta rvsave,x dex bpl :- ; Check for ProDOS. ldy $BF00 ; MLI call entry point cpy #$4C ; Is MLI present? (JMP opcode) php ; Remember whether we're running ProDOS 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 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 c_sp stx c_sp+1 ; 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 ; Call the module constructors. jsr initlib ; Copy the LC segment to its destination ; Switch in LC bank 2 for W/O. bit $C081 bit $C081 plp ; Are we running ProDOS? beq :+ ; Yes, no need to patch vectors lda #reset_6502 sta ROM_RST stx ROM_RST+1 lda #irq_6502 sta ROM_IRQ stx ROM_IRQ+1 : ; Set the source start address. ; Aka __LC_LOAD__ iff segment LC exists. lda #<(__ONCE_LOAD__ + __ONCE_SIZE__) ldy #>(__ONCE_LOAD__ + __ONCE_SIZE__) sta $9B sty $9C ; Set the source last address. ; Aka __LC_LOAD__ + __LC_SIZE__ iff segment LC exists. lda #<((__ONCE_LOAD__ + __ONCE_SIZE__) + (__LC_LAST__ - __LC_START__)) ldy #>((__ONCE_LOAD__ + __ONCE_SIZE__) + (__LC_LAST__ - __LC_START__)) sta $96 sty $97 ; Set the destination last address. ; Aka __LC_RUN__ + __LC_SIZE__ iff segment LC exists. 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 bltu2 ; Switch in LC bank 2 for R/O and return. bit $C080 rts ; ------------------------------------------------------------------------ .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 reset_6502: ; Used with DOS3.3 programs bit $C082 ; Switch in ROM jmp (ROM_RST) ; Jump to ROM's RESET vector irq_6502: ; Used with DOS3.3 programs bit $C082 ; Switch in ROM jmp (ROM_IRQ) ; Jump to ROM's IRQ/BRK vector ; ------------------------------------------------------------------------ .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 "INIT" zpsave: .res zpspace rvsave: .res 3