xref: /aosp_15_r20/external/pigweed/pw_assert_basic/basic_handler.cc (revision 61c4878ac05f98d0ceed94b57d316916de578985)
1*61c4878aSAndroid Build Coastguard Worker // Copyright 2021 The Pigweed Authors
2*61c4878aSAndroid Build Coastguard Worker //
3*61c4878aSAndroid Build Coastguard Worker // Licensed under the Apache License, Version 2.0 (the "License"); you may not
4*61c4878aSAndroid Build Coastguard Worker // use this file except in compliance with the License. You may obtain a copy of
5*61c4878aSAndroid Build Coastguard Worker // the License at
6*61c4878aSAndroid Build Coastguard Worker //
7*61c4878aSAndroid Build Coastguard Worker //     https://www.apache.org/licenses/LICENSE-2.0
8*61c4878aSAndroid Build Coastguard Worker //
9*61c4878aSAndroid Build Coastguard Worker // Unless required by applicable law or agreed to in writing, software
10*61c4878aSAndroid Build Coastguard Worker // distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
11*61c4878aSAndroid Build Coastguard Worker // WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
12*61c4878aSAndroid Build Coastguard Worker // License for the specific language governing permissions and limitations under
13*61c4878aSAndroid Build Coastguard Worker // the License.
14*61c4878aSAndroid Build Coastguard Worker 
15*61c4878aSAndroid Build Coastguard Worker // This is a very basic direct output log implementation with no buffering.
16*61c4878aSAndroid Build Coastguard Worker 
17*61c4878aSAndroid Build Coastguard Worker // #define PW_LOG_MODULE_NAME "ASRT"
18*61c4878aSAndroid Build Coastguard Worker // #include "pw_log/log.h"
19*61c4878aSAndroid Build Coastguard Worker 
20*61c4878aSAndroid Build Coastguard Worker #include <cstdio>
21*61c4878aSAndroid Build Coastguard Worker #include <cstring>
22*61c4878aSAndroid Build Coastguard Worker 
23*61c4878aSAndroid Build Coastguard Worker #include "pw_assert/config.h"
24*61c4878aSAndroid Build Coastguard Worker #include "pw_assert_basic/handler.h"
25*61c4878aSAndroid Build Coastguard Worker #include "pw_preprocessor/util.h"
26*61c4878aSAndroid Build Coastguard Worker #include "pw_string/string_builder.h"
27*61c4878aSAndroid Build Coastguard Worker #include "pw_sys_io/sys_io.h"
28*61c4878aSAndroid Build Coastguard Worker 
29*61c4878aSAndroid Build Coastguard Worker // ANSI color constants to control the terminal. Not Windows compatible.
30*61c4878aSAndroid Build Coastguard Worker // clang-format off
31*61c4878aSAndroid Build Coastguard Worker #if PW_ASSERT_BASIC_USE_COLORS
32*61c4878aSAndroid Build Coastguard Worker #define MAGENTA   "\033[35m"
33*61c4878aSAndroid Build Coastguard Worker #define YELLOW    "\033[33m"
34*61c4878aSAndroid Build Coastguard Worker #define RED       "\033[31m"
35*61c4878aSAndroid Build Coastguard Worker #define GREEN     "\033[32m"
36*61c4878aSAndroid Build Coastguard Worker #define BLUE      "\033[96m"
37*61c4878aSAndroid Build Coastguard Worker #define BLACK     "\033[30m"
38*61c4878aSAndroid Build Coastguard Worker #define YELLOW_BG "\033[43m"
39*61c4878aSAndroid Build Coastguard Worker #define WHITE_BG  "\033[47m"
40*61c4878aSAndroid Build Coastguard Worker #define RED_BG    "\033[41m"
41*61c4878aSAndroid Build Coastguard Worker #define BOLD      "\033[1m"
42*61c4878aSAndroid Build Coastguard Worker #define RESET     "\033[0m"
43*61c4878aSAndroid Build Coastguard Worker #else
44*61c4878aSAndroid Build Coastguard Worker #define MAGENTA   ""
45*61c4878aSAndroid Build Coastguard Worker #define YELLOW    ""
46*61c4878aSAndroid Build Coastguard Worker #define RED       ""
47*61c4878aSAndroid Build Coastguard Worker #define GREEN     ""
48*61c4878aSAndroid Build Coastguard Worker #define BLUE      ""
49*61c4878aSAndroid Build Coastguard Worker #define BLACK     ""
50*61c4878aSAndroid Build Coastguard Worker #define YELLOW_BG ""
51*61c4878aSAndroid Build Coastguard Worker #define WHITE_BG  ""
52*61c4878aSAndroid Build Coastguard Worker #define RED_BG    ""
53*61c4878aSAndroid Build Coastguard Worker #define BOLD      ""
54*61c4878aSAndroid Build Coastguard Worker #define RESET     ""
55*61c4878aSAndroid Build Coastguard Worker #endif  // PW_ASSERT_BASIC_USE_COLORS
56*61c4878aSAndroid Build Coastguard Worker // clang-format on
57*61c4878aSAndroid Build Coastguard Worker 
58*61c4878aSAndroid Build Coastguard Worker static const char* kCrashBanner[] = {
59*61c4878aSAndroid Build Coastguard Worker     " ",
60*61c4878aSAndroid Build Coastguard Worker     "   ▄████▄      ██▀███      ▄▄▄           ██████     ██░ ██    ",
61*61c4878aSAndroid Build Coastguard Worker     "  ▒██▀ ▀█     ▓██ ▒ ██▒   ▒████▄       ▒██    ▒    ▓██░ ██▒   ",
62*61c4878aSAndroid Build Coastguard Worker     "  ▒▓█ �� ▄    ▓██ ░▄█ ▒   ▒██  ▀█▄     ░ ▓██▄      ▒██▀▀██░   ",
63*61c4878aSAndroid Build Coastguard Worker     "  ▒▓▓▄ ▄██▒   ▒██▀▀█▄     ░██▄▄▄▄██      ▒   ██▒   ░▓█ ░██    ",
64*61c4878aSAndroid Build Coastguard Worker     "  ▒ ▓███▀ ░   ░██▓ ▒██▒    ▓█   ▓██▒   ▒██████▒▒   ░▓█▒░██▓   ",
65*61c4878aSAndroid Build Coastguard Worker     "  ░ ░▒ ▒  ░   ░ ▒▓ ░▒▓░    ▒▒   ▓▒█░   ▒ ▒▓▒ ▒ ░    ▒ ░░▒░▒   ",
66*61c4878aSAndroid Build Coastguard Worker     "    ░  ▒        ░▒ ░ ▒░     ▒   ▒▒ ░   ░ ░▒  ░ ░    ▒ ░▒░ ░   ",
67*61c4878aSAndroid Build Coastguard Worker     "  ░             ░░   ░      ░   ▒      ░  ░  ░      ░  ░░ ░   ",
68*61c4878aSAndroid Build Coastguard Worker     "  ░ ░            ░              ░  ░         ░      ░  ░  ░   ",
69*61c4878aSAndroid Build Coastguard Worker     "  ░",
70*61c4878aSAndroid Build Coastguard Worker     " ",
71*61c4878aSAndroid Build Coastguard Worker };
72*61c4878aSAndroid Build Coastguard Worker 
WriteLine(std::string_view s)73*61c4878aSAndroid Build Coastguard Worker static void WriteLine(std::string_view s) {
74*61c4878aSAndroid Build Coastguard Worker   pw::sys_io::WriteLine(s)
75*61c4878aSAndroid Build Coastguard Worker       .IgnoreError();  // TODO: b/242598609 - Handle Status properly
76*61c4878aSAndroid Build Coastguard Worker }
77*61c4878aSAndroid Build Coastguard Worker 
78*61c4878aSAndroid Build Coastguard Worker using Buffer = pw::StringBuffer<150>;
79*61c4878aSAndroid Build Coastguard Worker 
pw_assert_basic_HandleFailure(const char * file_name,int line_number,const char * function_name,const char * format,...)80*61c4878aSAndroid Build Coastguard Worker extern "C" void pw_assert_basic_HandleFailure(const char* file_name,
81*61c4878aSAndroid Build Coastguard Worker                                               int line_number,
82*61c4878aSAndroid Build Coastguard Worker                                               const char* function_name,
83*61c4878aSAndroid Build Coastguard Worker                                               const char* format,
84*61c4878aSAndroid Build Coastguard Worker                                               ...) {
85*61c4878aSAndroid Build Coastguard Worker   // As a matter of usability, crashes should be visible; make it so.
86*61c4878aSAndroid Build Coastguard Worker #if PW_ASSERT_BASIC_SHOW_BANNER
87*61c4878aSAndroid Build Coastguard Worker   WriteLine(RED);
88*61c4878aSAndroid Build Coastguard Worker   for (const char* line : kCrashBanner) {
89*61c4878aSAndroid Build Coastguard Worker     WriteLine(line);
90*61c4878aSAndroid Build Coastguard Worker   }
91*61c4878aSAndroid Build Coastguard Worker   WriteLine(RESET);
92*61c4878aSAndroid Build Coastguard Worker #endif  // PW_ASSERT_BASIC_SHOW_BANNER
93*61c4878aSAndroid Build Coastguard Worker 
94*61c4878aSAndroid Build Coastguard Worker   WriteLine(
95*61c4878aSAndroid Build Coastguard Worker       "  Welp, that didn't go as planned. "
96*61c4878aSAndroid Build Coastguard Worker       "It seems we crashed. Terribly sorry!");
97*61c4878aSAndroid Build Coastguard Worker   WriteLine("");
98*61c4878aSAndroid Build Coastguard Worker   WriteLine(YELLOW "  CRASH MESSAGE" RESET);
99*61c4878aSAndroid Build Coastguard Worker   WriteLine("");
100*61c4878aSAndroid Build Coastguard Worker   {
101*61c4878aSAndroid Build Coastguard Worker     Buffer buffer;
102*61c4878aSAndroid Build Coastguard Worker     buffer << "     ";
103*61c4878aSAndroid Build Coastguard Worker     va_list args;
104*61c4878aSAndroid Build Coastguard Worker     va_start(args, format);
105*61c4878aSAndroid Build Coastguard Worker     buffer.FormatVaList(format, args);
106*61c4878aSAndroid Build Coastguard Worker     va_end(args);
107*61c4878aSAndroid Build Coastguard Worker     WriteLine(buffer.view());
108*61c4878aSAndroid Build Coastguard Worker   }
109*61c4878aSAndroid Build Coastguard Worker 
110*61c4878aSAndroid Build Coastguard Worker   if (file_name != nullptr && line_number != -1) {
111*61c4878aSAndroid Build Coastguard Worker     WriteLine("");
112*61c4878aSAndroid Build Coastguard Worker     WriteLine(YELLOW "  CRASH FILE & LINE" RESET);
113*61c4878aSAndroid Build Coastguard Worker     WriteLine("");
114*61c4878aSAndroid Build Coastguard Worker     {
115*61c4878aSAndroid Build Coastguard Worker       Buffer buffer;
116*61c4878aSAndroid Build Coastguard Worker       buffer.Format("     %s:%d", file_name, line_number);
117*61c4878aSAndroid Build Coastguard Worker       WriteLine(buffer.view());
118*61c4878aSAndroid Build Coastguard Worker     }
119*61c4878aSAndroid Build Coastguard Worker   }
120*61c4878aSAndroid Build Coastguard Worker   if (function_name != nullptr) {
121*61c4878aSAndroid Build Coastguard Worker     WriteLine("");
122*61c4878aSAndroid Build Coastguard Worker     WriteLine(YELLOW "  CRASH FUNCTION" RESET);
123*61c4878aSAndroid Build Coastguard Worker     WriteLine("");
124*61c4878aSAndroid Build Coastguard Worker     {
125*61c4878aSAndroid Build Coastguard Worker       Buffer buffer;
126*61c4878aSAndroid Build Coastguard Worker       buffer.Format("     %s", function_name);
127*61c4878aSAndroid Build Coastguard Worker       WriteLine(buffer.view());
128*61c4878aSAndroid Build Coastguard Worker     }
129*61c4878aSAndroid Build Coastguard Worker   }
130*61c4878aSAndroid Build Coastguard Worker 
131*61c4878aSAndroid Build Coastguard Worker   // TODO(pwbug/95): Perhaps surprisingly, this doesn't actually crash the
132*61c4878aSAndroid Build Coastguard Worker   // device. At some point we'll have a reboot BSP function or similar, but for
133*61c4878aSAndroid Build Coastguard Worker   // now this is acceptable since no one is using this basic backend.
134*61c4878aSAndroid Build Coastguard Worker   if (!PW_ASSERT_BASIC_DISABLE_NORETURN) {
135*61c4878aSAndroid Build Coastguard Worker #if (PW_ASSERT_BASIC_ACTION == PW_ASSERT_BASIC_ACTION_ABORT)
136*61c4878aSAndroid Build Coastguard Worker     // abort() doesn't flush stderr/stdout, so manually flush them before
137*61c4878aSAndroid Build Coastguard Worker     // aborting. abort() is preferred to exit(1) because debuggers catch it.
138*61c4878aSAndroid Build Coastguard Worker     std::fflush(stderr);
139*61c4878aSAndroid Build Coastguard Worker     std::fflush(stdout);
140*61c4878aSAndroid Build Coastguard Worker     std::abort();
141*61c4878aSAndroid Build Coastguard Worker #elif (PW_ASSERT_BASIC_ACTION == PW_ASSERT_BASIC_ACTION_EXIT)
142*61c4878aSAndroid Build Coastguard Worker     // Use _Exit to not run destructors or atexit hooks in case they cause
143*61c4878aSAndroid Build Coastguard Worker     // further crashes.
144*61c4878aSAndroid Build Coastguard Worker     std::_Exit(-1);
145*61c4878aSAndroid Build Coastguard Worker #elif (PW_ASSERT_BASIC_ACTION == PW_ASSERT_BASIC_ACTION_LOOP)
146*61c4878aSAndroid Build Coastguard Worker     WriteLine("");
147*61c4878aSAndroid Build Coastguard Worker     WriteLine(MAGENTA "  HANG TIME" RESET);
148*61c4878aSAndroid Build Coastguard Worker     WriteLine("");
149*61c4878aSAndroid Build Coastguard Worker     WriteLine(
150*61c4878aSAndroid Build Coastguard Worker         "     ... until a debugger joins. System is waiting in a while(1)");
151*61c4878aSAndroid Build Coastguard Worker     while (true) {
152*61c4878aSAndroid Build Coastguard Worker     }
153*61c4878aSAndroid Build Coastguard Worker #else
154*61c4878aSAndroid Build Coastguard Worker #error PW_ASSERT_BASIC_ACTION Must be set to valid option.
155*61c4878aSAndroid Build Coastguard Worker #endif
156*61c4878aSAndroid Build Coastguard Worker     PW_UNREACHABLE;
157*61c4878aSAndroid Build Coastguard Worker   } else {
158*61c4878aSAndroid Build Coastguard Worker     WriteLine("");
159*61c4878aSAndroid Build Coastguard Worker     WriteLine(MAGENTA "  NOTE: YOU ARE IN ASSERT BASIC TEST MODE" RESET);
160*61c4878aSAndroid Build Coastguard Worker     WriteLine("");
161*61c4878aSAndroid Build Coastguard Worker     WriteLine("     This build returns from the crash handler for testing.");
162*61c4878aSAndroid Build Coastguard Worker     WriteLine("     If you see this message in production, your build is ");
163*61c4878aSAndroid Build Coastguard Worker     WriteLine("     incorrectly configured. Search for");
164*61c4878aSAndroid Build Coastguard Worker     WriteLine("     PW_ASSERT_BASIC_DISABLE_NORETURN to fix it.");
165*61c4878aSAndroid Build Coastguard Worker     WriteLine("");
166*61c4878aSAndroid Build Coastguard Worker   }
167*61c4878aSAndroid Build Coastguard Worker }
168