Sacado Package Browser (Single Doxygen Collection)  Version of the Day
 All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Groups Pages
gtest-param-util.h
Go to the documentation of this file.
1 // Copyright 2008 Google Inc.
2 // All Rights Reserved.
3 //
4 // Redistribution and use in source and binary forms, with or without
5 // modification, are permitted provided that the following conditions are
6 // met:
7 //
8 // * Redistributions of source code must retain the above copyright
9 // notice, this list of conditions and the following disclaimer.
10 // * Redistributions in binary form must reproduce the above
11 // copyright notice, this list of conditions and the following disclaimer
12 // in the documentation and/or other materials provided with the
13 // distribution.
14 // * Neither the name of Google Inc. nor the names of its
15 // contributors may be used to endorse or promote products derived from
16 // this software without specific prior written permission.
17 //
18 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
19 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
20 // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
21 // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
22 // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
23 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
24 // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
25 // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
26 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
28 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29 
30 
31 // Type and function utilities for implementing parameterized tests.
32 
33 // GOOGLETEST_CM0001 DO NOT DELETE
34 
35 #ifndef GTEST_INCLUDE_GTEST_INTERNAL_GTEST_PARAM_UTIL_H_
36 #define GTEST_INCLUDE_GTEST_INTERNAL_GTEST_PARAM_UTIL_H_
37 
38 #include <ctype.h>
39 
40 #include <cassert>
41 #include <iterator>
42 #include <memory>
43 #include <set>
44 #include <tuple>
45 #include <type_traits>
46 #include <utility>
47 #include <vector>
48 
51 #include "gtest/gtest-printers.h"
52 #include "gtest/gtest-test-part.h"
53 
54 namespace testing {
55 // Input to a parameterized test name generator, describing a test parameter.
56 // Consists of the parameter value and the integer parameter index.
57 template <class ParamType>
58 struct TestParamInfo {
59  TestParamInfo(const ParamType& a_param, size_t an_index) :
60  param(a_param),
61  index(an_index) {}
62  ParamType param;
63  size_t index;
64 };
65 
66 // A builtin parameterized test name generator which returns the result of
67 // testing::PrintToString.
69  template <class ParamType>
70  std::string operator()(const TestParamInfo<ParamType>& info) const {
71  return PrintToString(info.param);
72  }
73 };
74 
75 namespace internal {
76 
77 // INTERNAL IMPLEMENTATION - DO NOT USE IN USER CODE.
78 // Utility Functions
79 
80 // Outputs a message explaining invalid registration of different
81 // fixture class for the same test suite. This may happen when
82 // TEST_P macro is used to define two tests with the same name
83 // but in different namespaces.
84 GTEST_API_ void ReportInvalidTestSuiteType(const char* test_suite_name,
85  CodeLocation code_location);
86 
87 template <typename> class ParamGeneratorInterface;
88 template <typename> class ParamGenerator;
89 
90 // Interface for iterating over elements provided by an implementation
91 // of ParamGeneratorInterface<T>.
92 template <typename T>
94  public:
96  // A pointer to the base generator instance.
97  // Used only for the purposes of iterator comparison
98  // to make sure that two iterators belong to the same generator.
99  virtual const ParamGeneratorInterface<T>* BaseGenerator() const = 0;
100  // Advances iterator to point to the next element
101  // provided by the generator. The caller is responsible
102  // for not calling Advance() on an iterator equal to
103  // BaseGenerator()->End().
104  virtual void Advance() = 0;
105  // Clones the iterator object. Used for implementing copy semantics
106  // of ParamIterator<T>.
107  virtual ParamIteratorInterface* Clone() const = 0;
108  // Dereferences the current iterator and provides (read-only) access
109  // to the pointed value. It is the caller's responsibility not to call
110  // Current() on an iterator equal to BaseGenerator()->End().
111  // Used for implementing ParamGenerator<T>::operator*().
112  virtual const T* Current() const = 0;
113  // Determines whether the given iterator and other point to the same
114  // element in the sequence generated by the generator.
115  // Used for implementing ParamGenerator<T>::operator==().
116  virtual bool Equals(const ParamIteratorInterface& other) const = 0;
117 };
118 
119 // Class iterating over elements provided by an implementation of
120 // ParamGeneratorInterface<T>. It wraps ParamIteratorInterface<T>
121 // and implements the const forward iterator concept.
122 template <typename T>
124  public:
125  typedef T value_type;
126  typedef const T& reference;
127  typedef ptrdiff_t difference_type;
128 
129  // ParamIterator assumes ownership of the impl_ pointer.
130  ParamIterator(const ParamIterator& other) : impl_(other.impl_->Clone()) {}
132  if (this != &other)
133  impl_.reset(other.impl_->Clone());
134  return *this;
135  }
136 
137  const T& operator*() const { return *impl_->Current(); }
138  const T* operator->() const { return impl_->Current(); }
139  // Prefix version of operator++.
141  impl_->Advance();
142  return *this;
143  }
144  // Postfix version of operator++.
145  ParamIterator operator++(int /*unused*/) {
146  ParamIteratorInterface<T>* clone = impl_->Clone();
147  impl_->Advance();
148  return ParamIterator(clone);
149  }
150  bool operator==(const ParamIterator& other) const {
151  return impl_.get() == other.impl_.get() || impl_->Equals(*other.impl_);
152  }
153  bool operator!=(const ParamIterator& other) const {
154  return !(*this == other);
155  }
156 
157  private:
158  friend class ParamGenerator<T>;
159  explicit ParamIterator(ParamIteratorInterface<T>* impl) : impl_(impl) {}
160  std::unique_ptr<ParamIteratorInterface<T> > impl_;
161 };
162 
163 // ParamGeneratorInterface<T> is the binary interface to access generators
164 // defined in other translation units.
165 template <typename T>
167  public:
168  typedef T ParamType;
169 
171 
172  // Generator interface definition
173  virtual ParamIteratorInterface<T>* Begin() const = 0;
174  virtual ParamIteratorInterface<T>* End() const = 0;
175 };
176 
177 // Wraps ParamGeneratorInterface<T> and provides general generator syntax
178 // compatible with the STL Container concept.
179 // This class implements copy initialization semantics and the contained
180 // ParamGeneratorInterface<T> instance is shared among all copies
181 // of the original object. This is possible because that instance is immutable.
182 template<typename T>
183 class ParamGenerator {
184  public:
186 
188  ParamGenerator(const ParamGenerator& other) : impl_(other.impl_) {}
189 
191  impl_ = other.impl_;
192  return *this;
193  }
194 
195  iterator begin() const { return iterator(impl_->Begin()); }
196  iterator end() const { return iterator(impl_->End()); }
197 
198  private:
199  std::shared_ptr<const ParamGeneratorInterface<T> > impl_;
200 };
201 
202 // Generates values from a range of two comparable values. Can be used to
203 // generate sequences of user-defined types that implement operator+() and
204 // operator<().
205 // This class is used in the Range() function.
206 template <typename T, typename IncrementT>
208  public:
209  RangeGenerator(T begin, T end, IncrementT step)
210  : begin_(begin), end_(end),
211  step_(step), end_index_(CalculateEndIndex(begin, end, step)) {}
212  ~RangeGenerator() override {}
213 
214  ParamIteratorInterface<T>* Begin() const override {
215  return new Iterator(this, begin_, 0, step_);
216  }
217  ParamIteratorInterface<T>* End() const override {
218  return new Iterator(this, end_, end_index_, step_);
219  }
220 
221  private:
222  class Iterator : public ParamIteratorInterface<T> {
223  public:
224  Iterator(const ParamGeneratorInterface<T>* base, T value, int index,
225  IncrementT step)
226  : base_(base), value_(value), index_(index), step_(step) {}
227  ~Iterator() override {}
228 
229  const ParamGeneratorInterface<T>* BaseGenerator() const override {
230  return base_;
231  }
232  void Advance() override {
233  value_ = static_cast<T>(value_ + step_);
234  index_++;
235  }
236  ParamIteratorInterface<T>* Clone() const override {
237  return new Iterator(*this);
238  }
239  const T* Current() const override { return &value_; }
240  bool Equals(const ParamIteratorInterface<T>& other) const override {
241  // Having the same base generator guarantees that the other
242  // iterator is of the same type and we can downcast.
244  << "The program attempted to compare iterators "
245  << "from different generators." << std::endl;
246  const int other_index =
247  CheckedDowncastToActualType<const Iterator>(&other)->index_;
248  return index_ == other_index;
249  }
250 
251  private:
252  Iterator(const Iterator& other)
254  base_(other.base_), value_(other.value_), index_(other.index_),
255  step_(other.step_) {}
256 
257  // No implementation - assignment is unsupported.
258  void operator=(const Iterator& other);
259 
262  int index_;
263  const IncrementT step_;
264  }; // class RangeGenerator::Iterator
265 
266  static int CalculateEndIndex(const T& begin,
267  const T& end,
268  const IncrementT& step) {
269  int end_index = 0;
270  for (T i = begin; i < end; i = static_cast<T>(i + step))
271  end_index++;
272  return end_index;
273  }
274 
275  // No implementation - assignment is unsupported.
276  void operator=(const RangeGenerator& other);
277 
278  const T begin_;
279  const T end_;
280  const IncrementT step_;
281  // The index for the end() iterator. All the elements in the generated
282  // sequence are indexed (0-based) to aid iterator comparison.
283  const int end_index_;
284 }; // class RangeGenerator
285 
286 
287 // Generates values from a pair of STL-style iterators. Used in the
288 // ValuesIn() function. The elements are copied from the source range
289 // since the source can be located on the stack, and the generator
290 // is likely to persist beyond that stack frame.
291 template <typename T>
293  public:
294  template <typename ForwardIterator>
295  ValuesInIteratorRangeGenerator(ForwardIterator begin, ForwardIterator end)
296  : container_(begin, end) {}
298 
299  ParamIteratorInterface<T>* Begin() const override {
300  return new Iterator(this, container_.begin());
301  }
302  ParamIteratorInterface<T>* End() const override {
303  return new Iterator(this, container_.end());
304  }
305 
306  private:
307  typedef typename ::std::vector<T> ContainerType;
308 
309  class Iterator : public ParamIteratorInterface<T> {
310  public:
312  typename ContainerType::const_iterator iterator)
313  : base_(base), iterator_(iterator) {}
314  ~Iterator() override {}
315 
316  const ParamGeneratorInterface<T>* BaseGenerator() const override {
317  return base_;
318  }
319  void Advance() override {
320  ++iterator_;
321  value_.reset();
322  }
323  ParamIteratorInterface<T>* Clone() const override {
324  return new Iterator(*this);
325  }
326  // We need to use cached value referenced by iterator_ because *iterator_
327  // can return a temporary object (and of type other then T), so just
328  // having "return &*iterator_;" doesn't work.
329  // value_ is updated here and not in Advance() because Advance()
330  // can advance iterator_ beyond the end of the range, and we cannot
331  // detect that fact. The client code, on the other hand, is
332  // responsible for not calling Current() on an out-of-range iterator.
333  const T* Current() const override {
334  if (value_.get() == nullptr) value_.reset(new T(*iterator_));
335  return value_.get();
336  }
337  bool Equals(const ParamIteratorInterface<T>& other) const override {
338  // Having the same base generator guarantees that the other
339  // iterator is of the same type and we can downcast.
341  << "The program attempted to compare iterators "
342  << "from different generators." << std::endl;
343  return iterator_ ==
344  CheckedDowncastToActualType<const Iterator>(&other)->iterator_;
345  }
346 
347  private:
348  Iterator(const Iterator& other)
349  // The explicit constructor call suppresses a false warning
350  // emitted by gcc when supplied with the -Wextra option.
352  base_(other.base_),
354 
356  typename ContainerType::const_iterator iterator_;
357  // A cached value of *iterator_. We keep it here to allow access by
358  // pointer in the wrapping iterator's operator->().
359  // value_ needs to be mutable to be accessed in Current().
360  // Use of std::unique_ptr helps manage cached value's lifetime,
361  // which is bound by the lifespan of the iterator itself.
362  mutable std::unique_ptr<const T> value_;
363  }; // class ValuesInIteratorRangeGenerator::Iterator
364 
365  // No implementation - assignment is unsupported.
366  void operator=(const ValuesInIteratorRangeGenerator& other);
367 
369 }; // class ValuesInIteratorRangeGenerator
370 
371 // INTERNAL IMPLEMENTATION - DO NOT USE IN USER CODE.
372 //
373 // Default parameterized test name generator, returns a string containing the
374 // integer test parameter index.
375 template <class ParamType>
376 std::string DefaultParamName(const TestParamInfo<ParamType>& info) {
377  Message name_stream;
378  name_stream << info.index;
379  return name_stream.GetString();
380 }
381 
382 template <typename T = int>
383 void TestNotEmpty() {
384  static_assert(sizeof(T) == 0, "Empty arguments are not allowed.");
385 }
386 template <typename T = int>
387 void TestNotEmpty(const T&) {}
388 
389 // INTERNAL IMPLEMENTATION - DO NOT USE IN USER CODE.
390 //
391 // Stores a parameter value and later creates tests parameterized with that
392 // value.
393 template <class TestClass>
395  public:
396  typedef typename TestClass::ParamType ParamType;
397  explicit ParameterizedTestFactory(ParamType parameter) :
398  parameter_(parameter) {}
399  Test* CreateTest() override {
400  TestClass::SetParam(&parameter_);
401  return new TestClass();
402  }
403 
404  private:
406 
408 };
409 
410 // INTERNAL IMPLEMENTATION - DO NOT USE IN USER CODE.
411 //
412 // TestMetaFactoryBase is a base class for meta-factories that create
413 // test factories for passing into MakeAndRegisterTestInfo function.
414 template <class ParamType>
416  public:
417  virtual ~TestMetaFactoryBase() {}
418 
419  virtual TestFactoryBase* CreateTestFactory(ParamType parameter) = 0;
420 };
421 
422 // INTERNAL IMPLEMENTATION - DO NOT USE IN USER CODE.
423 //
424 // TestMetaFactory creates test factories for passing into
425 // MakeAndRegisterTestInfo function. Since MakeAndRegisterTestInfo receives
426 // ownership of test factory pointer, same factory object cannot be passed
427 // into that method twice. But ParameterizedTestSuiteInfo is going to call
428 // it for each Test/Parameter value combination. Thus it needs meta factory
429 // creator class.
430 template <class TestSuite>
432  : public TestMetaFactoryBase<typename TestSuite::ParamType> {
433  public:
434  using ParamType = typename TestSuite::ParamType;
435 
437 
439  return new ParameterizedTestFactory<TestSuite>(parameter);
440  }
441 
442  private:
444 };
445 
446 // INTERNAL IMPLEMENTATION - DO NOT USE IN USER CODE.
447 //
448 // ParameterizedTestSuiteInfoBase is a generic interface
449 // to ParameterizedTestSuiteInfo classes. ParameterizedTestSuiteInfoBase
450 // accumulates test information provided by TEST_P macro invocations
451 // and generators provided by INSTANTIATE_TEST_SUITE_P macro invocations
452 // and uses that information to register all resulting test instances
453 // in RegisterTests method. The ParameterizeTestSuiteRegistry class holds
454 // a collection of pointers to the ParameterizedTestSuiteInfo objects
455 // and calls RegisterTests() on each of them when asked.
457  public:
459 
460  // Base part of test suite name for display purposes.
461  virtual const std::string& GetTestSuiteName() const = 0;
462  // Test case id to verify identity.
463  virtual TypeId GetTestSuiteTypeId() const = 0;
464  // UnitTest class invokes this method to register tests in this
465  // test suite right before running them in RUN_ALL_TESTS macro.
466  // This method should not be called more than once on any single
467  // instance of a ParameterizedTestSuiteInfoBase derived class.
468  virtual void RegisterTests() = 0;
469 
470  protected:
472 
473  private:
475 };
476 
477 // INTERNAL IMPLEMENTATION - DO NOT USE IN USER CODE.
478 //
479 // Report a the name of a test_suit as safe to ignore
480 // as the side effect of construction of this type.
482  explicit MarkAsIgnored(const char* test_suite);
483 };
484 
485 GTEST_API_ void InsertSyntheticTestCase(const std::string& name,
486  CodeLocation location, bool has_test_p);
487 
488 // INTERNAL IMPLEMENTATION - DO NOT USE IN USER CODE.
489 //
490 // ParameterizedTestSuiteInfo accumulates tests obtained from TEST_P
491 // macro invocations for a particular test suite and generators
492 // obtained from INSTANTIATE_TEST_SUITE_P macro invocations for that
493 // test suite. It registers tests with all values generated by all
494 // generators when asked.
495 template <class TestSuite>
497  public:
498  // ParamType and GeneratorCreationFunc are private types but are required
499  // for declarations of public methods AddTestPattern() and
500  // AddTestSuiteInstantiation().
501  using ParamType = typename TestSuite::ParamType;
502  // A function that returns an instance of appropriate generator type.
503  typedef ParamGenerator<ParamType>(GeneratorCreationFunc)();
505 
506  explicit ParameterizedTestSuiteInfo(const char* name,
507  CodeLocation code_location)
508  : test_suite_name_(name), code_location_(code_location) {}
509 
510  // Test case base name for display purposes.
511  const std::string& GetTestSuiteName() const override {
512  return test_suite_name_;
513  }
514  // Test case id to verify identity.
515  TypeId GetTestSuiteTypeId() const override { return GetTypeId<TestSuite>(); }
516  // TEST_P macro uses AddTestPattern() to record information
517  // about a single test in a LocalTestInfo structure.
518  // test_suite_name is the base name of the test suite (without invocation
519  // prefix). test_base_name is the name of an individual test without
520  // parameter index. For the test SequenceA/FooTest.DoBar/1 FooTest is
521  // test suite base name and DoBar is test base name.
522  void AddTestPattern(const char* test_suite_name, const char* test_base_name,
523  TestMetaFactoryBase<ParamType>* meta_factory,
524  CodeLocation code_location) {
525  tests_.push_back(std::shared_ptr<TestInfo>(new TestInfo(
526  test_suite_name, test_base_name, meta_factory, code_location)));
527  }
528  // INSTANTIATE_TEST_SUITE_P macro uses AddGenerator() to record information
529  // about a generator.
530  int AddTestSuiteInstantiation(const std::string& instantiation_name,
531  GeneratorCreationFunc* func,
532  ParamNameGeneratorFunc* name_func,
533  const char* file, int line) {
534  instantiations_.push_back(
535  InstantiationInfo(instantiation_name, func, name_func, file, line));
536  return 0; // Return value used only to run this method in namespace scope.
537  }
538  // UnitTest class invokes this method to register tests in this test suite
539  // right before running tests in RUN_ALL_TESTS macro.
540  // This method should not be called more than once on any single
541  // instance of a ParameterizedTestSuiteInfoBase derived class.
542  // UnitTest has a guard to prevent from calling this method more than once.
543  void RegisterTests() override {
544  bool generated_instantiations = false;
545 
546  for (typename TestInfoContainer::iterator test_it = tests_.begin();
547  test_it != tests_.end(); ++test_it) {
548  std::shared_ptr<TestInfo> test_info = *test_it;
549  for (typename InstantiationContainer::iterator gen_it =
550  instantiations_.begin(); gen_it != instantiations_.end();
551  ++gen_it) {
552  const std::string& instantiation_name = gen_it->name;
553  ParamGenerator<ParamType> generator((*gen_it->generator)());
554  ParamNameGeneratorFunc* name_func = gen_it->name_func;
555  const char* file = gen_it->file;
556  int line = gen_it->line;
557 
558  std::string test_suite_name;
559  if ( !instantiation_name.empty() )
560  test_suite_name = instantiation_name + "/";
561  test_suite_name += test_info->test_suite_base_name;
562 
563  size_t i = 0;
564  std::set<std::string> test_param_names;
565  for (typename ParamGenerator<ParamType>::iterator param_it =
566  generator.begin();
567  param_it != generator.end(); ++param_it, ++i) {
568  generated_instantiations = true;
569 
570  Message test_name_stream;
571 
572  std::string param_name = name_func(
573  TestParamInfo<ParamType>(*param_it, i));
574 
575  GTEST_CHECK_(IsValidParamName(param_name))
576  << "Parameterized test name '" << param_name
577  << "' is invalid, in " << file
578  << " line " << line << std::endl;
579 
580  GTEST_CHECK_(test_param_names.count(param_name) == 0)
581  << "Duplicate parameterized test name '" << param_name
582  << "', in " << file << " line " << line << std::endl;
583 
584  test_param_names.insert(param_name);
585 
586  if (!test_info->test_base_name.empty()) {
587  test_name_stream << test_info->test_base_name << "/";
588  }
589  test_name_stream << param_name;
591  test_suite_name.c_str(), test_name_stream.GetString().c_str(),
592  nullptr, // No type parameter.
593  PrintToString(*param_it).c_str(), test_info->code_location,
597  test_info->test_meta_factory->CreateTestFactory(*param_it));
598  } // for param_it
599  } // for gen_it
600  } // for test_it
601 
602  if (!generated_instantiations) {
603  // There are no generaotrs, or they all generate nothing ...
605  !tests_.empty());
606  }
607  } // RegisterTests
608 
609  private:
610  // LocalTestInfo structure keeps information about a single test registered
611  // with TEST_P macro.
612  struct TestInfo {
613  TestInfo(const char* a_test_suite_base_name, const char* a_test_base_name,
614  TestMetaFactoryBase<ParamType>* a_test_meta_factory,
615  CodeLocation a_code_location)
616  : test_suite_base_name(a_test_suite_base_name),
617  test_base_name(a_test_base_name),
618  test_meta_factory(a_test_meta_factory),
619  code_location(a_code_location) {}
620 
621  const std::string test_suite_base_name;
622  const std::string test_base_name;
623  const std::unique_ptr<TestMetaFactoryBase<ParamType> > test_meta_factory;
625  };
626  using TestInfoContainer = ::std::vector<std::shared_ptr<TestInfo> >;
627  // Records data received from INSTANTIATE_TEST_SUITE_P macros:
628  // <Instantiation name, Sequence generator creation function,
629  // Name generator function, Source file, Source line>
631  InstantiationInfo(const std::string &name_in,
632  GeneratorCreationFunc* generator_in,
633  ParamNameGeneratorFunc* name_func_in,
634  const char* file_in,
635  int line_in)
636  : name(name_in),
637  generator(generator_in),
638  name_func(name_func_in),
639  file(file_in),
640  line(line_in) {}
641 
642  std::string name;
643  GeneratorCreationFunc* generator;
645  const char* file;
646  int line;
647  };
648  typedef ::std::vector<InstantiationInfo> InstantiationContainer;
649 
650  static bool IsValidParamName(const std::string& name) {
651  // Check for empty string
652  if (name.empty())
653  return false;
654 
655  // Check for invalid characters
656  for (std::string::size_type index = 0; index < name.size(); ++index) {
657  if (!isalnum(name[index]) && name[index] != '_')
658  return false;
659  }
660 
661  return true;
662  }
663 
664  const std::string test_suite_name_;
668 
670 }; // class ParameterizedTestSuiteInfo
671 
672 // Legacy API is deprecated but still available
673 #ifndef GTEST_REMOVE_LEGACY_TEST_CASEAPI_
674 template <class TestCase>
676 #endif // GTEST_REMOVE_LEGACY_TEST_CASEAPI_
677 
678 // INTERNAL IMPLEMENTATION - DO NOT USE IN USER CODE.
679 //
680 // ParameterizedTestSuiteRegistry contains a map of
681 // ParameterizedTestSuiteInfoBase classes accessed by test suite names. TEST_P
682 // and INSTANTIATE_TEST_SUITE_P macros use it to locate their corresponding
683 // ParameterizedTestSuiteInfo descriptors.
685  public:
688  for (auto& test_suite_info : test_suite_infos_) {
689  delete test_suite_info;
690  }
691  }
692 
693  // Looks up or creates and returns a structure containing information about
694  // tests and instantiations of a particular test suite.
695  template <class TestSuite>
697  const char* test_suite_name, CodeLocation code_location) {
698  ParameterizedTestSuiteInfo<TestSuite>* typed_test_info = nullptr;
699  for (auto& test_suite_info : test_suite_infos_) {
700  if (test_suite_info->GetTestSuiteName() == test_suite_name) {
701  if (test_suite_info->GetTestSuiteTypeId() != GetTypeId<TestSuite>()) {
702  // Complain about incorrect usage of Google Test facilities
703  // and terminate the program since we cannot guaranty correct
704  // test suite setup and tear-down in this case.
705  ReportInvalidTestSuiteType(test_suite_name, code_location);
706  posix::Abort();
707  } else {
708  // At this point we are sure that the object we found is of the same
709  // type we are looking for, so we downcast it to that type
710  // without further checks.
711  typed_test_info = CheckedDowncastToActualType<
712  ParameterizedTestSuiteInfo<TestSuite> >(test_suite_info);
713  }
714  break;
715  }
716  }
717  if (typed_test_info == nullptr) {
718  typed_test_info = new ParameterizedTestSuiteInfo<TestSuite>(
719  test_suite_name, code_location);
720  test_suite_infos_.push_back(typed_test_info);
721  }
722  return typed_test_info;
723  }
724  void RegisterTests() {
725  for (auto& test_suite_info : test_suite_infos_) {
726  test_suite_info->RegisterTests();
727  }
728  }
729 // Legacy API is deprecated but still available
730 #ifndef GTEST_REMOVE_LEGACY_TEST_CASEAPI_
731  template <class TestCase>
733  const char* test_case_name, CodeLocation code_location) {
734  return GetTestSuitePatternHolder<TestCase>(test_case_name, code_location);
735  }
736 
737 #endif // GTEST_REMOVE_LEGACY_TEST_CASEAPI_
738 
739  private:
740  using TestSuiteInfoContainer = ::std::vector<ParameterizedTestSuiteInfoBase*>;
741 
743 
745 };
746 
747 // Keep track of what type-parameterized test suite are defined and
748 // where as well as which are intatiated. This allows susequently
749 // identifying suits that are defined but never used.
751  public:
752  // Add a suite definition
753  void RegisterTestSuite(const char* test_suite_name,
754  CodeLocation code_location);
755 
756  // Add an instantiation of a suit.
757  void RegisterInstantiation(const char* test_suite_name);
758 
759  // For each suit repored as defined but not reported as instantiation,
760  // emit a test that reports that fact (configurably, as an error).
761  void CheckForInstantiations();
762 
763  private:
766  : code_location(c), instantiated(false) {}
767 
770  };
771 
772  std::map<std::string, TypeParameterizedTestSuiteInfo> suites_;
773 };
774 
775 } // namespace internal
776 
777 // Forward declarations of ValuesIn(), which is implemented in
778 // include/gtest/gtest-param-test.h.
779 template <class Container>
781  const Container& container);
782 
783 namespace internal {
784 // Used in the Values() function to provide polymorphic capabilities.
785 
786 template <typename... Ts>
787 class ValueArray {
788  public:
789  ValueArray(Ts... v) : v_{std::move(v)...} {}
790 
791  template <typename T>
792  operator ParamGenerator<T>() const { // NOLINT
793  return ValuesIn(MakeVector<T>(MakeIndexSequence<sizeof...(Ts)>()));
794  }
795 
796  private:
797  template <typename T, size_t... I>
798  std::vector<T> MakeVector(IndexSequence<I...>) const {
799  return std::vector<T>{static_cast<T>(v_.template Get<I>())...};
800  }
801 
802  FlatTuple<Ts...> v_;
803 };
804 
805 template <typename... T>
807  : public ParamGeneratorInterface<::std::tuple<T...>> {
808  public:
809  typedef ::std::tuple<T...> ParamType;
810 
812  : generators_(g) {}
814 
816  return new Iterator(this, generators_, false);
817  }
819  return new Iterator(this, generators_, true);
820  }
821 
822  private:
823  template <class I>
825  template <size_t... I>
827  : public ParamIteratorInterface<ParamType> {
828  public:
830  const std::tuple<ParamGenerator<T>...>& generators, bool is_end)
831  : base_(base),
832  begin_(std::get<I>(generators).begin()...),
833  end_(std::get<I>(generators).end()...),
834  current_(is_end ? end_ : begin_) {
835  ComputeCurrentValue();
836  }
837  ~IteratorImpl() override {}
838 
840  return base_;
841  }
842  // Advance should not be called on beyond-of-range iterators
843  // so no component iterators must be beyond end of range, either.
844  void Advance() override {
845  assert(!AtEnd());
846  // Advance the last iterator.
847  ++std::get<sizeof...(T) - 1>(current_);
848  // if that reaches end, propagate that up.
849  AdvanceIfEnd<sizeof...(T) - 1>();
850  ComputeCurrentValue();
851  }
853  return new IteratorImpl(*this);
854  }
855 
856  const ParamType* Current() const override { return current_value_.get(); }
857 
858  bool Equals(const ParamIteratorInterface<ParamType>& other) const override {
859  // Having the same base generator guarantees that the other
860  // iterator is of the same type and we can downcast.
861  GTEST_CHECK_(BaseGenerator() == other.BaseGenerator())
862  << "The program attempted to compare iterators "
863  << "from different generators." << std::endl;
864  const IteratorImpl* typed_other =
865  CheckedDowncastToActualType<const IteratorImpl>(&other);
866 
867  // We must report iterators equal if they both point beyond their
868  // respective ranges. That can happen in a variety of fashions,
869  // so we have to consult AtEnd().
870  if (AtEnd() && typed_other->AtEnd()) return true;
871 
872  bool same = true;
873  bool dummy[] = {
874  (same = same && std::get<I>(current_) ==
875  std::get<I>(typed_other->current_))...};
876  (void)dummy;
877  return same;
878  }
879 
880  private:
881  template <size_t ThisI>
882  void AdvanceIfEnd() {
883  if (std::get<ThisI>(current_) != std::get<ThisI>(end_)) return;
884 
885  bool last = ThisI == 0;
886  if (last) {
887  // We are done. Nothing else to propagate.
888  return;
889  }
890 
891  constexpr size_t NextI = ThisI - (ThisI != 0);
892  std::get<ThisI>(current_) = std::get<ThisI>(begin_);
893  ++std::get<NextI>(current_);
894  AdvanceIfEnd<NextI>();
895  }
896 
898  if (!AtEnd())
899  current_value_ = std::make_shared<ParamType>(*std::get<I>(current_)...);
900  }
901  bool AtEnd() const {
902  bool at_end = false;
903  bool dummy[] = {
904  (at_end = at_end || std::get<I>(current_) == std::get<I>(end_))...};
905  (void)dummy;
906  return at_end;
907  }
908 
910  std::tuple<typename ParamGenerator<T>::iterator...> begin_;
911  std::tuple<typename ParamGenerator<T>::iterator...> end_;
912  std::tuple<typename ParamGenerator<T>::iterator...> current_;
913  std::shared_ptr<ParamType> current_value_;
914  };
915 
916  using Iterator = IteratorImpl<typename MakeIndexSequence<sizeof...(T)>::type>;
917 
918  std::tuple<ParamGenerator<T>...> generators_;
919 };
920 
921 template <class... Gen>
923  public:
924  CartesianProductHolder(const Gen&... g) : generators_(g...) {}
925  template <typename... T>
926  operator ParamGenerator<::std::tuple<T...>>() const {
927  return ParamGenerator<::std::tuple<T...>>(
929  }
930 
931  private:
932  std::tuple<Gen...> generators_;
933 };
934 
935 } // namespace internal
936 } // namespace testing
937 
938 #endif // GTEST_INCLUDE_GTEST_INTERNAL_GTEST_PARAM_UTIL_H_
virtual ParamIteratorInterface< T > * End() const =0
const ParamGeneratorInterface< T > *const base_
ParamGenerator(const ParamGenerator &other)
static SetUpTearDownSuiteFuncType GetTearDownCaseOrSuite(const char *filename, int line_num)
void RegisterInstantiation(const char *test_suite_name)
Definition: gtest.cc:537
void operator=(const RangeGenerator &other)
const std::unique_ptr< TestMetaFactoryBase< ParamType > > test_meta_factory
Iterator(const ParamGeneratorInterface< T > *base, T value, int index, IncrementT step)
MarkAsIgnored(const char *test_suite)
Definition: gtest.cc:470
bool operator==(const ParamIterator &other) const
ParamIteratorInterface< T > * End() const override
void operator=(const Iterator &other)
::std::string PrintToString(const T &value)
virtual bool Equals(const ParamIteratorInterface &other) const =0
typename TestSuite::ParamType ParamType
GTEST_DISALLOW_COPY_AND_ASSIGN_(ParameterizedTestSuiteRegistry)
::std::vector< InstantiationInfo > InstantiationContainer
TestParamInfo(const ParamType &a_param, size_t an_index)
static bool IsValidParamName(const std::string &name)
std::tuple< ParamGenerator< T >...> generators_
#define GTEST_API_
Definition: gtest-port.h:775
RangeGenerator(T begin, T end, IncrementT step)
ValuesInIteratorRangeGenerator(ForwardIterator begin, ForwardIterator end)
internal::ParamGenerator< typename std::iterator_traits< ForwardIterator >::value_type > ValuesIn(ForwardIterator begin, ForwardIterator end)
GTEST_DISALLOW_COPY_AND_ASSIGN_(TestMetaFactory)
void AddTestPattern(const char *test_suite_name, const char *test_base_name, TestMetaFactoryBase< ParamType > *meta_factory, CodeLocation code_location)
ParamGenerator & operator=(const ParamGenerator &other)
bool Equals(const ParamIteratorInterface< T > &other) const override
ParamIterator(const ParamIterator &other)
ParamIteratorInterface< T > * Begin() const override
virtual const T * Current() const =0
ParamIteratorInterface< T > * Clone() const override
GTEST_API_ void ReportInvalidTestSuiteType(const char *test_suite_name, CodeLocation code_location)
Definition: gtest.cc:2774
virtual TestFactoryBase * CreateTestFactory(ParamType parameter)=0
const void * TypeId
#define T
Definition: Sacado_rad.hpp:553
const ParamGeneratorInterface< T > *const base_
TestFactoryBase * CreateTestFactory(ParamType parameter) override
static int CalculateEndIndex(const T &begin, const T &end, const IncrementT &step)
bool Equals(const ParamIteratorInterface< ParamType > &other) const override
expr expr1 expr1 expr1 c expr2 expr1 expr2 expr1 expr2 expr1 expr1 expr1 expr1 c expr2 expr1 expr2 expr1 expr2 expr1 expr1 expr1 expr1 c *expr2 expr1 expr2 expr1 expr2 expr1 expr1 expr1 expr1 c expr2 expr1 expr2 expr1 expr2 expr1 expr1 expr1 expr2 expr1 expr2 expr1 expr1 expr1 expr2 expr1 expr2 expr1 expr1 expr1 c
int AddTestSuiteInstantiation(const std::string &instantiation_name, GeneratorCreationFunc *func, ParamNameGeneratorFunc *name_func, const char *file, int line)
GTEST_API_ TestInfo * MakeAndRegisterTestInfo(const char *test_suite_name, const char *name, const char *type_param, const char *value_param, CodeLocation code_location, TypeId fixture_class_id, SetUpTestSuiteFunc set_up_tc, TearDownTestSuiteFunc tear_down_tc, TestFactoryBase *factory)
Definition: gtest.cc:2762
#define GTEST_CHECK_(condition)
Definition: gtest-port.h:1004
CartesianProductGenerator(const std::tuple< ParamGenerator< T >...> &g)
GTEST_DISALLOW_COPY_AND_ASSIGN_(ParameterizedTestFactory)
::std::vector< std::shared_ptr< TestInfo > > TestInfoContainer
std::string(const TestParamInfo< ParamType > &) ParamNameGeneratorFunc
GTEST_DISALLOW_COPY_AND_ASSIGN_(ParameterizedTestSuiteInfoBase)
void operator=(const ValuesInIteratorRangeGenerator &other)
std::map< std::string, TypeParameterizedTestSuiteInfo > suites_
const ParamGeneratorInterface< T > * BaseGenerator() const override
bool operator!=(const ParamIterator &other) const
void g()
ParameterizedTestSuiteInfo(const char *name, CodeLocation code_location)
std::string GetString() const
Definition: gtest.cc:1168
ParameterizedTestSuiteInfo< TestSuite > * GetTestSuitePatternHolder(const char *test_suite_name, CodeLocation code_location)
virtual ParamIteratorInterface< T > * Begin() const =0
void
Definition: uninit.c:105
std::string operator()(const TestParamInfo< ParamType > &info) const
InstantiationInfo(const std::string &name_in, GeneratorCreationFunc *generator_in, ParamNameGeneratorFunc *name_func_in, const char *file_in, int line_in)
::std::vector< ParameterizedTestSuiteInfoBase * > TestSuiteInfoContainer
GTEST_API_ void InsertSyntheticTestCase(const std::string &name, CodeLocation location, bool has_test_p)
Definition: gtest.cc:476
int value
GTEST_DISALLOW_COPY_AND_ASSIGN_(ParameterizedTestSuiteInfo)
std::vector< T > MakeVector(IndexSequence< I...>) const
std::string DefaultParamName(const TestParamInfo< ParamType > &info)
ParamIterator(ParamIteratorInterface< T > *impl)
virtual const ParamGeneratorInterface< T > * BaseGenerator() const =0
ParamIteratorInterface< T > * Begin() const override
ParamGenerator(ParamGeneratorInterface< T > *impl)
ParamIteratorInterface< T > * End() const override
void RegisterTestSuite(const char *test_suite_name, CodeLocation code_location)
Definition: gtest.cc:531
static SetUpTearDownSuiteFuncType GetSetUpCaseOrSuite(const char *filename, int line_num)
std::unique_ptr< ParamIteratorInterface< T > > impl_
ParamIteratorInterface< ParamType > * Begin() const override
ParameterizedTestCaseInfo< TestCase > * GetTestCasePatternHolder(const char *test_case_name, CodeLocation code_location)
const T func(int n, T *x)
Definition: ad_example.cpp:29
Iterator(const ParamGeneratorInterface< T > *base, typename ContainerType::const_iterator iterator)
const std::string & GetTestSuiteName() const override
TestInfo(const char *a_test_suite_base_name, const char *a_test_base_name, TestMetaFactoryBase< ParamType > *a_test_meta_factory, CodeLocation a_code_location)
virtual ParamIteratorInterface * Clone() const =0
ParamIteratorInterface< T > * Clone() const override
expr expr expr bar false
const ParamGeneratorInterface< T > * BaseGenerator() const override
Derived * CheckedDowncastToActualType(Base *base)
Definition: gtest-port.h:1112
IteratorImpl(const ParamGeneratorInterface< ParamType > *base, const std::tuple< ParamGenerator< T >...> &generators, bool is_end)
const ParamGeneratorInterface< ParamType > * BaseGenerator() const override
IteratorImpl< typename MakeIndexSequence< sizeof...(T)>::type > Iterator
ParamIterator & operator=(const ParamIterator &other)
bool Equals(const ParamIteratorInterface< T > &other) const override
std::shared_ptr< const ParamGeneratorInterface< T > > impl_
ParamIteratorInterface< ParamType > * End() const override
virtual const std::string & GetTestSuiteName() const =0