318 lines
7.7 KiB
C++
318 lines
7.7 KiB
C++
// Copyright (c) 2017 Google Inc.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
#include <algorithm>
|
|
#include <map>
|
|
#include <sstream>
|
|
#include <string>
|
|
#include <unordered_map>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
#include "gmock/gmock.h"
|
|
#include "source/comp/bit_stream.h"
|
|
#include "source/comp/huffman_codec.h"
|
|
|
|
namespace spvtools {
|
|
namespace comp {
|
|
namespace {
|
|
|
|
const std::map<std::string, uint32_t>& GetTestSet() {
|
|
static const std::map<std::string, uint32_t> hist = {
|
|
{"a", 4}, {"e", 7}, {"f", 3}, {"h", 2}, {"i", 3},
|
|
{"m", 2}, {"n", 2}, {"s", 2}, {"t", 2}, {"l", 1},
|
|
{"o", 2}, {"p", 1}, {"r", 1}, {"u", 1}, {"x", 1},
|
|
};
|
|
|
|
return hist;
|
|
}
|
|
|
|
class TestBitReader {
|
|
public:
|
|
TestBitReader(const std::string& bits) : bits_(bits) {}
|
|
|
|
bool ReadBit(bool* bit) {
|
|
if (pos_ < bits_.length()) {
|
|
*bit = bits_[pos_++] == '0' ? false : true;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
private:
|
|
std::string bits_;
|
|
size_t pos_ = 0;
|
|
};
|
|
|
|
TEST(Huffman, PrintTree) {
|
|
HuffmanCodec<std::string> huffman(GetTestSet());
|
|
std::stringstream ss;
|
|
huffman.PrintTree(ss);
|
|
|
|
// clang-format off
|
|
const std::string expected = std::string(R"(
|
|
15-----7------e
|
|
8------4------a
|
|
4------2------m
|
|
2------n
|
|
19-----8------4------2------o
|
|
2------s
|
|
4------2------t
|
|
2------1------l
|
|
1------p
|
|
11-----5------2------1------r
|
|
1------u
|
|
3------f
|
|
6------3------i
|
|
3------1------x
|
|
2------h
|
|
)").substr(1);
|
|
// clang-format on
|
|
|
|
EXPECT_EQ(expected, ss.str());
|
|
}
|
|
|
|
TEST(Huffman, PrintTable) {
|
|
HuffmanCodec<std::string> huffman(GetTestSet());
|
|
std::stringstream ss;
|
|
huffman.PrintTable(ss);
|
|
|
|
const std::string expected = std::string(R"(
|
|
e 7 11
|
|
a 4 101
|
|
i 3 0001
|
|
f 3 0010
|
|
t 2 0101
|
|
s 2 0110
|
|
o 2 0111
|
|
n 2 1000
|
|
m 2 1001
|
|
h 2 00000
|
|
x 1 00001
|
|
u 1 00110
|
|
r 1 00111
|
|
p 1 01000
|
|
l 1 01001
|
|
)")
|
|
.substr(1);
|
|
|
|
EXPECT_EQ(expected, ss.str());
|
|
}
|
|
|
|
TEST(Huffman, TestValidity) {
|
|
HuffmanCodec<std::string> huffman(GetTestSet());
|
|
const auto& encoding_table = huffman.GetEncodingTable();
|
|
std::vector<std::string> codes;
|
|
for (const auto& entry : encoding_table) {
|
|
codes.push_back(BitsToStream(entry.second.first, entry.second.second));
|
|
}
|
|
|
|
std::sort(codes.begin(), codes.end());
|
|
|
|
ASSERT_LT(codes.size(), 20u) << "Inefficient test ahead";
|
|
|
|
for (size_t i = 0; i < codes.size(); ++i) {
|
|
for (size_t j = i + 1; j < codes.size(); ++j) {
|
|
ASSERT_FALSE(codes[i] == codes[j].substr(0, codes[i].length()))
|
|
<< codes[i] << " is prefix of " << codes[j];
|
|
}
|
|
}
|
|
}
|
|
|
|
TEST(Huffman, TestEncode) {
|
|
HuffmanCodec<std::string> huffman(GetTestSet());
|
|
|
|
uint64_t bits = 0;
|
|
size_t num_bits = 0;
|
|
|
|
EXPECT_TRUE(huffman.Encode("e", &bits, &num_bits));
|
|
EXPECT_EQ(2u, num_bits);
|
|
EXPECT_EQ("11", BitsToStream(bits, num_bits));
|
|
|
|
EXPECT_TRUE(huffman.Encode("a", &bits, &num_bits));
|
|
EXPECT_EQ(3u, num_bits);
|
|
EXPECT_EQ("101", BitsToStream(bits, num_bits));
|
|
|
|
EXPECT_TRUE(huffman.Encode("x", &bits, &num_bits));
|
|
EXPECT_EQ(5u, num_bits);
|
|
EXPECT_EQ("00001", BitsToStream(bits, num_bits));
|
|
|
|
EXPECT_FALSE(huffman.Encode("y", &bits, &num_bits));
|
|
}
|
|
|
|
TEST(Huffman, TestDecode) {
|
|
HuffmanCodec<std::string> huffman(GetTestSet());
|
|
TestBitReader bit_reader(
|
|
"01001"
|
|
"0001"
|
|
"1000"
|
|
"00110"
|
|
"00001"
|
|
"00");
|
|
auto read_bit = [&bit_reader](bool* bit) { return bit_reader.ReadBit(bit); };
|
|
|
|
std::string decoded;
|
|
|
|
ASSERT_TRUE(huffman.DecodeFromStream(read_bit, &decoded));
|
|
EXPECT_EQ("l", decoded);
|
|
|
|
ASSERT_TRUE(huffman.DecodeFromStream(read_bit, &decoded));
|
|
EXPECT_EQ("i", decoded);
|
|
|
|
ASSERT_TRUE(huffman.DecodeFromStream(read_bit, &decoded));
|
|
EXPECT_EQ("n", decoded);
|
|
|
|
ASSERT_TRUE(huffman.DecodeFromStream(read_bit, &decoded));
|
|
EXPECT_EQ("u", decoded);
|
|
|
|
ASSERT_TRUE(huffman.DecodeFromStream(read_bit, &decoded));
|
|
EXPECT_EQ("x", decoded);
|
|
|
|
ASSERT_FALSE(huffman.DecodeFromStream(read_bit, &decoded));
|
|
}
|
|
|
|
TEST(Huffman, TestDecodeNumbers) {
|
|
const std::map<uint32_t, uint32_t> hist = {{1, 10}, {2, 5}, {3, 15}};
|
|
HuffmanCodec<uint32_t> huffman(hist);
|
|
|
|
TestBitReader bit_reader(
|
|
"1"
|
|
"1"
|
|
"01"
|
|
"00"
|
|
"01"
|
|
"1");
|
|
auto read_bit = [&bit_reader](bool* bit) { return bit_reader.ReadBit(bit); };
|
|
|
|
uint32_t decoded;
|
|
|
|
ASSERT_TRUE(huffman.DecodeFromStream(read_bit, &decoded));
|
|
EXPECT_EQ(3u, decoded);
|
|
|
|
ASSERT_TRUE(huffman.DecodeFromStream(read_bit, &decoded));
|
|
EXPECT_EQ(3u, decoded);
|
|
|
|
ASSERT_TRUE(huffman.DecodeFromStream(read_bit, &decoded));
|
|
EXPECT_EQ(2u, decoded);
|
|
|
|
ASSERT_TRUE(huffman.DecodeFromStream(read_bit, &decoded));
|
|
EXPECT_EQ(1u, decoded);
|
|
|
|
ASSERT_TRUE(huffman.DecodeFromStream(read_bit, &decoded));
|
|
EXPECT_EQ(2u, decoded);
|
|
|
|
ASSERT_TRUE(huffman.DecodeFromStream(read_bit, &decoded));
|
|
EXPECT_EQ(3u, decoded);
|
|
}
|
|
|
|
TEST(Huffman, SerializeToTextU64) {
|
|
const std::map<uint64_t, uint32_t> hist = {{1001, 10}, {1002, 5}, {1003, 15}};
|
|
HuffmanCodec<uint64_t> huffman(hist);
|
|
|
|
const std::string code = huffman.SerializeToText(2);
|
|
|
|
const std::string expected = R"((5, {
|
|
{0, 0, 0},
|
|
{1001, 0, 0},
|
|
{1002, 0, 0},
|
|
{1003, 0, 0},
|
|
{0, 1, 2},
|
|
{0, 4, 3},
|
|
}))";
|
|
|
|
ASSERT_EQ(expected, code);
|
|
}
|
|
|
|
TEST(Huffman, SerializeToTextString) {
|
|
const std::map<std::string, uint32_t> hist = {
|
|
{"aaa", 10}, {"bbb", 20}, {"ccc", 15}};
|
|
HuffmanCodec<std::string> huffman(hist);
|
|
|
|
const std::string code = huffman.SerializeToText(4);
|
|
|
|
const std::string expected = R"((5, {
|
|
{"", 0, 0},
|
|
{"aaa", 0, 0},
|
|
{"bbb", 0, 0},
|
|
{"ccc", 0, 0},
|
|
{"", 3, 1},
|
|
{"", 4, 2},
|
|
}))";
|
|
|
|
ASSERT_EQ(expected, code);
|
|
}
|
|
|
|
TEST(Huffman, CreateFromTextString) {
|
|
std::vector<HuffmanCodec<std::string>::Node> nodes = {
|
|
{},
|
|
{"root", 2, 3},
|
|
{"left", 0, 0},
|
|
{"right", 0, 0},
|
|
};
|
|
|
|
HuffmanCodec<std::string> huffman(1, std::move(nodes));
|
|
|
|
std::stringstream ss;
|
|
huffman.PrintTree(ss);
|
|
|
|
const std::string expected = std::string(R"(
|
|
0------right
|
|
0------left
|
|
)")
|
|
.substr(1);
|
|
|
|
EXPECT_EQ(expected, ss.str());
|
|
}
|
|
|
|
TEST(Huffman, CreateFromTextU64) {
|
|
HuffmanCodec<uint64_t> huffman(5, {
|
|
{0, 0, 0},
|
|
{1001, 0, 0},
|
|
{1002, 0, 0},
|
|
{1003, 0, 0},
|
|
{0, 1, 2},
|
|
{0, 4, 3},
|
|
});
|
|
|
|
std::stringstream ss;
|
|
huffman.PrintTree(ss);
|
|
|
|
const std::string expected = std::string(R"(
|
|
0------1003
|
|
0------0------1002
|
|
0------1001
|
|
)")
|
|
.substr(1);
|
|
|
|
EXPECT_EQ(expected, ss.str());
|
|
|
|
TestBitReader bit_reader("01");
|
|
auto read_bit = [&bit_reader](bool* bit) { return bit_reader.ReadBit(bit); };
|
|
|
|
uint64_t decoded = 0;
|
|
ASSERT_TRUE(huffman.DecodeFromStream(read_bit, &decoded));
|
|
EXPECT_EQ(1002u, decoded);
|
|
|
|
uint64_t bits = 0;
|
|
size_t num_bits = 0;
|
|
|
|
EXPECT_TRUE(huffman.Encode(1001, &bits, &num_bits));
|
|
EXPECT_EQ(2u, num_bits);
|
|
EXPECT_EQ("00", BitsToStream(bits, num_bits));
|
|
}
|
|
|
|
} // namespace
|
|
} // namespace comp
|
|
} // namespace spvtools
|