-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathdeshrink_np_r800.asm
234 lines (212 loc) · 4.92 KB
/
deshrink_np_r800.asm
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
; Madram/OVL version / modified roudoudou
; V1: replace D4 (32 bits) by A': 16.03 s
; V2: replace D3 by DE' 14.83 s / 217 bytes
; V3: exx' by default (prepare v4) 14.86 s / 234 bytes
; V4: replace D6 by HL' 14.23 s / 217 bytes #800 buffer
; v5: slight cleanup 14.10 s / 215 bytes
; v6: reintroduce MSB table trick ---------- idem Discarded
; v7: slight getkind factorization 14.23 s / 213 bytes
; v8: replace D5 by HL 14.23 s / 209 bytes
; v9: DE init optimisation 14.23 s / 208 bytes
; vA: no more parity context / 202 bytes
; r800: replace SLL A to ADC A / 202 bytes
; rom: remove self-modification / 202 bytes
; CALL shrinkler_decrunch
; input IX=source
; input DE=destination
shrinkler_decrunch
EXX
LD HL,8*256+probs
LD BC,#0880
XOR A ; start by LSB
init
DEC H
iniloop LD (HL),A
INC L
JR NZ,iniloop ; fill #100 row
XOR C
IFNDEF ROM
LD E,B
ENDIF
DJNZ init
LD D,B
IFNDEF ROM
ELSE
LD E,B
PUSH DE
POP IY
INC E
ENDIF
LD A,C
EX AF,AF' ; A'= #80
; Here HL=probs
literal
SCF
getlit CALL NC,getbit
RL L ; bits = context
JR NC,getlit
LD A,L
EXX
LD (DE),A
INC DE
;after literal
CALL getkind
JR NC,literal
;reference
; l=0 here
LD H,probs_ref / 256
CALL getbit
JR NC,readoffset
readlength
LD H,probs_length / 256
CALL getnumber
PUSH HL ; keep offset for next time
ADD HL,DE
LDIR
POP HL
CALL getkind
JR NC,literal
readoffset
LD H,probs_offset / 256
CALL getnumber
LD HL,#02
SBC HL,BC
EXX
JR NZ,readlength
;END!!!
RET
getnumber
;in: hl point to prob table
;out: bc = number
; EXX' !
_numberloop
INC L
INC L
CALL getbit
JR C,_numberloop
EXX
LD BC,1
EXX
_bitsloop
DEC L
CALL getbit
EXX
RL C
RL B
EXX
DEC L
JR NZ,_bitsloop
EXX
RET
;-----------------------------------
readbit
SLA E
RL D ; d3 <<= 1
EX AF,AF'
IFNDEF ROM
ELSE
ADD IY,IY
ENDIF
ADD A
JR NZ,_rbok
LD A,(IX)
INC IX
ADC A
_rbok
IFNDEF ROM
PUSH HL
d2 LD HL,#00 ; TODO replace by IY
ADC HL,HL
LD (d2+1),HL
POP HL
ELSE
JR NC,iync
INC IY
iync
ENDIF
EX AF,AF'
JR getbit
getkind
; do not use parity as context anymoar
EXX
LD HL,probs
getbit
;In: hl points to context prob
; de = d3
;Out: hl preserved
; de = new d3 value
LD A,D ; d3
ADD A
JR NC,readbit
LD B,(HL)
INC H
LD C,(HL)
PUSH HL
;d1 = one prob
LD L,C
LD H,B ; bc=hl=d1
PUSH BC
LD A,%11100001 ;MSQ: cpt 4
shift4
SRL B
RR C
ADD A
JR C,shift4
SBC HL,BC ; hl=d1-d1/16
POP BC
PUSH HL
;here, A = 16
SBC HL,HL
; bchl = bc*de
muluw
ADD HL,HL
RL C
RL B
JR NC,_mulcont
ADD HL,DE
JR NC,_mulcont
INC BC
_mulcont
DEC A
JR NZ,muluw
IFNDEF ROM
LD HL,(d2+1)
ELSE
PUSH IY
POP HL
ENDIF
OR A
SBC HL,BC
JR NC,zero
one
LD E,C
LD D,B ; d3 = d3*d1
POP BC
; + #fff
LD A,B
SUB #F0 ; force Carry
LD B,A
DEC BC
JR _probret
zero
IFNDEF ROM
LD (d2+1),HL
ELSE
PUSH HL
POP IY
ENDIF
EX DE,HL
SBC HL,BC ; d3. still NC after
EX DE,HL
POP BC
_probret
POP HL
LD (HL),C
DEC H
LD (HL),B
RET
probs=($+256)&#FF00
;+#200: for odd context
probs_ref = probs+#400 ; one word
probs_length = probs_ref ; starts at +1
probs_offset = probs_length + #200 ; could be + #40 by changing test