/*
* Copyright (c) Meta Platforms, Inc. and affiliates.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include <folly/detail/StaticSingletonManager.h>
#include <folly/lang/Keep.h>
#include <folly/portability/GTest.h>
namespace folly {
namespace detail {
FOLLY_ATTR_WEAK void check_doit() {}
namespace {
template <bool Noexcept>
struct MayThrow {
FOLLY_NOINLINE MayThrow() noexcept(Noexcept) { check_doit(); }
FOLLY_NOINLINE ~MayThrow() { check_doit(); }
};
} // namespace
extern "C" FOLLY_KEEP int* check() {
return &createGlobal<int, void>();
}
extern "C" FOLLY_KEEP void* check_throw() {
return &createGlobal<MayThrow<false>, void>();
}
extern "C" FOLLY_KEEP void* check_nothrow() {
return &createGlobal<MayThrow<true>, void>();
}
template <typename Impl>
struct StaticSingletonManagerTest : public testing::TestWithParam<Impl> {};
TYPED_TEST_SUITE_P(StaticSingletonManagerTest);
template <typename Impl, typename T>
struct Tag {};
template <int I>
using Int = std::integral_constant<int, I>;
TYPED_TEST_P(StaticSingletonManagerTest, example) {
using K = TypeParam;
using T = std::integral_constant<int, 3>;
auto& i = K::template create<T, Tag<K, char>>();
EXPECT_EQ(T::value, i);
auto& j = K::template create<T, Tag<K, char>>();
EXPECT_EQ(&i, &j);
EXPECT_EQ(T::value, j);
auto& k = K::template create<T, Tag<K, char*>>();
EXPECT_NE(&i, &k);
EXPECT_EQ(T::value, k);
typename K::template ArgCreate<true> m_arg{tag<T, Tag<K, int>>};
EXPECT_EQ(nullptr, K::template get_existing_cached<T>(m_arg));
EXPECT_EQ(nullptr, K::template get_existing<T>(m_arg));
auto& m = K::template create<T>(m_arg);
EXPECT_NE(&i, &m);
EXPECT_EQ(T::value, m);
EXPECT_EQ(&m, K::template get_existing_cached<T>(m_arg));
EXPECT_EQ(&m, K::template get_existing<T>(m_arg));
typename K::template ArgCreate<true> n_arg{tag<T, Tag<K, int>>};
EXPECT_EQ(nullptr, K::template get_existing_cached<T>(n_arg));
EXPECT_EQ(&m, K::template get_existing<T>(n_arg));
EXPECT_EQ(&m, &K::template create<T>(n_arg));
}
REGISTER_TYPED_TEST_SUITE_P( //
StaticSingletonManagerTest,
example);
INSTANTIATE_TYPED_TEST_SUITE_P(
sans_rtti, StaticSingletonManagerTest, StaticSingletonManagerSansRtti);
#if FOLLY_HAS_RTTI
INSTANTIATE_TYPED_TEST_SUITE_P(
with_rtti, StaticSingletonManagerTest, StaticSingletonManagerWithRtti);
#endif
struct StaticSingletonManagerTestType : StaticSingletonManager {};
INSTANTIATE_TYPED_TEST_SUITE_P(
selection, StaticSingletonManagerTest, StaticSingletonManagerTestType);
struct CreateGlobalTest : testing::Test {};
TEST_F(CreateGlobalTest, example) {
using T = std::integral_constant<int, 3>;
auto& i = createGlobal<T, Tag<void, char>>();
EXPECT_EQ(T::value, i);
auto& j = createGlobal<T, Tag<void, char>>();
EXPECT_EQ(&i, &j);
EXPECT_EQ(T::value, j);
auto& k = createGlobal<T, Tag<void, char*>>();
EXPECT_NE(&i, &k);
EXPECT_EQ(T::value, k);
}
} // namespace detail
} // namespace folly