1 /*
2 * Copyright 2022 The Android Open Source Project
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 <ftl/function.h>
18 #include <gtest/gtest.h>
19
20 #include <array>
21 #include <cstddef>
22 #include <cstdint>
23 #include <string_view>
24 #include <type_traits>
25
26 namespace android::test {
27 namespace {
28
29 // Create an alias to composite requirements defined by the trait class `T` for easier testing.
30 template <typename T, typename S>
31 inline constexpr bool is_opaquely_storable = (T::template require_trivially_copyable<S> &&
32 T::template require_trivially_destructible<S> &&
33 T::template require_will_fit_in_opaque_storage<S> &&
34 T::template require_alignment_compatible<S>);
35
36 // `I` gives a count of sizeof(std::intptr_t) bytes , and `J` gives a raw count of bytes
37 template <size_t I, size_t J = 0>
38 struct KnownSizeFunctionObject {
39 using Data = std::array<std::byte, sizeof(std::intptr_t) * I + J>;
operator ()android::test::__anon42ce56e90111::KnownSizeFunctionObject40 void operator()() const {};
41 Data data{};
42 };
43
44 } // namespace
45
46 // static_assert the expected type traits
47 static_assert(std::is_invocable_r_v<void, ftl::Function<void()>>);
48 static_assert(std::is_trivially_copyable_v<ftl::Function<void()>>);
49 static_assert(std::is_trivially_destructible_v<ftl::Function<void()>>);
50 static_assert(std::is_trivially_copy_constructible_v<ftl::Function<void()>>);
51 static_assert(std::is_trivially_move_constructible_v<ftl::Function<void()>>);
52 static_assert(std::is_trivially_copy_assignable_v<ftl::Function<void()>>);
53 static_assert(std::is_trivially_move_assignable_v<ftl::Function<void()>>);
54
55 template <typename T>
56 using function_traits = ftl::details::function_traits<T>;
57
58 // static_assert that the expected value of N is used for known function object sizes.
59 static_assert(function_traits<KnownSizeFunctionObject<0, 0>>::size == 0);
60 static_assert(function_traits<KnownSizeFunctionObject<0, 1>>::size == 0);
61 static_assert(function_traits<KnownSizeFunctionObject<1, 0>>::size == 0);
62 static_assert(function_traits<KnownSizeFunctionObject<1, 1>>::size == 1);
63 static_assert(function_traits<KnownSizeFunctionObject<2, 0>>::size == 1);
64 static_assert(function_traits<KnownSizeFunctionObject<2, 1>>::size == 2);
65
66 // Check that is_function_v works
67 static_assert(!ftl::is_function_v<KnownSizeFunctionObject<0>>);
68 static_assert(!ftl::is_function_v<std::function<void()>>);
69 static_assert(ftl::is_function_v<ftl::Function<void()>>);
70
71 // static_assert what can and cannot be stored inside the opaque storage
72
73 template <size_t N>
74 using function_opaque_storage = ftl::details::function_opaque_storage<N>;
75
76 // Function objects can be stored if they fit.
77 static_assert(is_opaquely_storable<function_opaque_storage<0>, KnownSizeFunctionObject<0>>);
78 static_assert(is_opaquely_storable<function_opaque_storage<0>, KnownSizeFunctionObject<1>>);
79 static_assert(!is_opaquely_storable<function_opaque_storage<0>, KnownSizeFunctionObject<2>>);
80
81 static_assert(is_opaquely_storable<function_opaque_storage<1>, KnownSizeFunctionObject<2>>);
82 static_assert(!is_opaquely_storable<function_opaque_storage<1>, KnownSizeFunctionObject<3>>);
83
84 static_assert(is_opaquely_storable<function_opaque_storage<2>, KnownSizeFunctionObject<3>>);
85 static_assert(!is_opaquely_storable<function_opaque_storage<2>, KnownSizeFunctionObject<4>>);
86
87 // Another opaque storage can be stored if it fits. This property is used to copy smaller
88 // ftl::Functions into larger ones.
89 static_assert(is_opaquely_storable<function_opaque_storage<2>, function_opaque_storage<0>::type>);
90 static_assert(is_opaquely_storable<function_opaque_storage<2>, function_opaque_storage<1>::type>);
91 static_assert(is_opaquely_storable<function_opaque_storage<2>, function_opaque_storage<2>::type>);
92 static_assert(!is_opaquely_storable<function_opaque_storage<2>, function_opaque_storage<3>::type>);
93
94 // Function objects that aren't trivially copyable or destroyable cannot be stored.
__anon42ce56e90202null95 auto lambda_capturing_unique_ptr = [ptr = std::unique_ptr<void*>()] { static_cast<void>(ptr); };
96 static_assert(
97 !is_opaquely_storable<function_opaque_storage<2>, decltype(lambda_capturing_unique_ptr)>);
98
99 // Keep in sync with "Example usage" in header file.
TEST(Function,Example)100 TEST(Function, Example) {
101 using namespace std::string_view_literals;
102
103 class MyClass {
104 public:
105 void on_event() const {}
106 int on_string(int*, std::string_view) { return 1; }
107
108 auto get_function() {
109 return ftl::make_function([this] { on_event(); });
110 }
111 } cls;
112
113 // A function container with no arguments, and returning no value.
114 ftl::Function<void()> f;
115
116 // Construct a ftl::Function containing a small lambda.
117 f = cls.get_function();
118
119 // Construct a ftl::Function that calls `cls.on_event()`.
120 f = ftl::make_function<&MyClass::on_event>(&cls);
121
122 // Create a do-nothing function.
123 f = ftl::no_op;
124
125 // Invoke the contained function.
126 f();
127
128 // Also invokes it.
129 std::invoke(f);
130
131 // Create a typedef to give a more meaningful name and bound the size.
132 using MyFunction = ftl::Function<int(std::string_view), 2>;
133 int* ptr = nullptr;
134 auto f1 =
135 MyFunction::make([cls = &cls, ptr](std::string_view sv) { return cls->on_string(ptr, sv); });
136 int r = f1("abc"sv);
137
138 // Returns a default-constructed int (0).
139 f1 = ftl::no_op;
140 r = f1("abc"sv);
141 EXPECT_EQ(r, 0);
142 }
143
TEST(Function,BasicOperations)144 TEST(Function, BasicOperations) {
145 // Default constructible.
146 ftl::Function<int()> f;
147
148 // Compares as empty
149 EXPECT_FALSE(f);
150 EXPECT_TRUE(f == nullptr);
151 EXPECT_FALSE(f != nullptr);
152 EXPECT_TRUE(ftl::Function<int()>() == f);
153 EXPECT_FALSE(ftl::Function<int()>() != f);
154
155 // Assigning no_op sets it to not empty.
156 f = ftl::no_op;
157
158 // Verify it can be called, and that it returns a default constructed value.
159 EXPECT_EQ(f(), 0);
160
161 // Comparable when non-empty.
162 EXPECT_TRUE(f);
163 EXPECT_FALSE(f == nullptr);
164 EXPECT_TRUE(f != nullptr);
165 EXPECT_FALSE(ftl::Function<int()>() == f);
166 EXPECT_TRUE(ftl::Function<int()>() != f);
167
168 // Constructing from nullptr means empty.
169 f = ftl::Function<int()>{nullptr};
170 EXPECT_FALSE(f);
171
172 // Assigning nullptr means it is empty.
173 f = nullptr;
174 EXPECT_FALSE(f);
175
176 // Move construction
177 f = ftl::no_op;
178 ftl::Function<int()> g{std::move(f)};
179 EXPECT_TRUE(g != nullptr);
180
181 // Move assignment
182 f = nullptr;
183 f = std::move(g);
184 EXPECT_TRUE(f != nullptr);
185
186 // Copy construction
187 ftl::Function<int()> h{f};
188 EXPECT_TRUE(h != nullptr);
189
190 // Copy assignment
191 g = h;
192 EXPECT_TRUE(g != nullptr);
193 }
194
TEST(Function,CanMoveConstructFromLambda)195 TEST(Function, CanMoveConstructFromLambda) {
196 auto lambda = [] {};
197 ftl::Function<void()> f{std::move(lambda)};
198 }
199
TEST(Function,TerseDeducedConstructAndAssignFromLambda)200 TEST(Function, TerseDeducedConstructAndAssignFromLambda) {
201 auto f = ftl::Function([] { return 1; });
202 EXPECT_EQ(f(), 1);
203
204 f = [] { return 2; };
205 EXPECT_EQ(f(), 2);
206 }
207
208 namespace {
209
210 struct ImplicitConversionsHelper {
exactandroid::test::__anon42ce56e90811::ImplicitConversionsHelper211 auto exact(int) -> int { return 0; }
inexactandroid::test::__anon42ce56e90811::ImplicitConversionsHelper212 auto inexact(long) -> short { return 0; }
213 // TODO: Switch to `auto templated(auto x)` with C++20
214 template <typename T>
templatedandroid::test::__anon42ce56e90811::ImplicitConversionsHelper215 T templated(T x) {
216 return x;
217 }
218
static_exactandroid::test::__anon42ce56e90811::ImplicitConversionsHelper219 static auto static_exact(int) -> int { return 0; }
static_inexactandroid::test::__anon42ce56e90811::ImplicitConversionsHelper220 static auto static_inexact(long) -> short { return 0; }
221 // TODO: Switch to `static auto static_templated(auto x)` with C++20
222 template <typename T>
static_templatedandroid::test::__anon42ce56e90811::ImplicitConversionsHelper223 static T static_templated(T x) {
224 return x;
225 }
226 };
227
228 } // namespace
229
TEST(Function,ImplicitConversions)230 TEST(Function, ImplicitConversions) {
231 using Function = ftl::Function<int(int)>;
232 auto check = [](Function f) { return f(0); };
233 auto exact = [](int) -> int { return 0; };
234 auto inexact = [](long) -> short { return 0; };
235 auto templated = [](auto x) { return x; };
236
237 ImplicitConversionsHelper helper;
238
239 // Note, `check(nullptr)` would crash, so we can only check if it would be invocable.
240 static_assert(std::is_invocable_v<decltype(check), decltype(nullptr)>);
241
242 // Note: We invoke each of these to fully expand all the templates involved.
243 EXPECT_EQ(check(ftl::no_op), 0);
244
245 EXPECT_EQ(check(exact), 0);
246 EXPECT_EQ(check(inexact), 0);
247 EXPECT_EQ(check(templated), 0);
248
249 EXPECT_EQ(check(Function::make<&ImplicitConversionsHelper::exact>(&helper)), 0);
250 EXPECT_EQ(check(Function::make<&ImplicitConversionsHelper::inexact>(&helper)), 0);
251 EXPECT_EQ(check(Function::make<&ImplicitConversionsHelper::templated<int>>(&helper)), 0);
252
253 EXPECT_EQ(check(Function::make<&ImplicitConversionsHelper::static_exact>()), 0);
254 EXPECT_EQ(check(Function::make<&ImplicitConversionsHelper::static_inexact>()), 0);
255 EXPECT_EQ(check(Function::make<&ImplicitConversionsHelper::static_templated<int>>()), 0);
256 }
257
TEST(Function,MakeWithNonConstMemberFunction)258 TEST(Function, MakeWithNonConstMemberFunction) {
259 struct Observer {
260 bool called = false;
261 void setCalled() { called = true; }
262 } observer;
263
264 auto f = ftl::make_function<&Observer::setCalled>(&observer);
265
266 f();
267
268 EXPECT_TRUE(observer.called);
269
270 EXPECT_TRUE(f == ftl::Function<void()>::make<&Observer::setCalled>(&observer));
271 }
272
TEST(Function,MakeWithConstMemberFunction)273 TEST(Function, MakeWithConstMemberFunction) {
274 struct Observer {
275 mutable bool called = false;
276 void setCalled() const { called = true; }
277 } observer;
278
279 const auto f = ftl::make_function<&Observer::setCalled>(&observer);
280
281 f();
282
283 EXPECT_TRUE(observer.called);
284
285 EXPECT_TRUE(f == ftl::Function<void()>::make<&Observer::setCalled>(&observer));
286 }
287
TEST(Function,MakeWithConstClassPointer)288 TEST(Function, MakeWithConstClassPointer) {
289 const struct Observer {
290 mutable bool called = false;
291 void setCalled() const { called = true; }
292 } observer;
293
294 const auto f = ftl::make_function<&Observer::setCalled>(&observer);
295
296 f();
297
298 EXPECT_TRUE(observer.called);
299
300 EXPECT_TRUE(f == ftl::Function<void()>::make<&Observer::setCalled>(&observer));
301 }
302
TEST(Function,MakeWithNonCapturingLambda)303 TEST(Function, MakeWithNonCapturingLambda) {
304 auto f = ftl::make_function([](int a, int b) { return a + b; });
305 EXPECT_EQ(f(1, 2), 3);
306 }
307
TEST(Function,MakeWithCapturingLambda)308 TEST(Function, MakeWithCapturingLambda) {
309 bool called = false;
310 auto f = ftl::make_function([&called](int a, int b) {
311 called = true;
312 return a + b;
313 });
314 EXPECT_EQ(f(1, 2), 3);
315 EXPECT_TRUE(called);
316 }
317
TEST(Function,MakeWithCapturingMutableLambda)318 TEST(Function, MakeWithCapturingMutableLambda) {
319 bool called = false;
320 auto f = ftl::make_function([&called](int a, int b) mutable {
321 called = true;
322 return a + b;
323 });
324 EXPECT_EQ(f(1, 2), 3);
325 EXPECT_TRUE(called);
326 }
327
TEST(Function,MakeWithThreePointerCapturingLambda)328 TEST(Function, MakeWithThreePointerCapturingLambda) {
329 bool my_bool = false;
330 int my_int = 0;
331 float my_float = 0.f;
332
333 auto f = ftl::make_function(
334 [ptr_bool = &my_bool, ptr_int = &my_int, ptr_float = &my_float](int a, int b) mutable {
335 *ptr_bool = true;
336 *ptr_int = 1;
337 *ptr_float = 1.f;
338
339 return a + b;
340 });
341
342 EXPECT_EQ(f(1, 2), 3);
343
344 EXPECT_TRUE(my_bool);
345 EXPECT_EQ(my_int, 1);
346 EXPECT_EQ(my_float, 1.f);
347 }
348
TEST(Function,MakeWithFreeFunction)349 TEST(Function, MakeWithFreeFunction) {
350 auto f = ftl::make_function<&std::make_unique<int, int>>();
351 std::unique_ptr<int> unique_int = f(1);
352 ASSERT_TRUE(unique_int);
353 EXPECT_EQ(*unique_int, 1);
354 }
355
TEST(Function,CopyToLarger)356 TEST(Function, CopyToLarger) {
357 int counter = 0;
358 ftl::Function<void()> a{[ptr_counter = &counter] { (*ptr_counter)++; }};
359 ftl::Function<void(), 1> b = a;
360 ftl::Function<void(), 2> c = a;
361
362 EXPECT_EQ(counter, 0);
363 a();
364 EXPECT_EQ(counter, 1);
365 b();
366 EXPECT_EQ(counter, 2);
367 c();
368 EXPECT_EQ(counter, 3);
369
370 b = [ptr_counter = &counter] { (*ptr_counter) += 2; };
371 c = [ptr_counter = &counter] { (*ptr_counter) += 3; };
372
373 b();
374 EXPECT_EQ(counter, 5);
375 c();
376 EXPECT_EQ(counter, 8);
377 }
378
379 } // namespace android::test
380