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