/* * Copyright (C) 2007-2016 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "logger_write.h" #include #include #include #include #include #include #ifdef __BIONIC__ #include #endif #include #include #include #include #include #include "android/log.h" #include "log/log_read.h" #include "logger.h" #include "uio.h" #ifdef __ANDROID__ #include "logd_writer.h" #include "pmsg_writer.h" #endif #if defined(__APPLE__) #include #elif defined(__linux__) && !defined(__ANDROID__) #include #elif defined(_WIN32) #include #endif using android::base::ErrnoRestorer; #define LOG_BUF_SIZE 1024 #if defined(__ANDROID__) static int check_log_uid_permissions() { uid_t uid = getuid(); /* Matches clientHasLogCredentials() in logd */ if ((uid != AID_SYSTEM) && (uid != AID_ROOT) && (uid != AID_LOG)) { uid = geteuid(); if ((uid != AID_SYSTEM) && (uid != AID_ROOT) && (uid != AID_LOG)) { gid_t gid = getgid(); if ((gid != AID_SYSTEM) && (gid != AID_ROOT) && (gid != AID_LOG)) { gid = getegid(); if ((gid != AID_SYSTEM) && (gid != AID_ROOT) && (gid != AID_LOG)) { int num_groups; gid_t* groups; num_groups = getgroups(0, NULL); if (num_groups <= 0) { return -EPERM; } groups = static_cast(calloc(num_groups, sizeof(gid_t))); if (!groups) { return -ENOMEM; } num_groups = getgroups(num_groups, groups); while (num_groups > 0) { if (groups[num_groups - 1] == AID_LOG) { break; } --num_groups; } free(groups); if (num_groups <= 0) { return -EPERM; } } } } } return 0; } #endif /* * Release any logger resources. A new log write will immediately re-acquire. */ void __android_log_close() { #ifdef __ANDROID__ LogdClose(); PmsgClose(); #endif } #if defined(__GLIBC__) || defined(_WIN32) static const char* getprogname() { #if defined(__GLIBC__) return program_invocation_short_name; #elif defined(_WIN32) static bool first = true; static char progname[MAX_PATH] = {}; if (first) { char path[PATH_MAX + 1]; DWORD result = GetModuleFileName(nullptr, path, sizeof(path) - 1); if (result == 0 || result == sizeof(path) - 1) return ""; path[PATH_MAX - 1] = 0; char* path_basename = basename(path); snprintf(progname, sizeof(progname), "%s", path_basename); first = false; } return progname; #endif } #endif // It's possible for logging to happen during static initialization before our globals are // initialized, so we place this std::string in a function such that it is initialized on the first // call. std::string& GetDefaultTag() { static std::string default_tag = getprogname(); return default_tag; } void __android_log_set_default_tag(const char* tag) { GetDefaultTag().assign(tag, 0, LOGGER_ENTRY_MAX_PAYLOAD); } static std::atomic_int32_t minimum_log_priority = ANDROID_LOG_DEFAULT; int32_t __android_log_set_minimum_priority(int32_t priority) { return minimum_log_priority.exchange(priority, std::memory_order_relaxed); } int32_t __android_log_get_minimum_priority() { return minimum_log_priority; } #ifdef __ANDROID__ static __android_logger_function logger_function = __android_log_logd_logger; #else static __android_logger_function logger_function = __android_log_stderr_logger; #endif void __android_log_set_logger(__android_logger_function logger) { logger_function = logger; } void __android_log_default_aborter(const char* abort_message) { #ifdef __ANDROID__ android_set_abort_message(abort_message); #else UNUSED(abort_message); #endif abort(); } static __android_aborter_function aborter_function = __android_log_default_aborter; void __android_log_set_aborter(__android_aborter_function aborter) { aborter_function = aborter; } void __android_log_call_aborter(const char* abort_message) { aborter_function(abort_message); } #ifdef __ANDROID__ static int write_to_log(log_id_t log_id, struct iovec* vec, size_t nr) { int ret; struct timespec ts; if (log_id == LOG_ID_KERNEL) { return -EINVAL; } clock_gettime(android_log_clockid(), &ts); if (log_id == LOG_ID_SECURITY) { if (vec[0].iov_len < 4) { return -EINVAL; } ret = check_log_uid_permissions(); if (ret < 0) { return ret; } if (!__android_log_security()) { /* If only we could reset downstream logd counter */ return -EPERM; } } else if (log_id == LOG_ID_EVENTS || log_id == LOG_ID_STATS) { if (vec[0].iov_len < 4) { return -EINVAL; } } ret = LogdWrite(log_id, &ts, vec, nr); PmsgWrite(log_id, &ts, vec, nr); return ret; } #else static int write_to_log(log_id_t, struct iovec*, size_t) { // Non-Android text logs should go to __android_log_stderr_logger, not here. // Non-Android binary logs are always dropped. return 1; } #endif // Copied from base/threads.cpp static uint64_t GetThreadId() { #if defined(__BIONIC__) return gettid(); #elif defined(__APPLE__) uint64_t tid; pthread_threadid_np(NULL, &tid); return tid; #elif defined(__linux__) return syscall(__NR_gettid); #elif defined(_WIN32) return GetCurrentThreadId(); #endif } void __android_log_stderr_logger(const struct __android_log_message* log_message) { struct tm now; time_t t = time(nullptr); #if defined(_WIN32) localtime_s(&now, &t); #else localtime_r(&t, &now); #endif char timestamp[32]; strftime(timestamp, sizeof(timestamp), "%m-%d %H:%M:%S", &now); static const char log_characters[] = "XXVDIWEF"; static_assert(arraysize(log_characters) - 1 == ANDROID_LOG_SILENT, "Mismatch in size of log_characters and values in android_LogPriority"); int32_t priority = log_message->priority > ANDROID_LOG_SILENT ? ANDROID_LOG_FATAL : log_message->priority; char priority_char = log_characters[priority]; uint64_t tid = GetThreadId(); if (log_message->file != nullptr) { fprintf(stderr, "%s %c %s %5d %5" PRIu64 " %s:%u] %s\n", log_message->tag ? log_message->tag : "nullptr", priority_char, timestamp, getpid(), tid, log_message->file, log_message->line, log_message->message); } else { fprintf(stderr, "%s %c %s %5d %5" PRIu64 " %s\n", log_message->tag ? log_message->tag : "nullptr", priority_char, timestamp, getpid(), tid, log_message->message); } } void __android_log_logd_logger(const struct __android_log_message* log_message) { int buffer_id = log_message->buffer_id == LOG_ID_DEFAULT ? LOG_ID_MAIN : log_message->buffer_id; struct iovec vec[3]; vec[0].iov_base = const_cast(reinterpret_cast(&log_message->priority)); vec[0].iov_len = 1; vec[1].iov_base = const_cast(static_cast(log_message->tag)); vec[1].iov_len = strlen(log_message->tag) + 1; vec[2].iov_base = const_cast(static_cast(log_message->message)); vec[2].iov_len = strlen(log_message->message) + 1; write_to_log(static_cast(buffer_id), vec, 3); } int __android_log_write(int prio, const char* tag, const char* msg) { return __android_log_buf_write(LOG_ID_MAIN, prio, tag, msg); } void __android_log_write_log_message(__android_log_message* log_message) { ErrnoRestorer errno_restorer; if (log_message->buffer_id != LOG_ID_DEFAULT && log_message->buffer_id != LOG_ID_MAIN && log_message->buffer_id != LOG_ID_SYSTEM && log_message->buffer_id != LOG_ID_RADIO && log_message->buffer_id != LOG_ID_CRASH) { return; } if (log_message->tag == nullptr) { log_message->tag = GetDefaultTag().c_str(); } #if __BIONIC__ if (log_message->priority == ANDROID_LOG_FATAL) { android_set_abort_message(log_message->message); } #endif logger_function(log_message); } int __android_log_buf_write(int bufID, int prio, const char* tag, const char* msg) { ErrnoRestorer errno_restorer; if (!__android_log_is_loggable(prio, tag, ANDROID_LOG_VERBOSE)) { return -EPERM; } __android_log_message log_message = { sizeof(__android_log_message), bufID, prio, tag, nullptr, 0, msg}; __android_log_write_log_message(&log_message); return 1; } int __android_log_vprint(int prio, const char* tag, const char* fmt, va_list ap) { ErrnoRestorer errno_restorer; if (!__android_log_is_loggable(prio, tag, ANDROID_LOG_VERBOSE)) { return -EPERM; } char buf[LOG_BUF_SIZE]; vsnprintf(buf, LOG_BUF_SIZE, fmt, ap); __android_log_message log_message = { sizeof(__android_log_message), LOG_ID_MAIN, prio, tag, nullptr, 0, buf}; __android_log_write_log_message(&log_message); return 1; } int __android_log_print(int prio, const char* tag, const char* fmt, ...) { ErrnoRestorer errno_restorer; if (!__android_log_is_loggable(prio, tag, ANDROID_LOG_VERBOSE)) { return -EPERM; } va_list ap; char buf[LOG_BUF_SIZE]; va_start(ap, fmt); vsnprintf(buf, LOG_BUF_SIZE, fmt, ap); va_end(ap); __android_log_message log_message = { sizeof(__android_log_message), LOG_ID_MAIN, prio, tag, nullptr, 0, buf}; __android_log_write_log_message(&log_message); return 1; } int __android_log_buf_print(int bufID, int prio, const char* tag, const char* fmt, ...) { ErrnoRestorer errno_restorer; if (!__android_log_is_loggable(prio, tag, ANDROID_LOG_VERBOSE)) { return -EPERM; } va_list ap; char buf[LOG_BUF_SIZE]; va_start(ap, fmt); vsnprintf(buf, LOG_BUF_SIZE, fmt, ap); va_end(ap); __android_log_message log_message = { sizeof(__android_log_message), bufID, prio, tag, nullptr, 0, buf}; __android_log_write_log_message(&log_message); return 1; } void __android_log_assert(const char* cond, const char* tag, const char* fmt, ...) { char buf[LOG_BUF_SIZE]; if (fmt) { va_list ap; va_start(ap, fmt); vsnprintf(buf, LOG_BUF_SIZE, fmt, ap); va_end(ap); } else { /* Msg not provided, log condition. N.B. Do not use cond directly as * format string as it could contain spurious '%' syntax (e.g. * "%d" in "blocks%devs == 0"). */ if (cond) snprintf(buf, LOG_BUF_SIZE, "Assertion failed: %s", cond); else strcpy(buf, "Unspecified assertion failed"); } // Log assertion failures to stderr for the benefit of "adb shell" users // and gtests (http://b/23675822). TEMP_FAILURE_RETRY(write(2, buf, strlen(buf))); TEMP_FAILURE_RETRY(write(2, "\n", 1)); __android_log_write(ANDROID_LOG_FATAL, tag, buf); __android_log_call_aborter(buf); abort(); } int __android_log_bwrite(int32_t tag, const void* payload, size_t len) { ErrnoRestorer errno_restorer; struct iovec vec[2]; vec[0].iov_base = &tag; vec[0].iov_len = sizeof(tag); vec[1].iov_base = (void*)payload; vec[1].iov_len = len; return write_to_log(LOG_ID_EVENTS, vec, 2); } int __android_log_stats_bwrite(int32_t tag, const void* payload, size_t len) { ErrnoRestorer errno_restorer; struct iovec vec[2]; vec[0].iov_base = &tag; vec[0].iov_len = sizeof(tag); vec[1].iov_base = (void*)payload; vec[1].iov_len = len; return write_to_log(LOG_ID_STATS, vec, 2); } int __android_log_security_bwrite(int32_t tag, const void* payload, size_t len) { ErrnoRestorer errno_restorer; struct iovec vec[2]; vec[0].iov_base = &tag; vec[0].iov_len = sizeof(tag); vec[1].iov_base = (void*)payload; vec[1].iov_len = len; return write_to_log(LOG_ID_SECURITY, vec, 2); } /* * Like __android_log_bwrite, but takes the type as well. Doesn't work * for the general case where we're generating lists of stuff, but very * handy if we just want to dump an integer into the log. */ int __android_log_btwrite(int32_t tag, char type, const void* payload, size_t len) { ErrnoRestorer errno_restorer; struct iovec vec[3]; vec[0].iov_base = &tag; vec[0].iov_len = sizeof(tag); vec[1].iov_base = &type; vec[1].iov_len = sizeof(type); vec[2].iov_base = (void*)payload; vec[2].iov_len = len; return write_to_log(LOG_ID_EVENTS, vec, 3); } /* * Like __android_log_bwrite, but used for writing strings to the * event log. */ int __android_log_bswrite(int32_t tag, const char* payload) { ErrnoRestorer errno_restorer; struct iovec vec[4]; char type = EVENT_TYPE_STRING; uint32_t len = strlen(payload); vec[0].iov_base = &tag; vec[0].iov_len = sizeof(tag); vec[1].iov_base = &type; vec[1].iov_len = sizeof(type); vec[2].iov_base = &len; vec[2].iov_len = sizeof(len); vec[3].iov_base = (void*)payload; vec[3].iov_len = len; return write_to_log(LOG_ID_EVENTS, vec, 4); } /* * Like __android_log_security_bwrite, but used for writing strings to the * security log. */ int __android_log_security_bswrite(int32_t tag, const char* payload) { ErrnoRestorer errno_restorer; struct iovec vec[4]; char type = EVENT_TYPE_STRING; uint32_t len = strlen(payload); vec[0].iov_base = &tag; vec[0].iov_len = sizeof(tag); vec[1].iov_base = &type; vec[1].iov_len = sizeof(type); vec[2].iov_base = &len; vec[2].iov_len = sizeof(len); vec[3].iov_base = (void*)payload; vec[3].iov_len = len; return write_to_log(LOG_ID_SECURITY, vec, 4); }