/*
* 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/synchronization/AtomicRef.h>
#include <folly/portability/GTest.h>
struct foo {};
static_assert(std::is_same_v<int*, std::atomic_ref<int*>::value_type>);
static_assert(std::is_same_v<int, std::atomic_ref<int>::value_type>);
static_assert(std::is_same_v<float, std::atomic_ref<float>::value_type>);
static_assert(std::is_same_v<foo, std::atomic_ref<foo>::value_type>);
class AtomicRefTest : public testing::Test {};
TEST_F(AtomicRefTest, deduction) {
long value = 17;
auto ref = folly::atomic_ref{value}; // use deduction guide
EXPECT_EQ(17, ref.load(std::memory_order_relaxed));
}
TEST_F(AtomicRefTest, integer) {
{
long value = 17;
auto ref = folly::make_atomic_ref(value);
EXPECT_EQ(17, ref.load(std::memory_order_relaxed));
ref.store(55, std::memory_order_relaxed);
EXPECT_EQ(55, ref.exchange(42, std::memory_order_relaxed));
EXPECT_EQ(42, ref.load(std::memory_order_relaxed));
}
{
long value = 17;
auto ref = folly::make_atomic_ref(value);
auto prev = ref.fetch_add(4, std::memory_order_relaxed);
EXPECT_EQ(17, prev);
EXPECT_EQ(21, value);
}
{
long value = 17;
auto ref = folly::make_atomic_ref(value);
auto prev = ref.fetch_sub(4, std::memory_order_relaxed);
EXPECT_EQ(17, prev);
EXPECT_EQ(13, value);
}
{
long value = 17;
auto ref = folly::make_atomic_ref(value);
auto prev = ref.fetch_and(18, std::memory_order_relaxed);
EXPECT_EQ(17, prev);
EXPECT_EQ(16, value);
}
{
long value = 17;
auto ref = folly::make_atomic_ref(value);
auto prev = ref.fetch_or(18, std::memory_order_relaxed);
EXPECT_EQ(17, prev);
EXPECT_EQ(19, value);
}
{
long value = 17;
auto ref = folly::make_atomic_ref(value);
auto prev = ref.fetch_xor(19, std::memory_order_relaxed);
EXPECT_EQ(17, prev);
EXPECT_EQ(2, value);
}
}
TEST_F(AtomicRefTest, integer_compare_exchange_weak) {
auto const relaxed = std::memory_order_relaxed;
{
long value = 17;
auto ref = folly::make_atomic_ref(value);
auto expected = value;
auto done = ref.compare_exchange_weak(expected, 19);
EXPECT_TRUE(done);
EXPECT_EQ(19, value);
EXPECT_EQ(17, expected);
done = ref.compare_exchange_weak(expected, 21);
EXPECT_FALSE(done);
EXPECT_EQ(19, value);
EXPECT_EQ(19, expected);
}
{
long value = 17;
auto ref = folly::make_atomic_ref(value);
auto expected = value;
auto done = ref.compare_exchange_weak(expected, 19, relaxed);
EXPECT_TRUE(done);
EXPECT_EQ(19, value);
EXPECT_EQ(17, expected);
done = ref.compare_exchange_weak(expected, 21, relaxed);
EXPECT_FALSE(done);
EXPECT_EQ(19, value);
EXPECT_EQ(19, expected);
}
{
long value = 17;
auto ref = folly::make_atomic_ref(value);
auto expected = value;
auto done = ref.compare_exchange_weak(expected, 19, relaxed, relaxed);
EXPECT_TRUE(done);
EXPECT_EQ(19, value);
EXPECT_EQ(17, expected);
done = ref.compare_exchange_weak(expected, 21, relaxed, relaxed);
EXPECT_FALSE(done);
EXPECT_EQ(19, value);
EXPECT_EQ(19, expected);
}
}
TEST_F(AtomicRefTest, integer_compare_exchange_strong) {
auto const relaxed = std::memory_order_relaxed;
{
long value = 17;
auto ref = folly::make_atomic_ref(value);
auto expected = value;
auto done = ref.compare_exchange_strong(expected, 19);
EXPECT_TRUE(done);
EXPECT_EQ(19, value);
EXPECT_EQ(17, expected);
done = ref.compare_exchange_strong(expected, 21);
EXPECT_FALSE(done);
EXPECT_EQ(19, value);
EXPECT_EQ(19, expected);
}
{
long value = 17;
auto ref = folly::make_atomic_ref(value);
auto expected = value;
auto done = ref.compare_exchange_strong(expected, 19, relaxed);
EXPECT_TRUE(done);
EXPECT_EQ(19, value);
EXPECT_EQ(17, expected);
done = ref.compare_exchange_strong(expected, 21, relaxed);
EXPECT_FALSE(done);
EXPECT_EQ(19, value);
EXPECT_EQ(19, expected);
}
{
long value = 17;
auto ref = folly::make_atomic_ref(value);
auto expected = value;
auto done = ref.compare_exchange_strong(expected, 19, relaxed, relaxed);
EXPECT_TRUE(done);
EXPECT_EQ(19, value);
EXPECT_EQ(17, expected);
done = ref.compare_exchange_strong(expected, 21, relaxed, relaxed);
EXPECT_FALSE(done);
EXPECT_EQ(19, value);
EXPECT_EQ(19, expected);
}
}