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