Coverage Report

Created: 2026-06-15 06:25

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
10.9k
  void  resetSigGroup   ()                      { m_sigCoeffGroupFlag.reset( m_subSetPos ); }
76
411k
  void  setSigGroup     ()                      { m_sigCoeffGroupFlag.set( m_subSetPos ); }
77
0
  bool  noneSigGroup    ()                      { return m_sigCoeffGroupFlag.none(); }
78
400k
  int   lastSubSet      ()                      { return ( maxNumCoeff() - 1 ) >> log2CGSize(); }
79
339k
  bool  isLastSubSet    ()                      { return lastSubSet() == m_subSetId; }
80
60.3k
  bool  only1stSigGroup ()                      { return m_sigCoeffGroupFlag.count()-m_sigCoeffGroupFlag[lastSubSet()]==0; }
81
795k
  void  setScanPosLast  ( int       posLast )   { m_scanPosLast = posLast; }
82
public:
83
0
  ComponentID     compID          ()                        const { return m_compID; }
84
567k
  int             subSetId        ()                        const { return m_subSetId; }
85
0
  int             subSetPos       ()                        const { return m_subSetPos; }
86
41.1k
  int             cgPosY          ()                        const { return m_subSetPosY; }
87
41.1k
  int             cgPosX          ()                        const { return m_subSetPosX; }
88
1.59M
  unsigned        width           ()                        const { return m_width; }
89
0
  unsigned        height          ()                        const { return m_height; }
90
5.32k
  unsigned        log2CGWidth     ()                        const { return m_log2CGWidth; }
91
0
  unsigned        log2CGHeight    ()                        const { return m_log2CGHeight; }
92
5.71M
  unsigned        log2CGSize      ()                        const { return m_log2CGSize; }
93
4.69M
  int             maxLog2TrDRange ()                        const { return m_maxLog2TrDynamicRange; }
94
6.44M
  unsigned        maxNumCoeff     ()                        const { return m_maxNumCoeff; }
95
11.3M
  int             scanPosLast     ()                        const { return m_scanPosLast; }
96
10.3M
  int             minSubPos       ()                        const { return m_minSubPos; }
97
364k
  int             maxSubPos       ()                        const { return m_maxSubPos; }
98
820k
  bool            isLast          ()                        const { return ( ( m_scanPosLast >> m_log2CGSize ) == m_subSetId ); }
99
41.3k
  bool            isNotFirst      ()                        const { return ( m_subSetId != 0 ); }
100
0
  bool            isSigGroup      ( int scanPosCG )         const { return m_sigCoeffGroupFlag[m_scanCG[scanPosCG].idx]; }
101
959k
  bool            isSigGroup      ()                        const { return m_sigCoeffGroupFlag[ m_subSetPos ]; }
102
8.52k
  bool            signHiding      ()                        const { return m_signHiding; }
103
  bool            hideSign        ( int       posFirst,
104
812k
                                    int       posLast   )   const { return ( m_signHiding && ( posLast - posFirst >= SBH_THRESHOLD ) ); }
105
51.0M
  unsigned        blockPos        ( int scanPos )           const { return m_scan[scanPos].idx; }
106
8.69M
  unsigned        posX            ( int scanPos )           const { return m_scan[scanPos].x; }
107
8.69M
  unsigned        posY            ( int scanPos )           const { return m_scan[scanPos].y; }
108
795k
  unsigned        maxLastPosX     ()                        const { return m_maxLastPosX; }
109
795k
  unsigned        maxLastPosY     ()                        const { return m_maxLastPosY; }
110
1.81M
  unsigned        lastXCtxId      ( unsigned  posLastX  )   const { return m_CtxSetLastX( m_lastOffsetX + ( posLastX >> m_lastShiftX ) ); }
111
1.86M
  unsigned        lastYCtxId      ( unsigned  posLastY  )   const { return m_CtxSetLastY( m_lastOffsetY + ( posLastY >> m_lastShiftY ) ); }
112
907k
  unsigned        sigGroupCtxId   ( bool ts = false     )   const { return ts ? m_sigGroupCtxIdTS : m_sigGroupCtxId; }
113
5.71M
  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
5.65M
  {
160
5.65M
    const auto scanEl       = m_scan[scanPos];
161
5.65M
    const uint32_t posY     = scanEl.y;
162
5.65M
    const uint32_t posX     = scanEl.x;
163
5.65M
    const int32_t blkPos    = scanEl.idx;
164
5.65M
    const int      diag     = posX + posY;
165
5.65M
    const int      tplVal   = m_tplBuf[-blkPos].ctxTpl;
166
5.65M
    const int      numPos   = tplVal >> 5u;
167
5.65M
    const int      sumAbs   = tplVal & 31;
168
169
5.65M
    int ctxOfs = std::min( ( sumAbs + 1 ) >> 1, 3 ) + ( diag < 2 ? 4 : 0 );
170
171
5.65M
    if( isLuma( m_chType ) )
172
401k
    {
173
401k
      ctxOfs += diag < 5 ? 4 : 0;
174
401k
    }
175
5.65M
    m_tmplCpDiag = diag;
176
5.65M
    m_tmplCpSum1 = sumAbs - numPos;
177
5.65M
    return m_sigFlagCtxSet[std::max( 0, state-1 )]( ctxOfs );
178
5.65M
  }
179
180
  void absVal1stPass( const int scanPos, const TCoeffSig absLevel1 )
181
5.94M
  {
182
5.94M
    CHECKD( !absLevel1, "Shound not be called if '0'!" );
183
184
5.94M
    const auto scanEl     = m_scan[scanPos];
185
5.94M
    const uint32_t posY   = scanEl.y;
186
5.94M
    const uint32_t posX   = scanEl.x;
187
5.94M
    const int32_t  blkPos = scanEl.idx;
188
189
5.94M
    auto update_deps = [&]( int offset )
190
15.5M
    {
191
15.5M
      auto& ctx   = m_tplBuf[-blkPos + offset];
192
15.5M
      ctx.ctxTpl += uint8_t( 32 + absLevel1 );
193
15.5M
    };
194
195
5.94M
    if( posY > 1 ) update_deps( 2 * m_width );
196
5.94M
    if( posY > 0
197
4.00M
     && posX > 0 ) update_deps( m_width + 1 );
198
5.94M
    if( posY > 0 ) update_deps( m_width );
199
5.94M
    if( posX > 1 ) update_deps( 2 );
200
5.94M
    if( posX > 0 ) update_deps( 1 );
201
5.94M
  }
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
5.94M
  {
229
5.94M
    int offset = 0;
230
5.94M
    if( m_tmplCpDiag != -1 )
231
5.15M
    {
232
5.15M
      offset  = std::min( m_tmplCpSum1, 4 ) + 1;
233
5.15M
      offset += ( !m_tmplCpDiag ? ( m_chType == CH_L ? 15 : 5 ) : m_chType == CH_L ? m_tmplCpDiag < 3 ? 10 : ( m_tmplCpDiag < 10 ? 5 : 0 ) : 0 );
234
5.15M
    }
235
5.94M
    return uint8_t(offset);
236
5.94M
  }
237
238
5.15M
  unsigned parityCtxIdAbs   ( uint8_t offset )  const { return m_parFlagCtxSet   ( offset ); }
239
5.94M
  unsigned greater1CtxIdAbs ( uint8_t offset )  const { return m_gtxFlagCtxSet[1]( offset ); }
240
5.15M
  unsigned greater2CtxIdAbs ( uint8_t offset )  const { return m_gtxFlagCtxSet[0]( offset ); }
241
242
  unsigned templateAbsSum( int scanPos, const TCoeffSig* coeff, int baseLevel )
243
4.21M
  {
244
4.21M
    const uint32_t   posY = m_scan[scanPos].y;
245
4.21M
    const uint32_t   posX = m_scan[scanPos].x;
246
4.21M
    const TCoeffSig* pData = coeff + posX + posY * m_width;
247
4.21M
    int              sum   = 0;
248
4.21M
    if (posX < m_width - 1)
249
4.13M
    {
250
4.13M
      sum += abs(pData[1]);
251
4.13M
      if (posX < m_width - 2)
252
3.97M
      {
253
3.97M
        sum += abs(pData[2]);
254
3.97M
      }
255
4.13M
      if (posY < m_height - 1)
256
4.00M
      {
257
4.00M
        sum += abs(pData[m_width + 1]);
258
4.00M
      }
259
4.13M
    }
260
4.21M
    if (posY < m_height - 1)
261
4.08M
    {
262
4.08M
      sum += abs(pData[m_width]);
263
4.08M
      if (posY < m_height - 2)
264
3.88M
      {
265
3.88M
        sum += abs(pData[m_width << 1]);
266
3.88M
      }
267
4.08M
    }
268
4.21M
    return std::max(std::min(sum - 5 * baseLevel, 31), 0);
269
4.21M
  }
270
271
  unsigned sigCtxIdAbsTS( int scanPos, const TCoeffSig* coeff )
272
11.0M
  {
273
11.0M
    const uint32_t   posY   = m_scan[scanPos].y;
274
11.0M
    const uint32_t   posX   = m_scan[scanPos].x;
275
11.0M
    const TCoeffSig* posC   = coeff + posX + posY * m_width;
276
11.0M
    int              numPos = 0;
277
19.5M
#define UPDATE(x) {int a=abs(x);numPos+=!!a;}
278
11.0M
    if( posX > 0 )
279
9.91M
    {
280
9.91M
      UPDATE( posC[-1] );
281
9.91M
    }
282
11.0M
    if( posY > 0 )
283
9.60M
    {
284
9.60M
      UPDATE( posC[-(int)m_width] );
285
9.60M
    }
286
11.0M
#undef UPDATE
287
288
11.0M
    return m_tsSigFlagCtxSet( numPos );
289
11.0M
  }
290
291
9.51M
  unsigned parityCtxIdAbsTS   ()                  const { return m_tsParFlagCtxSet(      0 ); }
292
3.23M
  unsigned greaterXCtxIdAbsTS ( uint8_t offset )  const { return m_tsGtxFlagCtxSet( offset ); }
293
294
  unsigned lrg1CtxIdAbsTS(int scanPos, const TCoeffSig* coeff, int bdpcm)
295
9.52M
  {
296
9.52M
    const uint32_t   posY = m_scan[scanPos].y;
297
9.52M
    const uint32_t   posX = m_scan[scanPos].x;
298
9.52M
    const TCoeffSig* posC = coeff + posX + posY * m_width;
299
300
9.52M
    int             numPos = 0;
301
9.52M
#define UPDATE(x) {int a=abs(x);numPos+=!!a;}
302
303
9.52M
    if (bdpcm)
304
9.06M
    {
305
9.06M
      numPos = 3;
306
9.06M
    }
307
460k
    else
308
460k
    {
309
460k
      if (posX > 0)
310
425k
      {
311
425k
        UPDATE(posC[-1]);
312
425k
      }
313
460k
      if (posY > 0)
314
423k
      {
315
423k
        UPDATE(posC[-(int)m_width]);
316
423k
      }
317
460k
    }
318
319
9.52M
#undef UPDATE
320
9.52M
    return m_tsLrg1FlagCtxSet(numPos);
321
9.52M
  }
322
323
  unsigned signCtxIdAbsTS(int scanPos, const TCoeffSig* coeff, int bdpcm)
324
9.52M
  {
325
9.52M
    const uint32_t   posY  = m_scan[scanPos].y;
326
9.52M
    const uint32_t   posX  = m_scan[scanPos].x;
327
9.52M
    const TCoeffSig* pData = coeff + posX + posY * m_width;
328
329
9.52M
    int rightSign = 0, belowSign = 0;
330
9.52M
    unsigned signCtx = 0;
331
332
9.52M
    if (posX > 0)
333
8.59M
    {
334
8.59M
      rightSign = pData[-1];
335
8.59M
    }
336
9.52M
    if (posY > 0)
337
8.12M
    {
338
8.12M
      belowSign = pData[-(int)m_width];
339
8.12M
    }
340
341
9.52M
    if ((rightSign == 0 && belowSign == 0) || ((rightSign*belowSign) < 0))
342
8.29M
    {
343
8.29M
      signCtx = 0;
344
8.29M
    }
345
1.23M
    else if (rightSign >= 0 && belowSign >= 0)
346
474
    {
347
474
      signCtx = 1;
348
474
    }
349
1.23M
    else
350
1.23M
    {
351
1.23M
      signCtx = 2;
352
1.23M
    }
353
9.52M
    if (bdpcm)
354
9.06M
    {
355
9.06M
      signCtx += 3;
356
9.06M
    }
357
9.52M
    return m_tsSignFlagCtxSet(signCtx);
358
9.52M
  }
359
360
  void neighTS(int& rightPixel, int& belowPixel, int scanPos, const TCoeffSig* coeff)
361
13.6M
  {
362
13.6M
    const uint32_t   posY = m_scan[scanPos].y;
363
13.6M
    const uint32_t   posX = m_scan[scanPos].x;
364
13.6M
    const TCoeffSig* data = coeff + posX + posY * m_width;
365
366
13.6M
    rightPixel = belowPixel = 0;
367
368
13.6M
    if (posX > 0)
369
12.2M
    {
370
12.2M
      rightPixel = data[-1];
371
12.2M
    }
372
13.6M
    if (posY > 0)
373
11.1M
    {
374
11.1M
      belowPixel = data[-(int)m_width];
375
11.1M
    }
376
13.6M
  }
377
378
  int deriveModCoeff(int rightPixel, int belowPixel, int absCoeff, int bdpcm = 0)
379
7.10M
  {
380
7.10M
    if (absCoeff == 0)
381
3.45M
      return 0;
382
383
3.64M
    int pred1, absBelow = abs(belowPixel), absRight = abs(rightPixel);
384
385
3.64M
    int absCoeffMod = absCoeff;
386
387
3.64M
    if (bdpcm == 0)
388
489k
    {
389
489k
      pred1 = std::max(absBelow, absRight);
390
391
489k
      if (absCoeff == pred1)
392
0
      {
393
0
        absCoeffMod = 1;
394
0
      }
395
489k
      else
396
489k
      {
397
489k
        absCoeffMod = absCoeff < pred1 ? absCoeff + 1 : absCoeff;
398
489k
      }
399
489k
    }
400
401
3.64M
    return(absCoeffMod);
402
7.10M
  }
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
392k
  CUCtx()              : isDQPCoded(false), isChromaQpAdjCoded(false),
488
392k
                         qgStart(false)
489
392k
                         {
490
392k
                           violatesLfnstConstrained[CH_L] = false;
491
392k
                           violatesLfnstConstrained[CH_C] = false;
492
392k
                           violatesMtsCoeffConstraint      = false;
493
392k
                           mtsLastScanPos                  = false;
494
392k
                           lfnstLastScanPos                = false;
495
392k
                         }
496
13.3k
  CUCtx(int _qp)       : isDQPCoded(false), isChromaQpAdjCoded(false),
497
13.3k
                         qgStart(false),
498
13.3k
                         qp(_qp)
499
13.3k
                         {
500
13.3k
                           violatesLfnstConstrained[CH_L] = false;
501
13.3k
                           violatesLfnstConstrained[CH_C] = false;
502
13.3k
                           violatesMtsCoeffConstraint      = false;
503
13.3k
                           mtsLastScanPos                  = false;
504
13.3k
                           lfnstLastScanPos                = false;
505
13.3k
                         }
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
720k
  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.30M
  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
85.6k
  {
579
85.6k
    const CodingUnit *cuLeft = cuRestrictedLeft[CH_L]; const CodingUnit *cuAbove = cuRestrictedAbove[CH_L];
580
85.6k
    return (( cuLeft && cuLeft->skip ) ? 1 : 0) + (( cuAbove && cuAbove->skip ) ? 1 : 0);
581
85.6k
  }
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
250k
  {
591
250k
    const CodingUnit *cuLeft = cuRestrictedLeft[cu.chType]; const CodingUnit *cuAbove = cuRestrictedAbove[cu.chType];
592
250k
    return ((cuLeft && cuLeft->predMode == MODE_IBC) ? 1 : 0) + ((cuAbove && cuAbove->predMode == MODE_IBC) ? 1 : 0);
593
250k
  }
594
595
};
596
597
} // namespace vvenc
598
599
//! \}
600