mirror of
https://github.com/HarbourMasters/Shipwright.git
synced 2024-08-13 17:03:47 -04:00
b3e299dbde
Some modifications to handle backslashes and forward slashes, along with some optimizations to speed up OTR generation.
144 lines
5.8 KiB
C++
144 lines
5.8 KiB
C++
/*****************************************************************************/
|
|
/* huffman.h Copyright (c) Ladislav Zezula 2003 */
|
|
/*---------------------------------------------------------------------------*/
|
|
/* Description : */
|
|
/*---------------------------------------------------------------------------*/
|
|
/* Date Ver Who Comment */
|
|
/* -------- ---- --- ------- */
|
|
/* xx.xx.xx 1.00 Lad The first version of huffman.h */
|
|
/* 03.05.03 2.00 Lad Added compression */
|
|
/* 08.12.03 2.01 Dan High-memory handling (> 0x80000000) */
|
|
/*****************************************************************************/
|
|
|
|
#ifndef __HUFFMAN_H__
|
|
#define __HUFFMAN_H__
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Defines
|
|
|
|
#define HUFF_ITEM_COUNT 0x203 // Number of items in the item pool
|
|
#define LINK_ITEM_COUNT 0x80 // Maximum number of quick-link items
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Structures and classes
|
|
|
|
// Input stream for Huffmann decompression
|
|
class TInputStream
|
|
{
|
|
public:
|
|
|
|
TInputStream(void * pvInBuffer, size_t cbInBuffer);
|
|
unsigned int Get1Bit();
|
|
unsigned int Peek7Bits();
|
|
unsigned int Get8Bits();
|
|
void SkipBits(unsigned int BitCount);
|
|
|
|
unsigned char * pbInBufferEnd; // End position in the the input buffer
|
|
unsigned char * pbInBuffer; // Current position in the the input buffer
|
|
unsigned int BitBuffer; // Input bit buffer
|
|
unsigned int BitCount; // Number of bits remaining in 'dwBitBuff'
|
|
};
|
|
|
|
|
|
// Output stream for Huffmann compression
|
|
class TOutputStream
|
|
{
|
|
public:
|
|
|
|
TOutputStream(void * pvOutBuffer, size_t cbOutLength);
|
|
void PutBits(unsigned int dwValue, unsigned int nBitCount);
|
|
void Flush();
|
|
|
|
unsigned char * pbOutBufferEnd; // End position in the output buffer
|
|
unsigned char * pbOutBuffer; // Current position in the output buffer
|
|
unsigned int BitBuffer; // Bit buffer
|
|
unsigned int BitCount; // Number of bits in the bit buffer
|
|
};
|
|
|
|
// A virtual tree item that represents the head of the item list
|
|
#define LIST_HEAD() ((THTreeItem *)(&pFirst))
|
|
|
|
enum TInsertPoint
|
|
{
|
|
InsertAfter = 1,
|
|
InsertBefore = 2
|
|
};
|
|
|
|
// Huffmann tree item
|
|
struct THTreeItem
|
|
{
|
|
THTreeItem() { pPrev = pNext = NULL; DecompressedValue = 0; Weight = 0; pParent = pChildLo = NULL; }
|
|
// ~THTreeItem() { RemoveItem(); }
|
|
|
|
void RemoveItem();
|
|
// void RemoveEntry();
|
|
|
|
THTreeItem * pNext; // Pointer to lower-weight tree item
|
|
THTreeItem * pPrev; // Pointer to higher-weight item
|
|
unsigned int DecompressedValue; // 08 - Decompressed byte value (also index in the array)
|
|
unsigned int Weight; // 0C - Weight
|
|
THTreeItem * pParent; // 10 - Pointer to parent item (NULL if none)
|
|
THTreeItem * pChildLo; // 14 - Pointer to the child with lower-weight child ("left child")
|
|
};
|
|
|
|
|
|
// Structure used for quick navigating in the huffmann tree.
|
|
// Allows skipping up to 7 bits in the compressed stream, thus
|
|
// decompressing a bit faster. Sometimes it can even get the decompressed
|
|
// byte directly.
|
|
struct TQuickLink
|
|
{
|
|
unsigned int ValidValue; // If greater than THuffmannTree::MinValidValue, the entry is valid
|
|
unsigned int ValidBits; // Number of bits that are valid for this item link
|
|
union
|
|
{
|
|
THTreeItem * pItem; // Pointer to the item within the Huffmann tree
|
|
unsigned int DecompressedValue; // Value for direct decompression
|
|
};
|
|
};
|
|
|
|
|
|
// Structure for Huffman tree (Size 0x3674 bytes). Because I'm not expert
|
|
// for the decompression, I do not know actually if the class is really a Hufmann
|
|
// tree. If someone knows the decompression details, please let me know
|
|
class THuffmannTree
|
|
{
|
|
public:
|
|
|
|
THuffmannTree(bool bCompression);
|
|
~THuffmannTree();
|
|
|
|
void LinkTwoItems(THTreeItem * pItem1, THTreeItem * pItem2);
|
|
void InsertItem(THTreeItem * item, TInsertPoint InsertPoint, THTreeItem * item2);
|
|
|
|
THTreeItem * FindHigherOrEqualItem(THTreeItem * pItem, unsigned int Weight);
|
|
THTreeItem * CreateNewItem(unsigned int DecompressedValue, unsigned int Weight, TInsertPoint InsertPoint);
|
|
|
|
unsigned int FixupItemPosByWeight(THTreeItem * pItem, unsigned int MaxWeight);
|
|
bool BuildTree(unsigned int CompressionType);
|
|
|
|
void IncWeightsAndRebalance(THTreeItem * pItem);
|
|
bool InsertNewBranchAndRebalance(unsigned int Value1, unsigned int Value2);
|
|
|
|
void EncodeOneByte(TOutputStream * os, THTreeItem * pItem);
|
|
unsigned int DecodeOneByte(TInputStream * is);
|
|
|
|
unsigned int Compress(TOutputStream * os, void * pvInBuffer, int cbInBuffer, int nCmpType);
|
|
unsigned int Decompress(void * pvOutBuffer, unsigned int cbOutLength, TInputStream * is);
|
|
|
|
THTreeItem ItemBuffer[HUFF_ITEM_COUNT]; // Buffer for tree items. No memory allocation is needed
|
|
unsigned int ItemsUsed; // Number of tree items used from ItemBuffer
|
|
|
|
// Head of the linear item list
|
|
THTreeItem * pFirst; // Pointer to the highest weight item
|
|
THTreeItem * pLast; // Pointer to the lowest weight item
|
|
|
|
THTreeItem * ItemsByByte[0x102]; // Array of item pointers, one for each possible byte value
|
|
TQuickLink QuickLinks[LINK_ITEM_COUNT]; // Array of quick-link items
|
|
|
|
unsigned int MinValidValue; // A minimum value of TQDecompress::ValidValue to be considered valid
|
|
unsigned int bIsCmp0; // 1 if compression type 0
|
|
};
|
|
|
|
#endif // __HUFFMAN_H__
|