Blob Blame History Raw
/*
 * libopenraw - unpack.cpp
 *
 * Copyright (C) 2008-2016 Hubert Figuiere
 * Copyright (C) 2008 Novell, Inc.
 *
 * This library is free software: you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public License
 * as published by the Free Software Foundation, either version 3 of
 * the License, or (at your option) any later version.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library.  If not, see
 * <http://www.gnu.org/licenses/>.
 */

#include <assert.h>

#include <libopenraw/consts.h>

#include "unpack.hpp"
#include "trace.hpp"
#include "ifd.hpp"

namespace OpenRaw {
namespace Internals {

using namespace Debug;

Unpack::Unpack(uint32_t w, uint32_t t)
  : m_w(w), m_type(t)
{
}

/* Return the size of an image row. */
size_t Unpack::block_size()
{
  size_t bs;
  if(m_type == IFD::COMPRESS_NIKON_PACK) {
    bs = (m_w / 2 * 3) + (m_w / 10);
  }
  else {
    bs = m_w / 2 * 3;
  }
  return bs;
}


/** source is in BE byte order
 * the output is always 16-bits values in native (host) byte order.
 * the source must correspond to an image row.
 */
or_error Unpack::unpack_be12to16(uint8_t *dest, size_t destsize, const uint8_t *src,
                                 size_t size, size_t & out)
{
  or_error err = OR_ERROR_NONE;
  uint16_t *dest16 = reinterpret_cast<uint16_t *>(dest);
  size_t pad = (m_type == IFD::COMPRESS_NIKON_PACK) ? 1 : 0;
  size_t n = size / (15 + pad);
  size_t rest = size % (15 + pad);
  size_t ret = n * 20 + rest / 3 * 4;

  out = 0;

  /* The inner loop advances 10 columns, which corresponds to 15 input
     bytes, 20 output bytes and, in a Nikon pack, one padding byte.*/
  if (pad) {
    if ((size % 16) != 0) {
      LOGERR("be12to16 incorrect padding.\n");
      return OR_ERROR_DECOMPRESSION;
    }
  }
  if ((rest % 3) != 0) {
    LOGERR("be12to16 incorrect rest.\n");
    return OR_ERROR_DECOMPRESSION;
  }

  for (size_t i = 0; i < n + 1; i++) {
    size_t m = (i == n) ? rest / 3 : 5;
    if((reinterpret_cast<uint8_t *>(dest16) - dest) + (m * 4) >  destsize) {
      err = OR_ERROR_DECOMPRESSION;
      LOGERR("overflow !\n");
      break;
    }
    for(size_t j = 0; j < m; j++) {
      /* Read 3 bytes */
      uint32_t t = *src++;
      t <<= 8;
      t |= *src++;
      t <<= 8;
      t |= *src++;

      /* Write two 16 bit values. */
      *dest16 = (t & (0xfff << 12)) >> 12;
      dest16++;

      *dest16 = t & 0xfff;
      dest16++;
    }

    src += pad;
  }

  out = ret;
  return err;
}

} }
/*
  Local Variables:
  mode:c++
  c-file-style:"stroustrup"
  c-file-offsets:((innamespace . 0))
  tab-width:2
  c-basic-offset:2
  indent-tabs-mode:nil
  fill-column:80
  End:
*/