blob: e76673f74846885eafa04a3f356a70ec0e05cd93 [file] [log] [blame]
Mark Salyzyn4d99c982017-02-28 12:59:01 -08001/*
2 * Copyright (C) 2017 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17/*
18 * stderr write handler. Output is logcat-like, and responds to
19 * logcat's environment variables ANDROID_PRINTF_LOG and
20 * ANDROID_LOG_TAGS to filter output.
21 *
22 * This transport only provides a writer, that means that it does not
23 * provide an End-To-End capability as the logs are effectively _lost_
24 * to the stderr file stream. The purpose of this transport is to
25 * supply a means for command line tools to report their logging
26 * to the stderr stream, in line with all other activities.
27 */
28
29#include <errno.h>
Mark Salyzyn4d99c982017-02-28 12:59:01 -080030#include <stdio.h>
31#include <stdlib.h>
32#include <string.h>
33#include <sys/types.h>
34#include <unistd.h>
35
36#include <log/event_tag_map.h>
37#include <log/log.h>
38#include <log/logprint.h>
Mark Salyzyn4d99c982017-02-28 12:59:01 -080039
40#include "log_portability.h"
41#include "logger.h"
Tom Cherry6f6ef392019-01-16 14:17:08 -080042#include "uio.h"
Mark Salyzyn4d99c982017-02-28 12:59:01 -080043
44static int stderrOpen();
45static void stderrClose();
46static int stderrAvailable(log_id_t logId);
Tom Cherry71ba1642019-01-10 10:37:36 -080047static int stderrWrite(log_id_t logId, struct timespec* ts, struct iovec* vec, size_t nr);
Mark Salyzyn4d99c982017-02-28 12:59:01 -080048
49struct stderrContext {
Mark Salyzyn2ed51d72017-03-09 08:09:43 -080050 AndroidLogFormat* logformat;
Mark Salyzyn4d99c982017-02-28 12:59:01 -080051#if defined(__ANDROID__)
Mark Salyzyn2ed51d72017-03-09 08:09:43 -080052 EventTagMap* eventTagMap;
Mark Salyzyn4d99c982017-02-28 12:59:01 -080053#endif
54};
55
Tom Cherry2d9779e2019-02-08 11:46:19 -080056struct android_log_transport_write stderrLoggerWrite = {
Tom Cherry71ba1642019-01-10 10:37:36 -080057 .node = {&stderrLoggerWrite.node, &stderrLoggerWrite.node},
58 .context.priv = NULL,
59 .name = "stderr",
60 .available = stderrAvailable,
61 .open = stderrOpen,
62 .close = stderrClose,
63 .write = stderrWrite,
Mark Salyzyn4d99c982017-02-28 12:59:01 -080064};
65
Mark Salyzyn2ed51d72017-03-09 08:09:43 -080066static int stderrOpen() {
67 struct stderrContext* ctx;
68 const char* envStr;
69 bool setFormat;
Mark Salyzyn4d99c982017-02-28 12:59:01 -080070
Mark Salyzyn2ed51d72017-03-09 08:09:43 -080071 if (!stderr || (fileno(stderr) < 0)) {
72 return -EBADF;
73 }
Mark Salyzyn4d99c982017-02-28 12:59:01 -080074
Mark Salyzyn04bbc8e2017-03-08 15:03:20 -080075 if (stderrLoggerWrite.context.priv) {
Mark Salyzyn4d99c982017-02-28 12:59:01 -080076 return fileno(stderr);
Mark Salyzyn2ed51d72017-03-09 08:09:43 -080077 }
78
Tom Cherry71ba1642019-01-10 10:37:36 -080079 ctx = static_cast<stderrContext*>(calloc(1, sizeof(stderrContext)));
Mark Salyzyn2ed51d72017-03-09 08:09:43 -080080 if (!ctx) {
81 return -ENOMEM;
82 }
83
84 ctx->logformat = android_log_format_new();
85 if (!ctx->logformat) {
86 free(ctx);
87 return -ENOMEM;
88 }
89
90 envStr = getenv("ANDROID_PRINTF_LOG");
91 setFormat = false;
92
93 if (envStr) {
94 char* formats = strdup(envStr);
95 char* sv = NULL;
96 char* arg = formats;
97 while (!!(arg = strtok_r(arg, ",:; \t\n\r\f", &sv))) {
98 AndroidLogPrintFormat format = android_log_formatFromString(arg);
99 arg = NULL;
100 if (format == FORMAT_OFF) {
101 continue;
102 }
103 if (android_log_setPrintFormat(ctx->logformat, format) <= 0) {
104 continue;
105 }
106 setFormat = true;
107 }
108 free(formats);
109 }
110 if (!setFormat) {
111 AndroidLogPrintFormat format = android_log_formatFromString("threadtime");
112 android_log_setPrintFormat(ctx->logformat, format);
113 }
114 envStr = getenv("ANDROID_LOG_TAGS");
115 if (envStr) {
116 android_log_addFilterString(ctx->logformat, envStr);
117 }
Mark Salyzyn04bbc8e2017-03-08 15:03:20 -0800118 stderrLoggerWrite.context.priv = ctx;
Mark Salyzyn2ed51d72017-03-09 08:09:43 -0800119
120 return fileno(stderr);
Mark Salyzyn4d99c982017-02-28 12:59:01 -0800121}
122
Mark Salyzyn2ed51d72017-03-09 08:09:43 -0800123static void stderrClose() {
Tom Cherry71ba1642019-01-10 10:37:36 -0800124 stderrContext* ctx = static_cast<stderrContext*>(stderrLoggerWrite.context.priv);
Mark Salyzyn4d99c982017-02-28 12:59:01 -0800125
Mark Salyzyn2ed51d72017-03-09 08:09:43 -0800126 if (ctx) {
Mark Salyzyn04bbc8e2017-03-08 15:03:20 -0800127 stderrLoggerWrite.context.priv = NULL;
Mark Salyzyn2ed51d72017-03-09 08:09:43 -0800128 if (ctx->logformat) {
129 android_log_format_free(ctx->logformat);
130 ctx->logformat = NULL;
131 }
Mark Salyzyn4d99c982017-02-28 12:59:01 -0800132#if defined(__ANDROID__)
Mark Salyzyn2ed51d72017-03-09 08:09:43 -0800133 if (ctx->eventTagMap) {
134 android_closeEventTagMap(ctx->eventTagMap);
135 ctx->eventTagMap = NULL;
136 }
Mark Salyzyn4d99c982017-02-28 12:59:01 -0800137#endif
Mark Salyzyn2ed51d72017-03-09 08:09:43 -0800138 }
Mark Salyzyn4d99c982017-02-28 12:59:01 -0800139}
140
Mark Salyzyn2ed51d72017-03-09 08:09:43 -0800141static int stderrAvailable(log_id_t logId) {
142 if ((logId >= LOG_ID_MAX) || (logId == LOG_ID_KERNEL)) {
143 return -EINVAL;
144 }
145 return 1;
Mark Salyzyn4d99c982017-02-28 12:59:01 -0800146}
147
Tom Cherry71ba1642019-01-10 10:37:36 -0800148static int stderrWrite(log_id_t logId, struct timespec* ts, struct iovec* vec, size_t nr) {
Mark Salyzyn2ed51d72017-03-09 08:09:43 -0800149 struct log_msg log_msg;
150 AndroidLogEntry entry;
151 char binaryMsgBuf[1024];
152 int err;
153 size_t i;
Tom Cherry71ba1642019-01-10 10:37:36 -0800154 stderrContext* ctx = static_cast<stderrContext*>(stderrLoggerWrite.context.priv);
Mark Salyzyn4d99c982017-02-28 12:59:01 -0800155
Mark Salyzyn2ed51d72017-03-09 08:09:43 -0800156 if (!ctx) return -EBADF;
157 if (!vec || !nr) return -EINVAL;
Mark Salyzyn4d99c982017-02-28 12:59:01 -0800158
Mark Salyzyn2ed51d72017-03-09 08:09:43 -0800159 log_msg.entry.len = 0;
160 log_msg.entry.hdr_size = sizeof(log_msg.entry);
161 log_msg.entry.pid = getpid();
Mark Salyzyn4d99c982017-02-28 12:59:01 -0800162#ifdef __BIONIC__
Mark Salyzyn2ed51d72017-03-09 08:09:43 -0800163 log_msg.entry.tid = gettid();
Mark Salyzyn4d99c982017-02-28 12:59:01 -0800164#else
Mark Salyzyn2ed51d72017-03-09 08:09:43 -0800165 log_msg.entry.tid = getpid();
Mark Salyzyn4d99c982017-02-28 12:59:01 -0800166#endif
Mark Salyzyn2ed51d72017-03-09 08:09:43 -0800167 log_msg.entry.sec = ts->tv_sec;
168 log_msg.entry.nsec = ts->tv_nsec;
169 log_msg.entry.lid = logId;
170 log_msg.entry.uid = __android_log_uid();
Mark Salyzyn4d99c982017-02-28 12:59:01 -0800171
Mark Salyzyn2ed51d72017-03-09 08:09:43 -0800172 for (i = 0; i < nr; ++i) {
173 size_t len = vec[i].iov_len;
174 if ((log_msg.entry.len + len) > LOGGER_ENTRY_MAX_PAYLOAD) {
175 len = LOGGER_ENTRY_MAX_PAYLOAD - log_msg.entry.len;
Mark Salyzyn4d99c982017-02-28 12:59:01 -0800176 }
Mark Salyzyn2ed51d72017-03-09 08:09:43 -0800177 if (!len) continue;
178 memcpy(log_msg.entry.msg + log_msg.entry.len, vec[i].iov_base, len);
179 log_msg.entry.len += len;
180 }
Mark Salyzyn4d99c982017-02-28 12:59:01 -0800181
Mark Salyzyn2ed51d72017-03-09 08:09:43 -0800182 if ((logId == LOG_ID_EVENTS) || (logId == LOG_ID_SECURITY)) {
Mark Salyzyn4d99c982017-02-28 12:59:01 -0800183#if defined(__ANDROID__)
Mark Salyzyn2ed51d72017-03-09 08:09:43 -0800184 if (!ctx->eventTagMap) {
185 ctx->eventTagMap = android_openEventTagMap(NULL);
186 }
Mark Salyzyn4d99c982017-02-28 12:59:01 -0800187#endif
Mark Salyzyn2ed51d72017-03-09 08:09:43 -0800188 err = android_log_processBinaryLogBuffer(&log_msg.entry_v1, &entry,
Mark Salyzyn4d99c982017-02-28 12:59:01 -0800189#if defined(__ANDROID__)
Mark Salyzyn2ed51d72017-03-09 08:09:43 -0800190 ctx->eventTagMap,
Mark Salyzyn4d99c982017-02-28 12:59:01 -0800191#else
Mark Salyzyn2ed51d72017-03-09 08:09:43 -0800192 NULL,
Mark Salyzyn4d99c982017-02-28 12:59:01 -0800193#endif
Mark Salyzyn2ed51d72017-03-09 08:09:43 -0800194 binaryMsgBuf, sizeof(binaryMsgBuf));
195 } else {
196 err = android_log_processLogBuffer(&log_msg.entry_v1, &entry);
197 }
Mark Salyzyn4d99c982017-02-28 12:59:01 -0800198
Mark Salyzyn2ed51d72017-03-09 08:09:43 -0800199 /* print known truncated data, in essence logcat --debug */
200 if ((err < 0) && !entry.message) return -EINVAL;
Mark Salyzyn4d99c982017-02-28 12:59:01 -0800201
Mark Salyzyn2ed51d72017-03-09 08:09:43 -0800202 if (!android_log_shouldPrintLine(ctx->logformat, entry.tag, entry.priority)) {
Mark Salyzyn4d99c982017-02-28 12:59:01 -0800203 return log_msg.entry.len;
Mark Salyzyn2ed51d72017-03-09 08:09:43 -0800204 }
205
206 err = android_log_printLogLine(ctx->logformat, fileno(stderr), &entry);
207 if (err < 0) return errno ? -errno : -EINVAL;
208 return log_msg.entry.len;
Mark Salyzyn4d99c982017-02-28 12:59:01 -0800209}