LCOV - code coverage report
Current view: top level - third_party/LercLib - BitStuffer2.h (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 32 39 82.1 %
Date: 2025-01-18 12:42:00 Functions: 7 8 87.5 %

          Line data    Source code
       1             : /*
       2             : Copyright 2015 Esri
       3             : 
       4             : Licensed under the Apache License, Version 2.0 (the "License");
       5             : you may not use this file except in compliance with the License.
       6             : You may obtain a copy of the License at
       7             : 
       8             : http://www.apache.org/licenses/LICENSE-2.0
       9             : 
      10             : Unless required by applicable law or agreed to in writing, software
      11             : distributed under the License is distributed on an "AS IS" BASIS,
      12             : WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
      13             : See the License for the specific language governing permissions and
      14             : limitations under the License.
      15             : 
      16             : A local copy of the license and additional notices are located with the
      17             : source distribution at:
      18             : 
      19             : http://github.com/Esri/lerc/
      20             : 
      21             : Contributors:  Thomas Maurer
      22             : */
      23             : 
      24             : #ifndef BITSTUFFER2_H
      25             : #define BITSTUFFER2_H
      26             : 
      27             : #include <vector>
      28             : #include <cstring>
      29             : #include <utility>
      30             : #include "Defines.h"
      31             : 
      32             : NAMESPACE_LERC_START
      33             : 
      34             : /** Bit stuffer, for writing unsigned int arrays compressed lossless
      35             :  *
      36             :  */
      37             : 
      38             : class BitStuffer2
      39             : {
      40             : public:
      41       12493 :   BitStuffer2()           {}
      42       12493 :   virtual ~BitStuffer2()  {}
      43             : 
      44             :   // dst buffer is already allocated. byte ptr is moved like a file pointer.
      45             :   bool EncodeSimple(Byte** ppByte, const std::vector<unsigned int>& dataVec, int lerc2Version) const;
      46             :   bool EncodeLut(Byte** ppByte, const std::vector<std::pair<unsigned int, unsigned int> >& sortedDataVec, int lerc2Version) const;
      47             :   bool Decode(const Byte** ppByte, size_t& nBytesRemaining, std::vector<unsigned int>& dataVec, size_t maxElementCount, int lerc2Version) const;
      48             : 
      49             :   static unsigned int ComputeNumBytesNeededSimple(unsigned int numElem, unsigned int maxElem);
      50             :   static unsigned int ComputeNumBytesNeededLut(const std::vector<std::pair<unsigned int, unsigned int> >& sortedDataVec, bool& doLut);
      51             : 
      52             : private:
      53             :   mutable std::vector<unsigned int>  m_tmpLutVec, m_tmpIndexVec, m_tmpBitStuffVec;
      54             : 
      55             :   static void BitStuff_Before_Lerc2v3(Byte** ppByte, const std::vector<unsigned int>& dataVec, int numBits);
      56             :   bool BitUnStuff_Before_Lerc2v3(const Byte** ppByte, size_t& nBytesRemaining, std::vector<unsigned int>& dataVec, unsigned int numElements, int numBits) const;
      57             :   void BitStuff(Byte** ppByte, const std::vector<unsigned int>& dataVec, int numBits) const;
      58             :   bool BitUnStuff(const Byte** ppByte, size_t& nBytesRemaining, std::vector<unsigned int>& dataVec, unsigned int numElements, int numBits) const;
      59             : 
      60             :   static bool EncodeUInt(Byte** ppByte, unsigned int k, int numBytes);     // numBytes = 1, 2, or 4
      61             :   static bool DecodeUInt(const Byte** ppByte, size_t& nBytesRemaining, unsigned int& k, int numBytes);
      62     1747160 :   static int NumBytesUInt(unsigned int k)  { return (k < 256) ? 1 : (k < (1 << 16)) ? 2 : 4; }
      63             :   static unsigned int NumTailBytesNotNeeded(unsigned int numElem, int numBits);
      64             : };
      65             : 
      66             : // -------------------------------------------------------------------------- ;
      67             : 
      68       64556 : inline unsigned int BitStuffer2::ComputeNumBytesNeededSimple(unsigned int numElem, unsigned int maxElem)
      69             : {
      70       64556 :   int numBits = 0;
      71      426583 :   while ((numBits < 32) && (maxElem >> numBits))
      72      362027 :     numBits++;
      73       64556 :   return 1 + NumBytesUInt(numElem) + ((numElem * numBits + 7) >> 3);
      74             : }
      75             : 
      76             : // -------------------------------------------------------------------------- ;
      77             : 
      78      263465 : inline bool BitStuffer2::EncodeUInt(Byte** ppByte, unsigned int k, int numBytes)
      79             : {
      80      263465 :   Byte* ptr = *ppByte;
      81             : 
      82      263465 :   if (numBytes == 1)
      83      258238 :     *ptr = (Byte)k;
      84        5227 :   else if (numBytes == 2)
      85             :   {
      86        5227 :     unsigned short kShort = (unsigned short)k;
      87        5227 :     memcpy(ptr, &kShort, sizeof(unsigned short));
      88             :   }
      89           0 :   else if (numBytes == 4)
      90           0 :     memcpy(ptr, &k, sizeof(unsigned int));
      91             :   else
      92           0 :     return false;
      93             : 
      94      263465 :   *ppByte += numBytes;
      95      263465 :   return true;
      96             : }
      97             : 
      98             : // -------------------------------------------------------------------------- ;
      99             : 
     100      116864 : inline bool BitStuffer2::DecodeUInt(const Byte** ppByte, size_t& nBytesRemaining, unsigned int& k, int numBytes)
     101             : {
     102      116864 :   if (nBytesRemaining < (size_t)numBytes)
     103           0 :     return false;
     104             : 
     105      116864 :   const Byte* ptr = *ppByte;
     106             : 
     107      116864 :   if (numBytes == 1)
     108      115105 :     k = *ptr;
     109        1759 :   else if (numBytes == 2)
     110             :   {
     111             :     unsigned short s;
     112        1759 :     memcpy(&s, ptr, sizeof(unsigned short));
     113        1759 :     k = s;
     114             :   }
     115           0 :   else if (numBytes == 4)
     116           0 :     memcpy(&k, ptr, sizeof(unsigned int));
     117             :   else
     118           0 :     return false;
     119             : 
     120      116864 :   *ppByte += numBytes;
     121      116864 :   nBytesRemaining -= numBytes;
     122      116864 :   return true;
     123             : }
     124             : 
     125             : // -------------------------------------------------------------------------- ;
     126             : 
     127      677721 : inline unsigned int BitStuffer2::NumTailBytesNotNeeded(unsigned int numElem, int numBits)
     128             : {
     129      677721 :   int numBitsTail = ((unsigned long long)numElem * numBits) & 31;
     130      677721 :   int numBytesTail = (numBitsTail + 7) >> 3;
     131      677721 :   return (numBytesTail > 0) ? 4 - numBytesTail : 0;
     132             : }
     133             : 
     134             : // -------------------------------------------------------------------------- ;
     135             : 
     136             : NAMESPACE_LERC_END
     137             : #endif

Generated by: LCOV version 1.14