Add call_once, wrapper around std::call_once with a fast path
[folly.git] / folly / test / CallOnceTest.cpp
1 /*
2  * Copyright 2016 Facebook, Inc.
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *   http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16
17 #include <deque>
18 #include <mutex>
19 #include <thread>
20
21 #include <folly/Benchmark.h>
22 #include <folly/CallOnce.h>
23 #include <gflags/gflags.h>
24 #include <glog/logging.h>
25 #include <gtest/gtest.h>
26
27 DEFINE_int32(threads, 16, "benchmark concurrency");
28
29 template <typename CallOnceFunc>
30 void bm_impl(CallOnceFunc&& fn, int64_t iters) {
31   std::deque<std::thread> threads;
32   for (int i = 0; i < FLAGS_threads; ++i) {
33     threads.emplace_back([&fn, iters] {
34       for (int64_t j = 0; j < iters; ++j) {
35         fn();
36       }
37     });
38   }
39   for (std::thread& t : threads) {
40     t.join();
41   }
42 }
43
44 BENCHMARK(StdCallOnceBench, iters) {
45   std::once_flag flag;
46   int out = 0;
47   bm_impl([&] { std::call_once(flag, [&] { ++out; }); }, iters);
48   ASSERT_EQ(1, out);
49 }
50
51 BENCHMARK(FollyCallOnceBench, iters) {
52   folly::once_flag flag;
53   int out = 0;
54   bm_impl([&] { folly::call_once(flag, [&] { ++out; }); }, iters);
55   ASSERT_EQ(1, out);
56 }
57
58 TEST(FollyCallOnce, Simple) {
59   folly::once_flag flag;
60   auto fn = [&](int* outp) { ++*outp; };
61   int out = 0;
62   folly::call_once(flag, fn, &out);
63   folly::call_once(flag, fn, &out);
64   ASSERT_EQ(1, out);
65 }
66
67 TEST(FollyCallOnce, Stress) {
68   for (int i = 0; i < 100; ++i) {
69     folly::once_flag flag;
70     int out = 0;
71     bm_impl([&] { folly::call_once(flag, [&] { ++out; }); }, 100);
72     ASSERT_EQ(1, out);
73   }
74 }
75
76 int main(int argc, char** argv) {
77   testing::InitGoogleTest(&argc, argv);
78   gflags::ParseCommandLineFlags(&argc, &argv, true);
79   if (FLAGS_benchmark) {
80     folly::runBenchmarksOnFlag();
81     return 0;
82   } else {
83     return RUN_ALL_TESTS();
84   }
85 }