xref: /aosp_15_r20/external/llvm-libc/test/src/stdlib/atexit_test.cpp (revision 71db0c75aadcf003ffe3238005f61d7618a3fead)
1*71db0c75SAndroid Build Coastguard Worker //===-- Unittests for atexit ----------------------------------------------===//
2*71db0c75SAndroid Build Coastguard Worker //
3*71db0c75SAndroid Build Coastguard Worker // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*71db0c75SAndroid Build Coastguard Worker // See https://llvm.org/LICENSE.txt for license information.
5*71db0c75SAndroid Build Coastguard Worker // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*71db0c75SAndroid Build Coastguard Worker //
7*71db0c75SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
8*71db0c75SAndroid Build Coastguard Worker 
9*71db0c75SAndroid Build Coastguard Worker #include "src/__support/CPP/array.h"
10*71db0c75SAndroid Build Coastguard Worker #include "src/__support/CPP/utility.h"
11*71db0c75SAndroid Build Coastguard Worker #include "src/stdlib/_Exit.h"
12*71db0c75SAndroid Build Coastguard Worker #include "src/stdlib/atexit.h"
13*71db0c75SAndroid Build Coastguard Worker #include "src/stdlib/exit.h"
14*71db0c75SAndroid Build Coastguard Worker #include "test/UnitTest/Test.h"
15*71db0c75SAndroid Build Coastguard Worker 
16*71db0c75SAndroid Build Coastguard Worker static int a;
TEST(LlvmLibcAtExit,Basic)17*71db0c75SAndroid Build Coastguard Worker TEST(LlvmLibcAtExit, Basic) {
18*71db0c75SAndroid Build Coastguard Worker   // In case tests ever run multiple times.
19*71db0c75SAndroid Build Coastguard Worker   a = 0;
20*71db0c75SAndroid Build Coastguard Worker 
21*71db0c75SAndroid Build Coastguard Worker   auto test = [] {
22*71db0c75SAndroid Build Coastguard Worker     int status = LIBC_NAMESPACE::atexit(+[] {
23*71db0c75SAndroid Build Coastguard Worker       if (a != 1)
24*71db0c75SAndroid Build Coastguard Worker         __builtin_trap();
25*71db0c75SAndroid Build Coastguard Worker     });
26*71db0c75SAndroid Build Coastguard Worker     status |= LIBC_NAMESPACE::atexit(+[] { a++; });
27*71db0c75SAndroid Build Coastguard Worker     if (status)
28*71db0c75SAndroid Build Coastguard Worker       __builtin_trap();
29*71db0c75SAndroid Build Coastguard Worker 
30*71db0c75SAndroid Build Coastguard Worker     LIBC_NAMESPACE::exit(0);
31*71db0c75SAndroid Build Coastguard Worker   };
32*71db0c75SAndroid Build Coastguard Worker   EXPECT_EXITS(test, 0);
33*71db0c75SAndroid Build Coastguard Worker }
34*71db0c75SAndroid Build Coastguard Worker 
TEST(LlvmLibcAtExit,AtExitCallsSysExit)35*71db0c75SAndroid Build Coastguard Worker TEST(LlvmLibcAtExit, AtExitCallsSysExit) {
36*71db0c75SAndroid Build Coastguard Worker   auto test = [] {
37*71db0c75SAndroid Build Coastguard Worker     LIBC_NAMESPACE::atexit(+[] { LIBC_NAMESPACE::_Exit(1); });
38*71db0c75SAndroid Build Coastguard Worker     LIBC_NAMESPACE::exit(0);
39*71db0c75SAndroid Build Coastguard Worker   };
40*71db0c75SAndroid Build Coastguard Worker   EXPECT_EXITS(test, 1);
41*71db0c75SAndroid Build Coastguard Worker }
42*71db0c75SAndroid Build Coastguard Worker 
43*71db0c75SAndroid Build Coastguard Worker static int size;
44*71db0c75SAndroid Build Coastguard Worker static LIBC_NAMESPACE::cpp::array<int, 256> arr;
45*71db0c75SAndroid Build Coastguard Worker 
46*71db0c75SAndroid Build Coastguard Worker template <int... Ts>
register_atexit_handlers(LIBC_NAMESPACE::cpp::integer_sequence<int,Ts...>)47*71db0c75SAndroid Build Coastguard Worker void register_atexit_handlers(
48*71db0c75SAndroid Build Coastguard Worker     LIBC_NAMESPACE::cpp::integer_sequence<int, Ts...>) {
49*71db0c75SAndroid Build Coastguard Worker   (LIBC_NAMESPACE::atexit(+[] { arr[size++] = Ts; }), ...);
50*71db0c75SAndroid Build Coastguard Worker }
51*71db0c75SAndroid Build Coastguard Worker 
getTest()52*71db0c75SAndroid Build Coastguard Worker template <int count> constexpr auto getTest() {
53*71db0c75SAndroid Build Coastguard Worker   return [] {
54*71db0c75SAndroid Build Coastguard Worker     LIBC_NAMESPACE::atexit(+[] {
55*71db0c75SAndroid Build Coastguard Worker       if (size != count)
56*71db0c75SAndroid Build Coastguard Worker         __builtin_trap();
57*71db0c75SAndroid Build Coastguard Worker       for (int i = 0; i < count; i++)
58*71db0c75SAndroid Build Coastguard Worker         if (arr[i] != count - 1 - i)
59*71db0c75SAndroid Build Coastguard Worker           __builtin_trap();
60*71db0c75SAndroid Build Coastguard Worker     });
61*71db0c75SAndroid Build Coastguard Worker     register_atexit_handlers(
62*71db0c75SAndroid Build Coastguard Worker         LIBC_NAMESPACE::cpp::make_integer_sequence<int, count>{});
63*71db0c75SAndroid Build Coastguard Worker     LIBC_NAMESPACE::exit(0);
64*71db0c75SAndroid Build Coastguard Worker   };
65*71db0c75SAndroid Build Coastguard Worker }
66*71db0c75SAndroid Build Coastguard Worker 
TEST(LlvmLibcAtExit,ReverseOrder)67*71db0c75SAndroid Build Coastguard Worker TEST(LlvmLibcAtExit, ReverseOrder) {
68*71db0c75SAndroid Build Coastguard Worker   // In case tests ever run multiple times.
69*71db0c75SAndroid Build Coastguard Worker   size = 0;
70*71db0c75SAndroid Build Coastguard Worker 
71*71db0c75SAndroid Build Coastguard Worker   auto test = getTest<32>();
72*71db0c75SAndroid Build Coastguard Worker   EXPECT_EXITS(test, 0);
73*71db0c75SAndroid Build Coastguard Worker }
74*71db0c75SAndroid Build Coastguard Worker 
TEST(LlvmLibcAtExit,Many)75*71db0c75SAndroid Build Coastguard Worker TEST(LlvmLibcAtExit, Many) {
76*71db0c75SAndroid Build Coastguard Worker   // In case tests ever run multiple times.
77*71db0c75SAndroid Build Coastguard Worker   size = 0;
78*71db0c75SAndroid Build Coastguard Worker 
79*71db0c75SAndroid Build Coastguard Worker   auto test = getTest<256>();
80*71db0c75SAndroid Build Coastguard Worker   EXPECT_EXITS(test, 0);
81*71db0c75SAndroid Build Coastguard Worker }
82*71db0c75SAndroid Build Coastguard Worker 
TEST(LlvmLibcAtExit,HandlerCallsAtExit)83*71db0c75SAndroid Build Coastguard Worker TEST(LlvmLibcAtExit, HandlerCallsAtExit) {
84*71db0c75SAndroid Build Coastguard Worker   auto test = [] {
85*71db0c75SAndroid Build Coastguard Worker     LIBC_NAMESPACE::atexit(
86*71db0c75SAndroid Build Coastguard Worker         +[] { LIBC_NAMESPACE::atexit(+[] { LIBC_NAMESPACE::exit(1); }); });
87*71db0c75SAndroid Build Coastguard Worker     LIBC_NAMESPACE::exit(0);
88*71db0c75SAndroid Build Coastguard Worker   };
89*71db0c75SAndroid Build Coastguard Worker   EXPECT_EXITS(test, 1);
90*71db0c75SAndroid Build Coastguard Worker }
91