Hamazing/source/framework/framework_multipart.asm

214 lines
6.3 KiB
NASM

;--------------------------------------------------------------------
; Multipart support for trackmos
;
; In my former demo system I used a6 as a base pointer to the demo
; framework and a4 as a pointer to my demo part variables. That
; seemed a bit wasteful and so the new system uses one pointer for
; both framework AND part variables for the current part. This,
; however, means that the framework base needs to be relocated
; depending on the part because, of course, the size of the
; local variables may vary.
;
; You must start your local variable space with
;
; STRUCTURE PartData,fw_SIZEOF
;
; and end it with
;
; LABEL pd_SIZEOF
;
;--------------------------------------------------------------------
; Allocate the part variables
;
; The first thing each demo part needs to do is to call InitPart.
; I repeat: Start your code with:
; IFD FW_DEMO_PART
; move.l #pd_SIZEOF,d0
; CALLFW InitPart
; ENDC
;
; Otherwise, no part variables are available via a6 and no other
; framework calls can be made (the jumping table is not available).
;
; After this call, a6 will be pointing to the new framework base
; with all your variable space cleared.
;
; Make sure that you don't free the memory allocated here until
; the end of your part! The demo framework will call
; RestoreFrameworkBase automatically at exit of your part.
;
; Flushes the caches (if available), too.
;
; In : d0 = pd_SIZEOF (part data structure size including fw_SIZEOF)
; Out: a6 = new base pointer
; Trashes: everything except a5/a6
;
fw_InitPart:
PUTMSG 10,<"%d: InitPart(%ld)">,fw_FrameCounterLong(a6),d0
PUSHM d0
bsr fw_RestoreFrameworkBase
POPM
DISABLE_INTS
move.l fw_PartFwBase(a6),a0
move.l d0,fw_PartDataSize(a6)
add.l #-FWLVOPOS,d0
bsr fw_AllocFast
lea -FWLVOPOS(a0),a1
PUTMSG 40,<"LVOs from %p to %p">,a1,a0
move.l a6,a0
move.l a6,a3
move.l a1,a6
move.w #(fw_SIZEOF/4)-1,d7
PUTMSG 40,<"Copying %d LW from %p to %p">,d7,a0,a1
.copynew
move.l (a0)+,(a1)+
dbra d7,.copynew
move.l a6,fw_PartFwBase(a6)
move.l fw_PartDataSize(a6),d7
sub.w #fw_SIZEOF,d7
ble.b .skipclr
addq.w #3,d7
lsr.w #2,d7
beq.s .skipclr
subq.w #1,d7
moveq.l #0,d0
.clrnew
move.l d0,(a1)+
dbra d7,.clrnew
.skipclr
move.l a6,a0
lea fw_lvo_offsets(pc),a1
.lvoloop
move.w (a1)+,d0
beq.s .lvodone
lea -2(a1,d0.w),a2
move.l a2,-(a0)
move.w #%0100111011111001,-(a0) ; $4ef9 jmp x.L
bra.s .lvoloop
.lvodone
IF FW_MULTITASKING_SUPPORT
; The task list header is now broken after copying! We need to fix it.
lea fw_Tasks(a6),a1
bsr fw_RelocateList
bsr fw_FixA6BaseInTaskStacks
ENDC
lea fw_BasePtr(pc),a0
move.l a6,(a0)
bsr fw_FlushCaches
ENABLE_INTS
PUTMSG 20,<"%d: InitPart done %p">,fw_FrameCounterLong(a6),a6
rts
;--------------------------------------------------------------------
; Restore global framework base
;
; Removes the custom framework variable space and makes the global
; one active again.
;
; Out: a6 = new base pointer
; Trashes: everything except a5/a6
;
fw_RestoreFrameworkBase:
PUTMSG 10,<"%d: RestoreFrameworkBase(%p)">,fw_FrameCounterLong(a6),a6
DISABLE_INTS
move.l fw_PartFwBase(a6),a0
move.l fw_PrimaryFwBase(a6),a1
PUTMSG 40,<"Part %p, Primary %p">,a0,a1
cmp.l a1,a0
beq.s .noold
move.l a0,a3
PUTMSG 40,<"Restore from %p to %p">,a0,a1
move.l a1,a6
move.w #(fw_SIZEOF/4)-1,d7
.copyold
move.l (a0)+,(a1)+
dbra d7,.copyold
IF FW_MULTITASKING_SUPPORT
bsr fw_FixA6BaseInTaskStacks
ENDC
move.l a6,fw_PartFwBase(a6)
clr.l fw_PartDataSize(a6)
IF FW_MULTITASKING_SUPPORT
; The task list header is now broken after copying! We need to fix it.
lea fw_Tasks(a6),a1
bsr.s fw_RelocateList
ENDC
lea fw_BasePtr(pc),a0
move.l a6,(a0)
.noold
ENABLE_INTS
rts
;--------------------------------------------------------------------
; Helper function to relocate a doubly-linked list
;
; When relocating the framework base structure, we need to fix the
; contents of lists, otherwise BadThings(TM) will happen.
; This function goes through the relocated list and fixes the linkage.
;
; In: a1 = ListHead
; Trashes: a0/a1
;
IF FW_MULTITASKING_SUPPORT
fw_RelocateList:
move.l LH_HEAD(a1),a0
subq.l #4,a0
cmp.l LH_TAILPRED(a1),a0
bne.s .nonewlist
PUTMSG 10,<"Fixup empty list %p">,a1
move.l a1,LH_TAILPRED(a1) ; fix empty list
addq.l #4,a1
move.l a1,-(a1)
bra.s .listfixed
.nonewlist
PUTMSG 10,<"Fixup existing list %p">,a1
move.l a1,4+LN_PRED(a0) ; fixup LN_PRED of first node (points to LH_HEAD)
move.l LH_TAILPRED(a1),a0 ; get last node
addq.l #4,a1 ; now points to LH_TAIL
move.l a1,LN_SUCC(a0) ; fixup LN_SUCC of last node (pointing to LH_TAIL)
.listfixed
rts
ENDC
;--------------------------------------------------------------------
; Running tasks will have the old framework base stored in their
; register context. We need to fix these occurrences. Note that
; this is not completely fool-proof, but should work for 99% of the
; cases (there shouldn't be framework-relative things).
;
; In: a3 = old base
; a6 = new base
fw_FixA6BaseInTaskStacks:
lea fw_Tasks(a3),a1
.loop
TSTNODE a1,a1
beq.s .done
move.l ft_USP(a1),a0
lea ft_StackEnd(a1),a2
.findloop
cmp.l a0,a2
beq.s .loop
cmp.l (a0),a3
bne.s .notmatched
PUTMSG 10,<"Replacing base at %p">,a0
move.l a6,(a0)
.notmatched
addq.l #2,a0
bra.s .findloop
.done rts
fw_lvo_offsets:
FWGENLVOTABLE SET 1
include "../framework/framework_lvos.i"
dc.w 0
FWGENLVOTABLE SET 0