xref: /aosp_15_r20/system/logging/logd/LogKlog.cpp (revision 598139dc91b21518d67c408eaea2644226490971)
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