// Copyright (c) 2006-2008 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "base/tuple.h" #include "base/compiler_specific.h" #include "testing/gtest/include/gtest/gtest.h" namespace base { namespace { void DoAdd(int a, int b, int c, int* res) { *res = a + b + c; } struct Addy { Addy() { } void DoAdd(int a, int b, int c, int d, int* res) { *res = a + b + c + d; } }; struct Addz { Addz() { } void DoAdd(int a, int b, int c, int d, int e, int* res) { *res = a + b + c + d + e; } }; } // namespace TEST(TupleTest, Basic) { std::tuple<> t0 = std::make_tuple(); ALLOW_UNUSED_LOCAL(t0); std::tuple<int> t1(1); std::tuple<int, const char*> t2 = std::make_tuple(1, static_cast<const char*>("wee")); ALLOW_UNUSED_LOCAL(t2); std::tuple<int, int, int> t3(1, 2, 3); std::tuple<int, int, int, int*> t4(1, 2, 3, &std::get<0>(t1)); std::tuple<int, int, int, int, int*> t5(1, 2, 3, 4, &std::get<0>(t4)); std::tuple<int, int, int, int, int, int*> t6(1, 2, 3, 4, 5, &std::get<0>(t4)); EXPECT_EQ(1, std::get<0>(t1)); DispatchToFunction(&DoAdd, t4); EXPECT_EQ(6, std::get<0>(t1)); int res = 0; DispatchToFunction(&DoAdd, std::make_tuple(9, 8, 7, &res)); EXPECT_EQ(24, res); Addy addy; EXPECT_EQ(1, std::get<0>(t4)); DispatchToMethod(&addy, &Addy::DoAdd, t5); EXPECT_EQ(10, std::get<0>(t4)); Addz addz; EXPECT_EQ(10, std::get<0>(t4)); DispatchToMethod(&addz, &Addz::DoAdd, t6); EXPECT_EQ(15, std::get<0>(t4)); } namespace { struct CopyLogger { CopyLogger() { ++TimesConstructed; } CopyLogger(const CopyLogger& tocopy) { ++TimesConstructed; ++TimesCopied; } ~CopyLogger() { } static int TimesCopied; static int TimesConstructed; }; void SomeLoggerMethRef(const CopyLogger& logy, const CopyLogger* ptr, bool* b) { *b = &logy == ptr; } void SomeLoggerMethCopy(CopyLogger logy, const CopyLogger* ptr, bool* b) { *b = &logy == ptr; } int CopyLogger::TimesCopied = 0; int CopyLogger::TimesConstructed = 0; } // namespace TEST(TupleTest, Copying) { CopyLogger logger; EXPECT_EQ(0, CopyLogger::TimesCopied); EXPECT_EQ(1, CopyLogger::TimesConstructed); bool res = false; // Creating the tuple should copy the class to store internally in the tuple. std::tuple<CopyLogger, CopyLogger*, bool*> tuple(logger, &logger, &res); std::get<1>(tuple) = &std::get<0>(tuple); EXPECT_EQ(2, CopyLogger::TimesConstructed); EXPECT_EQ(1, CopyLogger::TimesCopied); // Our internal Logger and the one passed to the function should be the same. res = false; DispatchToFunction(&SomeLoggerMethRef, tuple); EXPECT_TRUE(res); EXPECT_EQ(2, CopyLogger::TimesConstructed); EXPECT_EQ(1, CopyLogger::TimesCopied); // Now they should be different, since the function call will make a copy. res = false; DispatchToFunction(&SomeLoggerMethCopy, tuple); EXPECT_FALSE(res); EXPECT_EQ(3, CopyLogger::TimesConstructed); EXPECT_EQ(2, CopyLogger::TimesCopied); } TEST(TupleTest, Get) { int i = 1; int j = 2; std::tuple<int, int&, int&&> t(3, i, std::move(j)); EXPECT_TRUE((std::is_same<int&, decltype(base::get<0>(t))>::value)); EXPECT_EQ(3, base::get<0>(t)); EXPECT_TRUE((std::is_same<int&, decltype(base::get<1>(t))>::value)); EXPECT_EQ(1, base::get<1>(t)); EXPECT_TRUE((std::is_same<int&, decltype(base::get<2>(t))>::value)); EXPECT_EQ(2, base::get<2>(t)); EXPECT_TRUE((std::is_same<int&&, decltype(base::get<0>(std::move(t)))>::value)); EXPECT_EQ(3, base::get<0>(std::move(t))); EXPECT_TRUE((std::is_same<int&, decltype(base::get<1>(std::move(t)))>::value)); EXPECT_EQ(1, base::get<1>(std::move(t))); EXPECT_TRUE((std::is_same<int&&, decltype(base::get<2>(std::move(t)))>::value)); EXPECT_EQ(2, base::get<2>(std::move(t))); } } // namespace base