;Exomizer 2 Z80 decoder
; by Metalbrain
;
; optimized by Antonio Villena and Urusergi (169 bytes)
;
; compression algorithm by Magnus Lind

;input:         hl=compressed data start
;               de=uncompressed destination start
;
;               you may change exo_mapbasebits to point to any free buffer
;
;ATTENTION!
;A huge speed boost (around 14%) can be gained at the cost of only 5 bytes.
;If you want this, replace all instances of "call exo_getbit" with "srl a" followed by
;"call z,exo_getbit", and remove the first two instructions in exo_getbit routine.

_deexo::
				ld      iy, #exo_mapbasebits+11
                ld      a, (hl)
                inc     hl
                ld      b, #52
                push    de
                cp      a
exo_initbits:   ld      c, #16
                jr      nz, exo_get4bits
                .db 	0xDD 			;ld ixl, c
                ld		l,c
                ld      de, #1          ;DE=b2
exo_get4bits:   call    exo_getbit      ;get one bit
                rl      c
                jr      nc, exo_get4bits
                inc     c
                push    hl
                ld      hl, #1
                ld      41(iy), c      ;bits[i]=b1 (and opcode 41 == add hl,hl)
exo_setbit:     dec     c
                jr      nz, #exo_setbit-1 ;jump to add hl,hl instruction
                ld      -11(iy), e
                ld      93(iy), d      ;base[i]=b2
                add     hl, de
                ex      de, hl
                inc     iy
                pop     hl
                .db		0xDD
                dec     l
                djnz    exo_initbits
                pop     de
                jr      exo_mainloop
exo_literalrun: ld      e, c            ;DE=1
exo_getbits:    dec     b
                ret     z
exo_getbits1:   call    exo_getbit
                rl      e
                rl      d
                jr      nc, exo_getbits
                ld      b, d
                ld      c, e
                pop     de
exo_literalcopy:ldir
exo_mainloop:   inc     c
                call    exo_getbit      ;literal?
                jr      c, exo_literalcopy
                ld      c, #239
exo_getindex:   call    exo_getbit
                inc     c
                jr      nc,exo_getindex
                ret     z
                push    de
                ld      d, b
                jp      p, exo_literalrun
                ld      iy, #exo_mapbasebits-229
                call    exo_getpair
                push    de
                rlc     d
                jr      nz, exo_dontgo
                dec     e
                ld      bc, #512+32      ;2 bits, 48 offset
                jr      z, exo_goforit
                dec     e               ;2?
exo_dontgo:     ld      bc, #1024+16     ;4 bits, 32 offset
                jr      z, exo_goforit
                ld      de, #0
                ld      c, d            ;16 offset
exo_goforit:    call    exo_getbits1
                ld      iy, #exo_mapbasebits+27
                add     iy, de
                call    exo_getpair
                pop     bc
                ex      (sp), hl
                push    hl
                sbc     hl, de
                pop     de
                ldir
                pop     hl
                jr      exo_mainloop    ;Next!

exo_getpair:    add     iy, bc
                ld      e, d
                ld      b,41(iy)
                call    exo_getbits
                ex      de, hl
                ld      c,-11(iy)
                ld      b,93(iy)
                add     hl, bc          ;Always clear C flag
                ex      de, hl
                ret

exo_getbit:     srl     a
                ret     nz
                ld      a, (hl)
                inc     hl
                rra
                ret


exo_mapbasebits:
			    .db 0,0,0,0,0,0,0,0,0,0,0,0
			    .db 0,0,0,0,0,0,0,0,0,0,0,0
			    .db 0,0,0,0,0,0,0,0,0,0,0,0
			    .db 0,0,0,0,0,0,0,0,0,0,0,0
			    .db 0,0,0,0,0,0,0,0,0,0,0,0
			    .db 0,0,0,0,0,0,0,0,0,0,0,0
			    .db 0,0,0,0,0,0,0,0,0,0,0,0
			    .db 0,0,0,0,0,0,0,0,0,0,0,0
			    .db 0,0,0,0,0,0,0,0,0,0,0,0
			    .db 0,0,0,0,0,0,0,0,0,0,0,0
			    .db 0,0,0,0,0,0,0,0,0,0,0,0
			    .db 0,0,0,0,0,0,0,0,0,0,0,0
			    .db 0,0,0,0,0,0,0,0,0,0,0,0