.define .dvi2 ! 16 bits signed and unsigned integer divide and remainder routine ! Bit 0 of a-reg is set iff quotient has to be delivered ! Bit 7 of a-reg is set iff the operands are signed, so: ! Expects in a-reg: 0 if called by rmu 2 ! 1 if called by dvu 2 ! 128 if called by rmi 2 ! 129 if called by dvi 2 ! Expects on stack: divisor ! dividend ! Yields in de-reg: quotient or remainder .dvi2: pop h shld .retadr mov h,b mov l,c shld .bcreg sta .areg pop b ! bc = divisor mov a,b ! trap if divisor = 0 ora c cz eidivz pop d ! de = dividend mvi h,0 lda .areg ral jnc 0f ! jump if unsigned mov a,d ral jnc 1f ! jump if dividend >= 0 mvi h,129 ! indicate dividend is negative xra a ! negate dividend sub e mov e,a mvi a,0 sbb d mov d,a ! de is positive now 1: mov a,b ral jc 2f ! jump if divisor < 0 0: inr h ! indicate negation xra a ! negate divisor sub c mov c,a mvi a,0 sbb b mov b,a ! bc is negative now 2: push h ! save h-reg lxi h,0 ! initial value of remainder mvi a,16 ! initialize loop counter 3: push psw ! save loop counter dad h ! shift left: hl <- de <- 0 xchg dad h xchg jnc 4f inx h 4: push h ! save remainder dad b ! subtract divisor (add negative) jnc 5f xthl inx d 5: pop h pop psw ! restore loop counter dcr a jnz 3b pop b ! b-reg becomes what once was h-reg lda .areg rar ! what has to be delivered: quotient or remainder? jnc 6f ! for dvi 2 and dvu 2 only: mov a,b rar jc 8f ! jump if divisor and dividend had same sign xra a ! negate quotient sub e mov e,a mvi a,0 sbb d mov d,a jmp 8f ! for rmi 2 and rmu 2 only: 6: mov a,b ral jnc 7f ! negate remainder if dividend was negative xra a sub l mov l,a mvi a,0 sbb h mov h,a 7: mov d,h ! return remainder mov e,l 8: lhld .bcreg mov b,h mov c,l lhld .retadr pchl