2025-08-26 01:02:06 +01:00
|
|
|
|
/*$Source: /usr/home/dhesi/zoo/RCS/decode.c,v $*/
|
|
|
|
|
|
/*$Id: decode.c,v 1.6 91/07/09 01:39:49 dhesi Exp $*/
|
|
|
|
|
|
/***********************************************************
|
|
|
|
|
|
decode.c
|
|
|
|
|
|
|
2025-08-26 02:06:19 +01:00
|
|
|
|
Adapted from Haruhiko Okumura’s “ar” archiver. This
|
|
|
|
|
|
version has been modified in 2025 by Natalia Portillo
|
|
|
|
|
|
for in-memory decompression.
|
2025-08-26 01:02:06 +01:00
|
|
|
|
***********************************************************/
|
|
|
|
|
|
|
2025-08-26 02:06:19 +01:00
|
|
|
|
#include <limits.h> // for UCHAR_MAX
|
|
|
|
|
|
#include <stdint.h> // for fixed-width integer types
|
2025-08-26 01:02:06 +01:00
|
|
|
|
|
2025-08-26 02:06:19 +01:00
|
|
|
|
#include "ar.h" // archive format constants
|
|
|
|
|
|
#include "lzh.h" // LZH-specific constants (DICSIZ, THRESHOLD, etc.)
|
2025-08-26 01:02:06 +01:00
|
|
|
|
|
2025-08-26 02:06:19 +01:00
|
|
|
|
extern int decoded; // flag set by decode_c() when end-of-stream is reached
|
2025-08-26 01:02:06 +01:00
|
|
|
|
|
2025-08-26 02:06:19 +01:00
|
|
|
|
static int j; // number of literal/copy runs remaining from a match
|
2025-08-26 01:02:06 +01:00
|
|
|
|
|
2025-08-26 02:06:19 +01:00
|
|
|
|
/*
|
|
|
|
|
|
* decode_start()
|
|
|
|
|
|
*
|
|
|
|
|
|
* Prepare the decoder for a new file:
|
|
|
|
|
|
* - Initialize the Huffman bitstream (via huf_decode_start())
|
|
|
|
|
|
* - Reset the sliding-window copy counter `j`
|
|
|
|
|
|
* - Clear the end-of-data flag `decoded`
|
|
|
|
|
|
*/
|
2025-08-26 01:02:06 +01:00
|
|
|
|
void decode_start()
|
|
|
|
|
|
{
|
2025-08-26 02:06:19 +01:00
|
|
|
|
huf_decode_start(); // reset bit-reader state
|
|
|
|
|
|
j = 0; // no pending copy runs yet
|
|
|
|
|
|
decoded = 0; // not yet at end-of-stream
|
2025-08-26 01:02:06 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
2025-08-26 02:06:19 +01:00
|
|
|
|
* decode(count, buffer)
|
|
|
|
|
|
*
|
|
|
|
|
|
* Decode up to `count` bytes (usually DICSIZ) into `buffer[]`.
|
|
|
|
|
|
* Returns the actual number of bytes written, or 0 if `decoded` is set.
|
|
|
|
|
|
*
|
|
|
|
|
|
* Sliding‐window logic:
|
|
|
|
|
|
* 1. If `j` > 0, we are in the middle of copying a previous match:
|
|
|
|
|
|
* - Copy one byte from `buffer[i]` into `buffer[r]`
|
|
|
|
|
|
* - Advance `i` (circular within DICSIZ) and `r`
|
|
|
|
|
|
* - Decrement `j` and repeat until `j` = 0 or `r` = count
|
|
|
|
|
|
* 2. Otherwise, fetch the next symbol `c = decode_c()`:
|
|
|
|
|
|
* - If `c <= UCHAR_MAX`, it’s a literal byte: emit it directly
|
|
|
|
|
|
* - Else it’s a match:
|
|
|
|
|
|
* • compute `j = match_length = c - (UCHAR_MAX + 1 - THRESHOLD)`
|
|
|
|
|
|
* • compute `i = (r - match_offset - 1) mod DICSIZ`,
|
|
|
|
|
|
* where match_offset = decode_p()
|
|
|
|
|
|
* • enter copy loop from step 1
|
|
|
|
|
|
*/
|
2025-08-26 01:02:06 +01:00
|
|
|
|
int decode(uint32_t count, uint8_t *buffer)
|
|
|
|
|
|
{
|
2025-08-26 02:06:19 +01:00
|
|
|
|
static uint32_t i; // sliding-window read index (circular)
|
|
|
|
|
|
uint32_t r; // write position in buffer
|
|
|
|
|
|
uint32_t c; // symbol or match code
|
2025-08-26 01:02:06 +01:00
|
|
|
|
|
|
|
|
|
|
r = 0;
|
2025-08-26 02:06:19 +01:00
|
|
|
|
|
|
|
|
|
|
// Step 1: finish any pending copy from a previous match
|
2025-08-26 01:02:06 +01:00
|
|
|
|
while(--j >= 0)
|
|
|
|
|
|
{
|
2025-08-26 02:06:19 +01:00
|
|
|
|
buffer[r] = buffer[i]; // copy one byte from history
|
|
|
|
|
|
i = (i + 1) & (DICSIZ - 1); // wrap index within [0, DICSIZ)
|
|
|
|
|
|
if(++r == count) // if output buffer is full
|
|
|
|
|
|
return r; // return bytes written so far
|
2025-08-26 01:02:06 +01:00
|
|
|
|
}
|
2025-08-26 02:06:19 +01:00
|
|
|
|
|
|
|
|
|
|
// Step 2: decode new symbols until end-of-stream or buffer full
|
2025-08-26 01:02:06 +01:00
|
|
|
|
for(;;)
|
|
|
|
|
|
{
|
2025-08-26 02:06:19 +01:00
|
|
|
|
c = decode_c(); // get next Huffman symbol
|
|
|
|
|
|
if(decoded) // end-of-stream marker reached
|
|
|
|
|
|
return r; // no more bytes to decode
|
|
|
|
|
|
|
2025-08-26 01:02:06 +01:00
|
|
|
|
if(c <= UCHAR_MAX)
|
|
|
|
|
|
{
|
2025-08-26 02:06:19 +01:00
|
|
|
|
// Literal byte: emit it directly
|
|
|
|
|
|
buffer[r] = (uint8_t)c;
|
2025-08-26 01:02:06 +01:00
|
|
|
|
if(++r == count) return r;
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
2025-08-26 02:06:19 +01:00
|
|
|
|
// Match sequence: compute how many bytes to copy
|
|
|
|
|
|
// j = match length
|
2025-08-26 01:02:06 +01:00
|
|
|
|
j = c - (UCHAR_MAX + 1 - THRESHOLD);
|
2025-08-26 02:06:19 +01:00
|
|
|
|
|
|
|
|
|
|
// i = start position in sliding window:
|
|
|
|
|
|
// current output position minus offset minus 1, wrapped
|
2025-08-26 01:02:06 +01:00
|
|
|
|
i = (r - decode_p() - 1) & (DICSIZ - 1);
|
2025-08-26 02:06:19 +01:00
|
|
|
|
|
|
|
|
|
|
// Copy `j` bytes from history
|
2025-08-26 01:02:06 +01:00
|
|
|
|
while(--j >= 0)
|
|
|
|
|
|
{
|
|
|
|
|
|
buffer[r] = buffer[i];
|
|
|
|
|
|
i = (i + 1) & (DICSIZ - 1);
|
|
|
|
|
|
if(++r == count) return r;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2025-08-26 02:06:19 +01:00
|
|
|
|
}
|