1*14675a02SAndroid Build Coastguard Worker /*
2*14675a02SAndroid Build Coastguard Worker * Copyright 2017 Google LLC
3*14675a02SAndroid Build Coastguard Worker *
4*14675a02SAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*14675a02SAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*14675a02SAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*14675a02SAndroid Build Coastguard Worker *
8*14675a02SAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*14675a02SAndroid Build Coastguard Worker *
10*14675a02SAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*14675a02SAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*14675a02SAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*14675a02SAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*14675a02SAndroid Build Coastguard Worker * limitations under the License.
15*14675a02SAndroid Build Coastguard Worker */
16*14675a02SAndroid Build Coastguard Worker
17*14675a02SAndroid Build Coastguard Worker #include "fcp/base/monitoring.h"
18*14675a02SAndroid Build Coastguard Worker
19*14675a02SAndroid Build Coastguard Worker #include <stdlib.h> /* for abort() */
20*14675a02SAndroid Build Coastguard Worker
21*14675a02SAndroid Build Coastguard Worker #include <string>
22*14675a02SAndroid Build Coastguard Worker
23*14675a02SAndroid Build Coastguard Worker #ifndef FCP_BAREMETAL
24*14675a02SAndroid Build Coastguard Worker #include <stdarg.h>
25*14675a02SAndroid Build Coastguard Worker #include <stdio.h>
26*14675a02SAndroid Build Coastguard Worker
27*14675a02SAndroid Build Coastguard Worker #ifdef __ANDROID__
28*14675a02SAndroid Build Coastguard Worker #include <android/log.h>
29*14675a02SAndroid Build Coastguard Worker #endif
30*14675a02SAndroid Build Coastguard Worker
31*14675a02SAndroid Build Coastguard Worker #include <cstring>
32*14675a02SAndroid Build Coastguard Worker
33*14675a02SAndroid Build Coastguard Worker #include "absl/strings/str_format.h"
34*14675a02SAndroid Build Coastguard Worker #endif // FCP_BAREMETAL
35*14675a02SAndroid Build Coastguard Worker
36*14675a02SAndroid Build Coastguard Worker #include "fcp/base/base_name.h"
37*14675a02SAndroid Build Coastguard Worker
38*14675a02SAndroid Build Coastguard Worker namespace fcp {
39*14675a02SAndroid Build Coastguard Worker
40*14675a02SAndroid Build Coastguard Worker namespace internal {
41*14675a02SAndroid Build Coastguard Worker
42*14675a02SAndroid Build Coastguard Worker namespace {
43*14675a02SAndroid Build Coastguard Worker #ifdef __ANDROID__
44*14675a02SAndroid Build Coastguard Worker constexpr char kAndroidLogTag[] = "fcp";
45*14675a02SAndroid Build Coastguard Worker
AndroidLogLevel(LogSeverity severity)46*14675a02SAndroid Build Coastguard Worker int AndroidLogLevel(LogSeverity severity) {
47*14675a02SAndroid Build Coastguard Worker switch (severity) {
48*14675a02SAndroid Build Coastguard Worker case LogSeverity::kFatal:
49*14675a02SAndroid Build Coastguard Worker return ANDROID_LOG_FATAL;
50*14675a02SAndroid Build Coastguard Worker case LogSeverity::kError:
51*14675a02SAndroid Build Coastguard Worker return ANDROID_LOG_ERROR;
52*14675a02SAndroid Build Coastguard Worker case LogSeverity::kWarning:
53*14675a02SAndroid Build Coastguard Worker return ANDROID_LOG_WARN;
54*14675a02SAndroid Build Coastguard Worker default:
55*14675a02SAndroid Build Coastguard Worker return ANDROID_LOG_INFO;
56*14675a02SAndroid Build Coastguard Worker }
57*14675a02SAndroid Build Coastguard Worker }
58*14675a02SAndroid Build Coastguard Worker #endif
59*14675a02SAndroid Build Coastguard Worker
60*14675a02SAndroid Build Coastguard Worker } // namespace
61*14675a02SAndroid Build Coastguard Worker
62*14675a02SAndroid Build Coastguard Worker // Provides safe static initialization of the default global logger instance.
63*14675a02SAndroid Build Coastguard Worker // TODO(team): Improve the logger registration mechanism.
GetGlobalLogger()64*14675a02SAndroid Build Coastguard Worker Logger*& GetGlobalLogger() {
65*14675a02SAndroid Build Coastguard Worker static Logger* global_logger = new Logger();
66*14675a02SAndroid Build Coastguard Worker return global_logger;
67*14675a02SAndroid Build Coastguard Worker }
68*14675a02SAndroid Build Coastguard Worker
logger()69*14675a02SAndroid Build Coastguard Worker Logger* logger() { return GetGlobalLogger(); }
set_logger(Logger * logger)70*14675a02SAndroid Build Coastguard Worker void set_logger(Logger* logger) { GetGlobalLogger() = logger; }
71*14675a02SAndroid Build Coastguard Worker
Log(const char * file,int line,LogSeverity severity,const char * message)72*14675a02SAndroid Build Coastguard Worker void Logger::Log(const char* file, int line, LogSeverity severity,
73*14675a02SAndroid Build Coastguard Worker const char* message) {
74*14675a02SAndroid Build Coastguard Worker #ifndef FCP_BAREMETAL
75*14675a02SAndroid Build Coastguard Worker auto base_file_name = BaseName(file);
76*14675a02SAndroid Build Coastguard Worker #ifdef __ANDROID__
77*14675a02SAndroid Build Coastguard Worker bool log_to_logcat = true;
78*14675a02SAndroid Build Coastguard Worker #ifdef NDEBUG
79*14675a02SAndroid Build Coastguard Worker // We don't log INFO logs on Android if this is a production build, since
80*14675a02SAndroid Build Coastguard Worker // they're too verbose. We can't just log them at ANDROID_LOG_VERBOSE either,
81*14675a02SAndroid Build Coastguard Worker // since then they'd still show up in the logcat unless we first check
82*14675a02SAndroid Build Coastguard Worker // __android_log_is_loggable, but that function isn't available until Android
83*14675a02SAndroid Build Coastguard Worker // API level 30. So to keep things simple we only log warnings or above,
84*14675a02SAndroid Build Coastguard Worker // unless this is a debug build.
85*14675a02SAndroid Build Coastguard Worker log_to_logcat = severity != LogSeverity::kInfo;
86*14675a02SAndroid Build Coastguard Worker #endif // NDEBUG
87*14675a02SAndroid Build Coastguard Worker if (log_to_logcat) {
88*14675a02SAndroid Build Coastguard Worker int level = AndroidLogLevel(severity);
89*14675a02SAndroid Build Coastguard Worker __android_log_print(level, kAndroidLogTag, "%c %s:%d %s\n",
90*14675a02SAndroid Build Coastguard Worker absl::LogSeverityName(severity)[0],
91*14675a02SAndroid Build Coastguard Worker base_file_name.c_str(), line, message);
92*14675a02SAndroid Build Coastguard Worker }
93*14675a02SAndroid Build Coastguard Worker #endif // __ANDROID__
94*14675a02SAndroid Build Coastguard Worker // Note that on Android we print both to logcat *and* stderr. This allows
95*14675a02SAndroid Build Coastguard Worker // tests to use ASSERT_DEATH to test for fatal error messages, among other
96*14675a02SAndroid Build Coastguard Worker // uses.
97*14675a02SAndroid Build Coastguard Worker absl::FPrintF(stderr, "%c %s:%d %s\n", absl::LogSeverityName(severity)[0],
98*14675a02SAndroid Build Coastguard Worker base_file_name, line, message);
99*14675a02SAndroid Build Coastguard Worker #endif // FCP_BAREMETAL
100*14675a02SAndroid Build Coastguard Worker }
101*14675a02SAndroid Build Coastguard Worker
StatusBuilder(StatusCode code,const char * file,int line)102*14675a02SAndroid Build Coastguard Worker StatusBuilder::StatusBuilder(StatusCode code, const char* file, int line)
103*14675a02SAndroid Build Coastguard Worker : file_(file), line_(line), code_(code), message_() {}
104*14675a02SAndroid Build Coastguard Worker
StatusBuilder(StatusBuilder const & other)105*14675a02SAndroid Build Coastguard Worker StatusBuilder::StatusBuilder(StatusBuilder const& other)
106*14675a02SAndroid Build Coastguard Worker : file_(other.file_),
107*14675a02SAndroid Build Coastguard Worker line_(other.line_),
108*14675a02SAndroid Build Coastguard Worker code_(other.code_),
109*14675a02SAndroid Build Coastguard Worker message_(other.message_.str()) {}
110*14675a02SAndroid Build Coastguard Worker
operator Status()111*14675a02SAndroid Build Coastguard Worker StatusBuilder::operator Status() {
112*14675a02SAndroid Build Coastguard Worker auto message_str = message_.str();
113*14675a02SAndroid Build Coastguard Worker if (code_ != OK) {
114*14675a02SAndroid Build Coastguard Worker StringStream status_message;
115*14675a02SAndroid Build Coastguard Worker status_message << "(at " << BaseName(file_) << ":" << line_ << message_str;
116*14675a02SAndroid Build Coastguard Worker message_str = status_message.str();
117*14675a02SAndroid Build Coastguard Worker if (log_severity_ != kNoLog) {
118*14675a02SAndroid Build Coastguard Worker StringStream log_message;
119*14675a02SAndroid Build Coastguard Worker log_message << "[" << code_ << "] " << message_str;
120*14675a02SAndroid Build Coastguard Worker logger()->Log(file_, line_, log_severity_, log_message.str().c_str());
121*14675a02SAndroid Build Coastguard Worker if (log_severity_ == LogSeverity::kFatal) {
122*14675a02SAndroid Build Coastguard Worker abort();
123*14675a02SAndroid Build Coastguard Worker }
124*14675a02SAndroid Build Coastguard Worker }
125*14675a02SAndroid Build Coastguard Worker }
126*14675a02SAndroid Build Coastguard Worker return Status(code_, message_str);
127*14675a02SAndroid Build Coastguard Worker }
128*14675a02SAndroid Build Coastguard Worker
129*14675a02SAndroid Build Coastguard Worker } // namespace internal
130*14675a02SAndroid Build Coastguard Worker
131*14675a02SAndroid Build Coastguard Worker } // namespace fcp
132