mirror of
https://github.com/italicsjenga/gba.git
synced 2025-01-10 02:51:31 +11:00
91 lines
1.8 KiB
ArmAsm
91 lines
1.8 KiB
ArmAsm
.arm
|
|
__start:
|
|
b .Linit
|
|
|
|
@ ROM header will be filled in by gbafix
|
|
.fill 188, 1, 0
|
|
|
|
.Linit:
|
|
@ Set address of user IRQ handler
|
|
ldr r0, =MainIrqHandler
|
|
ldr r1, =0x03FFFFFC
|
|
str r0, [r1]
|
|
|
|
@ set IRQ stack pointer
|
|
mov r0, #0x12
|
|
msr CPSR_cf, r0
|
|
ldr sp, =0x3007fa0
|
|
|
|
@ set user stack pointer
|
|
mov r0, #0x1f
|
|
msr CPSR_cf, r0
|
|
ldr sp, =0x3007f00
|
|
|
|
@ copy .data section to IWRAM
|
|
ldr r0, =__data_lma @ source address
|
|
ldr r1, =__data_start @ destination address
|
|
ldr r3, =__data_end
|
|
sub r2, r3, r1
|
|
beq .Lskip @ don't try to copy an empty .data section
|
|
add r2, #3
|
|
mov r2, r2, asr #2 @ length (in words)
|
|
add r2, #0x04000000 @ copy by words
|
|
swi 0xb0000
|
|
|
|
.Lskip:
|
|
@ jump to user code
|
|
ldr r0, =main
|
|
bx r0
|
|
|
|
.arm
|
|
.global MainIrqHandler
|
|
.align 4, 0
|
|
MainIrqHandler:
|
|
@ Load base I/O register address
|
|
mov r2, #0x04000000
|
|
add r2, r2, #0x200
|
|
|
|
@ Save IRQ stack pointer and IME
|
|
mrs r0, spsr
|
|
ldrh r1, [r2, #8]
|
|
stmdb sp!, {r0-r2,lr}
|
|
|
|
@ Disable all interrupts by writing to IME
|
|
mov r0, #0
|
|
strh r0, [r2, #8]
|
|
|
|
@ Acknowledge all received interrupts that were enabled in IE
|
|
ldr r3, [r2, #0]
|
|
and r0, r3, r3, lsr #16
|
|
strh r0, [r2, #2]
|
|
|
|
@ Switch to system mode
|
|
mrs r2, cpsr
|
|
bic r2, r2, #0x1F
|
|
orr r2, r2, #0x1F
|
|
msr cpsr_cf, r2
|
|
|
|
@ Jump to user specified IRQ handler
|
|
ldr r2, =__IRQ_HANDLER
|
|
ldr r1, [r2]
|
|
stmdb sp!, {lr}
|
|
adr lr, MainIrqHandler_Return
|
|
bx r1
|
|
MainIrqHandler_Return:
|
|
ldmia sp!, {lr}
|
|
|
|
@ Switch to IRQ mode
|
|
mrs r2, cpsr
|
|
bic r2, r2, #0x1F
|
|
orr r2, r2, #0x92
|
|
msr cpsr_cf, r2
|
|
|
|
@ Restore IRQ stack pointer and IME
|
|
ldmia sp!, {r0-r2,lr}
|
|
strh r1, [r2, #8]
|
|
msr spsr_cf, r0
|
|
|
|
@ Return to BIOS IRQ handler
|
|
bx lr
|
|
.pool
|