Coverage Report

Created: 2026-05-27 06:16

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/minizip-ng/third_party/ppmd/C/Ppmd8Dec.c
Line
Count
Source
1
/* Ppmd8Dec.c -- Ppmd8 (PPMdI) Decoder
2
2023-09-07 : Igor Pavlov : Public domain
3
This code is based on:
4
  PPMd var.I (2002): Dmitry Shkarin : Public domain
5
  Carryless rangecoder (1999): Dmitry Subbotin : Public domain */
6
7
#include "Precomp.h"
8
9
#include "Ppmd8.h"
10
11
19.4M
#define kTop ((UInt32)1 << 24)
12
16.7M
#define kBot ((UInt32)1 << 15)
13
14
1.35M
#define READ_BYTE(p) IByteIn_Read((p)->Stream.In)
15
16
BoolInt Ppmd8_Init_RangeDec(CPpmd8 *p)
17
9.45k
{
18
9.45k
  unsigned i;
19
9.45k
  p->Code = 0;
20
9.45k
  p->Range = 0xFFFFFFFF;
21
9.45k
  p->Low = 0;
22
  
23
47.2k
  for (i = 0; i < 4; i++)
24
37.8k
    p->Code = (p->Code << 8) | READ_BYTE(p);
25
9.45k
  return (p->Code < 0xFFFFFFFF);
26
9.45k
}
27
28
#define RC_NORM(p) \
29
9.70M
  while ((p->Low ^ (p->Low + p->Range)) < kTop \
30
9.70M
    || (p->Range < kBot && ((p->Range = (0 - p->Low) & (kBot - 1)), 1))) { \
31
1.31M
      p->Code = (p->Code << 8) | READ_BYTE(p); \
32
1.31M
      p->Range <<= 8; p->Low <<= 8; }
33
34
// we must use only one type of Normalization from two: LOCAL or REMOTE
35
#define RC_NORM_LOCAL(p)    // RC_NORM(p)
36
3.26M
#define RC_NORM_REMOTE(p)   RC_NORM(p)
37
38
34.9M
#define R p
39
40
Z7_FORCE_INLINE
41
// Z7_NO_INLINE
42
static void Ppmd8_RD_Decode(CPpmd8 *p, UInt32 start, UInt32 size)
43
2.83M
{
44
2.83M
  start *= R->Range;
45
2.83M
  R->Low += start;
46
2.83M
  R->Code -= start;
47
2.83M
  R->Range *= size;
48
2.83M
  RC_NORM_LOCAL(R)
49
2.83M
}
50
51
2.83M
#define RC_Decode(start, size)  Ppmd8_RD_Decode(p, start, size);
52
2.03M
#define RC_DecodeFinal(start, size)  RC_Decode(start, size)  RC_NORM_REMOTE(R)
53
2.83M
#define RC_GetThreshold(total)  (R->Code / (R->Range /= (total)))
54
55
56
5.11M
#define CTX(ref) ((CPpmd8_Context *)Ppmd8_GetContext(p, ref))
57
// typedef CPpmd8_Context * CTX_PTR;
58
#define SUCCESSOR(p) Ppmd_GET_SUCCESSOR(p)
59
void Ppmd8_UpdateModel(CPpmd8 *p);
60
61
278M
#define MASK(sym)  ((Byte *)charMask)[sym]
62
63
64
int Ppmd8_DecodeSymbol(CPpmd8 *p)
65
7.15M
{
66
7.15M
  size_t charMask[256 / sizeof(size_t)];
67
68
7.15M
  if (p->MinContext->NumStats != 0)
69
1.60M
  {
70
1.60M
    CPpmd_State *s = Ppmd8_GetStats(p, p->MinContext);
71
1.60M
    unsigned i;
72
1.60M
    UInt32 count, hiCnt;
73
1.60M
    UInt32 summFreq = p->MinContext->Union2.SummFreq;
74
75
1.60M
    PPMD8_CORRECT_SUM_RANGE(p, summFreq)
76
77
78
1.60M
    count = RC_GetThreshold(summFreq);
79
1.60M
    hiCnt = count;
80
    
81
1.60M
    if ((Int32)(count -= s->Freq) < 0)
82
536k
    {
83
536k
      Byte sym;
84
536k
      RC_DecodeFinal(0, s->Freq)
85
536k
      p->FoundState = s;
86
536k
      sym = s->Symbol;
87
536k
      Ppmd8_Update1_0(p);
88
536k
      return sym;
89
536k
    }
90
    
91
1.06M
    p->PrevSuccess = 0;
92
1.06M
    i = p->MinContext->NumStats;
93
    
94
1.06M
    do
95
38.0M
    {
96
38.0M
      if ((Int32)(count -= (++s)->Freq) < 0)
97
573k
      {
98
573k
        Byte sym;
99
573k
        RC_DecodeFinal((hiCnt - count) - s->Freq, s->Freq)
100
573k
        p->FoundState = s;
101
573k
        sym = s->Symbol;
102
573k
        Ppmd8_Update1(p);
103
573k
        return sym;
104
573k
      }
105
38.0M
    }
106
37.4M
    while (--i);
107
    
108
491k
    if (hiCnt >= summFreq)
109
84
      return PPMD8_SYM_ERROR;
110
111
491k
    hiCnt -= count;
112
491k
    RC_Decode(hiCnt, summFreq - hiCnt)
113
    
114
    
115
491k
    PPMD_SetAllBitsIn256Bytes(charMask)
116
    // i = p->MinContext->NumStats - 1;
117
    // do { MASK((--s)->Symbol) = 0; } while (--i);
118
491k
    {
119
491k
      CPpmd_State *s2 = Ppmd8_GetStats(p, p->MinContext);
120
491k
      MASK(s->Symbol) = 0;
121
491k
      do
122
1.09M
      {
123
1.09M
        const unsigned sym0 = s2[0].Symbol;
124
1.09M
        const unsigned sym1 = s2[1].Symbol;
125
1.09M
        s2 += 2;
126
1.09M
        MASK(sym0) = 0;
127
1.09M
        MASK(sym1) = 0;
128
1.09M
      }
129
1.09M
      while (s2 < s);
130
491k
    }
131
491k
  }
132
5.55M
  else
133
5.55M
  {
134
5.55M
    CPpmd_State *s = Ppmd8Context_OneState(p->MinContext);
135
5.55M
    UInt16 *prob = Ppmd8_GetBinSumm(p);
136
5.55M
    UInt32 pr = *prob;
137
5.55M
    UInt32 size0 = (R->Range >> 14) * pr;
138
5.55M
    pr = PPMD_UPDATE_PROB_1(pr);
139
    
140
5.55M
    if (R->Code < size0)
141
5.11M
    {
142
5.11M
      Byte sym;
143
5.11M
      *prob = (UInt16)(pr + (1 << PPMD_INT_BITS));
144
      
145
      // RangeDec_DecodeBit0(size0);
146
5.11M
      R->Range = size0;
147
5.11M
      RC_NORM(R)
148
      
149
      
150
        
151
      // sym = (p->FoundState = Ppmd8Context_OneState(p->MinContext))->Symbol;
152
      // Ppmd8_UpdateBin(p);
153
5.11M
      {
154
5.11M
        unsigned freq = s->Freq;
155
5.11M
        CPpmd8_Context *c = CTX(SUCCESSOR(s));
156
5.11M
        sym = s->Symbol;
157
5.11M
        p->FoundState = s;
158
5.11M
        p->PrevSuccess = 1;
159
5.11M
        p->RunLength++;
160
5.11M
        s->Freq = (Byte)(freq + (freq < 196));
161
        // NextContext(p);
162
5.11M
        if (p->OrderFall == 0 && (const Byte *)c >= p->UnitsStart)
163
4.60M
          p->MaxContext = p->MinContext = c;
164
514k
        else
165
514k
          Ppmd8_UpdateModel(p);
166
5.11M
      }
167
5.11M
      return sym;
168
5.11M
    }
169
    
170
436k
    *prob = (UInt16)pr;
171
436k
    p->InitEsc = p->ExpEscape[pr >> 10];
172
    
173
    // RangeDec_DecodeBit1(rc2, size0);
174
436k
    R->Low += size0;
175
436k
    R->Code -= size0;
176
436k
    R->Range = (R->Range & ~((UInt32)PPMD_BIN_SCALE - 1)) - size0;
177
436k
    RC_NORM_LOCAL(R)
178
    
179
436k
    PPMD_SetAllBitsIn256Bytes(charMask)
180
436k
    MASK(Ppmd8Context_OneState(p->MinContext)->Symbol) = 0;
181
436k
    p->PrevSuccess = 0;
182
436k
  }
183
  
184
928k
  for (;;)
185
1.23M
  {
186
1.23M
    CPpmd_State *s, *s2;
187
1.23M
    UInt32 freqSum, count, hiCnt;
188
1.23M
    UInt32 freqSum2;
189
1.23M
    CPpmd_See *see;
190
1.23M
    CPpmd8_Context *mc;
191
1.23M
    unsigned numMasked;
192
1.23M
    RC_NORM_REMOTE(R)
193
1.23M
    mc = p->MinContext;
194
1.23M
    numMasked = mc->NumStats;
195
    
196
1.23M
    do
197
1.50M
    {
198
1.50M
      p->OrderFall++;
199
1.50M
      if (!mc->Suffix)
200
1.52k
        return PPMD8_SYM_END;
201
1.50M
      mc = Ppmd8_GetContext(p, mc->Suffix);
202
1.50M
    }
203
1.50M
    while (mc->NumStats == numMasked);
204
    
205
1.23M
    s = Ppmd8_GetStats(p, mc);
206
207
1.23M
    {
208
1.23M
      unsigned num = (unsigned)mc->NumStats + 1;
209
1.23M
      unsigned num2 = num / 2;
210
211
1.23M
      num &= 1;
212
1.23M
      hiCnt = (s->Freq & (UInt32)(MASK(s->Symbol))) & (0 - (UInt32)num);
213
1.23M
      s += num;
214
1.23M
      p->MinContext = mc;
215
216
1.23M
      do
217
92.1M
      {
218
92.1M
        const unsigned sym0 = s[0].Symbol;
219
92.1M
        const unsigned sym1 = s[1].Symbol;
220
92.1M
        s += 2;
221
92.1M
        hiCnt += (s[-2].Freq & (UInt32)(MASK(sym0)));
222
92.1M
        hiCnt += (s[-1].Freq & (UInt32)(MASK(sym1)));
223
92.1M
      }
224
92.1M
      while (--num2);
225
1.23M
    }
226
    
227
1.23M
    see = Ppmd8_MakeEscFreq(p, numMasked, &freqSum);
228
1.23M
    freqSum += hiCnt;
229
1.23M
    freqSum2 = freqSum;
230
1.23M
    PPMD8_CORRECT_SUM_RANGE(R, freqSum2)
231
232
233
1.23M
    count = RC_GetThreshold(freqSum2);
234
    
235
1.23M
    if (count < hiCnt)
236
926k
    {
237
926k
      Byte sym;
238
      // Ppmd_See_UPDATE(see) // new (see->Summ) value can overflow over 16-bits in some rare cases
239
926k
      s = Ppmd8_GetStats(p, p->MinContext);
240
926k
      hiCnt = count;
241
242
      
243
926k
      {
244
926k
        for (;;)
245
87.7M
        {
246
87.7M
          count -= s->Freq & (UInt32)(MASK((s)->Symbol)); s++; if ((Int32)count < 0) break;
247
          // count -= s->Freq & (UInt32)(MASK((s)->Symbol)); s++; if ((Int32)count < 0) break;
248
87.7M
        }
249
926k
      }
250
926k
      s--;
251
926k
      RC_DecodeFinal((hiCnt - count) - s->Freq, s->Freq)
252
253
      // new (see->Summ) value can overflow over 16-bits in some rare cases
254
926k
      Ppmd_See_UPDATE(see)
255
926k
      p->FoundState = s;
256
926k
      sym = s->Symbol;
257
926k
      Ppmd8_Update2(p);
258
926k
      return sym;
259
926k
    }
260
261
304k
    if (count >= freqSum2)
262
109
      return PPMD8_SYM_ERROR;
263
    
264
304k
    RC_Decode(hiCnt, freqSum2 - hiCnt)
265
    
266
    // We increase (see->Summ) for sum of Freqs of all non_Masked symbols.
267
    // new (see->Summ) value can overflow over 16-bits in some rare cases
268
304k
    see->Summ = (UInt16)(see->Summ + freqSum);
269
    
270
304k
    s = Ppmd8_GetStats(p, p->MinContext);
271
304k
    s2 = s + p->MinContext->NumStats + 1;
272
304k
    do
273
2.08M
    {
274
2.08M
      MASK(s->Symbol) = 0;
275
2.08M
      s++;
276
2.08M
    }
277
2.08M
    while (s != s2);
278
304k
  }
279
928k
}
280
281
#undef kTop
282
#undef kBot
283
#undef READ_BYTE
284
#undef RC_NORM_BASE
285
#undef RC_NORM_1
286
#undef RC_NORM
287
#undef RC_NORM_LOCAL
288
#undef RC_NORM_REMOTE
289
#undef R
290
#undef RC_Decode
291
#undef RC_DecodeFinal
292
#undef RC_GetThreshold
293
#undef CTX
294
#undef SUCCESSOR
295
#undef MASK