/* =============================================================================== FILE: decompressor.hpp CONTENTS: Integer decompressor PROGRAMMERS: martin.isenburg@rapidlasso.com - http://rapidlasso.com uday.karan@gmail.com - Hobu, Inc. COPYRIGHT: (c) 2007-2014, martin isenburg, rapidlasso - tools to catch reality (c) 2014, Uday Verma, Hobu, Inc. This is free software; you can redistribute and/or modify it under the terms of the GNU Lesser General Licence as published by the Free Software Foundation. See the COPYING file for more information. This software is distributed WITHOUT ANY WARRANTY and without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. CHANGE HISTORY: =============================================================================== */ #ifndef __decompressor_hpp__ #define __decompressor_hpp__ #include "model.hpp" #include #include #include namespace laszip { namespace decompressors { struct integer { integer(U32 bits = 16, U32 contexts = 1, U32 bits_high = 8, U32 range = 0): bits(bits), contexts(contexts), bits_high(bits_high), range(range) { if (range) { // the corrector's significant bits and range corr_bits = 0; corr_range = range; while (range) { range = range >> 1; corr_bits++; } if (corr_range == (1u << (corr_bits-1))) { corr_bits--; } // the corrector must fall into this interval corr_min = -((I32)(corr_range/2)); corr_max = corr_min + corr_range - 1; } else if (bits && bits < 32) { corr_bits = bits; corr_range = 1u << bits; // the corrector must fall into this interval corr_min = -((I32)(corr_range/2)); corr_max = corr_min + corr_range - 1; } else { corr_bits = 32; corr_range = 0; // the corrector must fall into this interval corr_min = I32_MIN; corr_max = I32_MAX; } k = 0; } void init() { using laszip::models::arithmetic; using laszip::models::arithmetic_bit; U32 i; // maybe create the models if (mBits.empty()) { for (i = 0; i < contexts; i++) mBits.push_back(arithmetic(corr_bits+1)); #ifndef COMPRESS_ONLY_K // mcorrector0 is already initialized for (i = 1; i <= corr_bits; i++) { U32 v = i <= bits_high ? 1 << i : 1 << bits_high; mCorrector.push_back(arithmetic(v)); } #endif } } template< typename TDecoder > I32 decompress(TDecoder& dec, I32 pred, U32 context) { I32 real = pred + readCorrector(dec, mBits[context]); if (real < 0) real += corr_range; else if ((U32)(real) >= corr_range) real -= corr_range; return real; } inline unsigned int getK() const { return k; } template< typename TDecoder, typename TEntroyModel > I32 readCorrector(TDecoder& dec, TEntroyModel& mBits) { I32 c; // decode within which interval the corrector is falling k = dec.decodeSymbol(mBits); // decode the exact location of the corrector within the interval #ifdef COMPRESS_ONLY_K if (k) // then c is either smaller than 0 or bigger than 1 { if (k < 32) { c = dec.readBits(k); if (c >= (1<<(k-1))) // if c is in the interval [ 2^(k-1) ... + 2^k - 1 ] { // so we translate c back into the interval [ 2^(k-1) + 1 ... 2^k ] by adding 1 c += 1; } else // otherwise c is in the interval [ 0 ... + 2^(k-1) - 1 ] { // so we translate c back into the interval [ - (2^k - 1) ... - (2^(k-1)) ] by subtracting (2^k - 1) c -= ((1<= (1<<(k-1))) // if c is in the interval [ 2^(k-1) ... + 2^k - 1 ] { // so we translate c back into the interval [ 2^(k-1) + 1 ... 2^k ] by adding 1 c += 1; } else // otherwise c is in the interval [ 0 ... + 2^(k-1) - 1 ] { // so we translate c back into the interval [ - (2^k - 1) ... - (2^(k-1)) ] by subtracting (2^k - 1) c -= ((1< mBits; laszip::models::arithmetic_bit mCorrector0; std::vector mCorrector; }; } } #endif // __decompressor_hpp__