Rewrite packed vector to also allow random access
This fixes issues #3952. The new approach pre-computes masks for fast access. Since elements can potentially span multiple words we need masks and offsets for each upper and lower word. Due to a bug in the C++14 standart the mask computation is not recognized as constexpr, but would work on C++17.
This commit is contained in:
committed by
Patrick Niklaus
parent
26a208529e
commit
6bd724fe24
@@ -1,9 +1,15 @@
|
||||
#include "util/packed_vector.hpp"
|
||||
#include "util/typedefs.hpp"
|
||||
|
||||
#include <boost/range/adaptor/reversed.hpp>
|
||||
#include <boost/range/iterator_range.hpp>
|
||||
#include <boost/test/test_case_template.hpp>
|
||||
#include <boost/test/unit_test.hpp>
|
||||
|
||||
#include <algorithm>
|
||||
#include <numeric>
|
||||
#include <random>
|
||||
|
||||
BOOST_AUTO_TEST_SUITE(packed_vector_test)
|
||||
|
||||
using namespace osrm;
|
||||
@@ -16,10 +22,15 @@ BOOST_AUTO_TEST_CASE(insert_and_retrieve_packed_test)
|
||||
std::vector<OSMNodeID> original_ids;
|
||||
|
||||
const constexpr std::size_t num_test_cases = 399;
|
||||
const constexpr std::uint64_t max_id = (1ULL << 33) - 1;
|
||||
|
||||
std::mt19937 rng;
|
||||
rng.seed(1337);
|
||||
std::uniform_int_distribution<std::mt19937::result_type> dist(0, max_id);
|
||||
|
||||
for (std::size_t i = 0; i < num_test_cases; i++)
|
||||
{
|
||||
OSMNodeID r{static_cast<std::uint64_t>(rand() % 2147483647)}; // max 33-bit uint
|
||||
OSMNodeID r{static_cast<std::uint64_t>(dist(rng))}; // max 33-bit uint
|
||||
|
||||
packed_ids.push_back(r);
|
||||
original_ids.push_back(r);
|
||||
@@ -44,4 +55,152 @@ BOOST_AUTO_TEST_CASE(packed_vector_capacity_test)
|
||||
BOOST_CHECK(packed_vec.capacity() >= 100);
|
||||
}
|
||||
|
||||
BOOST_AUTO_TEST_CASE(packed_vector_resize_test)
|
||||
{
|
||||
PackedVector<std::uint32_t, 33> packed_vec(100);
|
||||
|
||||
BOOST_CHECK_EQUAL(packed_vec.size(), 100);
|
||||
packed_vec[99] = 1337;
|
||||
packed_vec[0] = 42;
|
||||
BOOST_CHECK_EQUAL(packed_vec[99], 1337u);
|
||||
BOOST_CHECK_EQUAL(packed_vec[0], 42u);
|
||||
}
|
||||
|
||||
BOOST_AUTO_TEST_CASE(packed_vector_iterator_test)
|
||||
{
|
||||
PackedVector<std::uint32_t, 33> packed_vec(100);
|
||||
|
||||
std::iota(packed_vec.begin(), packed_vec.end(), 0);
|
||||
|
||||
BOOST_CHECK(std::is_sorted(packed_vec.begin(), packed_vec.end()));
|
||||
|
||||
auto idx = 0;
|
||||
for (auto value : packed_vec)
|
||||
{
|
||||
BOOST_CHECK_EQUAL(packed_vec[idx], value);
|
||||
idx++;
|
||||
}
|
||||
BOOST_CHECK_EQUAL(idx, packed_vec.size());
|
||||
|
||||
auto range = boost::make_iterator_range(packed_vec.cbegin(), packed_vec.cend());
|
||||
BOOST_CHECK_EQUAL(range.size(), packed_vec.size());
|
||||
for (auto idx : util::irange<std::size_t>(0, packed_vec.size()))
|
||||
{
|
||||
BOOST_CHECK_EQUAL(packed_vec[idx], range[idx]);
|
||||
}
|
||||
|
||||
auto reverse_range = boost::adaptors::reverse(
|
||||
boost::make_iterator_range(packed_vec.cbegin(), packed_vec.cend()));
|
||||
BOOST_CHECK_EQUAL(reverse_range.size(), packed_vec.size());
|
||||
for (auto idx : util::irange<std::size_t>(0, packed_vec.size()))
|
||||
{
|
||||
BOOST_CHECK_EQUAL(packed_vec[packed_vec.size() - 1 - idx], reverse_range[idx]);
|
||||
}
|
||||
|
||||
auto mut_range = boost::make_iterator_range(packed_vec.begin(), packed_vec.end());
|
||||
BOOST_CHECK_EQUAL(range.size(), packed_vec.size());
|
||||
for (auto idx : util::irange<std::size_t>(0, packed_vec.size()))
|
||||
{
|
||||
BOOST_CHECK_EQUAL(packed_vec[idx], mut_range[idx]);
|
||||
}
|
||||
|
||||
auto mut_reverse_range =
|
||||
boost::adaptors::reverse(boost::make_iterator_range(packed_vec.begin(), packed_vec.end()));
|
||||
BOOST_CHECK_EQUAL(reverse_range.size(), packed_vec.size());
|
||||
for (auto idx : util::irange<std::size_t>(0, packed_vec.size()))
|
||||
{
|
||||
BOOST_CHECK_EQUAL(packed_vec[packed_vec.size() - 1 - idx], mut_reverse_range[idx]);
|
||||
}
|
||||
}
|
||||
|
||||
BOOST_AUTO_TEST_CASE(packed_vector_10bit_small_test)
|
||||
{
|
||||
PackedVector<std::uint32_t, 10> vector = {10, 5, 8, 12, 254, 4, (1 << 10) - 1, 6};
|
||||
std::vector<std::uint32_t> reference = {10, 5, 8, 12, 254, 4, (1 << 10) - 1, 6};
|
||||
|
||||
BOOST_CHECK_EQUAL(vector[0], reference[0]);
|
||||
BOOST_CHECK_EQUAL(vector[1], reference[1]);
|
||||
BOOST_CHECK_EQUAL(vector[2], reference[2]);
|
||||
BOOST_CHECK_EQUAL(vector[3], reference[3]);
|
||||
BOOST_CHECK_EQUAL(vector[4], reference[4]);
|
||||
BOOST_CHECK_EQUAL(vector[5], reference[5]);
|
||||
BOOST_CHECK_EQUAL(vector[6], reference[6]);
|
||||
BOOST_CHECK_EQUAL(vector[7], reference[7]);
|
||||
}
|
||||
|
||||
BOOST_AUTO_TEST_CASE(packed_vector_33bit_small_test)
|
||||
{
|
||||
std::vector<std::uint64_t> reference = {1597322404,
|
||||
1939964443,
|
||||
2112255763,
|
||||
1432114613,
|
||||
1067854538,
|
||||
352118606,
|
||||
1782436840,
|
||||
1909002904,
|
||||
165344818};
|
||||
|
||||
PackedVector<std::uint64_t, 33> vector = {1597322404,
|
||||
1939964443,
|
||||
2112255763,
|
||||
1432114613,
|
||||
1067854538,
|
||||
352118606,
|
||||
1782436840,
|
||||
1909002904,
|
||||
165344818};
|
||||
|
||||
BOOST_CHECK_EQUAL(vector[0], reference[0]);
|
||||
BOOST_CHECK_EQUAL(vector[1], reference[1]);
|
||||
BOOST_CHECK_EQUAL(vector[2], reference[2]);
|
||||
BOOST_CHECK_EQUAL(vector[3], reference[3]);
|
||||
BOOST_CHECK_EQUAL(vector[4], reference[4]);
|
||||
BOOST_CHECK_EQUAL(vector[5], reference[5]);
|
||||
BOOST_CHECK_EQUAL(vector[6], reference[6]);
|
||||
BOOST_CHECK_EQUAL(vector[7], reference[7]);
|
||||
}
|
||||
|
||||
BOOST_AUTO_TEST_CASE(values_overflow)
|
||||
{
|
||||
const std::uint64_t mask = (1ull << 42) - 1;
|
||||
PackedVector<std::uint64_t, 42> vector(52, 0);
|
||||
|
||||
for (auto it = vector.begin(); it != vector.end(); ++it)
|
||||
{
|
||||
BOOST_CHECK_EQUAL(*it, 0);
|
||||
}
|
||||
|
||||
std::uint64_t value = 1;
|
||||
for (auto it = vector.begin(); it != vector.end(); ++it)
|
||||
{
|
||||
BOOST_CHECK_EQUAL(*it, 0);
|
||||
*it = value;
|
||||
BOOST_CHECK_EQUAL(*it, value & mask);
|
||||
value <<= 1;
|
||||
}
|
||||
|
||||
for (auto it = vector.rbegin(); it != vector.rend(); ++it)
|
||||
{
|
||||
value >>= 1;
|
||||
BOOST_CHECK_EQUAL(*it, value & mask);
|
||||
}
|
||||
|
||||
for (auto it = vector.cbegin(); it != vector.cend(); ++it)
|
||||
{
|
||||
BOOST_CHECK_EQUAL(*it, value & mask);
|
||||
value <<= 1;
|
||||
}
|
||||
}
|
||||
|
||||
BOOST_AUTO_TEST_CASE(packed_vector_33bit_continious)
|
||||
{
|
||||
PackedVector<std::uint64_t, 33> vector;
|
||||
|
||||
for (std::uint64_t i : osrm::util::irange(0, 400))
|
||||
{
|
||||
vector.push_back(i);
|
||||
BOOST_CHECK_EQUAL(vector.back(), i);
|
||||
}
|
||||
}
|
||||
|
||||
BOOST_AUTO_TEST_SUITE_END()
|
||||
|
||||
Reference in New Issue
Block a user