2019-04-01 16:04:56 +00:00
|
|
|
; decompress_small.S - space-efficient decompressor implementation for 8088
|
|
|
|
;
|
|
|
|
; Copyright (C) 2019 Emmanuel Marty
|
|
|
|
;
|
|
|
|
; This software is provided 'as-is', without any express or implied
|
|
|
|
; warranty. In no event will the authors be held liable for any damages
|
|
|
|
; arising from the use of this software.
|
|
|
|
;
|
|
|
|
; Permission is granted to anyone to use this software for any purpose,
|
|
|
|
; including commercial applications, and to alter it and redistribute it
|
|
|
|
; freely, subject to the following restrictions:
|
|
|
|
;
|
|
|
|
; 1. The origin of this software must not be misrepresented; you must not
|
|
|
|
; claim that you wrote the original software. If you use this software
|
|
|
|
; in a product, an acknowledgment in the product documentation would be
|
|
|
|
; appreciated but is not required.
|
|
|
|
; 2. Altered source versions must be plainly marked as such, and must not be
|
|
|
|
; misrepresented as being the original software.
|
|
|
|
; 3. This notice may not be removed or altered from any source distribution.
|
|
|
|
|
|
|
|
segment .text
|
|
|
|
bits 16
|
|
|
|
|
|
|
|
; ---------------------------------------------------------------------------
|
2019-05-09 14:51:29 +00:00
|
|
|
; Decompress raw LZSA1 block
|
2019-04-01 16:04:56 +00:00
|
|
|
; inputs:
|
2019-05-09 14:51:29 +00:00
|
|
|
; * ds:si: raw LZSA1 block
|
2019-04-01 16:04:56 +00:00
|
|
|
; * es:di: output buffer
|
|
|
|
; output:
|
|
|
|
; * ax: decompressed size
|
|
|
|
; ---------------------------------------------------------------------------
|
|
|
|
|
2019-05-09 14:51:29 +00:00
|
|
|
lzsa1_decompress:
|
2019-04-01 16:04:56 +00:00
|
|
|
push di ; remember decompression offset
|
|
|
|
cld ; make string operations (lods, movs, stos..) move forward
|
|
|
|
|
|
|
|
xor cx,cx
|
|
|
|
|
|
|
|
.decode_token:
|
2019-04-05 09:58:44 +00:00
|
|
|
mov ax,cx ; clear ah - cx is zero from above or from after rep movsb in .copy_match
|
2019-04-10 15:30:13 +00:00
|
|
|
lodsb ; read token byte: O|LLL|MMMM
|
2019-04-01 16:04:56 +00:00
|
|
|
mov dx,ax ; keep token in dl
|
|
|
|
|
2019-04-05 21:16:05 +00:00
|
|
|
and al,070H ; isolate literals length in token (LLL)
|
|
|
|
mov cl,4
|
|
|
|
shr al,cl ; shift literals length into place
|
2019-04-01 16:04:56 +00:00
|
|
|
|
|
|
|
cmp al,07H ; LITERALS_RUN_LEN?
|
2019-04-24 07:47:40 +00:00
|
|
|
jne .got_literals ; no, we have the full literals count from the token, go copy
|
2019-04-01 16:04:56 +00:00
|
|
|
|
2019-04-24 07:47:40 +00:00
|
|
|
lodsb ; grab extra length byte
|
|
|
|
add al,07H ; add LITERALS_RUN_LEN
|
|
|
|
jnc .got_literals ; if no overflow, we have the full literals count, go copy
|
|
|
|
jne .mid_literals
|
|
|
|
|
|
|
|
lodsw ; grab 16-bit extra length
|
2019-07-04 16:09:29 +00:00
|
|
|
db 81H ; mask inc ah/lodsb
|
2019-07-04 02:38:27 +00:00
|
|
|
; (*like jmp short .got_literals but faster)
|
2019-04-01 16:04:56 +00:00
|
|
|
|
2019-04-24 07:47:40 +00:00
|
|
|
.mid_literals:
|
|
|
|
inc ah ; add 256
|
2019-07-04 02:38:27 +00:00
|
|
|
lodsb ; grab single extra length byte
|
2019-04-24 07:47:40 +00:00
|
|
|
|
|
|
|
.got_literals:
|
|
|
|
xchg cx,ax
|
2019-04-01 16:04:56 +00:00
|
|
|
rep movsb ; copy cx literals from ds:si to es:di
|
|
|
|
|
2019-04-05 21:16:05 +00:00
|
|
|
test dl,dl ; check match offset size in token (O bit)
|
|
|
|
js .get_long_offset
|
2019-04-01 16:04:56 +00:00
|
|
|
|
2019-07-04 02:38:27 +00:00
|
|
|
dec cx
|
|
|
|
xchg cx,ax ; ah to 0xff - cx was zero from the rep movsb above
|
2019-04-01 16:04:56 +00:00
|
|
|
lodsb
|
2019-07-04 16:09:29 +00:00
|
|
|
db 3CH ; mask lodsw
|
2019-07-04 02:38:27 +00:00
|
|
|
; (*like jmp short .get_match_length but faster)
|
2019-04-01 16:04:56 +00:00
|
|
|
|
|
|
|
.get_long_offset:
|
|
|
|
lodsw ; Get 2-byte match offset
|
|
|
|
|
|
|
|
.get_match_length:
|
2019-07-04 02:38:27 +00:00
|
|
|
xchg dx,ax ; dx: match offset ax: original token
|
2019-04-01 16:04:56 +00:00
|
|
|
and al,0FH ; isolate match length in token (MMMM)
|
2019-04-24 07:47:40 +00:00
|
|
|
add al,3 ; add MIN_MATCH_SIZE
|
2019-04-01 16:04:56 +00:00
|
|
|
|
2019-04-24 07:47:40 +00:00
|
|
|
cmp al,012H ; MATCH_RUN_LEN?
|
|
|
|
jne .got_matchlen ; no, we have the full match length from the token, go copy
|
|
|
|
|
|
|
|
lodsb ; grab extra length byte
|
|
|
|
add al,012H ; add MIN_MATCH_SIZE + MATCH_RUN_LEN
|
|
|
|
jnc .got_matchlen ; if no overflow, we have the entire length
|
|
|
|
jne .mid_matchlen
|
|
|
|
|
|
|
|
lodsw ; grab 16-bit length
|
|
|
|
test ax,ax ; bail if we hit EOD
|
2019-07-07 19:05:02 +00:00
|
|
|
je short .done_decompressing
|
2019-04-24 07:47:40 +00:00
|
|
|
|
2019-07-07 19:05:02 +00:00
|
|
|
db 81H ; mask inc ah/lodsb
|
|
|
|
; (*like jmp short .got_literals but faster)
|
2019-04-24 07:47:40 +00:00
|
|
|
.mid_matchlen:
|
|
|
|
inc ah ; add 256
|
2019-07-07 19:05:02 +00:00
|
|
|
lodsb ; grab single extra length byte
|
2019-04-01 16:04:56 +00:00
|
|
|
|
2019-04-24 07:47:40 +00:00
|
|
|
.got_matchlen:
|
|
|
|
xchg cx,ax ; copy match length into cx
|
2019-04-01 16:04:56 +00:00
|
|
|
push ds ; save ds:si (current pointer to compressed data)
|
|
|
|
xchg si,ax
|
|
|
|
push es
|
|
|
|
pop ds
|
|
|
|
mov si,di ; ds:si now points at back reference in output data
|
2019-04-24 07:47:40 +00:00
|
|
|
add si,dx
|
2019-04-01 16:04:56 +00:00
|
|
|
rep movsb ; copy match
|
|
|
|
xchg si,ax ; restore ds:si
|
|
|
|
pop ds
|
|
|
|
jmp short .decode_token ; go decode another token
|
2019-07-07 19:05:02 +00:00
|
|
|
|
|
|
|
.done_decompressing:
|
|
|
|
pop ax ; retrieve the original decompression offset
|
|
|
|
xchg ax,di ; compute decompressed size
|
|
|
|
sub ax,di
|
|
|
|
ret ; done
|