source: SHVCSoftware/branches/SHM-dev/source/Lib/TLibCommon/TComPicYuv.cpp @ 1263

Last change on this file since 1263 was 1259, checked in by seregin, 9 years ago

port rev 4256

  • Property svn:eol-style set to native
File size: 9.9 KB
RevLine 
[313]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
[1029]4 * granted under this license.
[313]5 *
[1259]6 * Copyright (c) 2010-2015, ITU/ISO/IEC
[313]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.
17 *  * Neither the name of the ITU/ISO/IEC nor the names of its contributors may
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 */
33
34/** \file     TComPicYuv.cpp
35    \brief    picture YUV buffer class
36*/
37
38#include <cstdlib>
39#include <assert.h>
40#include <memory.h>
41
42#ifdef __APPLE__
43#include <malloc/malloc.h>
44#else
45#include <malloc.h>
46#endif
47
48#include "TComPicYuv.h"
[1029]49#include "TLibVideoIO/TVideoIOYuv.h"
[313]50
51//! \ingroup TLibCommon
52//! \{
53
54TComPicYuv::TComPicYuv()
55{
[1029]56  for(UInt i=0; i<MAX_NUM_COMPONENT; i++)
57  {
58    m_apiPicBuf[i]    = NULL;   // Buffer (including margin)
59    m_piPicOrg[i]     = NULL;    // m_apiPicBufY + m_iMarginLuma*getStride() + m_iMarginLuma
60  }
61
62  for(UInt i=0; i<MAX_NUM_CHANNEL_TYPE; i++)
63  {
64    m_ctuOffsetInBuffer[i]=0;
65    m_subCuOffsetInBuffer[i]=0;
66  }
67
[313]68  m_bIsBorderExtended = false;
69}
70
[1029]71
72
73
[313]74TComPicYuv::~TComPicYuv()
75{
76}
[815]77
78#if SVC_EXTENSION
[1235]79Void TComPicYuv::create( const Int iPicWidth, const Int iPicHeight, const ChromaFormat chromaFormatIDC, const UInt uiMaxCUWidth, const UInt uiMaxCUHeight, const UInt uiMaxCUDepth, const Window* conformanceWindow )
[869]80#else
[1029]81Void TComPicYuv::create( const Int  iPicWidth,    const  Int iPicHeight,    const ChromaFormat chromaFormatIDC,
82                         const UInt uiMaxCUWidth, const UInt uiMaxCUHeight, const UInt uiMaxCUDepth )
[494]83#endif
[313]84{
[1029]85  m_iPicWidth         = iPicWidth;
86  m_iPicHeight        = iPicHeight;
87  m_chromaFormatIDC   = chromaFormatIDC;
[313]88 
[815]89#if SVC_EXTENSION
[869]90  if(conformanceWindow != NULL)
91  {
92    m_conformanceWindow = *conformanceWindow;
93  }
[313]94#endif
95 
[494]96#if LAYER_CTB
[1029]97  m_iMarginX    = uiMaxCUWidth  + 16; // for 16-byte alignment
98  m_iMarginY    = uiMaxCUHeight + 16;  // margin for 8-tap filter and infinite padding
[494]99#else
[1029]100  m_iMarginX          = g_uiMaxCUWidth  + 16; // for 16-byte alignment
101  m_iMarginY          = g_uiMaxCUHeight + 16;  // margin for 8-tap filter and infinite padding
[494]102#endif
[313]103 
104  m_bIsBorderExtended = false;
[1029]105
106  // assign the picture arrays and set up the ptr to the top left of the original picture
[313]107  {
[1029]108    Int chan=0;
109    for(; chan<getNumberValidComponents(); chan++)
[313]110    {
[1029]111      const ComponentID ch=ComponentID(chan);
112      m_apiPicBuf[chan] = (Pel*)xMalloc( Pel, getStride(ch)       * getTotalHeight(ch));
113      m_piPicOrg[chan]  = m_apiPicBuf[chan] + (m_iMarginY >> getComponentScaleY(ch))   * getStride(ch)       + (m_iMarginX >> getComponentScaleX(ch));
[313]114    }
[1029]115    for(;chan<MAX_NUM_COMPONENT; chan++)
[313]116    {
[1029]117      m_apiPicBuf[chan] = NULL;
118      m_piPicOrg[chan]  = NULL;
[313]119    }
120  }
[1029]121
122
123  const Int numCuInWidth  = m_iPicWidth  / uiMaxCUWidth  + (m_iPicWidth  % uiMaxCUWidth  != 0);
124  const Int numCuInHeight = m_iPicHeight / uiMaxCUHeight + (m_iPicHeight % uiMaxCUHeight != 0);
125  for(Int chan=0; chan<2; chan++)
126  {
127    const ComponentID ch=ComponentID(chan);
128    const Int ctuHeight=uiMaxCUHeight>>getComponentScaleY(ch);
129    const Int ctuWidth=uiMaxCUWidth>>getComponentScaleX(ch);
130    const Int stride = getStride(ch);
131
132    m_ctuOffsetInBuffer[chan] = new Int[numCuInWidth * numCuInHeight];
133
134    for (Int cuRow = 0; cuRow < numCuInHeight; cuRow++)
[1246]135    {
[1029]136      for (Int cuCol = 0; cuCol < numCuInWidth; cuCol++)
[1246]137      {
[1029]138        m_ctuOffsetInBuffer[chan][cuRow * numCuInWidth + cuCol] = stride * cuRow * ctuHeight + cuCol * ctuWidth;
[1246]139      }
140    }
[1029]141
142    m_subCuOffsetInBuffer[chan] = new Int[(size_t)1 << (2 * uiMaxCUDepth)];
143
144    const Int numSubBlockPartitions=(1<<uiMaxCUDepth);
145    const Int minSubBlockHeight    =(ctuHeight >> uiMaxCUDepth);
146    const Int minSubBlockWidth     =(ctuWidth  >> uiMaxCUDepth);
147
148    for (Int buRow = 0; buRow < numSubBlockPartitions; buRow++)
[1246]149    {
[1029]150      for (Int buCol = 0; buCol < numSubBlockPartitions; buCol++)
[1246]151      {
[1029]152        m_subCuOffsetInBuffer[chan][(buRow << uiMaxCUDepth) + buCol] = stride  * buRow * minSubBlockHeight + buCol * minSubBlockWidth;
[1246]153      }
154    }
[1029]155  }
[313]156  return;
157}
158
159
160
[1029]161Void TComPicYuv::destroy()
[313]162{
[1029]163  for(Int chan=0; chan<MAX_NUM_COMPONENT; chan++)
[313]164  {
[1029]165    m_piPicOrg[chan] = NULL;
166
[1246]167    if( m_apiPicBuf[chan] )
168    {
169      xFree( m_apiPicBuf[chan] );
170      m_apiPicBuf[chan] = NULL;
171    }
[313]172  }
[1029]173
174  for(UInt chan=0; chan<MAX_NUM_CHANNEL_TYPE; chan++)
[313]175  {
[1246]176    if (m_ctuOffsetInBuffer[chan])
177    {
178      delete[] m_ctuOffsetInBuffer[chan];
179      m_ctuOffsetInBuffer[chan] = NULL;
180    }
181    if (m_subCuOffsetInBuffer[chan])
182    {
183      delete[] m_subCuOffsetInBuffer[chan];
184      m_subCuOffsetInBuffer[chan] = NULL;
185    }
[313]186  }
187}
188
189
190
[1029]191Void  TComPicYuv::copyToPic (TComPicYuv*  pcPicYuvDst) const
[313]192{
[1029]193  assert( m_iPicWidth  == pcPicYuvDst->getWidth(COMPONENT_Y)  );
194  assert( m_iPicHeight == pcPicYuvDst->getHeight(COMPONENT_Y) );
195  assert( m_chromaFormatIDC == pcPicYuvDst->getChromaFormat() );
[313]196
[1029]197  for(Int chan=0; chan<getNumberValidComponents(); chan++)
198  {
199    const ComponentID ch=ComponentID(chan);
200    ::memcpy ( pcPicYuvDst->getBuf(ch), m_apiPicBuf[ch], sizeof (Pel) * getStride(ch) * getTotalHeight(ch));
201  }
[313]202  return;
203}
204
205
206Void TComPicYuv::extendPicBorder ()
207{
[1246]208  if ( m_bIsBorderExtended )
209  {
210    return;
211  }
[313]212
[1029]213  for(Int chan=0; chan<getNumberValidComponents(); chan++)
[313]214  {
[1029]215    const ComponentID ch=ComponentID(chan);
216    Pel *piTxt=getAddr(ch); // piTxt = point to (0,0) of image within bigger picture.
217    const Int iStride=getStride(ch);
218    const Int iWidth=getWidth(ch);
219    const Int iHeight=getHeight(ch);
220    const Int iMarginX=getMarginX(ch);
221    const Int iMarginY=getMarginY(ch);
222
223    Pel*  pi = piTxt;
224    // do left and right margins
225    for (Int y = 0; y < iHeight; y++)
[313]226    {
[1029]227      for (Int x = 0; x < iMarginX; x++ )
228      {
229        pi[ -iMarginX + x ] = pi[0];
230        pi[    iWidth + x ] = pi[iWidth-1];
231      }
232      pi += iStride;
[313]233    }
[1029]234
235    // pi is now the (0,height) (bottom left of image within bigger picture
236    pi -= (iStride + iMarginX);
237    // pi is now the (-marginX, height-1)
238    for (Int y = 0; y < iMarginY; y++ )
239    {
240      ::memcpy( pi + (y+1)*iStride, pi, sizeof(Pel)*(iWidth + (iMarginX<<1)) );
241    }
242
243    // pi is still (-marginX, height-1)
244    pi -= ((iHeight-1) * iStride);
245    // pi is now (-marginX, 0)
246    for (Int y = 0; y < iMarginY; y++ )
247    {
248      ::memcpy( pi - (y+1)*iStride, pi, sizeof(Pel)*(iWidth + (iMarginX<<1)) );
249    }
[313]250  }
[1029]251
252  m_bIsBorderExtended = true;
[313]253}
254
255
[1029]256
257// NOTE: This function is never called, but may be useful for developers.
258Void TComPicYuv::dump (const Char* pFileName, Bool bAdd) const
[313]259{
260  FILE* pFile;
261  if (!bAdd)
262  {
263    pFile = fopen (pFileName, "wb");
264  }
265  else
266  {
267    pFile = fopen (pFileName, "ab");
268  }
[1029]269
270
271  for(Int chan = 0; chan < getNumberValidComponents(); chan++)
[313]272  {
[1029]273    const ComponentID  ch     = ComponentID(chan);
274    const Int          shift  = g_bitDepth[toChannelType(ch)] - 8;
275    const Int          offset = (shift>0)?(1<<(shift-1)):0;
276    const Pel         *pi     = getAddr(ch);
277    const Int          stride = getStride(ch);
278    const Int          height = getHeight(ch);
279    const Int          width  = getWidth(ch);
280
281    for (Int y = 0; y < height; y++ )
[313]282    {
[1029]283      for (Int x = 0; x < width; x++ )
284      {
285        UChar uc = (UChar)Clip3<Pel>(0, 255, (pi[x]+offset)>>shift);
286        fwrite( &uc, sizeof(UChar), 1, pFile );
287      }
288      pi += stride;
[313]289    }
290  }
291
292  fclose(pFile);
293}
294
[815]295Void TComPicYuv::dump( Char* pFileName, Bool bAdd, Int bitDepth )
296{
297  FILE* pFile;
298  if (!bAdd)
299  {
300    pFile = fopen (pFileName, "wb");
301  }
302  else
303  {
304    pFile = fopen (pFileName, "ab");
305  }
[313]306
[815]307  if( bitDepth == 8 )
308  {
309    dump( pFileName, bAdd );
310    return;
311  }
[1029]312
313  for(Int chan = 0; chan < getNumberValidComponents(); chan++)
[815]314  {
[1029]315    const ComponentID  ch     = ComponentID(chan);
316    const Pel         *pi     = getAddr(ch);
317    const Int          stride = getStride(ch);
318    const Int          height = getHeight(ch);
319    const Int          width  = getWidth(ch);
320
321    for (Int y = 0; y < height; y++ )
[815]322    {
[1029]323      for (Int x = 0; x < width; x++ )
324      {
[1089]325        Pel pix = pi[x];
[1029]326
327        UChar uc = pix & 0xff;     
328        fwrite( &uc, sizeof(UChar), 1, pFile );
329        uc = (pix >> 8) & 0xff;     
330        fwrite( &uc, sizeof(UChar), 1, pFile );
331      }
332      pi += stride;
[815]333    }
334  }
335 
[1029]336  fclose(pFile);
337}
338
339#if AUXILIARY_PICTURES
340Void TComPicYuv::convertToMonochrome()
341{
342  Pel grayVal = (1 << (g_bitDepth[CHANNEL_TYPE_CHROMA] - 1));
343
[1031]344  for( UInt comp = 1; comp < getNumberValidComponents(); comp++ )
[815]345  {
[1031]346    const ComponentID ch=ComponentID(comp);
347    for( UInt i = 0; i < getStride(ch) * getTotalHeight(ch); i++ )
348    {
349      m_apiPicBuf[ch][i] = grayVal;
350    }
[815]351  }
352}
[1029]353#endif
[815]354
[313]355//! \}
Note: See TracBrowser for help on using the repository browser.