Coverage Report

Created: 2026-05-30 06:10

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/work/vvenc/source/Lib/CommonLib/ContextModelling.h
Line
Count
Source
1
/* -----------------------------------------------------------------------------
2
The copyright in this software is being made available under the Clear BSD
3
License, included below. No patent rights, trademark rights and/or 
4
other Intellectual Property Rights other than the copyrights concerning 
5
the Software are granted under this license.
6
7
The Clear BSD License
8
9
Copyright (c) 2019-2026, Fraunhofer-Gesellschaft zur Förderung der angewandten Forschung e.V. & The VVenC Authors.
10
All rights reserved.
11
12
Redistribution and use in source and binary forms, with or without modification,
13
are permitted (subject to the limitations in the disclaimer below) provided that
14
the following conditions are met:
15
16
     * Redistributions of source code must retain the above copyright notice,
17
     this list of conditions and the following disclaimer.
18
19
     * Redistributions in binary form must reproduce the above copyright
20
     notice, this list of conditions and the following disclaimer in the
21
     documentation and/or other materials provided with the distribution.
22
23
     * Neither the name of the copyright holder nor the names of its
24
     contributors may be used to endorse or promote products derived from this
25
     software without specific prior written permission.
26
27
NO EXPRESS OR IMPLIED LICENSES TO ANY PARTY'S PATENT RIGHTS ARE GRANTED BY
28
THIS LICENSE. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND
29
CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
30
LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
31
PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
32
CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
33
EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
34
PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
35
BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER
36
IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
37
ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
38
POSSIBILITY OF SUCH DAMAGE.
39
40
41
------------------------------------------------------------------------------------------- */
42
/** \file     ContextModelling.h
43
 *  \brief    Classes providing probability descriptions and contexts (header)
44
 */
45
46
#pragma once
47
48
#include "CommonDef.h"
49
#include "Contexts.h"
50
#include "Slice.h"
51
#include "Unit.h"
52
#include "UnitPartitioner.h"
53
54
#include <bitset>
55
56
//! \ingroup CommonLib
57
//! \{
58
59
namespace vvenc
60
{
61
62
struct CtxTpl
63
{
64
  // lower 5 bits are absSum1, upper 3 bits are numPos
65
  uint8_t ctxTpl;
66
};
67
68
struct CoeffCodingContext
69
{
70
public:
71
  CoeffCodingContext( const TransformUnit& tu, ComponentID component, bool signHide, bool bdpcm = false, CtxTpl* tplBuf = nullptr );
72
public:
73
  void  initSubblock    ( int SubsetId, bool sigGroupFlag = false );
74
public:
75
12.3k
  void  resetSigGroup   ()                      { m_sigCoeffGroupFlag.reset( m_subSetPos ); }
76
482k
  void  setSigGroup     ()                      { m_sigCoeffGroupFlag.set( m_subSetPos ); }
77
0
  bool  noneSigGroup    ()                      { return m_sigCoeffGroupFlag.none(); }
78
475k
  int   lastSubSet      ()                      { return ( maxNumCoeff() - 1 ) >> log2CGSize(); }
79
405k
  bool  isLastSubSet    ()                      { return lastSubSet() == m_subSetId; }
80
70.1k
  bool  only1stSigGroup ()                      { return m_sigCoeffGroupFlag.count()-m_sigCoeffGroupFlag[lastSubSet()]==0; }
81
941k
  void  setScanPosLast  ( int       posLast )   { m_scanPosLast = posLast; }
82
public:
83
0
  ComponentID     compID          ()                        const { return m_compID; }
84
670k
  int             subSetId        ()                        const { return m_subSetId; }
85
0
  int             subSetPos       ()                        const { return m_subSetPos; }
86
46.9k
  int             cgPosY          ()                        const { return m_subSetPosY; }
87
46.9k
  int             cgPosX          ()                        const { return m_subSetPosX; }
88
1.88M
  unsigned        width           ()                        const { return m_width; }
89
0
  unsigned        height          ()                        const { return m_height; }
90
6.05k
  unsigned        log2CGWidth     ()                        const { return m_log2CGWidth; }
91
0
  unsigned        log2CGHeight    ()                        const { return m_log2CGHeight; }
92
6.75M
  unsigned        log2CGSize      ()                        const { return m_log2CGSize; }
93
5.49M
  int             maxLog2TrDRange ()                        const { return m_maxLog2TrDynamicRange; }
94
7.68M
  unsigned        maxNumCoeff     ()                        const { return m_maxNumCoeff; }
95
13.4M
  int             scanPosLast     ()                        const { return m_scanPosLast; }
96
12.1M
  int             minSubPos       ()                        const { return m_minSubPos; }
97
433k
  int             maxSubPos       ()                        const { return m_maxSubPos; }
98
969k
  bool            isLast          ()                        const { return ( ( m_scanPosLast >> m_log2CGSize ) == m_subSetId ); }
99
46.2k
  bool            isNotFirst      ()                        const { return ( m_subSetId != 0 ); }
100
0
  bool            isSigGroup      ( int scanPosCG )         const { return m_sigCoeffGroupFlag[m_scanCG[scanPosCG].idx]; }
101
1.13M
  bool            isSigGroup      ()                        const { return m_sigCoeffGroupFlag[ m_subSetPos ]; }
102
9.68k
  bool            signHiding      ()                        const { return m_signHiding; }
103
  bool            hideSign        ( int       posFirst,
104
959k
                                    int       posLast   )   const { return ( m_signHiding && ( posLast - posFirst >= SBH_THRESHOLD ) ); }
105
60.4M
  unsigned        blockPos        ( int scanPos )           const { return m_scan[scanPos].idx; }
106
10.2M
  unsigned        posX            ( int scanPos )           const { return m_scan[scanPos].x; }
107
10.2M
  unsigned        posY            ( int scanPos )           const { return m_scan[scanPos].y; }
108
941k
  unsigned        maxLastPosX     ()                        const { return m_maxLastPosX; }
109
941k
  unsigned        maxLastPosY     ()                        const { return m_maxLastPosY; }
110
2.15M
  unsigned        lastXCtxId      ( unsigned  posLastX  )   const { return m_CtxSetLastX( m_lastOffsetX + ( posLastX >> m_lastShiftX ) ); }
111
2.20M
  unsigned        lastYCtxId      ( unsigned  posLastY  )   const { return m_CtxSetLastY( m_lastOffsetY + ( posLastY >> m_lastShiftY ) ); }
112
1.07M
  unsigned        sigGroupCtxId   ( bool ts = false     )   const { return ts ? m_sigGroupCtxIdTS : m_sigGroupCtxId; }
113
6.73M
  bool            bdpcm           ()                        const { return m_bdpcm; }
114
115
  unsigned sigCtxIdAbs( int scanPos, const TCoeffSig* coeff, const int state )
116
0
  {
117
0
    const uint32_t   posY      = m_scan[scanPos].y;
118
0
    const uint32_t   posX      = m_scan[scanPos].x;
119
0
    const TCoeffSig* pData     = coeff + posX + posY * m_width;
120
0
    const int        diag      = posX + posY;
121
0
    int              numPos    = 0;
122
0
    int              sumAbs    = 0;
123
0
#define UPDATE(x) {int a=abs(x);sumAbs+=std::min(4+(a&1),a);numPos+=!!a;}
124
0
    if( posX < m_width-1 )
125
0
    {
126
0
      UPDATE( pData[1] );
127
0
      if( posX < m_width-2 )
128
0
      {
129
0
        UPDATE( pData[2] );
130
0
      }
131
0
      if( posY < m_height-1 )
132
0
      {
133
0
        UPDATE( pData[m_width+1] );
134
0
      }
135
0
    }
136
0
    if( posY < m_height-1 )
137
0
    {
138
0
      UPDATE( pData[m_width] );
139
0
      if( posY < m_height-2 )
140
0
      {
141
0
        UPDATE( pData[m_width<<1] );
142
0
      }
143
0
    }
144
0
#undef UPDATE
145
146
0
    int ctxOfs = std::min((sumAbs+1)>>1, 3) + ( diag < 2 ? 4 : 0 );
147
148
0
    if( m_chType == CH_L )
149
0
    {
150
0
      ctxOfs += diag < 5 ? 4 : 0;
151
0
    }
152
153
0
    m_tmplCpDiag = diag;
154
0
    m_tmplCpSum1 = sumAbs - numPos;
155
0
    return m_sigFlagCtxSet[std::max( 0, state-1 )]( ctxOfs );
156
0
  }
157
158
  unsigned sigCtxIdAbsWithAcc( const int scanPos, const int state )
159
6.71M
  {
160
6.71M
    const auto scanEl       = m_scan[scanPos];
161
6.71M
    const uint32_t posY     = scanEl.y;
162
6.71M
    const uint32_t posX     = scanEl.x;
163
6.71M
    const int32_t blkPos    = scanEl.idx;
164
6.71M
    const int      diag     = posX + posY;
165
6.71M
    const int      tplVal   = m_tplBuf[-blkPos].ctxTpl;
166
6.71M
    const int      numPos   = tplVal >> 5u;
167
6.71M
    const int      sumAbs   = tplVal & 31;
168
169
6.71M
    int ctxOfs = std::min( ( sumAbs + 1 ) >> 1, 3 ) + ( diag < 2 ? 4 : 0 );
170
171
6.71M
    if( isLuma( m_chType ) )
172
457k
    {
173
457k
      ctxOfs += diag < 5 ? 4 : 0;
174
457k
    }
175
6.71M
    m_tmplCpDiag = diag;
176
6.71M
    m_tmplCpSum1 = sumAbs - numPos;
177
6.71M
    return m_sigFlagCtxSet[std::max( 0, state-1 )]( ctxOfs );
178
6.71M
  }
179
180
  void absVal1stPass( const int scanPos, const TCoeffSig absLevel1 )
181
7.05M
  {
182
7.05M
    CHECKD( !absLevel1, "Shound not be called if '0'!" );
183
184
7.05M
    const auto scanEl     = m_scan[scanPos];
185
7.05M
    const uint32_t posY   = scanEl.y;
186
7.05M
    const uint32_t posX   = scanEl.x;
187
7.05M
    const int32_t  blkPos = scanEl.idx;
188
189
7.05M
    auto update_deps = [&]( int offset )
190
18.4M
    {
191
18.4M
      auto& ctx   = m_tplBuf[-blkPos + offset];
192
18.4M
      ctx.ctxTpl += uint8_t( 32 + absLevel1 );
193
18.4M
    };
194
195
7.05M
    if( posY > 1 ) update_deps( 2 * m_width );
196
7.05M
    if( posY > 0
197
4.74M
     && posX > 0 ) update_deps( m_width + 1 );
198
7.05M
    if( posY > 0 ) update_deps( m_width );
199
7.05M
    if( posX > 1 ) update_deps( 2 );
200
7.05M
    if( posX > 0 ) update_deps( 1 );
201
7.05M
  }
202
  
203
204
  void remAbsVal1stPass( const int scanPos, const TCoeffSig absLevel1 )
205
0
  {
206
0
    CHECKD( !absLevel1, "Shound not be called if '0'!" );
207
208
0
    const auto scanEl     = m_scan[scanPos];
209
0
    const uint32_t posY   = scanEl.y;
210
0
    const uint32_t posX   = scanEl.x;
211
0
    const int32_t  blkPos = scanEl.idx;
212
213
0
    auto update_deps = [&]( int offset )
214
0
    {
215
0
      auto& ctx   = m_tplBuf[-blkPos + offset];
216
0
      ctx.ctxTpl -= uint8_t( 32 + absLevel1 );
217
0
    };
218
219
0
    if( posY > 1 ) update_deps( 2 * m_width );
220
0
    if( posY > 0
221
0
     && posX > 0 ) update_deps( m_width + 1 );
222
0
    if( posY > 0 ) update_deps( m_width );
223
0
    if( posX > 1 ) update_deps( 2 );
224
0
    if( posX > 0 ) update_deps( 1 );
225
0
  }
226
227
  uint8_t ctxOffsetAbs()
228
7.05M
  {
229
7.05M
    int offset = 0;
230
7.05M
    if( m_tmplCpDiag != -1 )
231
6.11M
    {
232
6.11M
      offset  = std::min( m_tmplCpSum1, 4 ) + 1;
233
6.11M
      offset += ( !m_tmplCpDiag ? ( m_chType == CH_L ? 15 : 5 ) : m_chType == CH_L ? m_tmplCpDiag < 3 ? 10 : ( m_tmplCpDiag < 10 ? 5 : 0 ) : 0 );
234
6.11M
    }
235
7.05M
    return uint8_t(offset);
236
7.05M
  }
237
238
6.09M
  unsigned parityCtxIdAbs   ( uint8_t offset )  const { return m_parFlagCtxSet   ( offset ); }
239
7.05M
  unsigned greater1CtxIdAbs ( uint8_t offset )  const { return m_gtxFlagCtxSet[1]( offset ); }
240
6.09M
  unsigned greater2CtxIdAbs ( uint8_t offset )  const { return m_gtxFlagCtxSet[0]( offset ); }
241
242
  unsigned templateAbsSum( int scanPos, const TCoeffSig* coeff, int baseLevel )
243
4.94M
  {
244
4.94M
    const uint32_t   posY = m_scan[scanPos].y;
245
4.94M
    const uint32_t   posX = m_scan[scanPos].x;
246
4.94M
    const TCoeffSig* pData = coeff + posX + posY * m_width;
247
4.94M
    int              sum   = 0;
248
4.94M
    if (posX < m_width - 1)
249
4.85M
    {
250
4.85M
      sum += abs(pData[1]);
251
4.85M
      if (posX < m_width - 2)
252
4.67M
      {
253
4.67M
        sum += abs(pData[2]);
254
4.67M
      }
255
4.85M
      if (posY < m_height - 1)
256
4.71M
      {
257
4.71M
        sum += abs(pData[m_width + 1]);
258
4.71M
      }
259
4.85M
    }
260
4.94M
    if (posY < m_height - 1)
261
4.80M
    {
262
4.80M
      sum += abs(pData[m_width]);
263
4.80M
      if (posY < m_height - 2)
264
4.57M
      {
265
4.57M
        sum += abs(pData[m_width << 1]);
266
4.57M
      }
267
4.80M
    }
268
4.94M
    return std::max(std::min(sum - 5 * baseLevel, 31), 0);
269
4.94M
  }
270
271
  unsigned sigCtxIdAbsTS( int scanPos, const TCoeffSig* coeff )
272
13.0M
  {
273
13.0M
    const uint32_t   posY   = m_scan[scanPos].y;
274
13.0M
    const uint32_t   posX   = m_scan[scanPos].x;
275
13.0M
    const TCoeffSig* posC   = coeff + posX + posY * m_width;
276
13.0M
    int              numPos = 0;
277
23.0M
#define UPDATE(x) {int a=abs(x);numPos+=!!a;}
278
13.0M
    if( posX > 0 )
279
11.7M
    {
280
11.7M
      UPDATE( posC[-1] );
281
11.7M
    }
282
13.0M
    if( posY > 0 )
283
11.3M
    {
284
11.3M
      UPDATE( posC[-(int)m_width] );
285
11.3M
    }
286
13.0M
#undef UPDATE
287
288
13.0M
    return m_tsSigFlagCtxSet( numPos );
289
13.0M
  }
290
291
11.2M
  unsigned parityCtxIdAbsTS   ()                  const { return m_tsParFlagCtxSet(      0 ); }
292
3.78M
  unsigned greaterXCtxIdAbsTS ( uint8_t offset )  const { return m_tsGtxFlagCtxSet( offset ); }
293
294
  unsigned lrg1CtxIdAbsTS(int scanPos, const TCoeffSig* coeff, int bdpcm)
295
11.2M
  {
296
11.2M
    const uint32_t   posY = m_scan[scanPos].y;
297
11.2M
    const uint32_t   posX = m_scan[scanPos].x;
298
11.2M
    const TCoeffSig* posC = coeff + posX + posY * m_width;
299
300
11.2M
    int             numPos = 0;
301
11.2M
#define UPDATE(x) {int a=abs(x);numPos+=!!a;}
302
303
11.2M
    if (bdpcm)
304
10.7M
    {
305
10.7M
      numPos = 3;
306
10.7M
    }
307
509k
    else
308
509k
    {
309
509k
      if (posX > 0)
310
470k
      {
311
470k
        UPDATE(posC[-1]);
312
470k
      }
313
509k
      if (posY > 0)
314
468k
      {
315
468k
        UPDATE(posC[-(int)m_width]);
316
468k
      }
317
509k
    }
318
319
11.2M
#undef UPDATE
320
11.2M
    return m_tsLrg1FlagCtxSet(numPos);
321
11.2M
  }
322
323
  unsigned signCtxIdAbsTS(int scanPos, const TCoeffSig* coeff, int bdpcm)
324
11.2M
  {
325
11.2M
    const uint32_t   posY  = m_scan[scanPos].y;
326
11.2M
    const uint32_t   posX  = m_scan[scanPos].x;
327
11.2M
    const TCoeffSig* pData = coeff + posX + posY * m_width;
328
329
11.2M
    int rightSign = 0, belowSign = 0;
330
11.2M
    unsigned signCtx = 0;
331
332
11.2M
    if (posX > 0)
333
10.1M
    {
334
10.1M
      rightSign = pData[-1];
335
10.1M
    }
336
11.2M
    if (posY > 0)
337
9.60M
    {
338
9.60M
      belowSign = pData[-(int)m_width];
339
9.60M
    }
340
341
11.2M
    if ((rightSign == 0 && belowSign == 0) || ((rightSign*belowSign) < 0))
342
9.80M
    {
343
9.80M
      signCtx = 0;
344
9.80M
    }
345
1.44M
    else if (rightSign >= 0 && belowSign >= 0)
346
564
    {
347
564
      signCtx = 1;
348
564
    }
349
1.44M
    else
350
1.44M
    {
351
1.44M
      signCtx = 2;
352
1.44M
    }
353
11.2M
    if (bdpcm)
354
10.7M
    {
355
10.7M
      signCtx += 3;
356
10.7M
    }
357
11.2M
    return m_tsSignFlagCtxSet(signCtx);
358
11.2M
  }
359
360
  void neighTS(int& rightPixel, int& belowPixel, int scanPos, const TCoeffSig* coeff)
361
16.0M
  {
362
16.0M
    const uint32_t   posY = m_scan[scanPos].y;
363
16.0M
    const uint32_t   posX = m_scan[scanPos].x;
364
16.0M
    const TCoeffSig* data = coeff + posX + posY * m_width;
365
366
16.0M
    rightPixel = belowPixel = 0;
367
368
16.0M
    if (posX > 0)
369
14.4M
    {
370
14.4M
      rightPixel = data[-1];
371
14.4M
    }
372
16.0M
    if (posY > 0)
373
13.1M
    {
374
13.1M
      belowPixel = data[-(int)m_width];
375
13.1M
    }
376
16.0M
  }
377
378
  int deriveModCoeff(int rightPixel, int belowPixel, int absCoeff, int bdpcm = 0)
379
8.33M
  {
380
8.33M
    if (absCoeff == 0)
381
4.08M
      return 0;
382
383
4.24M
    int pred1, absBelow = abs(belowPixel), absRight = abs(rightPixel);
384
385
4.24M
    int absCoeffMod = absCoeff;
386
387
4.24M
    if (bdpcm == 0)
388
540k
    {
389
540k
      pred1 = std::max(absBelow, absRight);
390
391
540k
      if (absCoeff == pred1)
392
0
      {
393
0
        absCoeffMod = 1;
394
0
      }
395
540k
      else
396
540k
      {
397
540k
        absCoeffMod = absCoeff < pred1 ? absCoeff + 1 : absCoeff;
398
540k
      }
399
540k
    }
400
401
4.24M
    return(absCoeffMod);
402
8.33M
  }
403
404
  int decDeriveModCoeff(int rightPixel, int belowPixel, int absCoeff)
405
0
  {
406
0
    if (absCoeff == 0)
407
0
      return 0;
408
0
409
0
    int pred1, absBelow = abs(belowPixel), absRight = abs(rightPixel);
410
0
    pred1 = std::max(absBelow, absRight);
411
0
412
0
    int absCoeffMod;
413
0
414
0
    if (absCoeff == 1 && pred1 > 0)
415
0
    {
416
0
      absCoeffMod = pred1;
417
0
    }
418
0
    else
419
0
    {
420
0
      absCoeffMod = absCoeff - (absCoeff <= pred1);
421
0
    }
422
0
    return(absCoeffMod);
423
0
  }
424
425
  //unsigned templateAbsSumTS( int scanPos, const TCoeffSig* coeff )
426
  //{
427
  //  return 1;
428
  //}
429
430
  int                       remRegBins;
431
432
private:
433
  // constant
434
  const ComponentID         m_compID;
435
  const ChannelType         m_chType;
436
  const unsigned            m_width;
437
  const unsigned            m_height;
438
  const unsigned            m_log2CGWidth;
439
  const unsigned            m_log2CGHeight;
440
  const unsigned            m_log2CGSize;
441
  const unsigned            m_widthInGroups;
442
  const unsigned            m_heightInGroups;
443
  const unsigned            m_log2WidthInGroups;
444
  const unsigned            m_log2BlockWidth;
445
  const unsigned            m_log2BlockHeight;
446
  const unsigned            m_maxNumCoeff;
447
  const bool                m_signHiding;
448
  const int                 m_maxLog2TrDynamicRange;
449
  const ScanElement *       m_scan;
450
  const ScanElement *       m_scanCG;
451
  const CtxSet              m_CtxSetLastX;
452
  const CtxSet              m_CtxSetLastY;
453
  const unsigned            m_maxLastPosX;
454
  const unsigned            m_maxLastPosY;
455
  const int                 m_lastOffsetX;
456
  const int                 m_lastOffsetY;
457
  const int                 m_lastShiftX;
458
  const int                 m_lastShiftY;
459
  // modified
460
  int                       m_scanPosLast;
461
  int                       m_subSetId;
462
  int                       m_subSetPos;
463
  int                       m_subSetPosX;
464
  int                       m_subSetPosY;
465
  int                       m_minSubPos;
466
  int                       m_maxSubPos;
467
  unsigned                  m_sigGroupCtxId;
468
  int                       m_tmplCpSum1;
469
  int                       m_tmplCpDiag;
470
  CtxSet                    m_sigFlagCtxSet[3];
471
  CtxSet                    m_parFlagCtxSet;
472
  CtxSet                    m_gtxFlagCtxSet[2];
473
  unsigned                  m_sigGroupCtxIdTS;
474
  CtxSet                    m_tsSigFlagCtxSet;
475
  CtxSet                    m_tsParFlagCtxSet;
476
  CtxSet                    m_tsGtxFlagCtxSet;
477
  CtxSet                    m_tsLrg1FlagCtxSet;
478
  CtxSet                    m_tsSignFlagCtxSet;
479
  std::bitset<MLS_GRP_NUM>  m_sigCoeffGroupFlag;
480
  const bool                m_bdpcm;
481
  CtxTpl*                   m_tplBuf;
482
};
483
484
485
struct CUCtx
486
{
487
464k
  CUCtx()              : isDQPCoded(false), isChromaQpAdjCoded(false),
488
464k
                         qgStart(false)
489
464k
                         {
490
464k
                           violatesLfnstConstrained[CH_L] = false;
491
464k
                           violatesLfnstConstrained[CH_C] = false;
492
464k
                           violatesMtsCoeffConstraint      = false;
493
464k
                           mtsLastScanPos                  = false;
494
464k
                           lfnstLastScanPos                = false;
495
464k
                         }
496
16.2k
  CUCtx(int _qp)       : isDQPCoded(false), isChromaQpAdjCoded(false),
497
16.2k
                         qgStart(false),
498
16.2k
                         qp(_qp)
499
16.2k
                         {
500
16.2k
                           violatesLfnstConstrained[CH_L] = false;
501
16.2k
                           violatesLfnstConstrained[CH_C] = false;
502
16.2k
                           violatesMtsCoeffConstraint      = false;
503
16.2k
                           mtsLastScanPos                  = false;
504
16.2k
                           lfnstLastScanPos                = false;
505
16.2k
                         }
506
507
  bool      isDQPCoded;
508
  bool      isChromaQpAdjCoded;
509
  bool      qgStart;
510
  bool      lfnstLastScanPos;
511
  int8_t    qp;                   // used as a previous(last) QP and for QP prediction
512
  bool      violatesLfnstConstrained[MAX_NUM_CH];
513
  bool      violatesMtsCoeffConstraint;
514
  bool      mtsLastScanPos;
515
};
516
517
struct MergeCtx
518
{
519
861k
  MergeCtx() : numValidMergeCand( 0 ), hasMergedCandList( false ) { for( unsigned i = 0; i < MRG_MAX_NUM_CANDS; i++ ) mrgTypeNeighbours[i] = MRG_TYPE_DEFAULT_N; }
520
521
  void          setMmvdMergeCandiInfo ( CodingUnit& cu, const MmvdIdx candIdx) const;
522
  void          setMergeInfo          ( CodingUnit& cu, const int     candIdx) const;
523
  void          getMmvdDeltaMv        ( const Slice& slice, const MmvdIdx candIdx, Mv deltaMv[NUM_REF_PIC_LIST_01] ) const;
524
525
  MvField       mvFieldNeighbours [MRG_MAX_NUM_CANDS][NUM_REF_PIC_LIST_01];
526
  uint8_t       BcwIdx            [MRG_MAX_NUM_CANDS];
527
  unsigned char interDirNeighbours[MRG_MAX_NUM_CANDS];
528
  MergeType     mrgTypeNeighbours [MRG_MAX_NUM_CANDS];
529
  int           numValidMergeCand;
530
  bool          hasMergedCandList;
531
532
  MvField       mmvdBaseMv        [MMVD_BASE_MV_NUM][NUM_REF_PIC_LIST_01];
533
  bool          mmvdUseAltHpelIf  [MMVD_BASE_MV_NUM];
534
  bool          useAltHpelIf      [MRG_MAX_NUM_CANDS];
535
};
536
537
struct AffineMergeCtx
538
{
539
0
  AffineMergeCtx() : numValidMergeCand( 0 ) { for ( unsigned i = 0; i < AFFINE_MRG_MAX_NUM_CANDS; i++ ) affineType[i] = AFFINEMODEL_4PARAM; }
540
541
  MvField       mvFieldNeighbours [AFFINE_MRG_MAX_NUM_CANDS][NUM_REF_PIC_LIST_01][3];
542
  unsigned char interDirNeighbours[AFFINE_MRG_MAX_NUM_CANDS];
543
  EAffineModel  affineType        [AFFINE_MRG_MAX_NUM_CANDS];
544
  uint8_t       BcwIdx            [AFFINE_MRG_MAX_NUM_CANDS];
545
  int           numValidMergeCand;
546
  int           maxNumMergeCand;
547
548
  MergeType     mergeType[AFFINE_MRG_MAX_NUM_CANDS];
549
  MotionBuf     subPuMvpMiBuf;
550
};
551
552
553
struct DeriveCtx
554
{
555
  const CodingUnit* cuRestrictedLeft[MAX_NUM_CH];
556
  const CodingUnit* cuRestrictedAbove[MAX_NUM_CH];
557
558
  void determineNeighborCus       ( const CodingStructure& cs, const UnitArea& ua, const ChannelType ch, const TreeType treeType );
559
560
3.91M
  inline static unsigned CtxQtCbf ( const ComponentID compID, const bool prevCbf = false, const int ispIdx = 0 )   { return ( ispIdx && isLuma( compID ) ) ? (2 + (int)prevCbf) : (((compID == COMP_Cr) && prevCbf) ? 1 : 0); }
561
  void     CtxSplit               ( const Partitioner& partitioner, unsigned& ctxSpl, unsigned& ctxQt, unsigned& ctxHv, unsigned& ctxHorBt, unsigned& ctxVerBt, const bool canSplit[6] ) const;
562
  unsigned CtxMipFlag             ( const CodingUnit& cu ) const;
563
0
  unsigned CtxInterDir            ( const CodingUnit& cu ) const { return ( 7 - ((Log2(cu.lumaSize().area()) + 1) >> 1) ); }
564
565
  unsigned CtxModeConsFlag() const
566
0
  {
567
0
    const CodingUnit* cuLeft  = cuRestrictedLeft[CH_L]; const CodingUnit* cuAbove = cuRestrictedAbove[CH_L];
568
0
    return ((cuAbove && cuAbove->predMode == MODE_INTRA) || (cuLeft && cuLeft->predMode == MODE_INTRA)) ? 1 : 0;
569
0
  }
570
571
  unsigned CtxAffineFlag() const
572
0
  {
573
0
    const CodingUnit *cuLeft = cuRestrictedLeft[CH_L]; const CodingUnit *cuAbove = cuRestrictedAbove[CH_L];
574
0
    return (( cuLeft && cuLeft->affine ) ? 1 : 0) + (( cuAbove && cuAbove->affine ) ? 1 : 0);
575
0
  }
576
577
  unsigned CtxSkipFlag() const
578
102k
  {
579
102k
    const CodingUnit *cuLeft = cuRestrictedLeft[CH_L]; const CodingUnit *cuAbove = cuRestrictedAbove[CH_L];
580
102k
    return (( cuLeft && cuLeft->skip ) ? 1 : 0) + (( cuAbove && cuAbove->skip ) ? 1 : 0);
581
102k
  }
582
583
  unsigned CtxPredModeFlag() const
584
0
  {
585
0
    const CodingUnit *cuLeft = cuRestrictedLeft[CH_L]; const CodingUnit *cuAbove = cuRestrictedAbove[CH_L];
586
0
    return ((cuAbove && cuAbove->predMode == MODE_INTRA) || (cuLeft && cuLeft->predMode == MODE_INTRA)) ? 1 : 0;
587
0
  }
588
589
  unsigned CtxIBCFlag(const CodingUnit& cu) const
590
299k
  {
591
299k
    const CodingUnit *cuLeft = cuRestrictedLeft[cu.chType]; const CodingUnit *cuAbove = cuRestrictedAbove[cu.chType];
592
299k
    return ((cuLeft && cuLeft->predMode == MODE_IBC) ? 1 : 0) + ((cuAbove && cuAbove->predMode == MODE_IBC) ? 1 : 0);
593
299k
  }
594
595
};
596
597
} // namespace vvenc
598
599
//! \}
600