forked from KolibriOS/kolibrios
1486 lines
39 KiB
PHP
1486 lines
39 KiB
PHP
|
; PPMD decoder, ported from C++ sources of 7-Zip (c) Igor Pavlov
|
||
|
; C++ code is based on Dmitry Shkarin's PPMdH code
|
||
|
uglobal
|
||
|
ppmd_decoder.NS2Indx rb 256
|
||
|
ppmd_decoder.NS2BSIndx rb 256
|
||
|
ppmd_decoder.HB2Flag rb 256
|
||
|
ppmd_decoder.Indx2Units rb ppmd_decoder.N_INDEXES
|
||
|
ppmd_decoder.Units2Indx rb 128
|
||
|
endg
|
||
|
|
||
|
iglobal
|
||
|
label ppmd_decoder.InitBinEsc word
|
||
|
dw 0x3CDD, 0x1F3F, 0x59BF, 0x48F3, 0x64A1, 0x5ABC, 0x6632, 0x6051
|
||
|
ppmd_decoder.ExpEscape db 25, 14, 9, 7, 5, 5, 4, 4, 4, 3, 3, 3, 2, 2, 2, 2
|
||
|
endg
|
||
|
|
||
|
init_ppmd:
|
||
|
; NS2Indx table
|
||
|
mov edi, ppmd_decoder.NS2Indx
|
||
|
xor eax, eax
|
||
|
stosb
|
||
|
inc eax
|
||
|
stosb
|
||
|
inc eax
|
||
|
stosb
|
||
|
mov edx, 3-256
|
||
|
@@:
|
||
|
lea ecx, [eax-1]
|
||
|
inc eax
|
||
|
add edx, ecx
|
||
|
jc @f
|
||
|
rep stosb
|
||
|
jmp @b
|
||
|
@@:
|
||
|
sub ecx, edx
|
||
|
rep stosb
|
||
|
; NS2BSIndx table
|
||
|
xor eax, eax
|
||
|
stosb
|
||
|
add al, 2
|
||
|
stosb
|
||
|
add al, 2
|
||
|
mov cl, 9
|
||
|
rep stosb
|
||
|
add al, 2
|
||
|
mov cl, 256-11
|
||
|
rep stosb
|
||
|
; HB2Flag table
|
||
|
mov cl, 0x40/4
|
||
|
xor eax, eax
|
||
|
rep stosd
|
||
|
mov al, 8
|
||
|
mov cl, 0x100-0x40
|
||
|
rep stosb
|
||
|
; Indx2Units table
|
||
|
mov eax, 0x04030201
|
||
|
stosd
|
||
|
mov eax, 0x0C0A0806
|
||
|
stosd
|
||
|
mov eax, 0x1815120F
|
||
|
stosd
|
||
|
mov al, 0x1C
|
||
|
@@:
|
||
|
stosb
|
||
|
add al, 4
|
||
|
cmp al, 0x80
|
||
|
jbe @b
|
||
|
; Units2Indx table
|
||
|
xor eax, eax
|
||
|
xor edx, edx
|
||
|
inc edx
|
||
|
xor ecx, ecx
|
||
|
@@:
|
||
|
cmp [ppmd_decoder.Indx2Units+eax], dl
|
||
|
adc al, 0
|
||
|
stosb
|
||
|
inc edx
|
||
|
cmp dl, 0x80
|
||
|
jbe @b
|
||
|
ret
|
||
|
|
||
|
ppmd_decoder:
|
||
|
virtual at 0
|
||
|
; base is standard structure
|
||
|
.outStream rb streamInfo.size
|
||
|
.inStream dd ?
|
||
|
|
||
|
; RangeDecoder data
|
||
|
.inLen dd ?
|
||
|
.inPtr dd ?
|
||
|
.code dd ?
|
||
|
.range dd ?
|
||
|
|
||
|
.outSize dd ? ; number of bytes rest for output
|
||
|
|
||
|
; PPMD data
|
||
|
.order db ?
|
||
|
.GlueCount db ?
|
||
|
.bInited db ?
|
||
|
rb 1
|
||
|
.usedMemorySize dd ?
|
||
|
|
||
|
; CSubAllocator constants
|
||
|
.N1 = 4
|
||
|
.N2 = 4
|
||
|
.N3 = 4
|
||
|
.N4 = (128+3-1*.N1-2*.N2-3*.N3)/4
|
||
|
.UNIT_SIZE = 12
|
||
|
.N_INDEXES = .N1+.N2+.N3+.N4
|
||
|
.kExtraSize = .UNIT_SIZE*3
|
||
|
.kMaxMemBlockSize = 0xFFFFFFFF - .kExtraSize
|
||
|
|
||
|
; CSubAllocator data
|
||
|
.HeapStart dd ?
|
||
|
.LoUnit dd ?
|
||
|
.HiUnit dd ?
|
||
|
.pText dd ?
|
||
|
.UnitsStart dd ?
|
||
|
.FreeList rd .N_INDEXES
|
||
|
|
||
|
; Context constants
|
||
|
.INT_BITS = 7
|
||
|
.PERIOD_BITS = 7
|
||
|
.TOT_BITS = .INT_BITS + .PERIOD_BITS
|
||
|
.INTERVAL = 1 shl .INT_BITS
|
||
|
.BIN_SCALE = 1 shl .TOT_BITS
|
||
|
.MAX_FREQ = 124
|
||
|
|
||
|
.kMaxOrderCompress = 32
|
||
|
.MAX_O = 255
|
||
|
|
||
|
; CDecodeInfo (inherits from CInfo) data
|
||
|
; SEE2_CONTEXT is 4 bytes long
|
||
|
.SEE2Cont rd 25*16
|
||
|
.DummySEE2Cont dd ?
|
||
|
.MinContext dd ?
|
||
|
.MaxContext dd ?
|
||
|
.FoundState dd ? ; found next state transition
|
||
|
.NumMasked dd ?
|
||
|
.InitEsc dd ?
|
||
|
.OrderFall dd ?
|
||
|
.RunLength dd ?
|
||
|
.InitRL dd ?
|
||
|
.CharMask rb 256
|
||
|
.EscCount db ?
|
||
|
.PrintCount db ?
|
||
|
.PrevSuccess db ?
|
||
|
.HiBitsFlag db ?
|
||
|
.BinSumm rw 128*64
|
||
|
|
||
|
.basesize = $
|
||
|
.Base:
|
||
|
; rb .kExtraSize + [.usedMemorySize]
|
||
|
end virtual
|
||
|
|
||
|
.init:
|
||
|
mov eax, [eax+.inStream]
|
||
|
call fillBuf
|
||
|
mov esi, [eax+streamInfo.bufPtr]
|
||
|
mov eax, [eax+streamInfo.bufDataLen]
|
||
|
sub eax, 5
|
||
|
jb return.err
|
||
|
mov [ebp+.inLen], eax
|
||
|
inc esi
|
||
|
lodsd
|
||
|
mov [ebp+.inPtr], esi
|
||
|
bswap eax
|
||
|
mov [ebp+.code], eax
|
||
|
or [ebp+.range], -1
|
||
|
mov [ebp+.bInited], 1
|
||
|
call .StartModelRare
|
||
|
mov eax, ebp
|
||
|
jmp .mainloop
|
||
|
|
||
|
.fillBuf:
|
||
|
mov ebp, eax
|
||
|
mov [eax+.outSize], ecx
|
||
|
cmp [eax+.bInited], 0
|
||
|
jz .init
|
||
|
.mainloop:
|
||
|
sub [ebp+.outSize], 1
|
||
|
js .mainloopdone
|
||
|
; cmp edi, 0xde070+0x18
|
||
|
; jnz @f
|
||
|
; int3
|
||
|
;@@:
|
||
|
call .DecodeSymbol
|
||
|
jmp .mainloop
|
||
|
.mainloopdone:
|
||
|
popad
|
||
|
ret
|
||
|
|
||
|
.GetBinSumm:
|
||
|
; CInfo::GetBinSumm(ebx=rs, ecx=numStates)
|
||
|
movzx eax, [ebp+.PrevSuccess]
|
||
|
movzx edx, [.NS2BSIndx+ecx-1]
|
||
|
add eax, edx
|
||
|
mov edx, [ebp+.FoundState]
|
||
|
movzx edx, byte [edx]
|
||
|
movzx edx, [.HB2Flag+edx]
|
||
|
mov [ebp+.HiBitsFlag], dl
|
||
|
add eax, edx
|
||
|
movzx edx, byte [ebx]
|
||
|
movzx edx, [.HB2Flag+edx]
|
||
|
lea eax, [eax+edx*2]
|
||
|
mov edx, [ebp+.RunLength]
|
||
|
shr edx, 26
|
||
|
and edx, 0x20
|
||
|
add eax, edx
|
||
|
movzx edx, byte [ebx+1]
|
||
|
shl edx, 6
|
||
|
add eax, edx
|
||
|
lea ecx, [ebp+.BinSumm+eax*2-2*64]
|
||
|
ret
|
||
|
|
||
|
.StartModelRare:
|
||
|
; CInfo::StartModelRare(.order)
|
||
|
mov [ebp+.EscCount], 1
|
||
|
mov [ebp+.PrintCount], 1
|
||
|
; N.B.
|
||
|
; 1. Original code has some handling of [.order]<2, but this handling is incorrect
|
||
|
; and causes exception (access violation).
|
||
|
; 2. 7-Zip never generates archives with [.order]<2 due to input preprocessing
|
||
|
; (for PPMd method in switch -mo=<n> archiver checks that 2 <= n <= 32).
|
||
|
; 3. If manually created archive says [.order]<2, the exception will be generated
|
||
|
; in StartModelRare, but it will be handled in Code() resulting in "data error".
|
||
|
cmp [ebp+.order], 2
|
||
|
jb return.err
|
||
|
mov byte [ebp+.DummySEE2Cont+2], .PERIOD_BITS
|
||
|
|
||
|
.RestartModelRare:
|
||
|
; CInfo::RestartModelRare(void)
|
||
|
push edi
|
||
|
lea edi, [ebp+.CharMask]
|
||
|
xor eax, eax
|
||
|
push 0x40
|
||
|
pop ecx
|
||
|
rep stosd
|
||
|
; CSubAllocator::InitSubAllocator start
|
||
|
mov [ebp+.GlueCount], al
|
||
|
lea edi, [ebp+.FreeList]
|
||
|
mov cl, .N_INDEXES
|
||
|
rep stosd
|
||
|
mov ebx, [ebp+.HeapStart]
|
||
|
mov [ebp+.pText], ebx
|
||
|
add ebx, [ebp+.usedMemorySize]
|
||
|
mov [ebp+.HiUnit], ebx
|
||
|
mov eax, [ebp+.usedMemorySize]
|
||
|
xor edx, edx
|
||
|
mov cl, 8*.UNIT_SIZE
|
||
|
div ecx
|
||
|
imul eax, 7*.UNIT_SIZE
|
||
|
sub ebx, eax
|
||
|
mov [ebp+.LoUnit], ebx
|
||
|
mov [ebp+.UnitsStart], ebx
|
||
|
; CSubAllocator::InitSubAllocator end
|
||
|
pop edi
|
||
|
movzx eax, [ebp+.order]
|
||
|
cmp al, 12
|
||
|
jb @f
|
||
|
mov al, 12
|
||
|
@@:
|
||
|
neg eax
|
||
|
dec eax
|
||
|
mov [ebp+.InitRL], eax
|
||
|
mov [ebp+.RunLength], eax
|
||
|
call .AllocContext
|
||
|
mov [ebp+.MinContext], eax
|
||
|
mov [ebp+.MaxContext], eax
|
||
|
and dword [eax+8], 0
|
||
|
mov esi, eax
|
||
|
movzx edx, [ebp+.order]
|
||
|
mov [ebp+.OrderFall], edx
|
||
|
mov dword [eax], 257*10000h+256
|
||
|
mov ecx, 256/2
|
||
|
call .AllocUnits
|
||
|
mov [ebp+.FoundState], eax
|
||
|
mov [esi+4], eax
|
||
|
push edi
|
||
|
mov edi, eax
|
||
|
xor eax, eax
|
||
|
mov [ebp+.PrevSuccess], al
|
||
|
@@:
|
||
|
stosb
|
||
|
mov byte [edi], 1
|
||
|
and dword [edi+1], 0
|
||
|
add edi, 5
|
||
|
inc al
|
||
|
jnz @b
|
||
|
lea edi, [ebp+.BinSumm]
|
||
|
push 2
|
||
|
pop ecx
|
||
|
.rmr1:
|
||
|
mov esi, .InitBinEsc
|
||
|
@@:
|
||
|
lodsw
|
||
|
xor edx, edx
|
||
|
div ecx
|
||
|
sub eax, .BIN_SCALE
|
||
|
neg eax
|
||
|
mov [edi+2*8], ax
|
||
|
mov [edi+2*16], ax
|
||
|
mov [edi+2*24], ax
|
||
|
mov [edi+2*32], ax
|
||
|
mov [edi+2*40], ax
|
||
|
mov [edi+2*48], ax
|
||
|
mov [edi+2*56], ax
|
||
|
stosw
|
||
|
cmp esi, .InitBinEsc+2*8
|
||
|
jb @b
|
||
|
add edi, 128-16
|
||
|
inc ecx
|
||
|
cmp ecx, 128+2
|
||
|
jb .rmr1
|
||
|
lea edi, [ebp+.SEE2Cont]
|
||
|
mov eax, (10 shl (.PERIOD_BITS-4)) + ((.PERIOD_BITS-4) shl 16) + (4 shl 24)
|
||
|
push 25
|
||
|
pop edx
|
||
|
@@:
|
||
|
push 16
|
||
|
pop ecx
|
||
|
rep stosd
|
||
|
add ax, 5 shl (.PERIOD_BITS-4)
|
||
|
dec edx
|
||
|
jnz @b
|
||
|
pop edi
|
||
|
ret
|
||
|
|
||
|
.CreateSuccessors:
|
||
|
; CInfo::CreateSuccessors(bool al=skip,STATE* esi=p1)
|
||
|
push ebx edi
|
||
|
mov ebx, [ebp+.MinContext] ; ebx=pc
|
||
|
mov ecx, [ebp+.FoundState]
|
||
|
mov ecx, [ecx+2] ; ecx=UpBranch
|
||
|
sub esp, .MAX_O*4 ; esp=ps
|
||
|
mov edi, esp ; edi=pps
|
||
|
test al, al
|
||
|
jnz @f
|
||
|
mov eax, [ebp+.FoundState]
|
||
|
stosd
|
||
|
cmp dword [ebx+8], 0
|
||
|
jz .csnoloop
|
||
|
@@:
|
||
|
test esi, esi
|
||
|
jz .csloopstart
|
||
|
mov edx, esi ; edx=p
|
||
|
mov ebx, [ebx+8]
|
||
|
jmp .csloopentry
|
||
|
.csloopstart:
|
||
|
mov ebx, [ebx+8]
|
||
|
lea edx, [ebx+2]
|
||
|
cmp word [ebx], 1
|
||
|
jz .csloopentry
|
||
|
mov edx, [ebx+4]
|
||
|
mov eax, [ebp+.FoundState]
|
||
|
sub edx, 6
|
||
|
mov al, [eax]
|
||
|
@@:
|
||
|
add edx, 6
|
||
|
cmp al, [edx]
|
||
|
jnz @b
|
||
|
.csloopentry:
|
||
|
cmp ecx, [edx+2]
|
||
|
jz @f
|
||
|
mov ebx, [edx+2]
|
||
|
jmp .csnoloop
|
||
|
@@:
|
||
|
mov [edi], edx
|
||
|
add edi, 4
|
||
|
cmp dword [ebx+8], 0
|
||
|
jnz .csloopstart
|
||
|
.csnoloop:
|
||
|
cmp edi, esp
|
||
|
jz .csr
|
||
|
push eax
|
||
|
push eax
|
||
|
mov al, [ecx]
|
||
|
mov [esp], al
|
||
|
add ecx, 1
|
||
|
mov [esp+2], ecx
|
||
|
mov ah, [ebx+3]
|
||
|
cmp word [ebx], 1
|
||
|
jz .cs2
|
||
|
mov edx, [ebx+4]
|
||
|
sub edx, 6
|
||
|
@@:
|
||
|
add edx, 6
|
||
|
cmp [edx], al
|
||
|
jnz @b
|
||
|
movzx edx, byte [edx+1]
|
||
|
sub edx, 1 ; edx=cf
|
||
|
movzx ecx, word [ebx+2]
|
||
|
movzx eax, word [ebx]
|
||
|
sub ecx, eax
|
||
|
sub ecx, edx ; ecx=s0
|
||
|
lea eax, [edx+edx]
|
||
|
cmp eax, ecx
|
||
|
ja .cs0
|
||
|
lea eax, [edx*5]
|
||
|
cmp eax, ecx
|
||
|
seta ah
|
||
|
jmp .cs1
|
||
|
.cs0:
|
||
|
lea eax, [eax+ecx*2]
|
||
|
lea eax, [eax+ecx-1]
|
||
|
add ecx, ecx
|
||
|
xor edx, edx
|
||
|
div ecx
|
||
|
mov ah, al
|
||
|
.cs1:
|
||
|
add ah, 1
|
||
|
.cs2:
|
||
|
mov [esp+1], ah
|
||
|
sub edi, 8
|
||
|
.cs3:
|
||
|
; PPM_CONTEXT::createChild(this=ebx,pStats=[edi+4],FirstState=esp) begin
|
||
|
call .AllocContext
|
||
|
test eax, eax
|
||
|
jz .csr0
|
||
|
mov word [eax], 1
|
||
|
mov dx, [esp]
|
||
|
mov [eax+2], dx
|
||
|
mov edx, [esp+2]
|
||
|
mov [eax+4], edx
|
||
|
mov [eax+8], ebx
|
||
|
mov edx, [edi+4]
|
||
|
mov [edx+2], eax
|
||
|
; PPM_CONTEXT::createChild end
|
||
|
mov ebx, eax
|
||
|
sub edi, 4
|
||
|
cmp edi, esp
|
||
|
jnz .cs3
|
||
|
pop eax eax
|
||
|
.csr:
|
||
|
mov eax, ebx
|
||
|
@@:
|
||
|
add esp, .MAX_O*4
|
||
|
pop edi ebx
|
||
|
ret
|
||
|
.csr0:
|
||
|
pop eax eax
|
||
|
xor eax, eax
|
||
|
jmp @b
|
||
|
|
||
|
; CInfo::UpdateModel(void)
|
||
|
.UpdateModel:
|
||
|
mov ebx, [ebp+.FoundState]
|
||
|
xor esi, esi ; esi=p
|
||
|
movzx eax, word [ebx]
|
||
|
mov ebx, [ebx+2] ; ebx=fs.Successor
|
||
|
push eax
|
||
|
cmp ah, .MAX_FREQ/4
|
||
|
jae .um2
|
||
|
mov eax, [ebp+.MinContext]
|
||
|
mov eax, [eax+8]
|
||
|
test eax, eax
|
||
|
jz .um2
|
||
|
cmp word [eax], 1
|
||
|
jz .um1
|
||
|
push eax
|
||
|
mov esi, [eax+4]
|
||
|
mov al, [esp+4]
|
||
|
cmp al, [esi]
|
||
|
jz .um0
|
||
|
@@:
|
||
|
add esi, 6
|
||
|
cmp al, [esi]
|
||
|
jnz @b
|
||
|
mov al, [esi+1]
|
||
|
cmp al, [esi-6+1]
|
||
|
jb @f
|
||
|
mov eax, [esi]
|
||
|
xchg [esi-6], eax
|
||
|
mov [esi], eax
|
||
|
mov ax, [esi+4]
|
||
|
xchg [esi-6+4], ax
|
||
|
mov [esi+4], ax
|
||
|
sub esi, 6
|
||
|
@@:
|
||
|
.um0:
|
||
|
pop eax
|
||
|
cmp byte [esi+1], .MAX_FREQ-9
|
||
|
jae @f
|
||
|
add byte [esi+1], 2
|
||
|
add word [eax+2], 2
|
||
|
@@:
|
||
|
jmp .um2
|
||
|
.um1:
|
||
|
lea esi, [eax+2]
|
||
|
cmp byte [esi+1], 32
|
||
|
adc byte [esi+1], 0
|
||
|
.um2:
|
||
|
cmp [ebp+.OrderFall], 0
|
||
|
jnz .um3
|
||
|
pop eax
|
||
|
mov al, 1
|
||
|
call .CreateSuccessors
|
||
|
mov [ebp+.MinContext], eax
|
||
|
mov [ebp+.MaxContext], eax
|
||
|
mov edx, [ebp+.FoundState]
|
||
|
mov [edx+2], eax
|
||
|
test eax, eax
|
||
|
jz .RestartModel
|
||
|
ret
|
||
|
.um3:
|
||
|
mov edx, [ebp+.pText]
|
||
|
mov al, [esp]
|
||
|
mov [edx], al
|
||
|
add edx, 1 ; edx=Successor
|
||
|
mov [ebp+.pText], edx
|
||
|
cmp edx, [ebp+.UnitsStart]
|
||
|
jae .RestartModelPop
|
||
|
test ebx, ebx
|
||
|
jz .um4
|
||
|
cmp ebx, [ebp+.pText]
|
||
|
ja @f
|
||
|
push edx
|
||
|
xor eax, eax
|
||
|
call .CreateSuccessors
|
||
|
pop edx
|
||
|
mov ebx, eax
|
||
|
test eax, eax
|
||
|
jz .RestartModelPop
|
||
|
@@:
|
||
|
sub [ebp+.OrderFall], 1
|
||
|
jnz @f
|
||
|
mov edx, ebx
|
||
|
xor ecx, ecx
|
||
|
mov eax, [ebp+.MinContext]
|
||
|
cmp eax, [ebp+.MaxContext]
|
||
|
setnz cl
|
||
|
sub [ebp+.pText], ecx
|
||
|
@@:
|
||
|
jmp .um5
|
||
|
.um4:
|
||
|
mov eax, [ebp+.FoundState]
|
||
|
mov [eax+2], edx
|
||
|
mov ebx, [ebp+.MinContext]
|
||
|
.um5:
|
||
|
mov eax, [ebp+.MinContext]
|
||
|
movzx ecx, word [eax] ; ecx=ns
|
||
|
movzx eax, word [eax+2]
|
||
|
sub eax, ecx
|
||
|
push eax
|
||
|
movzx eax, byte [esp+5]
|
||
|
sub eax, 1
|
||
|
sub [esp], eax ; [esp]=s0
|
||
|
mov esi, [ebp+.MaxContext] ; ebx=pc
|
||
|
cmp esi, [ebp+.MinContext]
|
||
|
jz .um12
|
||
|
.um6:
|
||
|
movzx eax, word [esi]
|
||
|
cmp eax, 1
|
||
|
jz .um8
|
||
|
push eax
|
||
|
shr eax, 1
|
||
|
jc .um7
|
||
|
push esi
|
||
|
mov esi, [esi+4]
|
||
|
call .ExpandUnits
|
||
|
pop esi
|
||
|
mov [esi+4], eax
|
||
|
test eax, eax
|
||
|
jz .RestartModelPop3
|
||
|
.um7:
|
||
|
pop eax
|
||
|
add eax, eax
|
||
|
cmp eax, ecx
|
||
|
adc word [esi+2], 0
|
||
|
add eax, eax
|
||
|
cmp eax, ecx
|
||
|
ja @f
|
||
|
lea eax, [eax+eax+1]
|
||
|
cmp word [esi+2], ax
|
||
|
ja @f
|
||
|
add word [esi+2], 2
|
||
|
@@:
|
||
|
push edx
|
||
|
jmp .um9
|
||
|
.um8:
|
||
|
push edx ecx
|
||
|
mov ecx, 1
|
||
|
call .AllocUnits
|
||
|
pop ecx
|
||
|
test eax, eax
|
||
|
jz .RestartModelPop3
|
||
|
mov dx, [esi+2]
|
||
|
mov [eax], dx
|
||
|
mov edx, [esi+4]
|
||
|
mov [eax+2], edx
|
||
|
mov [esi+4], eax
|
||
|
movzx edx, byte [eax+1]
|
||
|
add edx, edx
|
||
|
cmp edx, (.MAX_FREQ/4-1)*2
|
||
|
jb @f
|
||
|
mov edx, .MAX_FREQ-4
|
||
|
@@:
|
||
|
mov [eax+1], dl
|
||
|
add edx, [ebp+.InitEsc]
|
||
|
cmp ecx, 4
|
||
|
sbb edx, -1
|
||
|
mov [esi+2], dx
|
||
|
.um9:
|
||
|
movzx edx, word [esi+2]
|
||
|
mov eax, [esp+4]
|
||
|
push ecx
|
||
|
lea ecx, [eax+edx] ; ecx=sf
|
||
|
add edx, 6
|
||
|
movzx eax, byte [esp+13]
|
||
|
add eax, eax
|
||
|
imul eax, edx ; eax=cf
|
||
|
lea edx, [ecx*3]
|
||
|
add edx, edx
|
||
|
cmp eax, edx
|
||
|
jae .um10
|
||
|
mov edx, 1
|
||
|
cmp ecx, eax
|
||
|
adc edx, 0
|
||
|
shl ecx, 2
|
||
|
add eax, 1
|
||
|
cmp ecx, eax
|
||
|
adc edx, 0
|
||
|
add word [esi+2], 3
|
||
|
jmp .um11
|
||
|
.um10:
|
||
|
lea ecx, [ecx*3]
|
||
|
lea edx, [ecx*3]
|
||
|
add eax, 1
|
||
|
push 4
|
||
|
cmp edx, eax
|
||
|
adc dword [esp], 0
|
||
|
add edx, ecx
|
||
|
cmp edx, eax
|
||
|
adc dword [esp], 0
|
||
|
add edx, ecx
|
||
|
cmp edx, eax
|
||
|
adc dword [esp], 0
|
||
|
pop edx
|
||
|
add [esi+2], dx
|
||
|
.um11:
|
||
|
movzx eax, word [esi]
|
||
|
lea eax, [eax*3]
|
||
|
add eax, eax
|
||
|
add eax, [esi+4]
|
||
|
mov ecx, [esp+4]
|
||
|
mov [eax+2], ecx
|
||
|
mov cl, [esp+12]
|
||
|
mov [eax], cl
|
||
|
mov [eax+1], dl
|
||
|
add word [esi], 1
|
||
|
pop ecx edx
|
||
|
mov esi, [esi+8]
|
||
|
cmp esi, [ebp+.MinContext]
|
||
|
jnz .um6
|
||
|
.um12:
|
||
|
pop eax
|
||
|
pop eax
|
||
|
mov [ebp+.MinContext], ebx
|
||
|
mov [ebp+.MaxContext], ebx
|
||
|
ret
|
||
|
.RestartModelPop3:
|
||
|
pop eax
|
||
|
pop eax
|
||
|
.RestartModelPop:
|
||
|
pop eax
|
||
|
.RestartModel:
|
||
|
call .RestartModelRare
|
||
|
mov [ebp+.EscCount], 0
|
||
|
mov [ebp+.PrintCount], 0xFF
|
||
|
ret
|
||
|
|
||
|
.rescale:
|
||
|
mov esi, [ebp+.MinContext]
|
||
|
movzx ecx, word [esi]
|
||
|
push ecx ; [esp]=OldNS
|
||
|
sub ecx, 1
|
||
|
mov ebx, [ebp+.FoundState]
|
||
|
cmp ebx, [esi+4]
|
||
|
jz .r1
|
||
|
.r0:
|
||
|
mov ax, [ebx]
|
||
|
xchg [ebx-6], ax
|
||
|
mov [ebx], ax
|
||
|
mov eax, [ebx+2]
|
||
|
xchg [ebx-6+2], eax
|
||
|
mov [ebx+2], eax
|
||
|
sub ebx, 6
|
||
|
cmp ebx, [esi+4]
|
||
|
jnz .r0
|
||
|
.r1:
|
||
|
add byte [ebx+1], 4
|
||
|
add word [esi+2], 4
|
||
|
movzx eax, byte [ebx+1]
|
||
|
movzx edx, word [esi+2]
|
||
|
sub edx, eax ; edx=EscFreq
|
||
|
cmp [ebp+.OrderFall], 1
|
||
|
sbb eax, -1
|
||
|
shr eax, 1
|
||
|
mov [ebx+1], al
|
||
|
mov [esi+2], ax
|
||
|
.r2:
|
||
|
add ebx, 6
|
||
|
movzx eax, byte [ebx+1]
|
||
|
sub edx, eax
|
||
|
cmp [ebp+.OrderFall], 1
|
||
|
sbb eax, -1
|
||
|
shr eax, 1
|
||
|
mov [ebx+1], al
|
||
|
add [esi+2], ax
|
||
|
cmp al, [ebx-6+1]
|
||
|
jbe .r3
|
||
|
push ecx
|
||
|
push ebx
|
||
|
push dword [ebx]
|
||
|
push word [ebx+4]
|
||
|
@@:
|
||
|
mov ecx, [ebx-6]
|
||
|
mov [ebx], ecx
|
||
|
mov cx, [ebx-6+4]
|
||
|
mov [ebx+4], cx
|
||
|
sub ebx, 6
|
||
|
cmp ebx, [esi+4]
|
||
|
jz @f
|
||
|
cmp al, [ebx-6+1]
|
||
|
ja @b
|
||
|
@@:
|
||
|
pop word [ebx+4]
|
||
|
pop dword [ebx]
|
||
|
pop ebx
|
||
|
pop ecx
|
||
|
.r3:
|
||
|
sub ecx, 1
|
||
|
jnz .r2
|
||
|
cmp byte [ebx+1], 0
|
||
|
jnz .r4
|
||
|
@@:
|
||
|
add ecx, 1
|
||
|
sub ebx, 6
|
||
|
cmp byte [ebx+1], 0
|
||
|
jz @b
|
||
|
add edx, ecx
|
||
|
sub word [esi], cx
|
||
|
cmp word [esi], 1
|
||
|
jnz .r4
|
||
|
pop ebx
|
||
|
mov eax, [esi+4]
|
||
|
movzx ecx, word [eax+4]
|
||
|
push ecx
|
||
|
push dword [eax]
|
||
|
movzx eax, byte [eax+1]
|
||
|
@@:
|
||
|
add eax, 1
|
||
|
shr eax, 1
|
||
|
shr edx, 1
|
||
|
cmp edx, 1
|
||
|
ja @b
|
||
|
mov [esp+1], al
|
||
|
add ebx, 1
|
||
|
shr ebx, 1
|
||
|
mov eax, [esi+4]
|
||
|
call .FreeUnits
|
||
|
lea ebx, [esi+2]
|
||
|
mov [ebp+.FoundState], ebx
|
||
|
pop dword [ebx]
|
||
|
pop eax
|
||
|
mov [ebx+4], ax
|
||
|
ret
|
||
|
.r4:
|
||
|
add edx, 1
|
||
|
shr edx, 1
|
||
|
add [esi+2], dx
|
||
|
pop ebx
|
||
|
add ebx, 1
|
||
|
shr ebx, 1
|
||
|
movzx ecx, word [esi]
|
||
|
add ecx, 1
|
||
|
shr ecx, 1
|
||
|
cmp ebx, ecx
|
||
|
jz @f
|
||
|
mov eax, [esi+4]
|
||
|
call .ShrinkUnits
|
||
|
mov [esi+4], eax
|
||
|
@@:
|
||
|
mov eax, [esi+4]
|
||
|
mov [ebp+.FoundState], eax
|
||
|
ret
|
||
|
|
||
|
.DecodeSymbol:
|
||
|
; CDecodeInfo::DecodeSymbol
|
||
|
mov esi, [ebp+.MinContext]
|
||
|
cmp word [esi], 1
|
||
|
jz .binsymbol
|
||
|
; CDecodeInfo::DecodeSymbol1 start
|
||
|
mov ebx, [esi+4] ; state
|
||
|
movzx ecx, word [esi+2]
|
||
|
mov eax, [ebp+.range]
|
||
|
xor edx, edx
|
||
|
div ecx
|
||
|
mov [ebp+.range], eax
|
||
|
mov ecx, eax
|
||
|
mov eax, [ebp+.code]
|
||
|
xor edx, edx
|
||
|
div ecx
|
||
|
movzx edx, byte [ebx+1]
|
||
|
cmp eax, edx
|
||
|
jae .ds0
|
||
|
push edx
|
||
|
add edx, edx
|
||
|
cmp dx, [esi+2]
|
||
|
pop edx
|
||
|
seta [ebp+.PrevSuccess]
|
||
|
movzx eax, [ebp+.PrevSuccess]
|
||
|
add [ebp+.RunLength], eax
|
||
|
xor eax, eax
|
||
|
call .RangeDecoder.Decode
|
||
|
mov [ebp+.FoundState], ebx
|
||
|
add edx, 4
|
||
|
mov [ebx+1], dl
|
||
|
add word [esi+2], 4
|
||
|
cmp edx, .MAX_FREQ
|
||
|
jbe @f
|
||
|
call .rescale
|
||
|
@@:
|
||
|
jmp .dscmn
|
||
|
.ds0:
|
||
|
mov [ebp+.PrevSuccess], 0
|
||
|
movzx ecx, word [esi]
|
||
|
sub ecx, 1
|
||
|
push eax
|
||
|
.ds1:
|
||
|
add ebx, 6
|
||
|
movzx eax, byte [ebx+1]
|
||
|
add edx, eax
|
||
|
cmp edx, [esp]
|
||
|
ja .ds2
|
||
|
sub ecx, 1
|
||
|
jnz .ds1
|
||
|
pop eax
|
||
|
mov eax, [ebp+.FoundState]
|
||
|
movzx eax, byte [eax]
|
||
|
mov al, [.HB2Flag+eax]
|
||
|
mov [ebp+.HiBitsFlag], al
|
||
|
mov eax, edx
|
||
|
movzx edx, word [esi+2]
|
||
|
sub edx, eax
|
||
|
call .RangeDecoder.Decode
|
||
|
mov al, [ebp+.EscCount]
|
||
|
movzx edx, byte [ebx]
|
||
|
mov [ebp+.CharMask+edx], al
|
||
|
movzx ecx, word [esi]
|
||
|
mov [ebp+.NumMasked], ecx
|
||
|
sub ecx, 1
|
||
|
@@:
|
||
|
sub ebx, 6
|
||
|
movzx edx, byte [ebx]
|
||
|
mov [ebp+.CharMask+edx], al
|
||
|
sub ecx, 1
|
||
|
jnz @b
|
||
|
mov [ebp+.FoundState], ecx
|
||
|
jmp .dscmn
|
||
|
.ds2:
|
||
|
pop eax
|
||
|
mov eax, edx
|
||
|
movzx edx, byte [ebx+1]
|
||
|
sub eax, edx
|
||
|
call .RangeDecoder.Decode
|
||
|
.update1:
|
||
|
mov [ebp+.FoundState], ebx
|
||
|
add byte [ebx+1], 4
|
||
|
add word [esi+2], 4
|
||
|
mov al, [ebx+1]
|
||
|
cmp al, [ebx-6+1]
|
||
|
jbe @f
|
||
|
mov eax, [ebx]
|
||
|
xchg eax, [ebx-6]
|
||
|
mov [ebx], eax
|
||
|
mov ax, [ebx+4]
|
||
|
xchg ax, [ebx-6+4]
|
||
|
mov [ebx+4], ax
|
||
|
sub ebx, 6
|
||
|
mov [ebp+.FoundState], ebx
|
||
|
cmp byte [ebx+1], .MAX_FREQ
|
||
|
jbe @f
|
||
|
call .rescale
|
||
|
@@:
|
||
|
jmp .dscmn
|
||
|
; CDecodeInfo::DecodeSymbol1 end
|
||
|
.binsymbol:
|
||
|
; CDecodeInfo::DecodeBinSymbol start
|
||
|
lea ebx, [esi+2]
|
||
|
mov ecx, [esi+8]
|
||
|
movzx ecx, word [ecx]
|
||
|
call .GetBinSumm
|
||
|
movzx eax, word [ecx]
|
||
|
call .RangeDecoder.DecodeBit
|
||
|
jc .ds3
|
||
|
mov [ebp+.FoundState], ebx
|
||
|
cmp byte [ebx+1], 128
|
||
|
adc byte [ebx+1], 0
|
||
|
movzx eax, word [ecx]
|
||
|
add eax, 1 shl (.PERIOD_BITS-2)
|
||
|
shr eax, .PERIOD_BITS
|
||
|
sub eax, .INTERVAL
|
||
|
sub [ecx], ax
|
||
|
mov [ebp+.PrevSuccess], 1
|
||
|
add [ebp+.RunLength], 1
|
||
|
jmp .dscmn
|
||
|
.ds3:
|
||
|
movzx eax, word [ecx]
|
||
|
add eax, 1 shl (.PERIOD_BITS-2)
|
||
|
shr eax, .PERIOD_BITS
|
||
|
sub [ecx], ax
|
||
|
movzx eax, word [ecx]
|
||
|
shr eax, 10
|
||
|
movzx eax, [.ExpEscape+eax]
|
||
|
mov [ebp+.InitEsc], eax
|
||
|
mov [ebp+.NumMasked], 1
|
||
|
mov al, [ebp+.EscCount]
|
||
|
movzx edx, byte [ebx]
|
||
|
mov [ebp+.CharMask+edx], al
|
||
|
mov [ebp+.PrevSuccess], 0
|
||
|
and [ebp+.FoundState], 0
|
||
|
; CDecodeInfo::DecodeBinSymbol end
|
||
|
.dscmn:
|
||
|
cmp [ebp+.FoundState], 0
|
||
|
jnz .dsfnd
|
||
|
.ds4:
|
||
|
add [ebp+.OrderFall], 1
|
||
|
mov eax, [ebp+.MinContext]
|
||
|
mov eax, [eax+8]
|
||
|
test eax, eax
|
||
|
jz return.err ; no end-of-stream mark
|
||
|
mov [ebp+.MinContext], eax
|
||
|
movzx ecx, word [eax]
|
||
|
sub ecx, [ebp+.NumMasked]
|
||
|
jz .ds4
|
||
|
; CDecodeInfo::DecodeSymbol2 start
|
||
|
call .makeEscFreq2
|
||
|
push eax
|
||
|
mov ebx, [esi+4]
|
||
|
sub ebx, 6
|
||
|
sub esp, 256*4
|
||
|
mov esi, esp
|
||
|
xor eax, eax
|
||
|
push eax
|
||
|
@@:
|
||
|
add ebx, 6
|
||
|
mov al, [ebx]
|
||
|
mov al, [ebp+.CharMask+eax]
|
||
|
cmp al, [ebp+.EscCount]
|
||
|
jz @b
|
||
|
mov al, [ebx+1]
|
||
|
add [esp], eax
|
||
|
mov [esi], ebx
|
||
|
add esi, 4
|
||
|
sub ecx, 1
|
||
|
jnz @b
|
||
|
add edx, [esp]
|
||
|
mov ecx, edx
|
||
|
mov eax, [ebp+.range]
|
||
|
xor edx, edx
|
||
|
div ecx
|
||
|
mov [ebp+.range], eax
|
||
|
mov eax, [ebp+.code]
|
||
|
xor edx, edx
|
||
|
div [ebp+.range]
|
||
|
cmp eax, [esp]
|
||
|
jae .ds5
|
||
|
pop ecx
|
||
|
mov esi, esp
|
||
|
xor ecx, ecx
|
||
|
@@:
|
||
|
mov ebx, [esi]
|
||
|
add esi, 4
|
||
|
movzx edx, byte [ebx+1]
|
||
|
add ecx, edx
|
||
|
cmp eax, ecx
|
||
|
jae @b
|
||
|
mov eax, ecx
|
||
|
movzx edx, byte [ebx+1]
|
||
|
sub eax, edx
|
||
|
call .RangeDecoder.Decode
|
||
|
add esp, 256*4
|
||
|
pop eax
|
||
|
mov cl, [eax+2]
|
||
|
cmp cl, .PERIOD_BITS
|
||
|
jae @f
|
||
|
sub byte [eax+3], 1
|
||
|
jnz @f
|
||
|
shl word [eax], 1
|
||
|
mov dl, 3
|
||
|
shl dl, cl
|
||
|
mov [eax+3], dl
|
||
|
add byte [eax+2], 1
|
||
|
@@:
|
||
|
.update2:
|
||
|
mov [ebp+.FoundState], ebx
|
||
|
add byte [ebx+1], 4
|
||
|
mov esi, [ebp+.MinContext]
|
||
|
add word [esi+2], 4
|
||
|
cmp byte [ebx+1], .MAX_FREQ
|
||
|
jbe @f
|
||
|
call .rescale
|
||
|
@@:
|
||
|
add [ebp+.EscCount], 1
|
||
|
mov eax, [ebp+.InitRL]
|
||
|
mov [ebp+.RunLength], eax
|
||
|
jmp .dsfnd
|
||
|
.ds5:
|
||
|
pop eax
|
||
|
mov edx, ecx
|
||
|
sub edx, eax
|
||
|
call .RangeDecoder.Decode
|
||
|
mov eax, [ebp+.MinContext]
|
||
|
movzx eax, word [eax]
|
||
|
mov ebx, eax
|
||
|
sub ebx, [ebp+.NumMasked]
|
||
|
mov [ebp+.NumMasked], eax
|
||
|
mov esi, esp
|
||
|
mov al, [ebp+.EscCount]
|
||
|
@@:
|
||
|
mov edx, [esi]
|
||
|
add esi, 4
|
||
|
movzx edx, byte [edx]
|
||
|
mov [ebp+.CharMask+edx], al
|
||
|
sub ebx, 1
|
||
|
jnz @b
|
||
|
add esp, 256*4
|
||
|
pop eax
|
||
|
add word [eax], cx
|
||
|
; CDecodeInfo::DecodeSymbol2 end
|
||
|
cmp [ebp+.FoundState], 0
|
||
|
jz .ds4
|
||
|
.dsfnd:
|
||
|
mov eax, [ebp+.FoundState]
|
||
|
mov al, [eax]
|
||
|
stosb
|
||
|
|
||
|
.NextContext:
|
||
|
; CInfo::NextContext(void)
|
||
|
mov ebx, [ebp+.FoundState]
|
||
|
mov ebx, [ebx+2]
|
||
|
cmp [ebp+.OrderFall], 0
|
||
|
jnz .nc0
|
||
|
cmp ebx, [ebp+.pText]
|
||
|
jbe .nc0
|
||
|
mov [ebp+.MinContext], ebx
|
||
|
mov [ebp+.MaxContext], ebx
|
||
|
ret
|
||
|
.nc0:
|
||
|
call .UpdateModel
|
||
|
cmp [ebp+.EscCount], 0
|
||
|
jz @f
|
||
|
ret
|
||
|
@@:
|
||
|
mov [ebp+.EscCount], 1
|
||
|
push edi
|
||
|
lea edi, [ebp+.CharMask]
|
||
|
mov ecx, 256/4
|
||
|
xor eax, eax
|
||
|
rep stosd
|
||
|
pop edi
|
||
|
ret
|
||
|
|
||
|
.makeEscFreq2:
|
||
|
; CInfo::makeEscFreq2(ecx=Diff)->{eax->SEE2_CONTEXT,edx=scale}
|
||
|
mov esi, [ebp+.MinContext]
|
||
|
cmp word [esi], 256
|
||
|
jz .mef0
|
||
|
movzx edx, [.NS2Indx+ecx-1]
|
||
|
shl edx, 4
|
||
|
mov eax, [esi+8]
|
||
|
movzx eax, word [eax]
|
||
|
sub ax, [esi]
|
||
|
cmp ecx, eax
|
||
|
adc edx, 0
|
||
|
movzx eax, word [esi]
|
||
|
push edx
|
||
|
lea edx, [eax*9]
|
||
|
lea edx, [edx+eax*2]
|
||
|
movzx eax, word [esi+2]
|
||
|
cmp eax, edx
|
||
|
pop edx
|
||
|
setc al
|
||
|
movzx eax, al
|
||
|
lea edx, [edx+eax*2]
|
||
|
cmp ecx, [ebp+.NumMasked]
|
||
|
setc al
|
||
|
lea edx, [edx+eax*4]
|
||
|
add dl, [ebp+.HiBitsFlag]
|
||
|
lea eax, [ebp+edx*4+.SEE2Cont]
|
||
|
movzx edx, word [eax]
|
||
|
push ecx
|
||
|
mov cl, [eax+2]
|
||
|
shr edx, cl
|
||
|
sub [eax], dx
|
||
|
pop ecx
|
||
|
cmp edx, 1
|
||
|
adc edx, 0
|
||
|
ret
|
||
|
.mef0:
|
||
|
lea eax, [ebp+.DummySEE2Cont]
|
||
|
mov edx, 1
|
||
|
ret
|
||
|
|
||
|
.RangeDecoder.DecodeBit:
|
||
|
; CRangeDecoder::DecodeBit(eax=size0,numTotalBits=.TOT_BITS)
|
||
|
mov edx, [ebp+.range]
|
||
|
shr edx, .TOT_BITS
|
||
|
imul eax, edx
|
||
|
cmp [ebp+.code], eax
|
||
|
jae .rddb
|
||
|
mov [ebp+.range], eax
|
||
|
call .RangeDecoder.Normalize
|
||
|
clc
|
||
|
ret
|
||
|
.rddb:
|
||
|
sub [ebp+.code], eax
|
||
|
sub [ebp+.range], eax
|
||
|
call .RangeDecoder.Normalize
|
||
|
stc
|
||
|
ret
|
||
|
|
||
|
.RangeDecoder.Decode:
|
||
|
imul eax, [ebp+.range]
|
||
|
sub [ebp+.code], eax
|
||
|
mov eax, [ebp+.range]
|
||
|
imul eax, edx
|
||
|
mov [ebp+.range], eax
|
||
|
.RangeDecoder.Normalize:
|
||
|
cmp byte [ebp+.range+3], 0
|
||
|
jz @f
|
||
|
ret
|
||
|
@@:
|
||
|
sub [ebp+.inLen], 1
|
||
|
js .refill
|
||
|
.filled:
|
||
|
shl [ebp+.range], 8
|
||
|
shl [ebp+.code], 8
|
||
|
mov eax, [ebp+.inPtr]
|
||
|
add [ebp+.inPtr], 1
|
||
|
mov al, [eax]
|
||
|
mov byte [ebp+.code], al
|
||
|
jmp .RangeDecoder.Normalize
|
||
|
.refill:
|
||
|
mov eax, [ebp+.inStream]
|
||
|
call fillBuf
|
||
|
push [eax+streamInfo.bufPtr]
|
||
|
pop [ebp+.inPtr]
|
||
|
mov eax, [eax+streamInfo.bufDataLen]
|
||
|
sub eax, 1
|
||
|
js return.err
|
||
|
mov [ebp+.inLen], eax
|
||
|
jmp .filled
|
||
|
|
||
|
.GlueFreeBlocks:
|
||
|
; CSubAllocator::GlueFreeBlocks, called from AllocUnitsRare
|
||
|
push eax
|
||
|
mov [ebp+.GlueCount], 255
|
||
|
mov edx, [ebp+.HeapStart]
|
||
|
add edx, [ebp+.usedMemorySize]
|
||
|
; we need add extra MEM_BLK with Stamp=0
|
||
|
and word [edx], 0
|
||
|
add edx, .UNIT_SIZE
|
||
|
mov eax, [ebp+.LoUnit]
|
||
|
cmp eax, [ebp+.HiUnit]
|
||
|
jz @f
|
||
|
mov byte [eax], 0
|
||
|
@@:
|
||
|
mov [edx+4], edx
|
||
|
mov [edx+8], edx
|
||
|
push ecx
|
||
|
xor ecx, ecx
|
||
|
.gfb1:
|
||
|
mov eax, [ebp+ecx*4+.FreeList]
|
||
|
test eax, eax
|
||
|
jz .gfb2
|
||
|
push dword [eax]
|
||
|
pop dword [ebp+ecx*4+.FreeList]
|
||
|
mov [eax+8], edx
|
||
|
push edx
|
||
|
mov edx, [edx+4]
|
||
|
mov [eax+4], edx
|
||
|
mov [edx+8], eax
|
||
|
or word [eax], 0xFFFF
|
||
|
movzx edx, [.Indx2Units+ecx]
|
||
|
mov [eax+2], dx
|
||
|
pop edx
|
||
|
mov [edx+4], eax
|
||
|
jmp .gfb1
|
||
|
.gfb2:
|
||
|
inc ecx
|
||
|
cmp ecx, .N_INDEXES
|
||
|
jb .gfb1
|
||
|
mov ecx, edx
|
||
|
.gfb3:
|
||
|
mov ecx, [ecx+4]
|
||
|
cmp ecx, edx
|
||
|
jz .gfb5
|
||
|
.gfb4:
|
||
|
movzx eax, word [ecx+2]
|
||
|
lea eax, [eax*3]
|
||
|
lea eax, [ecx+eax*4]
|
||
|
cmp word [eax], 0xFFFF
|
||
|
jnz .gfb3
|
||
|
push eax
|
||
|
mov ax, [eax+2]
|
||
|
add ax, [ecx+2]
|
||
|
pop eax
|
||
|
jc .gfb3
|
||
|
push edx
|
||
|
mov edx, [eax+4]
|
||
|
push dword [eax+8]
|
||
|
pop dword [edx+8]
|
||
|
mov edx, [eax+8]
|
||
|
push dword [eax+4]
|
||
|
pop dword [edx+4]
|
||
|
pop edx
|
||
|
mov ax, [eax+2]
|
||
|
add [ecx+2], ax
|
||
|
jmp .gfb4
|
||
|
.gfb5:
|
||
|
mov ecx, [edx+4]
|
||
|
cmp ecx, edx
|
||
|
jz .gfb8
|
||
|
mov eax, [ecx+4]
|
||
|
mov [eax+8], edx
|
||
|
mov [edx+4], eax
|
||
|
movzx eax, word [ecx+2]
|
||
|
push edx
|
||
|
.gfb6:
|
||
|
sub eax, 128
|
||
|
jbe .gfb7
|
||
|
mov edx, ecx
|
||
|
xchg edx, [ebp+.FreeList+(.N_INDEXES-1)*4]
|
||
|
mov [ecx], edx
|
||
|
add ecx, 128*.UNIT_SIZE
|
||
|
jmp .gfb6
|
||
|
.gfb7:
|
||
|
add eax, 128
|
||
|
movzx edx, [.Units2Indx+eax-1]
|
||
|
cmp [.Indx2Units+edx], al
|
||
|
jz @f
|
||
|
dec edx
|
||
|
push edx
|
||
|
movzx edx, [.Indx2Units+edx]
|
||
|
sub eax, edx
|
||
|
lea eax, [ebp+.FreeList+(eax-1)*4]
|
||
|
lea edx, [edx*3]
|
||
|
lea edx, [ecx+edx*4]
|
||
|
push dword [eax]
|
||
|
pop dword [edx]
|
||
|
mov [eax], edx
|
||
|
pop edx
|
||
|
@@:
|
||
|
mov eax, ecx
|
||
|
xchg eax, [ebp+.FreeList+edx*4]
|
||
|
mov [ecx], eax
|
||
|
pop edx
|
||
|
jmp .gfb5
|
||
|
.gfb8:
|
||
|
pop ecx
|
||
|
pop eax
|
||
|
mov edx, [ebp+.FreeList+eax*4]
|
||
|
test edx, edx
|
||
|
jz .aur.cont
|
||
|
push edx
|
||
|
mov edx, [edx]
|
||
|
mov [ebp+.FreeList+eax*4], edx
|
||
|
pop eax
|
||
|
ret
|
||
|
|
||
|
.AllocContext:
|
||
|
; CSubAllocator::AllocContext
|
||
|
mov eax, [ebp+.HiUnit]
|
||
|
cmp eax, [ebp+.LoUnit]
|
||
|
jz @f
|
||
|
sub eax, .UNIT_SIZE
|
||
|
mov [ebp+.HiUnit], eax
|
||
|
ret
|
||
|
@@:
|
||
|
mov eax, [ebp+.FreeList]
|
||
|
test eax, eax
|
||
|
jz @f
|
||
|
mov edx, [eax]
|
||
|
mov [ebp+.FreeList], edx
|
||
|
ret
|
||
|
@@:
|
||
|
xor eax, eax
|
||
|
jmp .AllocUnitsRare
|
||
|
|
||
|
.AllocUnits:
|
||
|
; CSubAllocator::AllocUnits(ecx)
|
||
|
movzx ecx, [.Units2Indx+ecx-1]
|
||
|
mov eax, [ebp+.FreeList+ecx*4]
|
||
|
test eax, eax
|
||
|
jz @f
|
||
|
mov edx, [eax]
|
||
|
mov [ebp+.FreeList+ecx*4], edx
|
||
|
ret
|
||
|
@@:
|
||
|
mov eax, [ebp+.LoUnit]
|
||
|
movzx edx, [.Indx2Units+ecx]
|
||
|
lea edx, [edx*3]
|
||
|
lea eax, [eax+edx*4]
|
||
|
cmp eax, [ebp+.HiUnit]
|
||
|
ja @f
|
||
|
xchg eax, [ebp+.LoUnit]
|
||
|
ret
|
||
|
@@:
|
||
|
mov eax, ecx
|
||
|
|
||
|
.AllocUnitsRare:
|
||
|
; CSubAllocator::AllocUnitsRare(eax)
|
||
|
cmp [ebp+.GlueCount], 0
|
||
|
jz .GlueFreeBlocks
|
||
|
.aur.cont:
|
||
|
push eax
|
||
|
.aur1:
|
||
|
inc eax
|
||
|
cmp eax, .N_INDEXES
|
||
|
jz .aur3
|
||
|
mov edx, [ebp+.FreeList+eax*4]
|
||
|
test edx, edx
|
||
|
jz .aur1
|
||
|
push edx
|
||
|
mov edx, [edx]
|
||
|
mov [ebp+.FreeList+eax*4], edx
|
||
|
call .SplitBlock
|
||
|
pop eax
|
||
|
pop edx
|
||
|
ret
|
||
|
.aur3:
|
||
|
dec [ebp+.GlueCount]
|
||
|
pop eax
|
||
|
movzx eax, [.Indx2Units+eax]
|
||
|
lea edx, [eax*3]
|
||
|
shl edx, 2
|
||
|
mov eax, [ebp+.UnitsStart]
|
||
|
sub eax, [ebp+.pText]
|
||
|
cmp eax, edx
|
||
|
jbe .aur4
|
||
|
mov eax, [ebp+.UnitsStart]
|
||
|
sub eax, edx
|
||
|
mov [ebp+.UnitsStart], eax
|
||
|
ret
|
||
|
.aur4:
|
||
|
xor eax, eax
|
||
|
ret
|
||
|
|
||
|
.SplitBlock:
|
||
|
; CSubAllocator::SplitBlock(pv=[esp+4],oldIndx=eax,newIndx=[esp+8])
|
||
|
push eax
|
||
|
mov edx, [esp+12]
|
||
|
movzx eax, [.Indx2Units+eax]
|
||
|
movzx edx, [.Indx2Units+edx]
|
||
|
sub eax, edx
|
||
|
lea edx, [edx*3]
|
||
|
push ecx
|
||
|
mov ecx, [esp+12]
|
||
|
lea ecx, [ecx+edx*4]
|
||
|
movzx edx, [.Units2Indx+eax-1]
|
||
|
cmp [.Indx2Units+edx], al
|
||
|
jz .aur2
|
||
|
push dword [ebp+.FreeList+(edx-1)*4]
|
||
|
pop dword [ecx]
|
||
|
mov [ebp+.FreeList+(edx-1)*4], ecx
|
||
|
movzx edx, [.Indx2Units+edx-1]
|
||
|
sub eax, edx
|
||
|
lea edx, [edx*3]
|
||
|
lea ecx, [ecx+edx*4]
|
||
|
.aur2:
|
||
|
movzx eax, [.Units2Indx+eax-1]
|
||
|
push dword [ebp+.FreeList+eax*4]
|
||
|
pop dword [ecx]
|
||
|
mov [ebp+.FreeList+eax*4], ecx
|
||
|
pop ecx
|
||
|
pop eax
|
||
|
ret
|
||
|
|
||
|
.ExpandUnits:
|
||
|
; CSubAllocator::ExpandUnits(void* oldPtr=esi, int oldNU=eax)
|
||
|
push edx
|
||
|
movzx edx, [.Units2Indx + eax - 1]
|
||
|
cmp dl, [.Units2Indx + eax]
|
||
|
jnz @f
|
||
|
pop edx
|
||
|
mov eax, esi
|
||
|
ret
|
||
|
@@:
|
||
|
push eax ecx edx
|
||
|
lea ecx, [eax+1]
|
||
|
call .AllocUnits
|
||
|
pop edx
|
||
|
test eax, eax
|
||
|
jz @f
|
||
|
push esi edi
|
||
|
mov edi, eax
|
||
|
mov ecx, [esp+8+4]
|
||
|
lea ecx, [ecx*3]
|
||
|
rep movsd
|
||
|
pop edi esi
|
||
|
mov ecx, [ebp+.FreeList+edx*4]
|
||
|
mov [esi], ecx
|
||
|
mov [ebp+.FreeList+edx*4], esi
|
||
|
@@:
|
||
|
pop ecx
|
||
|
add esp, 4
|
||
|
pop edx
|
||
|
ret
|
||
|
|
||
|
.ShrinkUnits:
|
||
|
; CSubAllocator::ShrinkUnits(void* oldPtr=eax, int oldNU=ebx, int newNU=ecx)
|
||
|
push ecx
|
||
|
movzx ebx, [.Units2Indx+ebx-1]
|
||
|
movzx ecx, [.Units2Indx+ecx-1]
|
||
|
cmp ebx, ecx
|
||
|
jnz @f
|
||
|
pop ecx
|
||
|
ret
|
||
|
@@:
|
||
|
mov edx, [ebp+.FreeList+ecx*4]
|
||
|
test edx, edx
|
||
|
jz @f
|
||
|
push dword [edx]
|
||
|
pop [ebp+.FreeList+ecx*4]
|
||
|
pop ecx
|
||
|
push esi edi
|
||
|
mov esi, eax
|
||
|
mov edi, edx
|
||
|
lea ecx, [ecx*3]
|
||
|
rep movsd
|
||
|
pop edi esi
|
||
|
mov ecx, [ebp+.FreeList+ebx*4]
|
||
|
mov [eax], ecx
|
||
|
mov [ebp+.FreeList+ebx*4], eax
|
||
|
mov eax, edx
|
||
|
ret
|
||
|
@@:
|
||
|
push ecx
|
||
|
push eax
|
||
|
mov eax, ebx
|
||
|
call .SplitBlock
|
||
|
pop eax
|
||
|
pop ecx
|
||
|
pop ecx
|
||
|
ret
|
||
|
|
||
|
.FreeUnits:
|
||
|
; CSubAllocator::FreeUnits(void* ptr=eax, int oldNU=ebx)
|
||
|
movzx ebx, [.Units2Indx+ebx-1]
|
||
|
push [ebp+.FreeList+ebx*4]
|
||
|
pop dword [eax]
|
||
|
mov [ebp+.FreeList+ebx*4], eax
|
||
|
ret
|
||
|
|
||
|
ppmd_get_buf_size:
|
||
|
cmp dword [esi-4], 5
|
||
|
jb return.err
|
||
|
lodsb
|
||
|
lodsd
|
||
|
cmp eax, ppmd_decoder.kMaxMemBlockSize
|
||
|
ja return.err
|
||
|
add eax, ppmd_decoder.basesize + ppmd_decoder.kExtraSize
|
||
|
mov edx, 0x4000
|
||
|
ret
|
||
|
|
||
|
ppmd_init_decoder:
|
||
|
mov [ebp+ppmd_decoder.bInited], 0
|
||
|
; CDecoder::SetDecoderProperties2
|
||
|
lodsb
|
||
|
mov [ebp+ppmd_decoder.order], al
|
||
|
lodsd
|
||
|
mov [ebp+ppmd_decoder.usedMemorySize], eax
|
||
|
; CSubAllocator::CSubAllocator
|
||
|
xor eax, eax
|
||
|
mov [ebp+ppmd_decoder.GlueCount], al
|
||
|
lea edi, [ebp+ppmd_decoder.LoUnit]
|
||
|
mov ecx, (ppmd_decoder.SEE2Cont - ppmd_decoder.LoUnit)/4
|
||
|
rep stosd
|
||
|
; CSubAllocator::StartSubAllocator
|
||
|
lea eax, [ebp+ppmd_decoder.Base+ppmd_decoder.UNIT_SIZE]
|
||
|
mov [ebp+ppmd_decoder.HeapStart], eax
|
||
|
mov [ebp+streamInfo.fillBuf], ppmd_decoder.fillBuf
|
||
|
ret
|