source: 3DVCSoftware/trunk/source/Lib/TLibDecoder/TDecGop.cpp @ 1296

Last change on this file since 1296 was 1179, checked in by tech, 10 years ago

Merged branch 13.1-dev0@1178.

  • Property svn:eol-style set to native
File size: 11.8 KB
RevLine 
[5]1/* The copyright in this software is being made available under the BSD
2 * License, included below. This software may be subject to other third party
3 * and contributor rights, including patent rights, and no such rights are
[56]4 * granted under this license. 
[5]5 *
[1179]6* Copyright (c) 2010-2015, ITU/ISO/IEC
[5]7 * All rights reserved.
8 *
9 * Redistribution and use in source and binary forms, with or without
10 * modification, are permitted provided that the following conditions are met:
11 *
12 *  * Redistributions of source code must retain the above copyright notice,
13 *    this list of conditions and the following disclaimer.
14 *  * Redistributions in binary form must reproduce the above copyright notice,
15 *    this list of conditions and the following disclaimer in the documentation
16 *    and/or other materials provided with the distribution.
[56]17 *  * Neither the name of the ITU/ISO/IEC nor the names of its contributors may
[5]18 *    be used to endorse or promote products derived from this software without
19 *    specific prior written permission.
20 *
21 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
22 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
23 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
24 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS
25 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
26 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
27 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
28 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
29 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
30 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
31 * THE POSSIBILITY OF SUCH DAMAGE.
32 */
[2]33
34/** \file     TDecGop.cpp
35    \brief    GOP decoder class
36*/
37
38#include "TDecGop.h"
39#include "TDecCAVLC.h"
40#include "TDecSbac.h"
41#include "TDecBinCoder.h"
42#include "TDecBinCoderCABAC.h"
[56]43#include "libmd5/MD5.h"
44#include "TLibCommon/SEI.h"
[2]45
46#include <time.h>
47
[608]48extern Bool g_md5_mismatch; ///< top level flag to signal when there is a decode problem
49
[56]50//! \ingroup TLibDecoder
51//! \{
[608]52static void calcAndPrintHashStatus(TComPicYuv& pic, const SEIDecodedPictureHash* pictureHashSEI);
[2]53// ====================================================================================================================
54// Constructor / destructor / initialization / destroy
55// ====================================================================================================================
56
57TDecGop::TDecGop()
58{
59  m_dDecTime = 0;
[56]60  m_pcSbacDecoders = NULL;
61  m_pcBinCABACs = NULL;
[2]62}
63
64TDecGop::~TDecGop()
65{
66 
67}
68
69Void TDecGop::create()
70{
71 
72}
73
74
75Void TDecGop::destroy()
76{
77}
78
79Void TDecGop::init( TDecEntropy*            pcEntropyDecoder, 
80                   TDecSbac*               pcSbacDecoder, 
81                   TDecBinCABAC*           pcBinCABAC,
82                   TDecCavlc*              pcCavlcDecoder, 
83                   TDecSlice*              pcSliceDecoder, 
[608]84                   TComLoopFilter*         pcLoopFilter,
85                   TComSampleAdaptiveOffset* pcSAO
[56]86                   )
[2]87{
88  m_pcEntropyDecoder      = pcEntropyDecoder;
89  m_pcSbacDecoder         = pcSbacDecoder;
90  m_pcBinCABAC            = pcBinCABAC;
91  m_pcCavlcDecoder        = pcCavlcDecoder;
92  m_pcSliceDecoder        = pcSliceDecoder;
93  m_pcLoopFilter          = pcLoopFilter;
[1179]94  m_pcSAO  = pcSAO; 
[2]95}
96
[56]97
[2]98// ====================================================================================================================
[56]99// Private member functions
100// ====================================================================================================================
101// ====================================================================================================================
[2]102// Public member functions
103// ====================================================================================================================
104
[608]105Void TDecGop::decompressSlice(TComInputBitstream* pcBitstream, TComPic*& rpcPic)
[2]106{
107  TComSlice*  pcSlice = rpcPic->getSlice(rpcPic->getCurrSliceIdx());
[56]108  // Table of extracted substreams.
109  // These must be deallocated AND their internal fifos, too.
110  TComInputBitstream **ppcSubstreams = NULL;
[2]111
112  //-- For time output for each slice
113  long iBeforeTime = clock();
[608]114  m_pcSbacDecoder->init( (TDecBinIf*)m_pcBinCABAC );
115  m_pcEntropyDecoder->setEntropyDecoder (m_pcSbacDecoder);
[56]116
[608]117  UInt uiNumSubstreams = pcSlice->getPPS()->getEntropyCodingSyncEnabledFlag() ? pcSlice->getNumEntryPointOffsets()+1 : pcSlice->getPPS()->getNumSubstreams();
[56]118
[608]119  // init each couple {EntropyDecoder, Substream}
120  UInt *puiSubstreamSizes = pcSlice->getSubstreamSizes();
121  ppcSubstreams    = new TComInputBitstream*[uiNumSubstreams];
122  m_pcSbacDecoders = new TDecSbac[uiNumSubstreams];
123  m_pcBinCABACs    = new TDecBinCABAC[uiNumSubstreams];
124  for ( UInt ui = 0 ; ui < uiNumSubstreams ; ui++ )
125  {
126    m_pcSbacDecoders[ui].init(&m_pcBinCABACs[ui]);
127    ppcSubstreams[ui] = pcBitstream->extractSubstream(ui+1 < uiNumSubstreams ? puiSubstreamSizes[ui] : pcBitstream->getNumBitsLeft());
128  }
[2]129
[608]130  for ( UInt ui = 0 ; ui+1 < uiNumSubstreams; ui++ )
131  {
132    m_pcEntropyDecoder->setEntropyDecoder ( &m_pcSbacDecoders[uiNumSubstreams - 1 - ui] );
133    m_pcEntropyDecoder->setBitstream      (  ppcSubstreams   [uiNumSubstreams - 1 - ui] );
[2]134    m_pcEntropyDecoder->resetEntropy      (pcSlice);
[608]135  }
[56]136
[608]137  m_pcEntropyDecoder->setEntropyDecoder ( m_pcSbacDecoder  );
138  m_pcEntropyDecoder->setBitstream      ( ppcSubstreams[0] );
139  m_pcEntropyDecoder->resetEntropy      (pcSlice);
140#if H_3D_NBDV
141  if(pcSlice->getViewIndex() && !pcSlice->getIsDepth()) //Notes from QC: this condition shall be changed once the configuration is completed, e.g. in pcSlice->getSPS()->getMultiviewMvPredMode() || ARP in prev. HTM. Remove this comment once it is done.
142  {
143    Int iColPoc = pcSlice->getRefPOC(RefPicList(1-pcSlice->getColFromL0Flag()), pcSlice->getColRefIdx());
144    rpcPic->setNumDdvCandPics(rpcPic->getDisCandRefPictures(iColPoc));
145  }
[655]146
[608]147  if(pcSlice->getViewIndex() && !pcSlice->getIsDepth() && !pcSlice->isIntra()) //Notes from QC: this condition shall be changed once the configuration is completed, e.g. in pcSlice->getSPS()->getMultiviewMvPredMode() || ARP in prev. HTM. Remove this comment once it is done.
148  {
149    rpcPic->checkTemporalIVRef();
150  }
[655]151
[608]152  if(pcSlice->getIsDepth())
153  {
154    rpcPic->checkTextureRef();
155  }
[5]156#endif
[608]157#if H_3D
158  pcSlice->setDepthToDisparityLUTs(); 
[189]159#endif
[608]160  m_pcSbacDecoders[0].load(m_pcSbacDecoder);
161  m_pcSliceDecoder->decompressSlice( ppcSubstreams, rpcPic, m_pcSbacDecoder, m_pcSbacDecoders);
162  m_pcEntropyDecoder->setBitstream(  ppcSubstreams[uiNumSubstreams-1] );
163  // deallocate all created substreams, including internal buffers.
164  for (UInt ui = 0; ui < uiNumSubstreams; ui++)
165  {
166    ppcSubstreams[ui]->deleteFifo();
167    delete ppcSubstreams[ui];
168  }
169  delete[] ppcSubstreams;
170  delete[] m_pcSbacDecoders; m_pcSbacDecoders = NULL;
171  delete[] m_pcBinCABACs; m_pcBinCABACs = NULL;
[56]172
[608]173  m_dDecTime += (Double)(clock()-iBeforeTime) / CLOCKS_PER_SEC;
174}
175
176Void TDecGop::filterPicture(TComPic*& rpcPic)
177{
178  TComSlice*  pcSlice = rpcPic->getSlice(rpcPic->getCurrSliceIdx());
179
180  //-- For time output for each slice
181  long iBeforeTime = clock();
182
183  // deblocking filter
184  Bool bLFCrossTileBoundary = pcSlice->getPPS()->getLoopFilterAcrossTilesEnabledFlag();
185  m_pcLoopFilter->setCfg(bLFCrossTileBoundary);
186  m_pcLoopFilter->loopFilterPic( rpcPic );
187  if( pcSlice->getSPS()->getUseSAO() )
188  {
[872]189    m_pcSAO->reconstructBlkSAOParams(rpcPic, rpcPic->getPicSym()->getSAOBlkParam());
190    m_pcSAO->SAOProcess(rpcPic);
191    m_pcSAO->PCMLFDisableProcess(rpcPic);
[2]192  }
[655]193#if H_3D
[608]194  rpcPic->compressMotion(2); 
[5]195#endif
[608]196#if !H_3D
197  rpcPic->compressMotion(); 
[5]198#endif
[608]199  Char c = (pcSlice->isIntra() ? 'I' : pcSlice->isInterP() ? 'P' : 'B');
200  if (!pcSlice->isReferenced()) c += 32;
[56]201
[608]202  //-- For time output for each slice
203#if H_MV
204  printf("\nLayer %2d   POC %4d TId: %1d ( %c-SLICE, QP%3d ) ", pcSlice->getLayerId(),
205                                                              pcSlice->getPOC(),
206                                                              pcSlice->getTLayer(),
207                                                              c,
208                                                              pcSlice->getSliceQp() );
[443]209#else
[608]210  printf("\nPOC %4d TId: %1d ( %c-SLICE, QP%3d ) ", pcSlice->getPOC(),
211                                                    pcSlice->getTLayer(),
212                                                    c,
213                                                    pcSlice->getSliceQp() );
[443]214#endif
[56]215
[608]216  m_dDecTime += (Double)(clock()-iBeforeTime) / CLOCKS_PER_SEC;
217  printf ("[DT %6.3f] ", m_dDecTime );
218  m_dDecTime  = 0;
[56]219
[608]220  for (Int iRefList = 0; iRefList < 2; iRefList++)
221  {
222    printf ("[L%d ", iRefList);
223    for (Int iRefIndex = 0; iRefIndex < pcSlice->getNumRefIdx(RefPicList(iRefList)); iRefIndex++)
[2]224    {
[608]225#if H_MV
226      if( pcSlice->getLayerId() != pcSlice->getRefLayerId( RefPicList(iRefList), iRefIndex ) )
[2]227      {
[608]228        printf( "V%d ", pcSlice->getRefLayerId( RefPicList(iRefList), iRefIndex ) );
[2]229      }
[608]230      else
[2]231      {
[608]232#endif
233      printf ("%d ", pcSlice->getRefPOC(RefPicList(iRefList), iRefIndex));
234#if H_MV
[2]235      }
[608]236#endif
[2]237    }
[608]238    printf ("] ");
239  }
240  if (m_decodedPictureHashSEIEnabled)
241  {
242    SEIMessages pictureHashes = getSeisByType(rpcPic->getSEIs(), SEI::DECODED_PICTURE_HASH );
243    const SEIDecodedPictureHash *hash = ( pictureHashes.size() > 0 ) ? (SEIDecodedPictureHash*) *(pictureHashes.begin()) : NULL;
244    if (pictureHashes.size() > 1)
[56]245    {
[608]246      printf ("Warning: Got multiple decoded picture hash SEI messages. Using first.");
[2]247    }
[608]248    calcAndPrintHashStatus(*rpcPic->getPicYuvRec(), hash);
249  }
[1084]250#if !H_MV
251#if SETTING_PIC_OUTPUT_MARK
252  rpcPic->setOutputMark(rpcPic->getSlice(0)->getPicOutputFlag() ? true : false);
253#else
254  rpcPic->setOutputMark(true);
255#endif
256  rpcPic->setReconMark(true);
257#endif
[2]258}
259
260/**
[608]261 * Calculate and print hash for pic, compare to picture_digest SEI if
262 * present in seis.  seis may be NULL.  Hash is printed to stdout, in
[2]263 * a manner suitable for the status line. Theformat is:
[608]264 *  [Hash_type:xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx,(yyy)]
265 * Where, x..x is the hash
[2]266 *        yyy has the following meanings:
[608]267 *            OK          - calculated hash matches the SEI message
268 *            ***ERROR*** - calculated hash does not match the SEI message
[2]269 *            unk         - no SEI message was available for comparison
270 */
[608]271static void calcAndPrintHashStatus(TComPicYuv& pic, const SEIDecodedPictureHash* pictureHashSEI)
[2]272{
273  /* calculate MD5sum for entire reconstructed picture */
[608]274  UChar recon_digest[3][16];
275  Int numChar=0;
276  const Char* hashType = "\0";
[2]277
[608]278  if (pictureHashSEI)
279  {
280    switch (pictureHashSEI->method)
281    {
282    case SEIDecodedPictureHash::MD5:
283      {
284        hashType = "MD5";
285        calcMD5(pic, recon_digest);
286        numChar = 16;
287        break;
288      }
289    case SEIDecodedPictureHash::CRC:
290      {
291        hashType = "CRC";
292        calcCRC(pic, recon_digest);
293        numChar = 2;
294        break;
295      }
296    case SEIDecodedPictureHash::CHECKSUM:
297      {
298        hashType = "Checksum";
299        calcChecksum(pic, recon_digest);
300        numChar = 4;
301        break;
302      }
303    default:
304      {
305        assert (!"unknown hash type");
306      }
307    }
308  }
309
[2]310  /* compare digest against received version */
[608]311  const Char* ok = "(unk)";
312  Bool mismatch = false;
[2]313
[608]314  if (pictureHashSEI)
[2]315  {
[608]316    ok = "(OK)";
317    for(Int yuvIdx = 0; yuvIdx < 3; yuvIdx++)
[2]318    {
[608]319      for (UInt i = 0; i < numChar; i++)
[2]320      {
[608]321        if (recon_digest[yuvIdx][i] != pictureHashSEI->digest[yuvIdx][i])
322        {
323          ok = "(***ERROR***)";
324          mismatch = true;
325        }
[2]326      }
327    }
328  }
329
[608]330  printf("[%s:%s,%s] ", hashType, digestToString(recon_digest, numChar), ok);
331
332  if (mismatch)
[2]333  {
334    g_md5_mismatch = true;
[608]335    printf("[rx%s:%s] ", hashType, digestToString(pictureHashSEI->digest, numChar));
[2]336  }
337}
[56]338//! \}
Note: See TracBrowser for help on using the repository browser.