/* The copyright in this software is being made available under the BSD * License, included below. This software may be subject to other third party * and contributor rights, including patent rights, and no such rights are * granted under this license. * * Copyright (c) 2010-2012, ITU/ISO/IEC * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * * Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * Neither the name of the ITU/ISO/IEC nor the names of its contributors may * be used to endorse or promote products derived from this software without * specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF * THE POSSIBILITY OF SUCH DAMAGE. */ /** \file TVideoIOYuv.cpp \brief YUV file I/O class */ #include #include #include #include #include #include #include "TLibCommon/TComRom.h" #include "TVideoIOYuv.h" using namespace std; /** * Perform division with rounding of all pixels in img by * 2shiftbits. All pixels are clipped to [minval, maxval] * * @param img pointer to image to be transformed * @param stride distance between vertically adjacent pixels of img. * @param width width of active area in img. * @param height height of active area in img. * @param shiftbits number of rounding bits * @param minval minimum clipping value * @param maxval maximum clipping value */ static void invScalePlane(Pel* img, unsigned int stride, unsigned int width, unsigned int height, unsigned int shiftbits, Pel minval, Pel maxval) { Pel offset = 1 << (shiftbits-1); for (unsigned int y = 0; y < height; y++) { for (unsigned int x = 0; x < width; x++) { Pel val = (img[x] + offset) >> shiftbits; img[x] = Clip3(minval, maxval, val); } img += stride; } } /** * Multiply all pixels in img by 2shiftbits. * * @param img pointer to image to be transformed * @param stride distance between vertically adjacent pixels of img. * @param width width of active area in img. * @param height height of active area in img. * @param shiftbits number of bits to shift */ static void scalePlane(Pel* img, unsigned int stride, unsigned int width, unsigned int height, unsigned int shiftbits) { for (unsigned int y = 0; y < height; y++) { for (unsigned int x = 0; x < width; x++) { img[x] <<= shiftbits; } img += stride; } } /** * Scale all pixels in img depending upon sign of shiftbits by a factor of * 2shiftbits. * * @param img pointer to image to be transformed * @param stride distance between vertically adjacent pixels of img. * @param width width of active area in img. * @param height height of active area in img. * @param shiftbits if zero, no operation performed * if > 0, multiply by 2shiftbits, see scalePlane() * if < 0, divide and round by 2shiftbits and clip, * see invScalePlane(). * @param minval minimum clipping value when dividing. * @param maxval maximum clipping value when dividing. */ static void scalePlane(Pel* img, unsigned int stride, unsigned int width, unsigned int height, int shiftbits, Pel minval, Pel maxval) { if (shiftbits == 0) { return; } if (shiftbits > 0) { scalePlane(img, stride, width, height, shiftbits); } else { invScalePlane(img, stride, width, height, -shiftbits, minval, maxval); } } // ==================================================================================================================== // Public member functions // ==================================================================================================================== /** * Open file for reading/writing Y'CbCr frames. * * Frames read/written have bitdepth fileBitDepth, and are automatically * formatted as 8 or 16 bit word values (see TVideoIOYuv::write()). * * Image data read or written is converted to/from internalBitDepth * (See scalePlane(), TVideoIOYuv::read() and TVideoIOYuv::write() for * further details). * * \param pchFile file name string * \param bWriteMode file open mode: true=read, false=write * \param fileBitDepth bit-depth of input/output file data. * \param internalBitDepth bit-depth to scale image data to/from when reading/writing. */ Void TVideoIOYuv::open( char* pchFile, Bool bWriteMode, unsigned int fileBitDepth, unsigned int internalBitDepth ) { m_bitdepthShift = internalBitDepth - fileBitDepth; m_fileBitdepth = fileBitDepth; if ( bWriteMode ) { m_cHandle.open( pchFile, ios::binary | ios::out ); if( m_cHandle.fail() ) { printf("\nfailed to write reconstructed YUV file\n"); exit(0); } } else { m_cHandle.open( pchFile, ios::binary | ios::in ); if( m_cHandle.fail() ) { printf("\nfailed to open Input YUV file\n"); exit(0); } } return; } Void TVideoIOYuv::close() { m_cHandle.close(); } Bool TVideoIOYuv::isEof() { return m_cHandle.eof(); } Bool TVideoIOYuv::isFail() { return m_cHandle.fail(); } /** * Skip numFrames in input. * * This function correctly handles cases where the input file is not * seekable, by consuming bytes. */ void TVideoIOYuv::skipFrames(unsigned int numFrames, unsigned int width, unsigned int height) { if (!numFrames) return; const unsigned int wordsize = m_fileBitdepth > 8 ? 2 : 1; const streamoff framesize = wordsize * width * height * 3 / 2; const streamoff offset = framesize * numFrames; /* attempt to seek */ if (!!m_cHandle.seekg(offset, ios::cur)) return; /* success */ m_cHandle.clear(); /* fall back to consuming the input */ char buf[512]; const unsigned offset_mod_bufsize = offset % sizeof(buf); for (streamoff i = 0; i < offset - offset_mod_bufsize; i += sizeof(buf)) { m_cHandle.read(buf, sizeof(buf)); } m_cHandle.read(buf, offset_mod_bufsize); } /** * Read width*height pixels from fd into dst, optionally * padding the left and right edges by edge-extension. Input may be * either 8bit or 16bit little-endian lsb-aligned words. * * @param dst destination image * @param fd input file stream * @param is16bit true if input file carries > 8bit data, false otherwise. * @param stride distance between vertically adjacent pixels of dst. * @param width width of active area in dst. * @param height height of active area in dst. * @param pad_x length of horizontal padding. * @param pad_y length of vertical padding. * @return true for success, false in case of error */ static bool readPlane(Pel* dst, istream& fd, bool is16bit, unsigned int stride, unsigned int width, unsigned int height, unsigned int pad_x, unsigned int pad_y) { int read_len = width * (is16bit ? 2 : 1); unsigned char *buf = new unsigned char[read_len]; for (int y = 0; y < height; y++) { fd.read(reinterpret_cast(buf), read_len); if (fd.eof() || fd.fail() ) { delete[] buf; return false; } if (!is16bit) { for (int x = 0; x < width; x++) { dst[x] = buf[x]; } } else { for (int x = 0; x < width; x++) { dst[x] = (buf[2*x+1] << 8) | buf[2*x]; } } for (int x = width; x < width + pad_x; x++) { dst[x] = dst[width - 1]; } dst += stride; } for (int y = height; y < height + pad_y; y++) { for (int x = 0; x < width + pad_x; x++) { dst[x] = (dst - stride)[x]; } dst += stride; } delete[] buf; return true; } /** * Write width*height pixels info fd from src. * * @param fd output file stream * @param src source image * @param is16bit true if input file carries > 8bit data, false otherwise. * @param stride distance between vertically adjacent pixels of src. * @param width width of active area in src. * @param height height of active area in src. * @return true for success, false in case of error */ static bool writePlane(ostream& fd, Pel* src, bool is16bit, unsigned int stride, unsigned int width, unsigned int height) { int write_len = width * (is16bit ? 2 : 1); unsigned char *buf = new unsigned char[write_len]; for (int y = 0; y < height; y++) { if (!is16bit) { for (int x = 0; x < width; x++) { buf[x] = (unsigned char) src[x]; } } else { for (int x = 0; x < width; x++) { buf[2*x] = src[x] & 0xff; buf[2*x+1] = (src[x] >> 8) & 0xff; } } fd.write(reinterpret_cast(buf), write_len); if (fd.eof() || fd.fail() ) { delete[] buf; return false; } src += stride; } delete[] buf; return true; } /** * Read one Y'CbCr frame, performing any required input scaling to change * from the bitdepth of the input file to the internal bit-depth. * * If a bit-depth reduction is required, and internalBitdepth >= 8, then * the input file is assumed to be ITU-R BT.601/709 compliant, and the * resulting data is clipped to the appropriate legal range, as if the * file had been provided at the lower-bitdepth compliant to Rec601/709. * * @param pPicYuv input picture YUV buffer class pointer * @param aiPad source padding size, aiPad[0] = horizontal, aiPad[1] = vertical * @return true for success, false in case of error */ bool TVideoIOYuv::read ( TComPicYuv* pPicYuv, Int aiPad[2], Bool bRewind ) { // check end-of-file if ( isEof() ) return false; Int iStride = pPicYuv->getStride(); // compute actual YUV width & height excluding padding size unsigned int pad_h = aiPad[0]; unsigned int pad_v = aiPad[1]; unsigned int width_full = pPicYuv->getWidth(); unsigned int height_full = pPicYuv->getHeight(); unsigned int width = width_full - pad_h; unsigned int height = height_full - pad_v; bool is16bit = m_fileBitdepth > 8; int desired_bitdepth = m_fileBitdepth + m_bitdepthShift; Pel minval = 0; Pel maxval = (1 << desired_bitdepth) - 1; #if CLIP_TO_709_RANGE if (m_bitdepthShift < 0 && desired_bitdepth >= 8) { /* ITU-R BT.709 compliant clipping for converting say 10b to 8b */ minval = 1 << (desired_bitdepth - 8); maxval = (0xff << (desired_bitdepth - 8)) -1; } #endif if (! readPlane(pPicYuv->getLumaAddr(), m_cHandle, is16bit, iStride, width, height, pad_h, pad_v)) return false; scalePlane(pPicYuv->getLumaAddr(), iStride, width_full, height_full, m_bitdepthShift, minval, maxval); iStride >>= 1; width_full >>= 1; height_full >>= 1; width >>= 1; height >>= 1; pad_h >>= 1; pad_v >>= 1; if (! readPlane(pPicYuv->getCbAddr(), m_cHandle, is16bit, iStride, width, height, pad_h, pad_v)) return false; scalePlane(pPicYuv->getCbAddr(), iStride, width_full, height_full, m_bitdepthShift, minval, maxval); if (! readPlane(pPicYuv->getCrAddr(), m_cHandle, is16bit, iStride, width, height, pad_h, pad_v)) return false; scalePlane(pPicYuv->getCrAddr(), iStride, width_full, height_full, m_bitdepthShift, minval, maxval); if( bRewind ) { Int iFrameSize = ( is16bit ? 2 : 1 ) * 6 * width * height; m_cHandle.seekg( -iFrameSize, std::ios_base::cur ); } return true; } /** * Write one Y'CbCr frame. No bit-depth conversion is performed, pcPicYuv is * assumed to be at TVideoIO::m_fileBitdepth depth. * * @param pPicYuv input picture YUV buffer class pointer * @param aiPad source padding size, aiPad[0] = horizontal, aiPad[1] = vertical * @return true for success, false in case of error */ #if PIC_CROPPING Bool TVideoIOYuv::write( TComPicYuv* pPicYuv, Int cropLeft, Int cropRight, Int cropTop, Int cropBottom ) #else bool TVideoIOYuv::write( TComPicYuv* pPicYuv, Int aiPad[2] ) #endif { // compute actual YUV frame size excluding padding size Int iStride = pPicYuv->getStride(); #if PIC_CROPPING UInt width = pPicYuv->getWidth() - cropLeft - cropRight; UInt height = pPicYuv->getHeight() - cropTop - cropBottom; #else unsigned int width = pPicYuv->getWidth() - aiPad[0]; unsigned int height = pPicYuv->getHeight() - aiPad[1]; #endif bool is16bit = m_fileBitdepth > 8; TComPicYuv *dstPicYuv = NULL; bool retval = true; if (m_bitdepthShift != 0) { dstPicYuv = new TComPicYuv; dstPicYuv->create( pPicYuv->getWidth(), pPicYuv->getHeight(), 1, 1, 0 ); pPicYuv->copyToPic(dstPicYuv); Pel minval = 0; Pel maxval = (1 << m_fileBitdepth) - 1; #if CLIP_TO_709_RANGE if (-m_bitdepthShift < 0 && m_fileBitdepth >= 8) { /* ITU-R BT.709 compliant clipping for converting say 10b to 8b */ minval = 1 << (m_fileBitdepth - 8); maxval = (0xff << (m_fileBitdepth - 8)) -1; } #endif scalePlane(dstPicYuv->getLumaAddr(), dstPicYuv->getStride(), dstPicYuv->getWidth(), dstPicYuv->getHeight(), -m_bitdepthShift, minval, maxval); scalePlane(dstPicYuv->getCbAddr(), dstPicYuv->getCStride(), dstPicYuv->getWidth()>>1, dstPicYuv->getHeight()>>1, -m_bitdepthShift, minval, maxval); scalePlane(dstPicYuv->getCrAddr(), dstPicYuv->getCStride(), dstPicYuv->getWidth()>>1, dstPicYuv->getHeight()>>1, -m_bitdepthShift, minval, maxval); } else { dstPicYuv = pPicYuv; } #if PIC_CROPPING // location of upper left pel in a plane Int planeOffset = 0; //cropLeft + cropTop * iStride; if (! writePlane(m_cHandle, dstPicYuv->getLumaAddr() + planeOffset, is16bit, iStride, width, height)) #else if (! writePlane(m_cHandle, dstPicYuv->getLumaAddr(), is16bit, iStride, width, height)) #endif { retval=false; goto exit; } width >>= 1; height >>= 1; iStride >>= 1; #if PIC_CROPPING cropLeft >>= 1; cropRight >>= 1; planeOffset = 0; // cropLeft + cropTop * iStride; if (! writePlane(m_cHandle, dstPicYuv->getCbAddr() + planeOffset, is16bit, iStride, width, height)) #else if (! writePlane(m_cHandle, dstPicYuv->getCbAddr(), is16bit, iStride, width, height)) #endif { retval=false; goto exit; } #if PIC_CROPPING if (! writePlane(m_cHandle, dstPicYuv->getCrAddr() + planeOffset, is16bit, iStride, width, height)) #else if (! writePlane(m_cHandle, dstPicYuv->getCrAddr(), is16bit, iStride, width, height)) #endif { retval=false; goto exit; } exit: if (m_bitdepthShift != 0) { dstPicYuv->destroy(); delete dstPicYuv; } return retval; }