Mercurial > hg > Members > kono > nitros9-code
view level2/modules/kernel/fmove.asm @ 1282:c1f7d686924e
Now back-ported to OS-9 Level One and assembled from there.
author | boisy |
---|---|
date | Wed, 27 Aug 2003 23:47:59 +0000 |
parents | 90bad6d8388c |
children | 1da8ab9ba433 |
line wrap: on
line source
* F$Move entry point * Entry: U=Register stack pointer IFEQ H6309 L0A01 clrb rts ENDC * A = source task # * B = destination task # * Y = bytes to move * X = source pointer * U = destination pointer FMove ldd R$D,u get source & destination task #'s L0B25 ldy R$Y,u Get # bytes to move beq L0A01 None, exit without error ldx R$X,u get source pointer ldu R$U,u get destination pointer L0B2C pshs d,x,y,u preserve it all pshs d,y save task #'s & byte count tfr a,b copy source task to B lbsr L0BF5 calculate block offset of source leay a,u point to block pshs x,y save source pointer & DAT pointer of source ldb 9,s get destination task # ldx 14,s get destination pointer lbsr L0BF5 calculate block offset leay a,u point to block pshs x,y save dest. pointer & DAT pointer to dest. * try ldq #$20002000/ subr x,w / pshsw (+3), take out ldd (-3) ldd #DAT.BlSz get block size subd ,s take off offset pshs d preserve ldd #DAT.BlSz init offset in block subd 6,s pshs d save distance to end?? ldx 8,s get source pointer leax -$6000,x make X point to where we'll map block ($a000) ldu 4,s get destination pointer leau -$4000,u make U point to where we'll map block ($c000) ldy <D.SysDAT Get ptr to system DAT image lda 11,y Get MMU block #5 ldb 13,y Get MMU block #6 tfr d,y Move to Y since unused in loop below * Main move loop * Stack: 0,s=distance to end of source block * 2,s=distance to end of destination block * 4,s=pointer to destination * 6,s=pointer to destination DAT image * 8,s=pointer to source * 10,s=pointer to source DAT image * 12,s=task # of source * 13,s=task # of destination * 14,s=total byte count of move * Registers: X=Source pointer * U=Destination pointer L0B6A equ * IFNE H6309 ldd [<6,s] [B]=Block # of source ldw [<10,s] [A]=Block # of destination tfr f,a * Calculate move length for this pass ldw 14,s get full byte count cmpw ,s we gonna overlap source? bls L0B82 no, skip ahead ldw ,s get remaining bytes in source block L0B82 cmpw 2,s we gonna overlap destination? bls L0B89 no, skip ahead ldw 2,s get remaining bytes in destination block L0B89 cmpw #$0100 less than 128 bytes? bls L0B92 yes, skip ahead ldw #$0100 force to 128 bytes L0B92 stw 12,s save count orcc #IntMasks Shut off interrupts std >$FFA5 map in the blocks tfm x+,u+ Copy up to 128 bytes sty >$FFA5 Restore system blocks 5&6 to normal andcc #^IntMasks ldd 14,s get full count subd 12,s done? beq L0BEF yes, return std 14,s save updated count ldd ,s get current offset in block subd 12,s need to switch source block? bne L0BD7 no, skip ahead lda #$20 B=0 from 'bne' above subr d,x reset source back to begining of block inc 11,s add 2 to source DAT pointer inc 11,s L0BD7 std ,s save updated source offset in block ldd 2,s get destination offset subd 12,s need to switch destination block? bne L0BEA no, skip ahead lda #$20 B=0 from 'bne', above subr d,u reset destination back to beginning of block inc 7,s add 2 to destination DAT pointer inc 7,s L0BEA std 2,s save updated destination offset in block bra L0B6A go do next block * Block move done, return L0BEF leas 16,s purge stack L0BF2 clrb clear errors puls d,x,y,u,pc return ELSE * Main move loop * Stack: 0,s=distance to end of source block * 2,s=distance to end of destination block * 4,s=pointer to destination * 6,s=pointer to destination DAT image * 8,s=pointer to source * 10,s=pointer to source DAT image * 12,s=task # of source * 13,s=task # of destination * 14,s=total byte count of move * Registers: X=Source pointer * U=Destination pointer L0BXA pshs cc ldd [<$07,s] pshs b ldd [<$0C,s] pshs b ldd <$11,s cmpd $03,s bls L0B82 ldd $03,s L0B82 cmpd $05,s bls L0B89 ldd $05,s L0B89 cmpd #$0040 bls L0B84 ldd #$0040 L0B84 std $0F,s puls y orcc #IntMasks sty >$FFA5 andb #$07 beq L0B99 L0B92 lda ,x+ sta ,u+ decb bne L0B92 L0B99 ldb $0E,s lsrb lsrb lsrb beq L0BBC pshs b exg x,u L0BA4 pulu y,b,a std ,x sty $02,x pulu y,b,a std $04,x sty $06,x leax $08,x dec ,s bne L0BA4 leas $01,s exg x,u L0BBC ldy <D.SysDAT lda $0B,y ldb $0D,y std >$FFA5 puls cc ldd $0E,s subd $0C,s beq L0BEF std $0E,s ldd ,s subd $0C,s bne L0BD7 ldd #DAT.BlSz leax >-DAT.BlSz,x inc $0B,s inc $0B,s L0BD7 std ,s ldd $02,s subd $0C,s bne L0BEA ldd #DAT.BlSz leau >-DAT.BlSz,u inc $07,s inc $07,s L0BEA std $02,s lbra L0BXA L0BEF leas <$10,s L0BF2 clrb puls pc,u,y,x,b,a ENDC L0BF3 tfr u,y save a copy of U for later * Calculate offset within DAT image * Entry: B=Task # * X=Pointer to data * Exit : A=Offset into DAT image * X=Offset within block from original pointer * Possible bug: No one ever checks if the DAT image, in fact, exists. L0BF5 ldu <D.TskIPt get task image ptr table lslb ldu b,u get ptr to this task's DAT image tfr x,d copy logical address to D anda #%11100000 Keep only which 8K bank it's in beq L0C07 Bank 0, no further calcs needed clrb force it to start on an 8K boundary IFNE H6309 subr d,x now X=offset into the block ELSE pshs d tfr x,d subd ,s tfr d,x puls d ENDC lsra Calculate offset into DAT image to get proper lsra 8K bank (remember that each entry in a DAT image lsra is 2 bytes) lsra L0C07 rts