xref: /aosp_15_r20/external/pigweed/pw_log_basic/log_basic.cc (revision 61c4878ac05f98d0ceed94b57d316916de578985)
1*61c4878aSAndroid Build Coastguard Worker // Copyright 2020 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 #include "pw_log_basic/log_basic.h"
18*61c4878aSAndroid Build Coastguard Worker 
19*61c4878aSAndroid Build Coastguard Worker #include <cstring>
20*61c4878aSAndroid Build Coastguard Worker 
21*61c4878aSAndroid Build Coastguard Worker #include "pw_log/levels.h"
22*61c4878aSAndroid Build Coastguard Worker #include "pw_log_basic_private/config.h"
23*61c4878aSAndroid Build Coastguard Worker #include "pw_string/string_builder.h"
24*61c4878aSAndroid Build Coastguard Worker #include "pw_sys_io/sys_io.h"
25*61c4878aSAndroid Build Coastguard Worker 
26*61c4878aSAndroid Build Coastguard Worker // ANSI color constants to control the terminal. Not Windows compatible.
27*61c4878aSAndroid Build Coastguard Worker // clang-format off
28*61c4878aSAndroid Build Coastguard Worker #define MAGENTA   "\033[35m"
29*61c4878aSAndroid Build Coastguard Worker #define YELLOW    "\033[33m"
30*61c4878aSAndroid Build Coastguard Worker #define RED       "\033[31m"
31*61c4878aSAndroid Build Coastguard Worker #define GREEN     "\033[32m"
32*61c4878aSAndroid Build Coastguard Worker #define BLUE      "\033[96m"
33*61c4878aSAndroid Build Coastguard Worker #define BLACK     "\033[30m"
34*61c4878aSAndroid Build Coastguard Worker #define YELLOW_BG "\033[43m"
35*61c4878aSAndroid Build Coastguard Worker #define WHITE_BG  "\033[47m"
36*61c4878aSAndroid Build Coastguard Worker #define RED_BG    "\033[41m"
37*61c4878aSAndroid Build Coastguard Worker #define BOLD      "\033[1m"
38*61c4878aSAndroid Build Coastguard Worker #define RESET     "\033[0m"
39*61c4878aSAndroid Build Coastguard Worker // clang-format on
40*61c4878aSAndroid Build Coastguard Worker 
41*61c4878aSAndroid Build Coastguard Worker namespace pw::log_basic {
42*61c4878aSAndroid Build Coastguard Worker namespace {
43*61c4878aSAndroid Build Coastguard Worker 
LogLevelToLogLevelName(int level)44*61c4878aSAndroid Build Coastguard Worker const char* LogLevelToLogLevelName(int level) {
45*61c4878aSAndroid Build Coastguard Worker   switch (level) {
46*61c4878aSAndroid Build Coastguard Worker     // clang-format off
47*61c4878aSAndroid Build Coastguard Worker #if PW_EMOJI
48*61c4878aSAndroid Build Coastguard Worker     case PW_LOG_LEVEL_DEBUG    : return "��" RESET;
49*61c4878aSAndroid Build Coastguard Worker     case PW_LOG_LEVEL_INFO     : return "ℹ️ " RESET;
50*61c4878aSAndroid Build Coastguard Worker     case PW_LOG_LEVEL_WARN     : return "⚠️ " RESET;
51*61c4878aSAndroid Build Coastguard Worker     case PW_LOG_LEVEL_ERROR    : return "❌" RESET;
52*61c4878aSAndroid Build Coastguard Worker     case PW_LOG_LEVEL_CRITICAL : return "☠️ " RESET;
53*61c4878aSAndroid Build Coastguard Worker     default: return "❔" RESET;
54*61c4878aSAndroid Build Coastguard Worker #else
55*61c4878aSAndroid Build Coastguard Worker     case PW_LOG_LEVEL_DEBUG    : return BLUE     BOLD        "DBG" RESET;
56*61c4878aSAndroid Build Coastguard Worker     case PW_LOG_LEVEL_INFO     : return MAGENTA  BOLD        "INF" RESET;
57*61c4878aSAndroid Build Coastguard Worker     case PW_LOG_LEVEL_WARN     : return YELLOW   BOLD        "WRN" RESET;
58*61c4878aSAndroid Build Coastguard Worker     case PW_LOG_LEVEL_ERROR    : return RED      BOLD        "ERR" RESET;
59*61c4878aSAndroid Build Coastguard Worker     case PW_LOG_LEVEL_CRITICAL : return BLACK    BOLD RED_BG "FTL" RESET;
60*61c4878aSAndroid Build Coastguard Worker     default                    : return GREEN    BOLD        "UNK" RESET;
61*61c4878aSAndroid Build Coastguard Worker #endif
62*61c4878aSAndroid Build Coastguard Worker       // clang-format on
63*61c4878aSAndroid Build Coastguard Worker   }
64*61c4878aSAndroid Build Coastguard Worker }
65*61c4878aSAndroid Build Coastguard Worker 
66*61c4878aSAndroid Build Coastguard Worker #if PW_LOG_SHOW_FILENAME
GetFileBasename(const char * filename)67*61c4878aSAndroid Build Coastguard Worker const char* GetFileBasename(const char* filename) {
68*61c4878aSAndroid Build Coastguard Worker   int length = std::strlen(filename);
69*61c4878aSAndroid Build Coastguard Worker   if (length == 0) {
70*61c4878aSAndroid Build Coastguard Worker     return filename;
71*61c4878aSAndroid Build Coastguard Worker   }
72*61c4878aSAndroid Build Coastguard Worker 
73*61c4878aSAndroid Build Coastguard Worker   // Start on the last character.
74*61c4878aSAndroid Build Coastguard Worker   // TODO(pwbug/38): This part of the function doesn't work for Windows paths.
75*61c4878aSAndroid Build Coastguard Worker   const char* basename = filename + std::strlen(filename) - 1;
76*61c4878aSAndroid Build Coastguard Worker   while (basename != filename && *basename != '/') {
77*61c4878aSAndroid Build Coastguard Worker     basename--;
78*61c4878aSAndroid Build Coastguard Worker   }
79*61c4878aSAndroid Build Coastguard Worker   if (*basename == '/') {
80*61c4878aSAndroid Build Coastguard Worker     basename++;
81*61c4878aSAndroid Build Coastguard Worker   }
82*61c4878aSAndroid Build Coastguard Worker   return basename;
83*61c4878aSAndroid Build Coastguard Worker }
84*61c4878aSAndroid Build Coastguard Worker #endif  // PW_LOG_SHOW_FILENAME
85*61c4878aSAndroid Build Coastguard Worker 
__anon1f1b52ea0202(std::string_view log) 86*61c4878aSAndroid Build Coastguard Worker void (*write_log)(std::string_view) = [](std::string_view log) {
87*61c4878aSAndroid Build Coastguard Worker   sys_io::WriteLine(log)
88*61c4878aSAndroid Build Coastguard Worker       .IgnoreError();  // TODO: b/242598609 - Handle Status properly
89*61c4878aSAndroid Build Coastguard Worker };
90*61c4878aSAndroid Build Coastguard Worker 
91*61c4878aSAndroid Build Coastguard Worker }  // namespace
92*61c4878aSAndroid Build Coastguard Worker 
93*61c4878aSAndroid Build Coastguard Worker // This is a fully loaded, inefficient-at-the-callsite, log implementation.
pw_Log(int level,unsigned int flags,const char * module_name,const char * file_name,int line_number,const char * function_name,const char * message,...)94*61c4878aSAndroid Build Coastguard Worker extern "C" void pw_Log(int level,
95*61c4878aSAndroid Build Coastguard Worker                        unsigned int flags,
96*61c4878aSAndroid Build Coastguard Worker                        const char* module_name,
97*61c4878aSAndroid Build Coastguard Worker                        const char* file_name,
98*61c4878aSAndroid Build Coastguard Worker                        int line_number,
99*61c4878aSAndroid Build Coastguard Worker                        const char* function_name,
100*61c4878aSAndroid Build Coastguard Worker                        const char* message,
101*61c4878aSAndroid Build Coastguard Worker                        ...) {
102*61c4878aSAndroid Build Coastguard Worker   va_list args;
103*61c4878aSAndroid Build Coastguard Worker   va_start(args, message);
104*61c4878aSAndroid Build Coastguard Worker   pw_Log_HandleMessageVaList(level,
105*61c4878aSAndroid Build Coastguard Worker                              flags,
106*61c4878aSAndroid Build Coastguard Worker                              module_name,
107*61c4878aSAndroid Build Coastguard Worker                              file_name,
108*61c4878aSAndroid Build Coastguard Worker                              line_number,
109*61c4878aSAndroid Build Coastguard Worker                              function_name,
110*61c4878aSAndroid Build Coastguard Worker                              message,
111*61c4878aSAndroid Build Coastguard Worker                              args);
112*61c4878aSAndroid Build Coastguard Worker   va_end(args);
113*61c4878aSAndroid Build Coastguard Worker }
114*61c4878aSAndroid Build Coastguard Worker 
pw_Log_HandleMessageVaList(int level,unsigned int flags,const char * module_name,const char * file_name,int line_number,const char * function_name,const char * message,va_list args)115*61c4878aSAndroid Build Coastguard Worker extern "C" void pw_Log_HandleMessageVaList(int level,
116*61c4878aSAndroid Build Coastguard Worker                                            unsigned int flags,
117*61c4878aSAndroid Build Coastguard Worker                                            const char* module_name,
118*61c4878aSAndroid Build Coastguard Worker                                            const char* file_name,
119*61c4878aSAndroid Build Coastguard Worker                                            int line_number,
120*61c4878aSAndroid Build Coastguard Worker                                            const char* function_name,
121*61c4878aSAndroid Build Coastguard Worker                                            const char* message,
122*61c4878aSAndroid Build Coastguard Worker                                            va_list args) {
123*61c4878aSAndroid Build Coastguard Worker   // Accumulate the log message in this buffer, then output it.
124*61c4878aSAndroid Build Coastguard Worker   pw::StringBuffer<PW_LOG_BASIC_ENTRY_SIZE> buffer;
125*61c4878aSAndroid Build Coastguard Worker 
126*61c4878aSAndroid Build Coastguard Worker   // Column: Timestamp
127*61c4878aSAndroid Build Coastguard Worker   // Note that this macro method defaults to a no-op.
128*61c4878aSAndroid Build Coastguard Worker   PW_LOG_APPEND_TIMESTAMP(buffer);
129*61c4878aSAndroid Build Coastguard Worker 
130*61c4878aSAndroid Build Coastguard Worker   // Column: Filename
131*61c4878aSAndroid Build Coastguard Worker #if PW_LOG_SHOW_FILENAME
132*61c4878aSAndroid Build Coastguard Worker   buffer.Format(" %-30s:%4d |", GetFileBasename(file_name), line_number);
133*61c4878aSAndroid Build Coastguard Worker #else
134*61c4878aSAndroid Build Coastguard Worker   static_cast<void>(file_name);
135*61c4878aSAndroid Build Coastguard Worker   static_cast<void>(line_number);
136*61c4878aSAndroid Build Coastguard Worker #endif
137*61c4878aSAndroid Build Coastguard Worker 
138*61c4878aSAndroid Build Coastguard Worker   // Column: Function
139*61c4878aSAndroid Build Coastguard Worker #if PW_LOG_SHOW_FUNCTION
140*61c4878aSAndroid Build Coastguard Worker   buffer.Format(" %20s |", function_name);
141*61c4878aSAndroid Build Coastguard Worker #else
142*61c4878aSAndroid Build Coastguard Worker   static_cast<void>(function_name);
143*61c4878aSAndroid Build Coastguard Worker #endif
144*61c4878aSAndroid Build Coastguard Worker 
145*61c4878aSAndroid Build Coastguard Worker   // Column: Module
146*61c4878aSAndroid Build Coastguard Worker #if PW_LOG_SHOW_MODULE
147*61c4878aSAndroid Build Coastguard Worker   buffer << " " BOLD;
148*61c4878aSAndroid Build Coastguard Worker   buffer.Format("%3s", module_name);
149*61c4878aSAndroid Build Coastguard Worker   buffer << RESET " ";
150*61c4878aSAndroid Build Coastguard Worker #else
151*61c4878aSAndroid Build Coastguard Worker   static_cast<void>(module_name);
152*61c4878aSAndroid Build Coastguard Worker #endif  // PW_LOG_SHOW_MODULE
153*61c4878aSAndroid Build Coastguard Worker 
154*61c4878aSAndroid Build Coastguard Worker   // Column: Flag
155*61c4878aSAndroid Build Coastguard Worker #if PW_LOG_SHOW_FLAG
156*61c4878aSAndroid Build Coastguard Worker #if PW_EMOJI
157*61c4878aSAndroid Build Coastguard Worker   buffer << (flags ? "��" : "  ");
158*61c4878aSAndroid Build Coastguard Worker #else
159*61c4878aSAndroid Build Coastguard Worker   buffer << (flags ? "*" : "|");
160*61c4878aSAndroid Build Coastguard Worker #endif  // PW_EMOJI
161*61c4878aSAndroid Build Coastguard Worker   buffer << " ";
162*61c4878aSAndroid Build Coastguard Worker #else
163*61c4878aSAndroid Build Coastguard Worker   static_cast<void>(flags);
164*61c4878aSAndroid Build Coastguard Worker #endif  // PW_LOG_SHOW_FLAG
165*61c4878aSAndroid Build Coastguard Worker 
166*61c4878aSAndroid Build Coastguard Worker   // Column: Level
167*61c4878aSAndroid Build Coastguard Worker   buffer << LogLevelToLogLevelName(level) << "  ";
168*61c4878aSAndroid Build Coastguard Worker 
169*61c4878aSAndroid Build Coastguard Worker   // Column: Message
170*61c4878aSAndroid Build Coastguard Worker   buffer.FormatVaList(message, args);
171*61c4878aSAndroid Build Coastguard Worker 
172*61c4878aSAndroid Build Coastguard Worker   // All done; flush the log.
173*61c4878aSAndroid Build Coastguard Worker   write_log(buffer);
174*61c4878aSAndroid Build Coastguard Worker }
175*61c4878aSAndroid Build Coastguard Worker 
SetOutput(void (* log_output)(std::string_view log))176*61c4878aSAndroid Build Coastguard Worker void SetOutput(void (*log_output)(std::string_view log)) {
177*61c4878aSAndroid Build Coastguard Worker   write_log = log_output;
178*61c4878aSAndroid Build Coastguard Worker }
179*61c4878aSAndroid Build Coastguard Worker 
180*61c4878aSAndroid Build Coastguard Worker }  // namespace pw::log_basic
181