Skip to content
Snippets Groups Projects
Commit 84c19378 authored by Ben Cumming's avatar Ben Cumming Committed by GitHub
Browse files

Merge pull request #94 from halfflat/bugfix/gcc-4.9.3-type-test-workaround

Bugfix/gcc 4.9.3 type test workaround
parents f7e4ff88 46be68fd
No related branches found
No related tags found
No related merge requests found
......@@ -44,7 +44,7 @@ upto(I iter, E end) {
}
template <typename I, typename E,
typename C = typename common_random_access_iterator<I,E>::type>
typename C = common_random_access_iterator_t<I,E>>
enable_if_t<std::is_same<I, E>::value ||
(has_common_random_access_iterator<I,E>::value &&
is_forward_iterator<I>::value),
......
......@@ -171,13 +171,22 @@ struct common_random_access_iterator<
>;
};
template <typename I, typename E, typename = void>
struct has_common_random_access_iterator: public std::false_type {};
template <typename I, typename E>
using common_random_access_iterator_t = typename common_random_access_iterator<I, E>::type;
// NB: using this version of SFINAE instead of a void default template
// parameter because of a bug in gcc 4.9.3.
template <typename I, typename E>
struct has_common_random_access_iterator<
I, E, void_t<typename common_random_access_iterator<I, E>::type>
> : public std::true_type {};
struct has_common_random_access_iterator {
private:
constexpr static bool test(...) { return false; }
template <typename U = I>
constexpr static bool test(typename common_random_access_iterator<U, E>::type*) { return true; }
public:
constexpr static bool value = test(nullptr);
};
......
......@@ -96,33 +96,30 @@ TEST(range, empty) {
EXPECT_EQ(0u, empty_range_rr.size());
}
template<typename I, typename E, typename = void>
struct util_distance_enabled : public std::false_type {};
// This is the same test for enabling util::distance
template<typename I, typename E>
struct util_distance_enabled<
I, E, util::void_t<
util::enable_if_t<
!util::has_common_random_access_iterator<I, E>::value &&
util::is_forward_iterator<I>::value
>>> : public std::true_type {};
TEST(range, size) {
static_assert(util_distance_enabled<
typename std::list<int>::iterator,
typename std::list<int>::const_iterator>::value,
"util::distance not enabled");
static_assert(!util_distance_enabled<
typename std::vector<int>::const_iterator,
typename std::vector<int>::iterator>::value,
"util::distance erroneously enabled");
static_assert(!util_distance_enabled<int*, int*>::value,
"util::distance erroneously enabled");
static_assert(!util_distance_enabled<int*, const int*>::value,
"util::distance erroneously enabled");
static_assert(!util_distance_enabled<const int*, int*>::value,
"util::distance erroneously enabled");
// This is the same test for enabling the incrementing util::distance
// implementation.
template <typename I, typename E>
constexpr bool uses_incrementing_distance() {
return !util::has_common_random_access_iterator<I, E>::value &&
util::is_forward_iterator<I>::value;
}
TEST(range, size_implementation) {
ASSERT_TRUE((uses_incrementing_distance<
std::list<int>::iterator,
std::list<int>::const_iterator
>()));
ASSERT_FALSE((uses_incrementing_distance<
std::vector<int>::iterator,
std::vector<int>::const_iterator
>()));
ASSERT_FALSE((uses_incrementing_distance<int*, int*>()));
ASSERT_FALSE((uses_incrementing_distance<const int*, int*>()));
ASSERT_FALSE((uses_incrementing_distance<int*, const int*>()));
}
TEST(range, input_iterator) {
......@@ -228,7 +225,7 @@ TYPED_TEST_P(counter_range, size) {
auto r = counter{signed_int_type{3}};
auto range = util::make_range(l, r);
EXPECT_FALSE(range.empty());
EXPECT_EQ(6, range.size());
EXPECT_EQ(6u, range.size());
}
......
0% or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment