mirror of https://github.com/bsnes-emu/bsnes.git
43 lines
961 B
C++
43 lines
961 B
C++
#pragma once
|
|
|
|
namespace nall { namespace Decode {
|
|
|
|
inline auto Huffman(const void* data) -> vector<uint8_t> {
|
|
auto input = (const uint8_t*)data;
|
|
vector<uint8_t> output;
|
|
|
|
uint size = 0;
|
|
for(uint byte : range(8)) size |= *input++ << byte * 8;
|
|
output.reserve(size);
|
|
|
|
uint byte = 0, bits = 0;
|
|
auto read = [&]() -> bool {
|
|
if(bits == 0) bits = 8, byte = *input++;
|
|
return byte >> --bits & 1;
|
|
};
|
|
|
|
uint nodes[256][2] = {};
|
|
for(uint offset : range(256)) {
|
|
for(uint index : range(9)) nodes[offset][0] = nodes[offset][0] << 1 | read();
|
|
for(uint index : range(9)) nodes[offset][1] = nodes[offset][1] << 1 | read();
|
|
}
|
|
|
|
uint node = 511;
|
|
while(output.size() < size) {
|
|
node = nodes[node - 256][read()];
|
|
if(node < 256) {
|
|
output.append(node);
|
|
node = 511;
|
|
}
|
|
}
|
|
|
|
return output;
|
|
}
|
|
|
|
template<typename T>
|
|
inline auto Huffman(const vector<T>& buffer) -> vector<uint8_t> {
|
|
return move(Huffman(buffer.data()));
|
|
}
|
|
|
|
}}
|