Skip to content

Commit

Permalink
Fix types in generators: decay types, properly constrain forwarding
Browse files Browse the repository at this point in the history
Fixes #2040
Closes #2012
  • Loading branch information
horenmar committed Oct 20, 2020
1 parent faffc29 commit 6ffac61
Show file tree
Hide file tree
Showing 2 changed files with 64 additions and 19 deletions.
53 changes: 34 additions & 19 deletions src/catch2/generators/catch_generators.hpp
Expand Up @@ -70,8 +70,11 @@ namespace Detail {
class SingleValueGenerator final : public IGenerator<T> {
T m_value;
public:
SingleValueGenerator(T const& value) :
m_value(value)
{}
SingleValueGenerator(T&& value):
m_value(std::forward<T>(value))
m_value(std::move(value))
{}

T const& get() const override {
Expand Down Expand Up @@ -101,9 +104,11 @@ namespace Detail {
}
};

template <typename T>
GeneratorWrapper<T> value(T&& value) {
return GeneratorWrapper<T>(Catch::Detail::make_unique<SingleValueGenerator<T>>(std::forward<T>(value)));
template <typename T, typename DecayedT = std::decay_t<T>>
GeneratorWrapper<DecayedT> value( T&& value ) {
return GeneratorWrapper<DecayedT>(
Catch::Detail::make_unique<SingleValueGenerator<DecayedT>>(
std::forward<T>( value ) ) );
}
template <typename T>
GeneratorWrapper<T> values(std::initializer_list<T> values) {
Expand All @@ -115,27 +120,36 @@ namespace Detail {
std::vector<GeneratorWrapper<T>> m_generators;
size_t m_current = 0;

void populate(GeneratorWrapper<T>&& generator) {
m_generators.emplace_back(std::move(generator));
void add_generator( GeneratorWrapper<T>&& generator ) {
m_generators.emplace_back( std::move( generator ) );
}
void add_generator( T const& val ) {
m_generators.emplace_back( value( val ) );
}
void populate(T&& val) {
m_generators.emplace_back(value(std::forward<T>(val)));
void add_generator( T&& val ) {
m_generators.emplace_back( value( std::move( val ) ) );
}
template<typename U>
void populate(U&& val) {
populate(T(std::forward<U>(val)));
template <typename U>
std::enable_if_t<!std::is_same<std::decay_t<U>, T>::value>
add_generator( U&& val ) {
add_generator( T( std::forward<U>( val ) ) );
}
template<typename U, typename... Gs>
void populate(U&& valueOrGenerator, Gs &&... moreGenerators) {
populate(std::forward<U>(valueOrGenerator));
populate(std::forward<Gs>(moreGenerators)...);

template <typename U> void add_generators( U&& valueOrGenerator ) {
add_generator( std::forward<U>( valueOrGenerator ) );
}

template <typename U, typename... Gs>
void add_generators( U&& valueOrGenerator, Gs&&... moreGenerators ) {
add_generator( std::forward<U>( valueOrGenerator ) );
add_generators( std::forward<Gs>( moreGenerators )... );
}

public:
template <typename... Gs>
Generators(Gs &&... moreGenerators) {
m_generators.reserve(sizeof...(Gs));
populate(std::forward<Gs>(moreGenerators)...);
add_generators(std::forward<Gs>(moreGenerators)...);
}

T const& get() const override {
Expand All @@ -155,8 +169,9 @@ namespace Detail {
};


template<typename... Ts>
GeneratorWrapper<std::tuple<Ts...>> table( std::initializer_list<std::tuple<std::decay_t<Ts>...>> tuples ) {
template <typename... Ts>
GeneratorWrapper<std::tuple<std::decay_t<Ts>...>>
table( std::initializer_list<std::tuple<std::decay_t<Ts>...>> tuples ) {
return values<std::tuple<Ts...>>( tuples );
}

Expand All @@ -173,7 +188,7 @@ namespace Detail {
return Generators<T>(std::move(generator));
}
template<typename T, typename... Gs>
auto makeGenerators( T&& val, Gs &&... moreGenerators ) -> Generators<T> {
auto makeGenerators( T&& val, Gs &&... moreGenerators ) -> Generators<std::decay_t<T>> {
return makeGenerators( value( std::forward<T>( val ) ), std::forward<Gs>( moreGenerators )... );
}
template<typename T, typename U, typename... Gs>
Expand Down
30 changes: 30 additions & 0 deletions tests/SelfTest/IntrospectiveTests/GeneratorsImpl.tests.cpp
Expand Up @@ -358,3 +358,33 @@ TEST_CASE("Multiple random generators in one test case output different values",
REQUIRE(same < 200);
}
}

TEST_CASE("#2040 - infinite compilation recursion in GENERATE with MSVC", "[generators][compilation][approvals]") {
int x = 42;
auto test = GENERATE_COPY(1, x, 2 * x);
CHECK(test < 100);
}

namespace {
static bool always_true(int) {
return true;
}

static bool is_even(int n) {
return n % 2 == 0;
}

static bool is_multiple_of_3(int n) {
return n % 3 == 0;
}
}

TEST_CASE("GENERATE handles function (pointers)", "[generators][compilation][approvals]") {
auto f = GENERATE(always_true, is_even, is_multiple_of_3);
REQUIRE(f(6));
}

TEST_CASE("GENERATE decays arrays", "[generators][compilation][approvals]") {
auto str = GENERATE("abc", "def", "gh");
STATIC_REQUIRE(std::is_same<decltype(str), const char*>::value);
}

0 comments on commit 6ffac61

Please sign in to comment.