LZ: line routine


[Prev][Next][Index][Thread]

LZ: line routine



I got that line routine off of ticalc.org but am having some trouble with
it. Can somebody explain to me what I'm doing wrong?




H = TEXT_MEM              ; x value of point on horizen
I = TEXT_MEM+3            ; x value of car on horizen
endgame = TEXT_MEM+6      ;check bit for quiting sequence.


arg1 = TEXT_MEM+7   ;\
arg2 = TEXT_MEM+8   ; )---used in math routines at very bottom of prog.
res1 = TEXT_MEM+9   ;/


temp1 = TEXT_MEM+10 ;\___Additional temp. space for storage.
temp2 = TEXT_MEM+13 ;/   LOTS OF SPACE HERE


ty = TEXT_MEM+41    ;<----ty=1 turn point on, else, turn point off.


Way = TEXT_MEM+42   ;<-- used in line routine




;  __________________________________________
; /                                          \
;|   SET FIRST VALUES OF VARIABLES AS NEEDED  |
; \__________________________________________/










        ld a, 50      ;\__Setup starting position of road, slightly
        ld (H), a     ;/  turning right.










;  ___________________________________________________________________
; /                                                                   \
;|    FIRST EXECUTION OF CODE........BEGIN GAME...THEN QUIT TO ZSHELL  |
; \___________________________________________________________________/










        CALL_(game)   ;<---start actual execution of game, afterwards
        ret           ; return to ZShell, but you'll never want
                      ;  to do that.








;  _______________________________________________________
; /                                                       \
;|   MAIN GAME LOOP........DRAW ROAD....MOVE CAR...QUIT    |
; \_______________________________________________________/










game:


        ROM_CALL(CLEARLCD)
        CALL_(drawroad)    ;<--Main substructure that draws road.
        CALL_(getter)  ;<--Get feedback of what player wants to do.


        ld a, (endgame)      ;\
        cp $37               ; )---if "endgame" is $37 quit game.
        ret z                ;/    (exit has been pressed)


        jr game              ;<--Don't want to quit? Keep playing!!


        ret


drawroad:     ; Routine used to draw road on screen.


        ld a, 40
        ld (temp1), a
        CALL_(drd)
        ld a, 86
        ld (temp1), a
        CALL_(drd)
        ret


drd:          ; routine that actually draws the road.
        ld a, 40
        ld b, a
        ld a, (H)
        sub b
        ld (arg1), a
        ld a, 2
        ld (arg2), a
        CALL_(Div32)
        ld a, (arg1)
        ld b, a
        ld a, 40
        add a, b
        ld l, a
        ld a, 27
        ld h, a
        ld a, 0
        ld d, a
        ld a, 40
        ld e, a
        CALL_(Line)


















        ret   ; Return to basic game stem after road has been drawn.




getter:                     ; Routine used to get and comprehend last
                            ; keystroke.
        call GET_KEY         ;<----Get last keystroke.
        ld (endgame), a      ;<---Store value in endgame.
        cp K_LEFT
        CALL_Z(mleft)
        cp K_RIGHT
        CALL_Z(mright)
        ret                  ;<---Return to basic game stem.("game")


mleft:
        ld a, (H)
        dec a
        ld (H), a
        ret


mright:
        ld a, (H)
        inc a
        ld (H), a
        ret




;  __________________________________________________________________
; /                                                                  \
;|   Some math routines  (multiplication, division, absolute value)   |
; \__________________________________________________________________/


Div32:                                  ; arg1 / arg2 -> arg1 , res1 = remainder
        ld      de,0                    ; uses de, b, hl
        ld      (res1),de
        ld      (res1+2),de
        ld      b,32
        and     a                       ; clear carry
Div32Loop:
        ld      de,(arg1)               ; arg1 >> 1 -> arg1
        rl      e
        rl      d
        ld      (arg1),de
        ld      de,(arg1+2)
        rl      e
        rl      d
        ld      (arg1+2),de
        ld      de,(res1)               ; res1 >> 1 -> res1
        rl      e
        rl      d
        ld      (res1),de
        ld      de,(res1+2)
        rl      e
        rl      d
        ld      (res1+2),de
        ld      hl,(res1)               ; res1 - arg2 -> res1
        ld      de,(arg2)
        sbc     hl,de
        ld      (res1),hl
        ld      hl,(res1+2)
        ld      de,(arg2+2)
        sbc     hl,de
        ld      (res1+2),hl
        jr      nc,Div32NoAdd
        ld      hl,(res1)               ; res1 + arg2 -> res1
        ld      de,(arg2)
        add     hl,de
        ld      (res1),hl
        ld      hl,(res1+2)
        ld      de,(arg2+2)
        adc     hl,de
        ld      (res1+2),hl
Div32NoAdd:
        ccf
        djnz    Div32Loop
        ld      de,(arg1)               ; arg1 >> 1 -> arg1
        rl      e
        rl      d
        ld      (arg1),de
        ld      de,(arg1+2)
        rl      e
        rl      d
        ld      (arg1+2),de
        ret


                   ;This is Jimmy Mardell, thanks again.
                   ;
abs:                          ; input: arg1  output: |arg1| => a
        ld a, (arg1)          ; load value into a
        bit 7, a              ; is it positive?
        ret z                 ; Yes, return.
        neg                   ; abs value.
        ret                   ; abs taken, now return.




integer:           ; takes integer value of arg1 and outputs in arg1.
        ld bc, (arg1)
        ld a, b
        ld (arg1), a
        ret










;          Optimized Implementation of Bresenham's line algorithm
;         by Stephane JANTZEN (Stephane.Jantzen@scinfo.u-nancy.fr)




Line:


        ld      A,D
        sub     H
        jr      NC,DXPositive    ;delta_x > 0
DXNegative:
        neg
DXPositive:
        ld      B,A              ;B <- |delta_x|
        ld      A,E
        sub     L
        jr      NC,DYPositive    ;delta_y > 0
DYNegative:
        neg
DYPositive:
        sub     B               ;|delta_y|
        push    AF
        jr      C,DeltaX        ;|delta_x| > |delta_y|
DeltaY:
        ld      A,H             ;if |delta_x| < |delta_y| then
        ld      H,L             ;then values x and y are swapped
        ld      L,A             ;so the loop will always be performed on the
        ld      A,D             ;x value. A flag must be set to
        ld      D,E             ;remind that data must be drawn (y,x)
        ld      E,A             ;instead of (x,y)


        ;ld      A,$01
        ;ld      (Flag),A        ;the Flag !!!
DeltaX:
        ld      A,D
        sub     H
        jr      NC,TestDY        ;x1 < x2
TestDX:
        ex	DE,HL
TestDY:
        ld      A,E
        sub     L
        ld      A,$01
        jr      NC,StoreA
        neg                     ;y1 > y2 : in case2 the 'y' variable
StoreA:
        ld      (Way),A
InitLine:
        ld      B,H
        ld      C,L
        ld      A,E
        sub     L
        jr      NC,EndInit
        ld      A,L
        ld      L,E
        ld      E,A
EndInit:
        ld      A,E
        sub     L
        rla
        ld      L,A             ;value to add in case1 (d < 0)
        add     A,H
        sub     D
        ld      E,A             ;'d' variable is initialised
        add     A,H
        sub     D
        ld      H,A             ;value to add in case2 (d >= 0)
Loop:
        ld      A,B
        sub     D
        jr      NC,EndLine       ;the line is completely drawn.
        pop     AF
        ;ld      A,(Flag)
        bit     7,A
        push    AF
        push    AF
        push    BC
        jr      Z,DrawPoint
        ld      A,B
        ld      B,C
        ld      C,A
DrawPoint:
        CALL_(PointOn)
        pop     BC
        pop     AF
TestD:
        bit     7,E
        jr      NZ,Case1
Case2:                          ;d >= 0
        ld      A,E
        add     A,H
        ld      E,A
        ld      A,(Way)
        add     A,C
        ld      C,A
        jr      EndLoop
Case1:                          ;d < 0
        ld      A,E
        add     A,L
        ld      E,A
EndLoop:
        inc     B
        jr      Loop
EndLine:
        pop     AF              ;MUST NOT BE REMOVED
        pop     HL              ;can be removed
        pop     DE              ;can be removed
        ret


PointOn:
        push    DE              ;MUST NOT BE REMOVED
        push    HL              ;MUST NOT BE REMOVED
        ROM_CALL(FIND_PIXEL)
        ld      DE,$FC00
        add     HL,DE
        or      (HL)
        ld      (HL),A
        pop     HL              ;MUST NOT BE REMOVED
        pop     DE              ;MUST NOT BE REMOVED
        ret














SetPixel:


        ld a, (ty)
        cp 1
        jr z, Pon
        jr Poff


Pon:
 ROM_CALL(FIND_PIXEL)
 ld de,VIDEO_MEM
 add hl,de
 or (hl)
 ld (hl),a
 ret


Poff:
 ROM_CALL(FIND_PIXEL)
 ld de,VIDEO_MEM
 add hl,de
 xor 255
 and (hl)
 ld (hl),a
 ret












.end       ;<---End label used for compilation...NOTHING BELOW HERE!!!!


References: