1*598139dcSAndroid Build Coastguard Worker /*
2*598139dcSAndroid Build Coastguard Worker * Copyright (C) 2014 The Android Open Source Project
3*598139dcSAndroid Build Coastguard Worker *
4*598139dcSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*598139dcSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*598139dcSAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*598139dcSAndroid Build Coastguard Worker *
8*598139dcSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*598139dcSAndroid Build Coastguard Worker *
10*598139dcSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*598139dcSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*598139dcSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*598139dcSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*598139dcSAndroid Build Coastguard Worker * limitations under the License.
15*598139dcSAndroid Build Coastguard Worker */
16*598139dcSAndroid Build Coastguard Worker
17*598139dcSAndroid Build Coastguard Worker #include "LogKlog.h"
18*598139dcSAndroid Build Coastguard Worker
19*598139dcSAndroid Build Coastguard Worker #include <ctype.h>
20*598139dcSAndroid Build Coastguard Worker #include <errno.h>
21*598139dcSAndroid Build Coastguard Worker #include <inttypes.h>
22*598139dcSAndroid Build Coastguard Worker #include <limits.h>
23*598139dcSAndroid Build Coastguard Worker #include <stdarg.h>
24*598139dcSAndroid Build Coastguard Worker #include <stdlib.h>
25*598139dcSAndroid Build Coastguard Worker #include <string.h>
26*598139dcSAndroid Build Coastguard Worker #include <sys/prctl.h>
27*598139dcSAndroid Build Coastguard Worker #include <sys/uio.h>
28*598139dcSAndroid Build Coastguard Worker #include <syslog.h>
29*598139dcSAndroid Build Coastguard Worker
30*598139dcSAndroid Build Coastguard Worker #include <private/android_filesystem_config.h>
31*598139dcSAndroid Build Coastguard Worker #include <private/android_logger.h>
32*598139dcSAndroid Build Coastguard Worker
33*598139dcSAndroid Build Coastguard Worker #include "LogBuffer.h"
34*598139dcSAndroid Build Coastguard Worker
35*598139dcSAndroid Build Coastguard Worker #define KMSG_PRIORITY(PRI) \
36*598139dcSAndroid Build Coastguard Worker '<', '0' + (LOG_SYSLOG | (PRI)) / 10, '0' + (LOG_SYSLOG | (PRI)) % 10, '>'
37*598139dcSAndroid Build Coastguard Worker
38*598139dcSAndroid Build Coastguard Worker static const char priority_message[] = { KMSG_PRIORITY(LOG_INFO), '\0' };
39*598139dcSAndroid Build Coastguard Worker
40*598139dcSAndroid Build Coastguard Worker // List of the _only_ needles we supply here to android::strnstr
41*598139dcSAndroid Build Coastguard Worker static const char suspendStr[] = "PM: suspend entry ";
42*598139dcSAndroid Build Coastguard Worker static const char resumeStr[] = "PM: suspend exit ";
43*598139dcSAndroid Build Coastguard Worker static const char suspendedStr[] = "suspended for ";
44*598139dcSAndroid Build Coastguard Worker static const char auditStr[] = " audit(";
45*598139dcSAndroid Build Coastguard Worker static const char klogdStr[] = "logd.klogd: ";
46*598139dcSAndroid Build Coastguard Worker
47*598139dcSAndroid Build Coastguard Worker // Parsing is hard
48*598139dcSAndroid Build Coastguard Worker
49*598139dcSAndroid Build Coastguard Worker // called if we see a '<', s is the next character, returns pointer after '>'
is_prio(char * s,ssize_t len)50*598139dcSAndroid Build Coastguard Worker static char* is_prio(char* s, ssize_t len) {
51*598139dcSAndroid Build Coastguard Worker if ((len <= 0) || !isdigit(*s++)) return nullptr;
52*598139dcSAndroid Build Coastguard Worker --len;
53*598139dcSAndroid Build Coastguard Worker static const size_t max_prio_len = (len < 4) ? len : 4;
54*598139dcSAndroid Build Coastguard Worker size_t priolen = 0;
55*598139dcSAndroid Build Coastguard Worker char c;
56*598139dcSAndroid Build Coastguard Worker while (((c = *s++)) && (++priolen <= max_prio_len)) {
57*598139dcSAndroid Build Coastguard Worker if (!isdigit(c)) return ((c == '>') && (*s == '[')) ? s : nullptr;
58*598139dcSAndroid Build Coastguard Worker }
59*598139dcSAndroid Build Coastguard Worker return nullptr;
60*598139dcSAndroid Build Coastguard Worker }
61*598139dcSAndroid Build Coastguard Worker
62*598139dcSAndroid Build Coastguard Worker // called if we see a '[', s is the next character, returns pointer after ']'
is_timestamp(char * s,ssize_t len)63*598139dcSAndroid Build Coastguard Worker static char* is_timestamp(char* s, ssize_t len) {
64*598139dcSAndroid Build Coastguard Worker while ((len > 0) && (*s == ' ')) {
65*598139dcSAndroid Build Coastguard Worker ++s;
66*598139dcSAndroid Build Coastguard Worker --len;
67*598139dcSAndroid Build Coastguard Worker }
68*598139dcSAndroid Build Coastguard Worker if ((len <= 0) || !isdigit(*s++)) return nullptr;
69*598139dcSAndroid Build Coastguard Worker --len;
70*598139dcSAndroid Build Coastguard Worker bool first_period = true;
71*598139dcSAndroid Build Coastguard Worker char c;
72*598139dcSAndroid Build Coastguard Worker while ((len > 0) && ((c = *s++))) {
73*598139dcSAndroid Build Coastguard Worker --len;
74*598139dcSAndroid Build Coastguard Worker if ((c == '.') && first_period) {
75*598139dcSAndroid Build Coastguard Worker first_period = false;
76*598139dcSAndroid Build Coastguard Worker } else if (!isdigit(c)) {
77*598139dcSAndroid Build Coastguard Worker return ((c == ']') && !first_period && (*s == ' ')) ? s : nullptr;
78*598139dcSAndroid Build Coastguard Worker }
79*598139dcSAndroid Build Coastguard Worker }
80*598139dcSAndroid Build Coastguard Worker return nullptr;
81*598139dcSAndroid Build Coastguard Worker }
82*598139dcSAndroid Build Coastguard Worker
83*598139dcSAndroid Build Coastguard Worker // Like strtok_r with "\r\n" except that we look for log signatures (regex)
84*598139dcSAndroid Build Coastguard Worker // \(\(<[0-9]\{1,4\}>\)\([[] *[0-9]+[.][0-9]+[]] \)\{0,1\}\|[[]
85*598139dcSAndroid Build Coastguard Worker // *[0-9]+[.][0-9]+[]] \)
86*598139dcSAndroid Build Coastguard Worker // and split if we see a second one without a newline.
87*598139dcSAndroid Build Coastguard Worker // We allow nuls in content, monitoring the overall length and sub-length of
88*598139dcSAndroid Build Coastguard Worker // the discovered tokens.
89*598139dcSAndroid Build Coastguard Worker
90*598139dcSAndroid Build Coastguard Worker #define SIGNATURE_MASK 0xF0
91*598139dcSAndroid Build Coastguard Worker // <digit> following ('0' to '9' masked with ~SIGNATURE_MASK) added to signature
92*598139dcSAndroid Build Coastguard Worker #define LESS_THAN_SIG SIGNATURE_MASK
93*598139dcSAndroid Build Coastguard Worker #define OPEN_BRACKET_SIG ((SIGNATURE_MASK << 1) & SIGNATURE_MASK)
94*598139dcSAndroid Build Coastguard Worker // space is one more than <digit> of 9
95*598139dcSAndroid Build Coastguard Worker #define OPEN_BRACKET_SPACE ((char)(OPEN_BRACKET_SIG | 10))
96*598139dcSAndroid Build Coastguard Worker
log_strntok_r(char * s,ssize_t & len,char * & last,ssize_t & sublen)97*598139dcSAndroid Build Coastguard Worker char* android::log_strntok_r(char* s, ssize_t& len, char*& last,
98*598139dcSAndroid Build Coastguard Worker ssize_t& sublen) {
99*598139dcSAndroid Build Coastguard Worker sublen = 0;
100*598139dcSAndroid Build Coastguard Worker if (len <= 0) return nullptr;
101*598139dcSAndroid Build Coastguard Worker if (!s) {
102*598139dcSAndroid Build Coastguard Worker if (!(s = last)) return nullptr;
103*598139dcSAndroid Build Coastguard Worker // fixup for log signature split <,
104*598139dcSAndroid Build Coastguard Worker // LESS_THAN_SIG + <digit>
105*598139dcSAndroid Build Coastguard Worker if ((*s & SIGNATURE_MASK) == LESS_THAN_SIG) {
106*598139dcSAndroid Build Coastguard Worker *s = (*s & ~SIGNATURE_MASK) + '0';
107*598139dcSAndroid Build Coastguard Worker *--s = '<';
108*598139dcSAndroid Build Coastguard Worker ++len;
109*598139dcSAndroid Build Coastguard Worker }
110*598139dcSAndroid Build Coastguard Worker // fixup for log signature split [,
111*598139dcSAndroid Build Coastguard Worker // OPEN_BRACKET_SPACE is space, OPEN_BRACKET_SIG + <digit>
112*598139dcSAndroid Build Coastguard Worker if ((*s & SIGNATURE_MASK) == OPEN_BRACKET_SIG) {
113*598139dcSAndroid Build Coastguard Worker *s = (*s == OPEN_BRACKET_SPACE) ? ' ' : (*s & ~SIGNATURE_MASK) + '0';
114*598139dcSAndroid Build Coastguard Worker *--s = '[';
115*598139dcSAndroid Build Coastguard Worker ++len;
116*598139dcSAndroid Build Coastguard Worker }
117*598139dcSAndroid Build Coastguard Worker }
118*598139dcSAndroid Build Coastguard Worker
119*598139dcSAndroid Build Coastguard Worker while ((len > 0) && ((*s == '\r') || (*s == '\n'))) {
120*598139dcSAndroid Build Coastguard Worker ++s;
121*598139dcSAndroid Build Coastguard Worker --len;
122*598139dcSAndroid Build Coastguard Worker }
123*598139dcSAndroid Build Coastguard Worker
124*598139dcSAndroid Build Coastguard Worker if (len <= 0) return last = nullptr;
125*598139dcSAndroid Build Coastguard Worker char *peek, *tok = s;
126*598139dcSAndroid Build Coastguard Worker
127*598139dcSAndroid Build Coastguard Worker for (;;) {
128*598139dcSAndroid Build Coastguard Worker if (len <= 0) {
129*598139dcSAndroid Build Coastguard Worker last = nullptr;
130*598139dcSAndroid Build Coastguard Worker return tok;
131*598139dcSAndroid Build Coastguard Worker }
132*598139dcSAndroid Build Coastguard Worker char c = *s++;
133*598139dcSAndroid Build Coastguard Worker --len;
134*598139dcSAndroid Build Coastguard Worker ssize_t adjust;
135*598139dcSAndroid Build Coastguard Worker switch (c) {
136*598139dcSAndroid Build Coastguard Worker case '\r':
137*598139dcSAndroid Build Coastguard Worker case '\n':
138*598139dcSAndroid Build Coastguard Worker s[-1] = '\0';
139*598139dcSAndroid Build Coastguard Worker last = s;
140*598139dcSAndroid Build Coastguard Worker return tok;
141*598139dcSAndroid Build Coastguard Worker
142*598139dcSAndroid Build Coastguard Worker case '<':
143*598139dcSAndroid Build Coastguard Worker peek = is_prio(s, len);
144*598139dcSAndroid Build Coastguard Worker if (!peek) break;
145*598139dcSAndroid Build Coastguard Worker if (s != (tok + 1)) { // not first?
146*598139dcSAndroid Build Coastguard Worker s[-1] = '\0';
147*598139dcSAndroid Build Coastguard Worker *s &= ~SIGNATURE_MASK;
148*598139dcSAndroid Build Coastguard Worker *s |= LESS_THAN_SIG; // signature for '<'
149*598139dcSAndroid Build Coastguard Worker last = s;
150*598139dcSAndroid Build Coastguard Worker return tok;
151*598139dcSAndroid Build Coastguard Worker }
152*598139dcSAndroid Build Coastguard Worker adjust = peek - s;
153*598139dcSAndroid Build Coastguard Worker if (adjust > len) {
154*598139dcSAndroid Build Coastguard Worker adjust = len;
155*598139dcSAndroid Build Coastguard Worker }
156*598139dcSAndroid Build Coastguard Worker sublen += adjust;
157*598139dcSAndroid Build Coastguard Worker len -= adjust;
158*598139dcSAndroid Build Coastguard Worker s = peek;
159*598139dcSAndroid Build Coastguard Worker if ((*s == '[') && ((peek = is_timestamp(s + 1, len - 1)))) {
160*598139dcSAndroid Build Coastguard Worker adjust = peek - s;
161*598139dcSAndroid Build Coastguard Worker if (adjust > len) {
162*598139dcSAndroid Build Coastguard Worker adjust = len;
163*598139dcSAndroid Build Coastguard Worker }
164*598139dcSAndroid Build Coastguard Worker sublen += adjust;
165*598139dcSAndroid Build Coastguard Worker len -= adjust;
166*598139dcSAndroid Build Coastguard Worker s = peek;
167*598139dcSAndroid Build Coastguard Worker }
168*598139dcSAndroid Build Coastguard Worker break;
169*598139dcSAndroid Build Coastguard Worker
170*598139dcSAndroid Build Coastguard Worker case '[':
171*598139dcSAndroid Build Coastguard Worker peek = is_timestamp(s, len);
172*598139dcSAndroid Build Coastguard Worker if (!peek) break;
173*598139dcSAndroid Build Coastguard Worker if (s != (tok + 1)) { // not first?
174*598139dcSAndroid Build Coastguard Worker s[-1] = '\0';
175*598139dcSAndroid Build Coastguard Worker if (*s == ' ') {
176*598139dcSAndroid Build Coastguard Worker *s = OPEN_BRACKET_SPACE;
177*598139dcSAndroid Build Coastguard Worker } else {
178*598139dcSAndroid Build Coastguard Worker *s &= ~SIGNATURE_MASK;
179*598139dcSAndroid Build Coastguard Worker *s |= OPEN_BRACKET_SIG; // signature for '['
180*598139dcSAndroid Build Coastguard Worker }
181*598139dcSAndroid Build Coastguard Worker last = s;
182*598139dcSAndroid Build Coastguard Worker return tok;
183*598139dcSAndroid Build Coastguard Worker }
184*598139dcSAndroid Build Coastguard Worker adjust = peek - s;
185*598139dcSAndroid Build Coastguard Worker if (adjust > len) {
186*598139dcSAndroid Build Coastguard Worker adjust = len;
187*598139dcSAndroid Build Coastguard Worker }
188*598139dcSAndroid Build Coastguard Worker sublen += adjust;
189*598139dcSAndroid Build Coastguard Worker len -= adjust;
190*598139dcSAndroid Build Coastguard Worker s = peek;
191*598139dcSAndroid Build Coastguard Worker break;
192*598139dcSAndroid Build Coastguard Worker }
193*598139dcSAndroid Build Coastguard Worker ++sublen;
194*598139dcSAndroid Build Coastguard Worker }
195*598139dcSAndroid Build Coastguard Worker // NOTREACHED
196*598139dcSAndroid Build Coastguard Worker }
197*598139dcSAndroid Build Coastguard Worker
198*598139dcSAndroid Build Coastguard Worker log_time LogKlog::correction = (log_time(CLOCK_REALTIME) < log_time(CLOCK_MONOTONIC))
199*598139dcSAndroid Build Coastguard Worker ? log_time(log_time::EPOCH)
200*598139dcSAndroid Build Coastguard Worker : (log_time(CLOCK_REALTIME) - log_time(CLOCK_MONOTONIC));
201*598139dcSAndroid Build Coastguard Worker
LogKlog(LogBuffer * buf,int fdWrite,int fdRead,bool auditd,LogStatistics * stats)202*598139dcSAndroid Build Coastguard Worker LogKlog::LogKlog(LogBuffer* buf, int fdWrite, int fdRead, bool auditd, LogStatistics* stats)
203*598139dcSAndroid Build Coastguard Worker : SocketListener(fdRead, false),
204*598139dcSAndroid Build Coastguard Worker logbuf(buf),
205*598139dcSAndroid Build Coastguard Worker signature(CLOCK_MONOTONIC),
206*598139dcSAndroid Build Coastguard Worker initialized(false),
207*598139dcSAndroid Build Coastguard Worker auditd(auditd),
208*598139dcSAndroid Build Coastguard Worker stats_(stats) {
209*598139dcSAndroid Build Coastguard Worker static const char klogd_message[] = "%s%s%" PRIu64 "\n";
210*598139dcSAndroid Build Coastguard Worker char buffer[strlen(priority_message) + strlen(klogdStr) +
211*598139dcSAndroid Build Coastguard Worker strlen(klogd_message) + 20];
212*598139dcSAndroid Build Coastguard Worker snprintf(buffer, sizeof(buffer), klogd_message, priority_message, klogdStr,
213*598139dcSAndroid Build Coastguard Worker signature.nsec());
214*598139dcSAndroid Build Coastguard Worker write(fdWrite, buffer, strlen(buffer));
215*598139dcSAndroid Build Coastguard Worker }
216*598139dcSAndroid Build Coastguard Worker
onDataAvailable(SocketClient * cli)217*598139dcSAndroid Build Coastguard Worker bool LogKlog::onDataAvailable(SocketClient* cli) {
218*598139dcSAndroid Build Coastguard Worker if (!initialized) {
219*598139dcSAndroid Build Coastguard Worker prctl(PR_SET_NAME, "logd.klogd");
220*598139dcSAndroid Build Coastguard Worker initialized = true;
221*598139dcSAndroid Build Coastguard Worker }
222*598139dcSAndroid Build Coastguard Worker
223*598139dcSAndroid Build Coastguard Worker char buffer[LOGGER_ENTRY_MAX_PAYLOAD];
224*598139dcSAndroid Build Coastguard Worker ssize_t len = 0;
225*598139dcSAndroid Build Coastguard Worker
226*598139dcSAndroid Build Coastguard Worker for (;;) {
227*598139dcSAndroid Build Coastguard Worker ssize_t retval = 0;
228*598139dcSAndroid Build Coastguard Worker if (len < (ssize_t)(sizeof(buffer) - 1)) {
229*598139dcSAndroid Build Coastguard Worker retval =
230*598139dcSAndroid Build Coastguard Worker read(cli->getSocket(), buffer + len, sizeof(buffer) - 1 - len);
231*598139dcSAndroid Build Coastguard Worker }
232*598139dcSAndroid Build Coastguard Worker if ((retval == 0) && (len <= 0)) {
233*598139dcSAndroid Build Coastguard Worker break;
234*598139dcSAndroid Build Coastguard Worker }
235*598139dcSAndroid Build Coastguard Worker if (retval < 0) {
236*598139dcSAndroid Build Coastguard Worker return false;
237*598139dcSAndroid Build Coastguard Worker }
238*598139dcSAndroid Build Coastguard Worker len += retval;
239*598139dcSAndroid Build Coastguard Worker bool full = len == (sizeof(buffer) - 1);
240*598139dcSAndroid Build Coastguard Worker char* ep = buffer + len;
241*598139dcSAndroid Build Coastguard Worker *ep = '\0';
242*598139dcSAndroid Build Coastguard Worker ssize_t sublen;
243*598139dcSAndroid Build Coastguard Worker for (char *ptr = nullptr, *tok = buffer;
244*598139dcSAndroid Build Coastguard Worker !!(tok = android::log_strntok_r(tok, len, ptr, sublen));
245*598139dcSAndroid Build Coastguard Worker tok = nullptr) {
246*598139dcSAndroid Build Coastguard Worker if (((tok + sublen) >= ep) && (retval != 0) && full) {
247*598139dcSAndroid Build Coastguard Worker if (sublen > 0) memmove(buffer, tok, sublen);
248*598139dcSAndroid Build Coastguard Worker len = sublen;
249*598139dcSAndroid Build Coastguard Worker break;
250*598139dcSAndroid Build Coastguard Worker }
251*598139dcSAndroid Build Coastguard Worker if ((sublen > 0) && *tok) {
252*598139dcSAndroid Build Coastguard Worker log(tok, sublen);
253*598139dcSAndroid Build Coastguard Worker }
254*598139dcSAndroid Build Coastguard Worker }
255*598139dcSAndroid Build Coastguard Worker }
256*598139dcSAndroid Build Coastguard Worker
257*598139dcSAndroid Build Coastguard Worker return true;
258*598139dcSAndroid Build Coastguard Worker }
259*598139dcSAndroid Build Coastguard Worker
calculateCorrection(const log_time & monotonic,const char * real_string,ssize_t len)260*598139dcSAndroid Build Coastguard Worker void LogKlog::calculateCorrection(const log_time& monotonic,
261*598139dcSAndroid Build Coastguard Worker const char* real_string, ssize_t len) {
262*598139dcSAndroid Build Coastguard Worker static const char real_format[] = "%Y-%m-%d %H:%M:%S.%09q UTC";
263*598139dcSAndroid Build Coastguard Worker if (len < (ssize_t)(strlen(real_format) + 5)) return;
264*598139dcSAndroid Build Coastguard Worker
265*598139dcSAndroid Build Coastguard Worker log_time real(log_time::EPOCH);
266*598139dcSAndroid Build Coastguard Worker const char* ep = real.strptime(real_string, real_format);
267*598139dcSAndroid Build Coastguard Worker if (!ep || (ep > &real_string[len]) || (real > log_time(CLOCK_REALTIME))) {
268*598139dcSAndroid Build Coastguard Worker return;
269*598139dcSAndroid Build Coastguard Worker }
270*598139dcSAndroid Build Coastguard Worker // kernel report UTC, log_time::strptime is localtime from calendar.
271*598139dcSAndroid Build Coastguard Worker // Bionic and liblog strptime does not support %z or %Z to pick up
272*598139dcSAndroid Build Coastguard Worker // timezone so we are calculating our own correction.
273*598139dcSAndroid Build Coastguard Worker time_t now = real.tv_sec;
274*598139dcSAndroid Build Coastguard Worker struct tm tm = {.tm_isdst = -1};
275*598139dcSAndroid Build Coastguard Worker localtime_r(&now, &tm);
276*598139dcSAndroid Build Coastguard Worker if ((tm.tm_gmtoff < 0) && ((-tm.tm_gmtoff) > (long)real.tv_sec)) {
277*598139dcSAndroid Build Coastguard Worker real = log_time(log_time::EPOCH);
278*598139dcSAndroid Build Coastguard Worker } else {
279*598139dcSAndroid Build Coastguard Worker real.tv_sec += tm.tm_gmtoff;
280*598139dcSAndroid Build Coastguard Worker }
281*598139dcSAndroid Build Coastguard Worker if (monotonic > real) {
282*598139dcSAndroid Build Coastguard Worker correction = log_time(log_time::EPOCH);
283*598139dcSAndroid Build Coastguard Worker } else {
284*598139dcSAndroid Build Coastguard Worker correction = real - monotonic;
285*598139dcSAndroid Build Coastguard Worker }
286*598139dcSAndroid Build Coastguard Worker }
287*598139dcSAndroid Build Coastguard Worker
sniffTime(const char * & buf,ssize_t len,bool reverse)288*598139dcSAndroid Build Coastguard Worker log_time LogKlog::sniffTime(const char*& buf, ssize_t len, bool reverse) {
289*598139dcSAndroid Build Coastguard Worker log_time now(log_time::EPOCH);
290*598139dcSAndroid Build Coastguard Worker if (len <= 0) return now;
291*598139dcSAndroid Build Coastguard Worker
292*598139dcSAndroid Build Coastguard Worker const char* cp = nullptr;
293*598139dcSAndroid Build Coastguard Worker if ((len > 10) && (*buf == '[')) {
294*598139dcSAndroid Build Coastguard Worker cp = now.strptime(buf, "[ %s.%q]"); // can index beyond buffer bounds
295*598139dcSAndroid Build Coastguard Worker if (cp && (cp > &buf[len - 1])) cp = nullptr;
296*598139dcSAndroid Build Coastguard Worker }
297*598139dcSAndroid Build Coastguard Worker if (cp) {
298*598139dcSAndroid Build Coastguard Worker len -= cp - buf;
299*598139dcSAndroid Build Coastguard Worker if ((len > 0) && isspace(*cp)) {
300*598139dcSAndroid Build Coastguard Worker ++cp;
301*598139dcSAndroid Build Coastguard Worker --len;
302*598139dcSAndroid Build Coastguard Worker }
303*598139dcSAndroid Build Coastguard Worker buf = cp;
304*598139dcSAndroid Build Coastguard Worker
305*598139dcSAndroid Build Coastguard Worker const char* b;
306*598139dcSAndroid Build Coastguard Worker if (((b = android::strnstr(cp, len, suspendStr))) &&
307*598139dcSAndroid Build Coastguard Worker (((b += strlen(suspendStr)) - cp) < len)) {
308*598139dcSAndroid Build Coastguard Worker len -= b - cp;
309*598139dcSAndroid Build Coastguard Worker calculateCorrection(now, b, len);
310*598139dcSAndroid Build Coastguard Worker } else if (((b = android::strnstr(cp, len, resumeStr))) &&
311*598139dcSAndroid Build Coastguard Worker (((b += strlen(resumeStr)) - cp) < len)) {
312*598139dcSAndroid Build Coastguard Worker len -= b - cp;
313*598139dcSAndroid Build Coastguard Worker calculateCorrection(now, b, len);
314*598139dcSAndroid Build Coastguard Worker } else if (((b = android::strnstr(cp, len, suspendedStr))) &&
315*598139dcSAndroid Build Coastguard Worker (((b += strlen(suspendedStr)) - cp) < len)) {
316*598139dcSAndroid Build Coastguard Worker len -= b - cp;
317*598139dcSAndroid Build Coastguard Worker log_time real(log_time::EPOCH);
318*598139dcSAndroid Build Coastguard Worker char* endp;
319*598139dcSAndroid Build Coastguard Worker real.tv_sec = strtol(b, &endp, 10);
320*598139dcSAndroid Build Coastguard Worker if ((*endp == '.') && ((endp - b) < len)) {
321*598139dcSAndroid Build Coastguard Worker unsigned long multiplier = NS_PER_SEC;
322*598139dcSAndroid Build Coastguard Worker real.tv_nsec = 0;
323*598139dcSAndroid Build Coastguard Worker len -= endp - b;
324*598139dcSAndroid Build Coastguard Worker while (--len && isdigit(*++endp) && (multiplier /= 10)) {
325*598139dcSAndroid Build Coastguard Worker real.tv_nsec += (*endp - '0') * multiplier;
326*598139dcSAndroid Build Coastguard Worker }
327*598139dcSAndroid Build Coastguard Worker if (reverse) {
328*598139dcSAndroid Build Coastguard Worker if (real > correction) {
329*598139dcSAndroid Build Coastguard Worker correction = log_time(log_time::EPOCH);
330*598139dcSAndroid Build Coastguard Worker } else {
331*598139dcSAndroid Build Coastguard Worker correction -= real;
332*598139dcSAndroid Build Coastguard Worker }
333*598139dcSAndroid Build Coastguard Worker } else {
334*598139dcSAndroid Build Coastguard Worker correction += real;
335*598139dcSAndroid Build Coastguard Worker }
336*598139dcSAndroid Build Coastguard Worker }
337*598139dcSAndroid Build Coastguard Worker } else {
338*598139dcSAndroid Build Coastguard Worker static time_t last_correction_time_utc = 0;
339*598139dcSAndroid Build Coastguard Worker time_t current_time_utc = time(nullptr);
340*598139dcSAndroid Build Coastguard Worker if (current_time_utc < last_correction_time_utc ||
341*598139dcSAndroid Build Coastguard Worker current_time_utc - last_correction_time_utc > 60) {
342*598139dcSAndroid Build Coastguard Worker log_time real(CLOCK_REALTIME);
343*598139dcSAndroid Build Coastguard Worker log_time mono(CLOCK_MONOTONIC);
344*598139dcSAndroid Build Coastguard Worker correction = (real < mono) ? log_time(log_time::EPOCH) : (real - mono);
345*598139dcSAndroid Build Coastguard Worker
346*598139dcSAndroid Build Coastguard Worker last_correction_time_utc = current_time_utc;
347*598139dcSAndroid Build Coastguard Worker }
348*598139dcSAndroid Build Coastguard Worker }
349*598139dcSAndroid Build Coastguard Worker
350*598139dcSAndroid Build Coastguard Worker convertMonotonicToReal(now);
351*598139dcSAndroid Build Coastguard Worker } else {
352*598139dcSAndroid Build Coastguard Worker now = log_time(CLOCK_REALTIME);
353*598139dcSAndroid Build Coastguard Worker }
354*598139dcSAndroid Build Coastguard Worker return now;
355*598139dcSAndroid Build Coastguard Worker }
356*598139dcSAndroid Build Coastguard Worker
sniffPid(const char * & buf,ssize_t len)357*598139dcSAndroid Build Coastguard Worker pid_t LogKlog::sniffPid(const char*& buf, ssize_t len) {
358*598139dcSAndroid Build Coastguard Worker if (len <= 0) return 0;
359*598139dcSAndroid Build Coastguard Worker
360*598139dcSAndroid Build Coastguard Worker const char* cp = buf;
361*598139dcSAndroid Build Coastguard Worker // sscanf does a strlen, let's check if the string is not nul terminated.
362*598139dcSAndroid Build Coastguard Worker // pseudo out-of-bounds access since we always have an extra char on buffer.
363*598139dcSAndroid Build Coastguard Worker if (((ssize_t)strnlen(cp, len) == len) && cp[len]) {
364*598139dcSAndroid Build Coastguard Worker return 0;
365*598139dcSAndroid Build Coastguard Worker }
366*598139dcSAndroid Build Coastguard Worker while (len) {
367*598139dcSAndroid Build Coastguard Worker // Mediatek kernels with modified printk
368*598139dcSAndroid Build Coastguard Worker if (*cp == '[') {
369*598139dcSAndroid Build Coastguard Worker int pid = 0;
370*598139dcSAndroid Build Coastguard Worker char placeholder;
371*598139dcSAndroid Build Coastguard Worker if (sscanf(cp, "[%d:%*[a-z_./0-9:A-Z]]%c", &pid, &placeholder) == 2) {
372*598139dcSAndroid Build Coastguard Worker return pid;
373*598139dcSAndroid Build Coastguard Worker }
374*598139dcSAndroid Build Coastguard Worker break; // Only the first one
375*598139dcSAndroid Build Coastguard Worker }
376*598139dcSAndroid Build Coastguard Worker ++cp;
377*598139dcSAndroid Build Coastguard Worker --len;
378*598139dcSAndroid Build Coastguard Worker }
379*598139dcSAndroid Build Coastguard Worker if (len > 8 && cp[0] == '[' && cp[7] == ']' && isdigit(cp[6])) {
380*598139dcSAndroid Build Coastguard Worker // Linux 5.10 and above, e.g. "[ T1] init: init first stage started!"
381*598139dcSAndroid Build Coastguard Worker int i = 5;
382*598139dcSAndroid Build Coastguard Worker while (i > 1 && isdigit(cp[i])) {
383*598139dcSAndroid Build Coastguard Worker --i;
384*598139dcSAndroid Build Coastguard Worker }
385*598139dcSAndroid Build Coastguard Worker int pos = i + 1;
386*598139dcSAndroid Build Coastguard Worker if (cp[i] != 'T') {
387*598139dcSAndroid Build Coastguard Worker return 0;
388*598139dcSAndroid Build Coastguard Worker }
389*598139dcSAndroid Build Coastguard Worker while (i > 1) {
390*598139dcSAndroid Build Coastguard Worker --i;
391*598139dcSAndroid Build Coastguard Worker if (cp[i] != ' ') {
392*598139dcSAndroid Build Coastguard Worker return 0;
393*598139dcSAndroid Build Coastguard Worker }
394*598139dcSAndroid Build Coastguard Worker }
395*598139dcSAndroid Build Coastguard Worker buf = cp + 8;
396*598139dcSAndroid Build Coastguard Worker return atoi(cp + pos);
397*598139dcSAndroid Build Coastguard Worker }
398*598139dcSAndroid Build Coastguard Worker return 0;
399*598139dcSAndroid Build Coastguard Worker }
400*598139dcSAndroid Build Coastguard Worker
401*598139dcSAndroid Build Coastguard Worker // kernel log prefix, convert to a kernel log priority number
parseKernelPrio(const char * & buf,ssize_t len)402*598139dcSAndroid Build Coastguard Worker static int parseKernelPrio(const char*& buf, ssize_t len) {
403*598139dcSAndroid Build Coastguard Worker int pri = LOG_USER | LOG_INFO;
404*598139dcSAndroid Build Coastguard Worker const char* cp = buf;
405*598139dcSAndroid Build Coastguard Worker if ((len > 0) && (*cp == '<')) {
406*598139dcSAndroid Build Coastguard Worker pri = 0;
407*598139dcSAndroid Build Coastguard Worker while (--len && isdigit(*++cp)) {
408*598139dcSAndroid Build Coastguard Worker pri = (pri * 10) + *cp - '0';
409*598139dcSAndroid Build Coastguard Worker }
410*598139dcSAndroid Build Coastguard Worker if ((len > 0) && (*cp == '>')) {
411*598139dcSAndroid Build Coastguard Worker ++cp;
412*598139dcSAndroid Build Coastguard Worker } else {
413*598139dcSAndroid Build Coastguard Worker cp = buf;
414*598139dcSAndroid Build Coastguard Worker pri = LOG_USER | LOG_INFO;
415*598139dcSAndroid Build Coastguard Worker }
416*598139dcSAndroid Build Coastguard Worker buf = cp;
417*598139dcSAndroid Build Coastguard Worker }
418*598139dcSAndroid Build Coastguard Worker return pri;
419*598139dcSAndroid Build Coastguard Worker }
420*598139dcSAndroid Build Coastguard Worker
421*598139dcSAndroid Build Coastguard Worker // Convert kernel log priority number into an Android Logger priority number
convertKernelPrioToAndroidPrio(int pri)422*598139dcSAndroid Build Coastguard Worker static int convertKernelPrioToAndroidPrio(int pri) {
423*598139dcSAndroid Build Coastguard Worker switch (pri & LOG_PRIMASK) {
424*598139dcSAndroid Build Coastguard Worker case LOG_EMERG:
425*598139dcSAndroid Build Coastguard Worker case LOG_ALERT:
426*598139dcSAndroid Build Coastguard Worker case LOG_CRIT:
427*598139dcSAndroid Build Coastguard Worker return ANDROID_LOG_FATAL;
428*598139dcSAndroid Build Coastguard Worker
429*598139dcSAndroid Build Coastguard Worker case LOG_ERR:
430*598139dcSAndroid Build Coastguard Worker return ANDROID_LOG_ERROR;
431*598139dcSAndroid Build Coastguard Worker
432*598139dcSAndroid Build Coastguard Worker case LOG_WARNING:
433*598139dcSAndroid Build Coastguard Worker return ANDROID_LOG_WARN;
434*598139dcSAndroid Build Coastguard Worker
435*598139dcSAndroid Build Coastguard Worker default:
436*598139dcSAndroid Build Coastguard Worker case LOG_NOTICE:
437*598139dcSAndroid Build Coastguard Worker case LOG_INFO:
438*598139dcSAndroid Build Coastguard Worker break;
439*598139dcSAndroid Build Coastguard Worker
440*598139dcSAndroid Build Coastguard Worker case LOG_DEBUG:
441*598139dcSAndroid Build Coastguard Worker return ANDROID_LOG_DEBUG;
442*598139dcSAndroid Build Coastguard Worker }
443*598139dcSAndroid Build Coastguard Worker
444*598139dcSAndroid Build Coastguard Worker return ANDROID_LOG_INFO;
445*598139dcSAndroid Build Coastguard Worker }
446*598139dcSAndroid Build Coastguard Worker
strnrchr(const char * s,ssize_t len,char c)447*598139dcSAndroid Build Coastguard Worker static const char* strnrchr(const char* s, ssize_t len, char c) {
448*598139dcSAndroid Build Coastguard Worker const char* save = nullptr;
449*598139dcSAndroid Build Coastguard Worker for (; len > 0; ++s, len--) {
450*598139dcSAndroid Build Coastguard Worker if (*s == c) {
451*598139dcSAndroid Build Coastguard Worker save = s;
452*598139dcSAndroid Build Coastguard Worker }
453*598139dcSAndroid Build Coastguard Worker }
454*598139dcSAndroid Build Coastguard Worker return save;
455*598139dcSAndroid Build Coastguard Worker }
456*598139dcSAndroid Build Coastguard Worker
457*598139dcSAndroid Build Coastguard Worker //
458*598139dcSAndroid Build Coastguard Worker // log a message into the kernel log buffer
459*598139dcSAndroid Build Coastguard Worker //
460*598139dcSAndroid Build Coastguard Worker // Filter rules to parse <PRI> <TIME> <tag> and <message> in order for
461*598139dcSAndroid Build Coastguard Worker // them to appear correct in the logcat output:
462*598139dcSAndroid Build Coastguard Worker //
463*598139dcSAndroid Build Coastguard Worker // LOG_KERN (0):
464*598139dcSAndroid Build Coastguard Worker // <PRI>[<TIME>] <tag> ":" <message>
465*598139dcSAndroid Build Coastguard Worker // <PRI>[<TIME>] <tag> <tag> ":" <message>
466*598139dcSAndroid Build Coastguard Worker // <PRI>[<TIME>] <tag> <tag>_work ":" <message>
467*598139dcSAndroid Build Coastguard Worker // <PRI>[<TIME>] <tag> '<tag>.<num>' ":" <message>
468*598139dcSAndroid Build Coastguard Worker // <PRI>[<TIME>] <tag> '<tag><num>' ":" <message>
469*598139dcSAndroid Build Coastguard Worker // <PRI>[<TIME>] <tag>_host '<tag>.<num>' ":" <message>
470*598139dcSAndroid Build Coastguard Worker // (unimplemented) <PRI>[<TIME>] <tag> '<num>.<tag>' ":" <message>
471*598139dcSAndroid Build Coastguard Worker // <PRI>[<TIME>] "[INFO]"<tag> : <message>
472*598139dcSAndroid Build Coastguard Worker // <PRI>[<TIME>] "------------[ cut here ]------------" (?)
473*598139dcSAndroid Build Coastguard Worker // <PRI>[<TIME>] "---[ end trace 3225a3070ca3e4ac ]---" (?)
474*598139dcSAndroid Build Coastguard Worker // LOG_USER, LOG_MAIL, LOG_DAEMON, LOG_AUTH, LOG_SYSLOG, LOG_LPR, LOG_NEWS
475*598139dcSAndroid Build Coastguard Worker // LOG_UUCP, LOG_CRON, LOG_AUTHPRIV, LOG_FTP:
476*598139dcSAndroid Build Coastguard Worker // <PRI+TAG>[<TIME>] (see sys/syslog.h)
477*598139dcSAndroid Build Coastguard Worker // Observe:
478*598139dcSAndroid Build Coastguard Worker // Minimum tag length = 3 NB: drops things like r5:c00bbadf, but allow PM:
479*598139dcSAndroid Build Coastguard Worker // Maximum tag words = 2
480*598139dcSAndroid Build Coastguard Worker // Maximum tag length = 16 NB: we are thinking of how ugly logcat can get.
481*598139dcSAndroid Build Coastguard Worker // Not a Tag if there is no message content.
482*598139dcSAndroid Build Coastguard Worker // leading additional spaces means no tag, inherit last tag.
483*598139dcSAndroid Build Coastguard Worker // Not a Tag if <tag>: is "ERROR:", "WARNING:", "INFO:" or "CPU:"
484*598139dcSAndroid Build Coastguard Worker // Drop:
485*598139dcSAndroid Build Coastguard Worker // empty messages
486*598139dcSAndroid Build Coastguard Worker // messages with ' audit(' in them if auditd is running
487*598139dcSAndroid Build Coastguard Worker // logd.klogd:
488*598139dcSAndroid Build Coastguard Worker // return -1 if message logd.klogd: <signature>
489*598139dcSAndroid Build Coastguard Worker //
log(const char * buf,ssize_t len)490*598139dcSAndroid Build Coastguard Worker int LogKlog::log(const char* buf, ssize_t len) {
491*598139dcSAndroid Build Coastguard Worker if (auditd && android::strnstr(buf, len, auditStr)) {
492*598139dcSAndroid Build Coastguard Worker return 0;
493*598139dcSAndroid Build Coastguard Worker }
494*598139dcSAndroid Build Coastguard Worker
495*598139dcSAndroid Build Coastguard Worker const char* p = buf;
496*598139dcSAndroid Build Coastguard Worker int pri = parseKernelPrio(p, len);
497*598139dcSAndroid Build Coastguard Worker
498*598139dcSAndroid Build Coastguard Worker log_time now = sniffTime(p, len - (p - buf), false);
499*598139dcSAndroid Build Coastguard Worker const char* start;
500*598139dcSAndroid Build Coastguard Worker
501*598139dcSAndroid Build Coastguard Worker // Parse pid, tid and uid
502*598139dcSAndroid Build Coastguard Worker const pid_t pid = sniffPid(p, len - (p - buf));
503*598139dcSAndroid Build Coastguard Worker const pid_t tid = pid;
504*598139dcSAndroid Build Coastguard Worker uid_t uid = AID_ROOT;
505*598139dcSAndroid Build Coastguard Worker if (pid) {
506*598139dcSAndroid Build Coastguard Worker uid = stats_->PidToUid(pid);
507*598139dcSAndroid Build Coastguard Worker }
508*598139dcSAndroid Build Coastguard Worker
509*598139dcSAndroid Build Coastguard Worker // Parse (rules at top) to pull out a tag from the incoming kernel message.
510*598139dcSAndroid Build Coastguard Worker // Some may view the following as an ugly heuristic, the desire is to
511*598139dcSAndroid Build Coastguard Worker // beautify the kernel logs into an Android Logging format; the goal is
512*598139dcSAndroid Build Coastguard Worker // admirable but costly.
513*598139dcSAndroid Build Coastguard Worker while ((p < &buf[len]) && (isspace(*p) || !*p)) {
514*598139dcSAndroid Build Coastguard Worker ++p;
515*598139dcSAndroid Build Coastguard Worker }
516*598139dcSAndroid Build Coastguard Worker if (p >= &buf[len]) { // timestamp, no content
517*598139dcSAndroid Build Coastguard Worker return 0;
518*598139dcSAndroid Build Coastguard Worker }
519*598139dcSAndroid Build Coastguard Worker start = p;
520*598139dcSAndroid Build Coastguard Worker const char* tag = "";
521*598139dcSAndroid Build Coastguard Worker const char* etag = tag;
522*598139dcSAndroid Build Coastguard Worker ssize_t taglen = len - (p - buf);
523*598139dcSAndroid Build Coastguard Worker const char* bt = p;
524*598139dcSAndroid Build Coastguard Worker
525*598139dcSAndroid Build Coastguard Worker static const char infoBrace[] = "[INFO]";
526*598139dcSAndroid Build Coastguard Worker static const ssize_t infoBraceLen = strlen(infoBrace);
527*598139dcSAndroid Build Coastguard Worker if ((taglen >= infoBraceLen) &&
528*598139dcSAndroid Build Coastguard Worker !fastcmp<strncmp>(p, infoBrace, infoBraceLen)) {
529*598139dcSAndroid Build Coastguard Worker // <PRI>[<TIME>] "[INFO]"<tag> ":" message
530*598139dcSAndroid Build Coastguard Worker bt = p + infoBraceLen;
531*598139dcSAndroid Build Coastguard Worker taglen -= infoBraceLen;
532*598139dcSAndroid Build Coastguard Worker }
533*598139dcSAndroid Build Coastguard Worker
534*598139dcSAndroid Build Coastguard Worker const char* et;
535*598139dcSAndroid Build Coastguard Worker for (et = bt; (taglen > 0) && *et && (*et != ':') && !isspace(*et);
536*598139dcSAndroid Build Coastguard Worker ++et, --taglen) {
537*598139dcSAndroid Build Coastguard Worker // skip ':' within [ ... ]
538*598139dcSAndroid Build Coastguard Worker if (*et == '[') {
539*598139dcSAndroid Build Coastguard Worker while ((taglen > 0) && *et && *et != ']') {
540*598139dcSAndroid Build Coastguard Worker ++et;
541*598139dcSAndroid Build Coastguard Worker --taglen;
542*598139dcSAndroid Build Coastguard Worker }
543*598139dcSAndroid Build Coastguard Worker if (taglen <= 0) {
544*598139dcSAndroid Build Coastguard Worker break;
545*598139dcSAndroid Build Coastguard Worker }
546*598139dcSAndroid Build Coastguard Worker }
547*598139dcSAndroid Build Coastguard Worker }
548*598139dcSAndroid Build Coastguard Worker const char* cp;
549*598139dcSAndroid Build Coastguard Worker for (cp = et; (taglen > 0) && isspace(*cp); ++cp, --taglen) {
550*598139dcSAndroid Build Coastguard Worker }
551*598139dcSAndroid Build Coastguard Worker
552*598139dcSAndroid Build Coastguard Worker // Validate tag
553*598139dcSAndroid Build Coastguard Worker ssize_t size = et - bt;
554*598139dcSAndroid Build Coastguard Worker if ((taglen > 0) && (size > 0)) {
555*598139dcSAndroid Build Coastguard Worker if (*cp == ':') {
556*598139dcSAndroid Build Coastguard Worker // ToDo: handle case insensitive colon separated logging stutter:
557*598139dcSAndroid Build Coastguard Worker // <tag> : <tag>: ...
558*598139dcSAndroid Build Coastguard Worker
559*598139dcSAndroid Build Coastguard Worker // One Word
560*598139dcSAndroid Build Coastguard Worker tag = bt;
561*598139dcSAndroid Build Coastguard Worker etag = et;
562*598139dcSAndroid Build Coastguard Worker p = cp + 1;
563*598139dcSAndroid Build Coastguard Worker } else if ((taglen > size) && (tolower(*bt) == tolower(*cp))) {
564*598139dcSAndroid Build Coastguard Worker // clean up any tag stutter
565*598139dcSAndroid Build Coastguard Worker if (!fastcmp<strncasecmp>(bt + 1, cp + 1, size - 1)) { // no match
566*598139dcSAndroid Build Coastguard Worker // <PRI>[<TIME>] <tag> <tag> : message
567*598139dcSAndroid Build Coastguard Worker // <PRI>[<TIME>] <tag> <tag>: message
568*598139dcSAndroid Build Coastguard Worker // <PRI>[<TIME>] <tag> '<tag>.<num>' : message
569*598139dcSAndroid Build Coastguard Worker // <PRI>[<TIME>] <tag> '<tag><num>' : message
570*598139dcSAndroid Build Coastguard Worker // <PRI>[<TIME>] <tag> '<tag><stuff>' : message
571*598139dcSAndroid Build Coastguard Worker const char* b = cp;
572*598139dcSAndroid Build Coastguard Worker cp += size;
573*598139dcSAndroid Build Coastguard Worker taglen -= size;
574*598139dcSAndroid Build Coastguard Worker while ((--taglen > 0) && !isspace(*++cp) && (*cp != ':')) {
575*598139dcSAndroid Build Coastguard Worker }
576*598139dcSAndroid Build Coastguard Worker const char* e;
577*598139dcSAndroid Build Coastguard Worker for (e = cp; (taglen > 0) && isspace(*cp); ++cp, --taglen) {
578*598139dcSAndroid Build Coastguard Worker }
579*598139dcSAndroid Build Coastguard Worker if ((taglen > 0) && (*cp == ':')) {
580*598139dcSAndroid Build Coastguard Worker tag = b;
581*598139dcSAndroid Build Coastguard Worker etag = e;
582*598139dcSAndroid Build Coastguard Worker p = cp + 1;
583*598139dcSAndroid Build Coastguard Worker }
584*598139dcSAndroid Build Coastguard Worker } else {
585*598139dcSAndroid Build Coastguard Worker // what about <PRI>[<TIME>] <tag>_host '<tag><stuff>' : message
586*598139dcSAndroid Build Coastguard Worker static const char host[] = "_host";
587*598139dcSAndroid Build Coastguard Worker static const ssize_t hostlen = strlen(host);
588*598139dcSAndroid Build Coastguard Worker if ((size > hostlen) &&
589*598139dcSAndroid Build Coastguard Worker !fastcmp<strncmp>(bt + size - hostlen, host, hostlen) &&
590*598139dcSAndroid Build Coastguard Worker !fastcmp<strncmp>(bt + 1, cp + 1, size - hostlen - 1)) {
591*598139dcSAndroid Build Coastguard Worker const char* b = cp;
592*598139dcSAndroid Build Coastguard Worker cp += size - hostlen;
593*598139dcSAndroid Build Coastguard Worker taglen -= size - hostlen;
594*598139dcSAndroid Build Coastguard Worker if (*cp == '.') {
595*598139dcSAndroid Build Coastguard Worker while ((--taglen > 0) && !isspace(*++cp) &&
596*598139dcSAndroid Build Coastguard Worker (*cp != ':')) {
597*598139dcSAndroid Build Coastguard Worker }
598*598139dcSAndroid Build Coastguard Worker const char* e;
599*598139dcSAndroid Build Coastguard Worker for (e = cp; (taglen > 0) && isspace(*cp);
600*598139dcSAndroid Build Coastguard Worker ++cp, --taglen) {
601*598139dcSAndroid Build Coastguard Worker }
602*598139dcSAndroid Build Coastguard Worker if ((taglen > 0) && (*cp == ':')) {
603*598139dcSAndroid Build Coastguard Worker tag = b;
604*598139dcSAndroid Build Coastguard Worker etag = e;
605*598139dcSAndroid Build Coastguard Worker p = cp + 1;
606*598139dcSAndroid Build Coastguard Worker }
607*598139dcSAndroid Build Coastguard Worker }
608*598139dcSAndroid Build Coastguard Worker } else {
609*598139dcSAndroid Build Coastguard Worker goto twoWord;
610*598139dcSAndroid Build Coastguard Worker }
611*598139dcSAndroid Build Coastguard Worker }
612*598139dcSAndroid Build Coastguard Worker } else {
613*598139dcSAndroid Build Coastguard Worker // <PRI>[<TIME>] <tag> <stuff>' : message
614*598139dcSAndroid Build Coastguard Worker twoWord:
615*598139dcSAndroid Build Coastguard Worker while ((--taglen > 0) && !isspace(*++cp) && (*cp != ':')) {
616*598139dcSAndroid Build Coastguard Worker }
617*598139dcSAndroid Build Coastguard Worker const char* e;
618*598139dcSAndroid Build Coastguard Worker for (e = cp; (taglen > 0) && isspace(*cp); ++cp, --taglen) {
619*598139dcSAndroid Build Coastguard Worker }
620*598139dcSAndroid Build Coastguard Worker // Two words
621*598139dcSAndroid Build Coastguard Worker if ((taglen > 0) && (*cp == ':')) {
622*598139dcSAndroid Build Coastguard Worker tag = bt;
623*598139dcSAndroid Build Coastguard Worker etag = e;
624*598139dcSAndroid Build Coastguard Worker p = cp + 1;
625*598139dcSAndroid Build Coastguard Worker }
626*598139dcSAndroid Build Coastguard Worker }
627*598139dcSAndroid Build Coastguard Worker } // else no tag
628*598139dcSAndroid Build Coastguard Worker
629*598139dcSAndroid Build Coastguard Worker static const char cpu[] = "CPU";
630*598139dcSAndroid Build Coastguard Worker static const ssize_t cpuLen = strlen(cpu);
631*598139dcSAndroid Build Coastguard Worker static const char warning[] = "WARNING";
632*598139dcSAndroid Build Coastguard Worker static const ssize_t warningLen = strlen(warning);
633*598139dcSAndroid Build Coastguard Worker static const char error[] = "ERROR";
634*598139dcSAndroid Build Coastguard Worker static const ssize_t errorLen = strlen(error);
635*598139dcSAndroid Build Coastguard Worker static const char info[] = "INFO";
636*598139dcSAndroid Build Coastguard Worker static const ssize_t infoLen = strlen(info);
637*598139dcSAndroid Build Coastguard Worker
638*598139dcSAndroid Build Coastguard Worker size = etag - tag;
639*598139dcSAndroid Build Coastguard Worker if ((size <= 1) ||
640*598139dcSAndroid Build Coastguard Worker // register names like x9
641*598139dcSAndroid Build Coastguard Worker ((size == 2) && (isdigit(tag[0]) || isdigit(tag[1]))) ||
642*598139dcSAndroid Build Coastguard Worker // register names like x18 but not driver names like en0
643*598139dcSAndroid Build Coastguard Worker ((size == 3) && (isdigit(tag[1]) && isdigit(tag[2]))) ||
644*598139dcSAndroid Build Coastguard Worker // ignore
645*598139dcSAndroid Build Coastguard Worker ((size == cpuLen) && !fastcmp<strncmp>(tag, cpu, cpuLen)) ||
646*598139dcSAndroid Build Coastguard Worker ((size == warningLen) && !fastcmp<strncasecmp>(tag, warning, warningLen)) ||
647*598139dcSAndroid Build Coastguard Worker ((size == errorLen) && !fastcmp<strncasecmp>(tag, error, errorLen)) ||
648*598139dcSAndroid Build Coastguard Worker ((size == infoLen) && !fastcmp<strncasecmp>(tag, info, infoLen))) {
649*598139dcSAndroid Build Coastguard Worker p = start;
650*598139dcSAndroid Build Coastguard Worker etag = tag = "";
651*598139dcSAndroid Build Coastguard Worker }
652*598139dcSAndroid Build Coastguard Worker
653*598139dcSAndroid Build Coastguard Worker // Suppress additional stutter in tag:
654*598139dcSAndroid Build Coastguard Worker // eg: [143:healthd]healthd -> [143:healthd]
655*598139dcSAndroid Build Coastguard Worker taglen = etag - tag;
656*598139dcSAndroid Build Coastguard Worker // Mediatek-special printk induced stutter
657*598139dcSAndroid Build Coastguard Worker const char* mp = strnrchr(tag, taglen, ']');
658*598139dcSAndroid Build Coastguard Worker if (mp && (++mp < etag)) {
659*598139dcSAndroid Build Coastguard Worker ssize_t s = etag - mp;
660*598139dcSAndroid Build Coastguard Worker if (((s + s) < taglen) && !fastcmp<memcmp>(mp, mp - 1 - s, s)) {
661*598139dcSAndroid Build Coastguard Worker taglen = mp - tag;
662*598139dcSAndroid Build Coastguard Worker }
663*598139dcSAndroid Build Coastguard Worker }
664*598139dcSAndroid Build Coastguard Worker // Deal with sloppy and simplistic harmless p = cp + 1 etc above.
665*598139dcSAndroid Build Coastguard Worker if (len < (p - buf)) {
666*598139dcSAndroid Build Coastguard Worker p = &buf[len];
667*598139dcSAndroid Build Coastguard Worker }
668*598139dcSAndroid Build Coastguard Worker // skip leading space
669*598139dcSAndroid Build Coastguard Worker while ((p < &buf[len]) && (isspace(*p) || !*p)) {
670*598139dcSAndroid Build Coastguard Worker ++p;
671*598139dcSAndroid Build Coastguard Worker }
672*598139dcSAndroid Build Coastguard Worker // truncate trailing space or nuls
673*598139dcSAndroid Build Coastguard Worker ssize_t b = len - (p - buf);
674*598139dcSAndroid Build Coastguard Worker while ((b > 0) && (isspace(p[b - 1]) || !p[b - 1])) {
675*598139dcSAndroid Build Coastguard Worker --b;
676*598139dcSAndroid Build Coastguard Worker }
677*598139dcSAndroid Build Coastguard Worker // trick ... allow tag with empty content to be logged. log() drops empty
678*598139dcSAndroid Build Coastguard Worker if ((b <= 0) && (taglen > 0)) {
679*598139dcSAndroid Build Coastguard Worker p = " ";
680*598139dcSAndroid Build Coastguard Worker b = 1;
681*598139dcSAndroid Build Coastguard Worker }
682*598139dcSAndroid Build Coastguard Worker // This shouldn't happen, but clamp the size if it does.
683*598139dcSAndroid Build Coastguard Worker if (b > LOGGER_ENTRY_MAX_PAYLOAD) {
684*598139dcSAndroid Build Coastguard Worker b = LOGGER_ENTRY_MAX_PAYLOAD;
685*598139dcSAndroid Build Coastguard Worker }
686*598139dcSAndroid Build Coastguard Worker if (taglen > LOGGER_ENTRY_MAX_PAYLOAD) {
687*598139dcSAndroid Build Coastguard Worker taglen = LOGGER_ENTRY_MAX_PAYLOAD;
688*598139dcSAndroid Build Coastguard Worker }
689*598139dcSAndroid Build Coastguard Worker // calculate buffer copy requirements
690*598139dcSAndroid Build Coastguard Worker ssize_t n = 1 + taglen + 1 + b + 1;
691*598139dcSAndroid Build Coastguard Worker // Extra checks for likely impossible cases.
692*598139dcSAndroid Build Coastguard Worker if ((taglen > n) || (b > n) || (n > (ssize_t)USHRT_MAX) || (n <= 0)) {
693*598139dcSAndroid Build Coastguard Worker return -EINVAL;
694*598139dcSAndroid Build Coastguard Worker }
695*598139dcSAndroid Build Coastguard Worker
696*598139dcSAndroid Build Coastguard Worker // Careful.
697*598139dcSAndroid Build Coastguard Worker // We are using the stack to house the log buffer for speed reasons.
698*598139dcSAndroid Build Coastguard Worker // If we malloc'd this buffer, we could get away without n's USHRT_MAX
699*598139dcSAndroid Build Coastguard Worker // test above, but we would then required a max(n, USHRT_MAX) as
700*598139dcSAndroid Build Coastguard Worker // truncating length argument to logbuf->log() below. Gain is protection
701*598139dcSAndroid Build Coastguard Worker // against stack corruption and speedup, loss is truncated long-line content.
702*598139dcSAndroid Build Coastguard Worker char newstr[n];
703*598139dcSAndroid Build Coastguard Worker char* np = newstr;
704*598139dcSAndroid Build Coastguard Worker
705*598139dcSAndroid Build Coastguard Worker // Convert priority into single-byte Android logger priority
706*598139dcSAndroid Build Coastguard Worker *np = convertKernelPrioToAndroidPrio(pri);
707*598139dcSAndroid Build Coastguard Worker ++np;
708*598139dcSAndroid Build Coastguard Worker
709*598139dcSAndroid Build Coastguard Worker // Copy parsed tag following priority
710*598139dcSAndroid Build Coastguard Worker memcpy(np, tag, taglen);
711*598139dcSAndroid Build Coastguard Worker np += taglen;
712*598139dcSAndroid Build Coastguard Worker *np = '\0';
713*598139dcSAndroid Build Coastguard Worker ++np;
714*598139dcSAndroid Build Coastguard Worker
715*598139dcSAndroid Build Coastguard Worker // Copy main message to the remainder
716*598139dcSAndroid Build Coastguard Worker memcpy(np, p, b);
717*598139dcSAndroid Build Coastguard Worker np[b] = '\0';
718*598139dcSAndroid Build Coastguard Worker
719*598139dcSAndroid Build Coastguard Worker {
720*598139dcSAndroid Build Coastguard Worker // Watch out for singular race conditions with timezone causing near
721*598139dcSAndroid Build Coastguard Worker // integer quarter-hour jumps in the time and compensate accordingly.
722*598139dcSAndroid Build Coastguard Worker // Entries will be temporal within near_seconds * 2. b/21868540
723*598139dcSAndroid Build Coastguard Worker static uint32_t vote_time[3];
724*598139dcSAndroid Build Coastguard Worker vote_time[2] = vote_time[1];
725*598139dcSAndroid Build Coastguard Worker vote_time[1] = vote_time[0];
726*598139dcSAndroid Build Coastguard Worker vote_time[0] = now.tv_sec;
727*598139dcSAndroid Build Coastguard Worker
728*598139dcSAndroid Build Coastguard Worker if (vote_time[1] && vote_time[2]) {
729*598139dcSAndroid Build Coastguard Worker static const unsigned near_seconds = 10;
730*598139dcSAndroid Build Coastguard Worker static const unsigned timezones_seconds = 900;
731*598139dcSAndroid Build Coastguard Worker int diff0 = (vote_time[0] - vote_time[1]) / near_seconds;
732*598139dcSAndroid Build Coastguard Worker unsigned abs0 = (diff0 < 0) ? -diff0 : diff0;
733*598139dcSAndroid Build Coastguard Worker int diff1 = (vote_time[1] - vote_time[2]) / near_seconds;
734*598139dcSAndroid Build Coastguard Worker unsigned abs1 = (diff1 < 0) ? -diff1 : diff1;
735*598139dcSAndroid Build Coastguard Worker if ((abs1 <= 1) && // last two were in agreement on timezone
736*598139dcSAndroid Build Coastguard Worker ((abs0 + 1) % (timezones_seconds / near_seconds)) <= 2) {
737*598139dcSAndroid Build Coastguard Worker abs0 = (abs0 + 1) / (timezones_seconds / near_seconds) *
738*598139dcSAndroid Build Coastguard Worker timezones_seconds;
739*598139dcSAndroid Build Coastguard Worker now.tv_sec -= (diff0 < 0) ? -abs0 : abs0;
740*598139dcSAndroid Build Coastguard Worker }
741*598139dcSAndroid Build Coastguard Worker }
742*598139dcSAndroid Build Coastguard Worker }
743*598139dcSAndroid Build Coastguard Worker
744*598139dcSAndroid Build Coastguard Worker // Log message
745*598139dcSAndroid Build Coastguard Worker int rc = logbuf->Log(LOG_ID_KERNEL, now, uid, pid, tid, newstr, (uint16_t)n);
746*598139dcSAndroid Build Coastguard Worker
747*598139dcSAndroid Build Coastguard Worker return rc;
748*598139dcSAndroid Build Coastguard Worker }
749