Compression Using the Splay-tree technique

Compress and decompress files using the "splay tree" technique. Based on the article by Douglas W. Jones, Application of Splay Trees to Data Compression in Communications of the ACM, August 1988.

Usage:
SPLAYIT (no filenames or parameters)
SPLAYIT simply compresses a predefined file to a predefined file and then uncompresses back to another predefined file. You can change the filenames in the code or add a command-line parser.

**Caution**
This routine has NO error checking. It is given to show how to program a splay tree compression routine. You can simple add error checking by testing each file read buffer and file write buffer routines as well as the open and close file routines.


SPLAY.ASM Copyright 1984-2025
by Forever Young Software
Benjamin David Lunt
Version 1.02
09 Dec 1998
Assembled with: NBASM

Comment |=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
  Compress and decompress files using the "splay tree" technique.
  Based on the article by Douglas W. Jones, "Application of Splay Trees to
  Data Compression" in Communications of the ACM, August 1988.

  Usage:
    SPLAYIT     (no filenames or parameters)

  SPLAYIT simply compresses a predefined file to a predefined file and then
  uncompresses back to another predefined file.  You can change the filenames
  in the code or add a command-line parser.
-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=|

.model tiny
.186
.code
           org  100h

           mov  dx,offset CompStr       ; ** compress file
           mov  ah,09
           int  21h
           mov  dx,offset InName
           call OpenSFile
           mov  dx,offset OutName
           call OpenTFile
           call InitSplay
           call CompFile
           mov  bx,Handle1
           call CloseFile
           mov  bx,Handle2
           call CloseFile               ; ** done compress file


           mov  dx,offset ExpndStr      ; ** expand file
           mov  ah,09
           int  21h
           mov  dx,offset OutName
           call OpenSFile
           mov  dx,offset OrgName
           call OpenTFile
           call InitSplay
           call ExpndFile
           mov  bx,Handle1
           call CloseFile
           mov  bx,Handle2
           call CloseFile               ; ** done expand file

           mov  ax,4C00h
           int  21h

InitSplay  proc near

           ; nbasm does not support the local directive (yet)
           ;local I:word,K:word,J:byte

           jmp  short ISkipLocal
I          dw 00h
J          dw 00h
K          dw 00h
ISkipLocal:

           mov  cx,512
           xor  ax,ax
           mov  di,offset Up
           stosb
           inc  ax
UpLoop:    push ax
           dec  ax
           shr  ax,1
           stosb
           pop  ax
           inc  ax
           loop UpLoop

           mov  cx,256
           xor  ax,ax
           mov  si,offset Left
           mov  di,offset Right
LRLoop:    push ax
           inc  ax
           shl  ax,1
           stosw
           dec  ax
           mov  [si],ax
           inc  si
           inc  si
           pop  ax
           inc  ax
           loop LRLoop
           ret
InitSplay  endp

OpenSFile  proc near
           mov  ax,3D00h                ; open RO file
           int  21h                     ;
           mov  Handle1,ax
           ret
OpenSFile  endp

OpenTFile  proc near
           mov  ah,5Bh                  ; create new file
           xor  ch,ch                   ; normal file
           mov  cl,00100000b            ;
           int  21h                     ;
           mov  Handle2,ax
           ret
OpenTFile  endp

CloseFile  proc near
           mov  ah,3Eh                  ; Close File
           int  21h                     ;
           ret
CloseFile  endp

SPlay      proc near uses ax bx         ; uses passed ax
           
           ; nbasm does not support the local directive (yet)
           ;local A:word,B:word,C:byte,D:byte

           jmp  short SSkipLocal
A          dw 00h
B          dw 00h
C          dw 00h
D          dw 00h
SSkipLocal:

           add  ax,256                  ; A = Plain + 256;
           mov  A,ax                    ;
                                        ;
SLoop:     mov  bx,offset Up            ; C = Up[A]
           add  bx,ax                   ;
           mov  al,[bx]                 ;
           mov  C,al                    ;

           or   al,al                   ; if C <> Root then begin
           jnz  NotSElse
           jmp  SElse                   ;
NotSElse:  mov  bx,offset Up            ; D = Up[C]
           xor  ah,ah                   ;
           add  bx,ax                   ;
           mov  al,[bx]                 ;
           mov  D,al                    ;

           mov  bx,offset Left          ; B = Left[D]
           xor  ah,ah                   ;
           shl  ax,1                    ; double for word
           add  bx,ax                   ;
           mov  ax,[bx]                 ;
           mov  B,ax                    ;

           mov  al,C                    ; if C = B then begin
           xor  ah,ah                   ;
           cmp  ax,B                    ;
           jne  short NotCBE            ;
           mov  bx,offset Right         ; B = Right[D]
           mov  al,D                    ;
           xor  ah,ah                   ;
           shl  ax,1                    ; double for word
           add  bx,ax                   ;
           mov  ax,[bx]                 ;
           mov  B,ax                    ;
           mov  ax,A                    ; Right[D] = A
           mov  [bx],ax                 ;
           jmp  short NextIf            ;
NotCBE:    mov  al,D                    ; Left[D] = A
           xor  ah,ah                   ;
           shl  ax,1                    ; double for word
           mov  bx,offset Left          ;
           add  bx,ax                   ;
           mov  ax,A                    ;
           mov  [bx],ax                 ;
NextIf:    mov  al,C                    ; if A = Left[C] then
           xor  ah,ah                   ;
           shl  ax,1                    ; double for word
           mov  bx,offset Left          ;
           add  bx,ax                   ;
           mov  ax,[bx]                 ;
           cmp  ax,A                    ;
           jne  short RCEB              ;
           mov  ax,B                    ; Left[C] = B
           mov  [bx],ax                 ;
           jmp  short DoUps             ;
RCEB:      mov  al,C                    ; Right[C] = B
           xor  ah,ah                   ;
           shl  ax,1                    ; double for word
           mov  bx,offset Right         ;
           add  bx,ax                   ;
           mov  ax,B                    ;
           mov  [bx],ax                 ;
DoUps:     mov  ax,A                    ; Up[A] = D
           mov  bx,offset UP            ;
           add  bx,ax                   ;
           mov  al,D                    ;
           mov  [bx],al                 ;
           mov  ax,B                    ; Up[B] = C
           mov  bx,offset UP            ;
           add  bx,ax                   ;
           mov  al,C                    ;
           mov  [bx],al                 ;
           mov  al,D                    ; A = D
SElse:     xor  ah,ah                   ; A = C
           mov  A,ax                    ;
           or   ax,ax                   ; until A = Root   (see next comment)
           jz   short SPlayRet
           jmp  SLoop                   ;     (A=ax)(faster to test ax then A)
SPlayRet:  ret
SPlay      endp

FOBuffer   proc near uses ax bx cx dx

           cmp  word OutSize,00h        ; if OutSize > 0 then begin
           je   short DFO               ;
           mov  ah,40h                  ; write to file
           mov  bx,Handle2
           mov  cx,OutSize
           mov  dx,offset BufferOut
           int  21h
           mov  word OutSize,00h
DFO:       ret
FOBuffer   endp

WriteByte  proc near uses ax bx

           cmp  word OutSize,16384           ; if OutSize = 16384 then
           jb   short NoFLO
           call FOBuffer                ; FlushOutBuffer
NoFLO:     mov  bx,offset BufferOut     ; OutBuffer[OutSize] = OutByte
           add  bx,OutSize              ;
           mov  al,OutByte              ;
           mov  [bx],al                 ;
           inc  word OutSize            ; inc OutSize
           ret
WriteByte  endp

Compress   proc near uses cx si di      ; uses passed ax
           
           ; nbasm does not support the local directive (yet)
           ;local PSDAX:word,A1:word,U:byte

           jmp  short CSkipLocal
PSDAX      dw 00h
A1         dw 00h
U          dw 00h
CSkipLocal:

           mov  PSDAX,ax                ; A1 = Plain + 256
           add  ax,256                  ;
           mov  A1,ax                   ;

           push ax                      ; save A1
           mov  di,offset Stck          ; our stack
           mov  al,0FFh                 ; bottom of stack flag
           stosb                        ;
           pop  ax                      ; restore A1
                                        ; repeat
RCLoop:    mov  bx,offset Up            ; U = Up[A1]
           add  bx,ax                   ;
           mov  al,[bx]                 ;
           mov  U,al                    ;

           mov  bx,offset Right         ; Stack[Sp] = (Right[U] = A1)
           xor  ah,ah                   ;
           shl  ax,1                    ; double for word
           add  bx,ax                   ;
           mov  ax,[bx]                 ;
           cmp  ax,A1                   ;
           jne  short NotTrue           ;
           mov  al,01h                  ; TRUE
           jmp  short IsTrue            ;
NotTrue:   xor  al,al                   ; FALSE
IsTrue:    stosb                        ; save on stack
           mov  al,U                    ; A1 = U
           xor  ah,ah                   ;
           mov  A1,ax                   ;
           or   ax,ax                   ; until A1 = Root
           jnz  short RCLoop            ;

UnStack:   dec  di                      ; if Stack[Sp] then
           mov  al,[di]                 ;
           cmp  al,0FFh
           je   short DoneStck
           cmp  al,01h                  ; TRUE
           jne  short NotOrIt           ;
           mov  bl,OutByte              ;
           or   bl,OrIt                 ;
           mov  OutByte,bl              ;
NotOrIt:   cmp  byte OrIt,10000000b
           jne  short NotPos7
           call WriteByte               ;
           mov  byte OutByte,00h        ;
NotPos7:   rol  byte OrIt,1
           jmp  short UnStack           ; go to the next one
DoneStck:  mov  ax,PSDAX
           call SPlay                   ; Splay(Plain)
           ret
Compress   endp

CompFile   proc near
           mov  byte OrIt,00000001b
DoIt:      mov  ah,3Fh                  ; BlockRead.....
           mov  cx,16384                ;
           mov  bx,Handle1              ;
           mov  dx,offset BufferIn      ;
           int  21h                     ;
           or   ax,ax                   ; if none read finish up
           jz   short NoMore            ;
           push ax                      ;
           mov  cx,ax                   ; Index = 1 to InSize
           mov  si,offset BufferIn      ;
DoFor:     lodsb                        ;
           xor  ah,ah                   ;
           call Compress                ; Compress(InBuffer[index])
           loop DoFor                   ;
           pop  ax                      ;
           cmp  ax,16384                ; until InSize < 16384
           je   short DoIt              ;

NoMore:    mov  ax,256                  ; Compress(EofChar)
           call Compress                ;
           cmp  byte OrIt,00000001b          ; if BitPos <> 0 then
           je   short FlushIt
           call WriteByte
FlushIt:   call FOBuffer
           ret
CompFile   endp


;; **  now begins the expansion part  ** ;;
GetByte    proc near uses bx cx dx

           inc  word Index
           mov  ax,Index
           cmp  ax,InSize
           jb   short GetIt
           mov  ah,3Fh                  ; BlockRead.....
           mov  cx,16384                ;
           mov  bx,Handle1              ;
           mov  dx,offset BufferIn      ;
           int  21h                     ;
           mov  InSize,ax
           mov  word Index,00h
GetIt:     mov  bx,offset BufferIn
           add  bx,Index
           mov  al,[bx]
           ret
GetByte    endp

Expand     proc near uses bx 

           xor  dx,dx                   ; A = Root;
ExLoop:    cmp  byte OrIt,10000000b
           jne  short ENot07
           call GetByte                 ; returns in al
           mov  InByte,al
ENot07:    rol  byte OrIt,1
           mov  al,InByte
           and  al,OrIt
           or   al,al
           jnz  short NBitEZ
           mov  bx,offset Left          ; A = left[A]
           jmp  short DoUntA
NBitEZ:    mov  bx,offset Right         ; A = right[A]
DoUntA:    mov  ax,dx
           shl  ax,1
           add  bx,ax
           mov  ax,[bx]
           mov  dx,ax                   ; A
           cmp  ax,255                  ; until A > PredMax
           jbe  short ExLoop

           sub  ax,256
           call Splay
ReadErr:   ret
Expand     endp

ExpndFile  proc near uses ax bx cx dx

           mov  byte OrIt,10000000b
EoCLoop:   call Expand                  ; returns in ax
           cmp  ax,256
           je   short XEndIt
           mov  OutByte,al
           call WriteByte
           jmp  short EoCLoop
XEndIt:    call FOBuffer
           ret
ExpndFile  endp

Handle1    dw  00h        ; handle for source file
Handle2    dw  00h        ; handle for target file
OutByte    db  00h        ; the byte to write
InByte     db  00h        ; the byte read
OrIt       db  00h        ; AND var
InSize     dw  00h        ; len of data in source buffer (amount read in)
Index      dw  00h        ; location to put next byte in buffer
OutSize    dw  00h        ; len of data in target buffer (amount to write)
InName     db  'name.org',0   ; *** filename to use here (source)(orig. file)
OutName    db  'name.spl',0   ; ***  and here  (compressed file)(splayed)
OrgName    db  'name.uns',0   ; ***  and here  (uncompressed file)(unsplayed)
CompStr    db  13,10,'Compressing...$'
ExpndStr   db  13,10,'Expanding...$'

Stck       dup 256,?
Left       dup 512,?
Right      dup 512,?
Up         dup 514,?  ; (2 extra for safety)
BufferIn   dup 16384,?
BufferOut  dup 16390,?
.end