Merge changes I96998c4b,I161bf03b am: dd0cd8d88f
am: f17500474a
am: 77a1fa9070
Change-Id: I5b296f6c1b01a8b2dc51c7ebbd44d599a3aa49c1
This commit is contained in:
commit
02ce4262dc
6 changed files with 241 additions and 226 deletions
|
@ -17,6 +17,13 @@
|
|||
#ifndef _ANDROID_UTILS_FASTSTRCMP_H__
|
||||
#define _ANDROID_UTILS_FASTSTRCMP_H__
|
||||
|
||||
#include <ctype.h>
|
||||
#include <string.h>
|
||||
|
||||
#ifndef __predict_true
|
||||
#define __predict_true(exp) __builtin_expect((exp) != 0, 1)
|
||||
#endif
|
||||
|
||||
#ifdef __cplusplus
|
||||
|
||||
// Optimized for instruction cache locality
|
||||
|
@ -28,25 +35,41 @@
|
|||
//
|
||||
// fastcmp<strncmp>(str1, str2, len)
|
||||
//
|
||||
// NB: Does not work for the case insensitive str*cmp functions.
|
||||
// NB: use fasticmp for the case insensitive str*cmp functions.
|
||||
// NB: Returns boolean, do not use if expecting to check negative value.
|
||||
// Thus not semantically identical to the expected function behavior.
|
||||
|
||||
template <int (*cmp)(const char *l, const char *r, const size_t s)>
|
||||
static inline int fastcmp(const char *l, const char *r, const size_t s) {
|
||||
return (*l != *r) || cmp(l + 1, r + 1, s - 1);
|
||||
template <int (*cmp)(const char* l, const char* r, const size_t s)>
|
||||
static inline int fastcmp(const char* l, const char* r, const size_t s) {
|
||||
const ssize_t n = s; // To help reject negative sizes, treat like zero
|
||||
return __predict_true(n > 0) &&
|
||||
((*l != *r) || (__predict_true(n > 1) && cmp(l + 1, r + 1, n - 1)));
|
||||
}
|
||||
|
||||
template <int (*cmp)(const void *l, const void *r, const size_t s)>
|
||||
static inline int fastcmp(const void *lv, const void *rv, const size_t s) {
|
||||
const char *l = static_cast<const char *>(lv);
|
||||
const char *r = static_cast<const char *>(rv);
|
||||
return (*l != *r) || cmp(l + 1, r + 1, s - 1);
|
||||
template <int (*cmp)(const char* l, const char* r, const size_t s)>
|
||||
static inline int fasticmp(const char* l, const char* r, const size_t s) {
|
||||
const ssize_t n = s; // To help reject negative sizes, treat like zero
|
||||
return __predict_true(n > 0) &&
|
||||
((tolower(*l) != tolower(*r)) || (__predict_true(n > 1) && cmp(l + 1, r + 1, n - 1)));
|
||||
}
|
||||
|
||||
template <int (*cmp)(const char *l, const char *r)>
|
||||
static inline int fastcmp(const char *l, const char *r) {
|
||||
return (*l != *r) || cmp(l + 1, r + 1);
|
||||
template <int (*cmp)(const void* l, const void* r, const size_t s)>
|
||||
static inline int fastcmp(const void* lv, const void* rv, const size_t s) {
|
||||
const char* l = static_cast<const char*>(lv);
|
||||
const char* r = static_cast<const char*>(rv);
|
||||
const ssize_t n = s; // To help reject negative sizes, treat like zero
|
||||
return __predict_true(n > 0) &&
|
||||
((*l != *r) || (__predict_true(n > 1) && cmp(l + 1, r + 1, n - 1)));
|
||||
}
|
||||
|
||||
template <int (*cmp)(const char* l, const char* r)>
|
||||
static inline int fastcmp(const char* l, const char* r) {
|
||||
return (*l != *r) || (__predict_true(*l) && cmp(l + 1, r + 1));
|
||||
}
|
||||
|
||||
template <int (*cmp)(const char* l, const char* r)>
|
||||
static inline int fasticmp(const char* l, const char* r) {
|
||||
return (tolower(*l) != tolower(*r)) || (__predict_true(*l) && cmp(l + 1, r + 1));
|
||||
}
|
||||
|
||||
#endif
|
||||
|
|
|
@ -132,11 +132,11 @@ static enum match_type identical(LogBufferElement* elem,
|
|||
LogBufferElement* last) {
|
||||
// is it mostly identical?
|
||||
// if (!elem) return DIFFERENT;
|
||||
unsigned short lenl = elem->getMsgLen();
|
||||
if (!lenl) return DIFFERENT;
|
||||
ssize_t lenl = elem->getMsgLen();
|
||||
if (lenl <= 0) return DIFFERENT; // value if this represents a chatty elem
|
||||
// if (!last) return DIFFERENT;
|
||||
unsigned short lenr = last->getMsgLen();
|
||||
if (!lenr) return DIFFERENT;
|
||||
ssize_t lenr = last->getMsgLen();
|
||||
if (lenr <= 0) return DIFFERENT; // value if this represents a chatty elem
|
||||
// if (elem->getLogId() != last->getLogId()) return DIFFERENT;
|
||||
if (elem->getUid() != last->getUid()) return DIFFERENT;
|
||||
if (elem->getPid() != last->getPid()) return DIFFERENT;
|
||||
|
@ -163,8 +163,6 @@ static enum match_type identical(LogBufferElement* elem,
|
|||
}
|
||||
|
||||
// audit message (except sequence number) identical?
|
||||
static const char avc[] = "): avc: ";
|
||||
|
||||
if (last->isBinary()) {
|
||||
if (fastcmp<memcmp>(msgl, msgr, sizeof(android_log_event_string_t) -
|
||||
sizeof(int32_t))) {
|
||||
|
@ -175,6 +173,7 @@ static enum match_type identical(LogBufferElement* elem,
|
|||
msgr += sizeof(android_log_event_string_t);
|
||||
lenr -= sizeof(android_log_event_string_t);
|
||||
}
|
||||
static const char avc[] = "): avc: ";
|
||||
const char* avcl = android::strnstr(msgl, lenl, avc);
|
||||
if (!avcl) return DIFFERENT;
|
||||
lenl -= avcl - msgl;
|
||||
|
@ -182,10 +181,7 @@ static enum match_type identical(LogBufferElement* elem,
|
|||
if (!avcr) return DIFFERENT;
|
||||
lenr -= avcr - msgr;
|
||||
if (lenl != lenr) return DIFFERENT;
|
||||
// TODO: After b/35468874 is addressed, revisit "lenl > strlen(avc)"
|
||||
// condition, it might become superfluous.
|
||||
if (lenl > strlen(avc) &&
|
||||
fastcmp<memcmp>(avcl + strlen(avc), avcr + strlen(avc),
|
||||
if (fastcmp<memcmp>(avcl + strlen(avc), avcr + strlen(avc),
|
||||
lenl - strlen(avc))) {
|
||||
return DIFFERENT;
|
||||
}
|
||||
|
@ -1094,12 +1090,12 @@ log_time LogBuffer::flushTo(
|
|||
// client wants to start from the beginning
|
||||
it = mLogElements.begin();
|
||||
} else {
|
||||
LogBufferElementCollection::iterator last = mLogElements.begin();
|
||||
LogBufferElementCollection::iterator last;
|
||||
// 30 second limit to continue search for out-of-order entries.
|
||||
log_time min = start - log_time(30, 0);
|
||||
// Client wants to start from some specified time. Chances are
|
||||
// we are better off starting from the end of the time sorted list.
|
||||
for (it = mLogElements.end(); it != mLogElements.begin();
|
||||
for (last = it = mLogElements.end(); it != mLogElements.begin();
|
||||
/* do nothing */) {
|
||||
--it;
|
||||
LogBufferElement* element = *it;
|
||||
|
|
329
logd/LogKlog.cpp
329
logd/LogKlog.cpp
|
@ -37,46 +37,49 @@
|
|||
|
||||
static const char priority_message[] = { KMSG_PRIORITY(LOG_INFO), '\0' };
|
||||
|
||||
// List of the _only_ needles we supply here to android::strnstr
|
||||
static const char suspendStr[] = "PM: suspend entry ";
|
||||
static const char resumeStr[] = "PM: suspend exit ";
|
||||
static const char suspendedStr[] = "Suspended for ";
|
||||
static const char healthdStr[] = "healthd";
|
||||
static const char batteryStr[] = ": battery ";
|
||||
static const char auditStr[] = " audit(";
|
||||
static const char klogdStr[] = "logd.klogd: ";
|
||||
|
||||
// Parsing is hard
|
||||
|
||||
// called if we see a '<', s is the next character, returns pointer after '>'
|
||||
static char* is_prio(char* s, size_t len) {
|
||||
if (!len || !isdigit(*s++)) {
|
||||
return NULL;
|
||||
}
|
||||
static char* is_prio(char* s, ssize_t len) {
|
||||
if ((len <= 0) || !isdigit(*s++)) return nullptr;
|
||||
--len;
|
||||
static const size_t max_prio_len = (len < 4) ? len : 4;
|
||||
size_t priolen = 0;
|
||||
char c;
|
||||
while (((c = *s++)) && (++priolen <= max_prio_len)) {
|
||||
if (!isdigit(c)) {
|
||||
return ((c == '>') && (*s == '[')) ? s : NULL;
|
||||
}
|
||||
if (!isdigit(c)) return ((c == '>') && (*s == '[')) ? s : nullptr;
|
||||
}
|
||||
return NULL;
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
// called if we see a '[', s is the next character, returns pointer after ']'
|
||||
static char* is_timestamp(char* s, size_t len) {
|
||||
while (len && (*s == ' ')) {
|
||||
static char* is_timestamp(char* s, ssize_t len) {
|
||||
while ((len > 0) && (*s == ' ')) {
|
||||
++s;
|
||||
--len;
|
||||
}
|
||||
if (!len || !isdigit(*s++)) {
|
||||
return NULL;
|
||||
}
|
||||
if ((len <= 0) || !isdigit(*s++)) return nullptr;
|
||||
--len;
|
||||
bool first_period = true;
|
||||
char c;
|
||||
while (len && ((c = *s++))) {
|
||||
while ((len > 0) && ((c = *s++))) {
|
||||
--len;
|
||||
if ((c == '.') && first_period) {
|
||||
first_period = false;
|
||||
} else if (!isdigit(c)) {
|
||||
return ((c == ']') && !first_period && (*s == ' ')) ? s : NULL;
|
||||
return ((c == ']') && !first_period && (*s == ' ')) ? s : nullptr;
|
||||
}
|
||||
}
|
||||
return NULL;
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
// Like strtok_r with "\r\n" except that we look for log signatures (regex)
|
||||
|
@ -93,96 +96,82 @@ static char* is_timestamp(char* s, size_t len) {
|
|||
// space is one more than <digit> of 9
|
||||
#define OPEN_BRACKET_SPACE ((char)(OPEN_BRACKET_SIG | 10))
|
||||
|
||||
char* log_strntok_r(char* s, size_t* len, char** last, size_t* sublen) {
|
||||
*sublen = 0;
|
||||
if (!*len) {
|
||||
return NULL;
|
||||
}
|
||||
char* android::log_strntok_r(char* s, ssize_t& len, char*& last,
|
||||
ssize_t& sublen) {
|
||||
sublen = 0;
|
||||
if (len <= 0) return nullptr;
|
||||
if (!s) {
|
||||
if (!(s = *last)) {
|
||||
return NULL;
|
||||
}
|
||||
if (!(s = last)) return nullptr;
|
||||
// fixup for log signature split <,
|
||||
// LESS_THAN_SIG + <digit>
|
||||
if ((*s & SIGNATURE_MASK) == LESS_THAN_SIG) {
|
||||
*s = (*s & ~SIGNATURE_MASK) + '0';
|
||||
*--s = '<';
|
||||
++*len;
|
||||
++len;
|
||||
}
|
||||
// fixup for log signature split [,
|
||||
// OPEN_BRACKET_SPACE is space, OPEN_BRACKET_SIG + <digit>
|
||||
if ((*s & SIGNATURE_MASK) == OPEN_BRACKET_SIG) {
|
||||
if (*s == OPEN_BRACKET_SPACE) {
|
||||
*s = ' ';
|
||||
} else {
|
||||
*s = (*s & ~SIGNATURE_MASK) + '0';
|
||||
}
|
||||
*s = (*s == OPEN_BRACKET_SPACE) ? ' ' : (*s & ~SIGNATURE_MASK) + '0';
|
||||
*--s = '[';
|
||||
++*len;
|
||||
++len;
|
||||
}
|
||||
}
|
||||
|
||||
while (*len && ((*s == '\r') || (*s == '\n'))) {
|
||||
while ((len > 0) && ((*s == '\r') || (*s == '\n'))) {
|
||||
++s;
|
||||
--*len;
|
||||
--len;
|
||||
}
|
||||
|
||||
if (!*len) {
|
||||
*last = NULL;
|
||||
return NULL;
|
||||
}
|
||||
if (len <= 0) return last = nullptr;
|
||||
char *peek, *tok = s;
|
||||
|
||||
for (;;) {
|
||||
if (*len == 0) {
|
||||
*last = NULL;
|
||||
if (len <= 0) {
|
||||
last = nullptr;
|
||||
return tok;
|
||||
}
|
||||
char c = *s++;
|
||||
--*len;
|
||||
size_t adjust;
|
||||
--len;
|
||||
ssize_t adjust;
|
||||
switch (c) {
|
||||
case '\r':
|
||||
case '\n':
|
||||
s[-1] = '\0';
|
||||
*last = s;
|
||||
last = s;
|
||||
return tok;
|
||||
|
||||
case '<':
|
||||
peek = is_prio(s, *len);
|
||||
if (!peek) {
|
||||
break;
|
||||
}
|
||||
peek = is_prio(s, len);
|
||||
if (!peek) break;
|
||||
if (s != (tok + 1)) { // not first?
|
||||
s[-1] = '\0';
|
||||
*s &= ~SIGNATURE_MASK;
|
||||
*s |= LESS_THAN_SIG; // signature for '<'
|
||||
*last = s;
|
||||
last = s;
|
||||
return tok;
|
||||
}
|
||||
adjust = peek - s;
|
||||
if (adjust > *len) {
|
||||
adjust = *len;
|
||||
if (adjust > len) {
|
||||
adjust = len;
|
||||
}
|
||||
*sublen += adjust;
|
||||
*len -= adjust;
|
||||
sublen += adjust;
|
||||
len -= adjust;
|
||||
s = peek;
|
||||
if ((*s == '[') && ((peek = is_timestamp(s + 1, *len - 1)))) {
|
||||
if ((*s == '[') && ((peek = is_timestamp(s + 1, len - 1)))) {
|
||||
adjust = peek - s;
|
||||
if (adjust > *len) {
|
||||
adjust = *len;
|
||||
if (adjust > len) {
|
||||
adjust = len;
|
||||
}
|
||||
*sublen += adjust;
|
||||
*len -= adjust;
|
||||
sublen += adjust;
|
||||
len -= adjust;
|
||||
s = peek;
|
||||
}
|
||||
break;
|
||||
|
||||
case '[':
|
||||
peek = is_timestamp(s, *len);
|
||||
if (!peek) {
|
||||
break;
|
||||
}
|
||||
peek = is_timestamp(s, len);
|
||||
if (!peek) break;
|
||||
if (s != (tok + 1)) { // not first?
|
||||
s[-1] = '\0';
|
||||
if (*s == ' ') {
|
||||
|
@ -191,19 +180,19 @@ char* log_strntok_r(char* s, size_t* len, char** last, size_t* sublen) {
|
|||
*s &= ~SIGNATURE_MASK;
|
||||
*s |= OPEN_BRACKET_SIG; // signature for '['
|
||||
}
|
||||
*last = s;
|
||||
last = s;
|
||||
return tok;
|
||||
}
|
||||
adjust = peek - s;
|
||||
if (adjust > *len) {
|
||||
adjust = *len;
|
||||
if (adjust > len) {
|
||||
adjust = len;
|
||||
}
|
||||
*sublen += adjust;
|
||||
*len -= adjust;
|
||||
sublen += adjust;
|
||||
len -= adjust;
|
||||
s = peek;
|
||||
break;
|
||||
}
|
||||
++*sublen;
|
||||
++sublen;
|
||||
}
|
||||
// NOTREACHED
|
||||
}
|
||||
|
@ -222,9 +211,10 @@ LogKlog::LogKlog(LogBuffer* buf, LogReader* reader, int fdWrite, int fdRead,
|
|||
initialized(false),
|
||||
enableLogging(true),
|
||||
auditd(auditd) {
|
||||
static const char klogd_message[] = "%slogd.klogd: %" PRIu64 "\n";
|
||||
char buffer[sizeof(priority_message) + sizeof(klogd_message) + 20 - 4];
|
||||
snprintf(buffer, sizeof(buffer), klogd_message, priority_message,
|
||||
static const char klogd_message[] = "%s%s%" PRIu64 "\n";
|
||||
char buffer[strlen(priority_message) + strlen(klogdStr) +
|
||||
strlen(klogd_message) + 20];
|
||||
snprintf(buffer, sizeof(buffer), klogd_message, priority_message, klogdStr,
|
||||
signature.nsec());
|
||||
write(fdWrite, buffer, strlen(buffer));
|
||||
}
|
||||
|
@ -237,15 +227,15 @@ bool LogKlog::onDataAvailable(SocketClient* cli) {
|
|||
}
|
||||
|
||||
char buffer[LOGGER_ENTRY_MAX_PAYLOAD];
|
||||
size_t len = 0;
|
||||
ssize_t len = 0;
|
||||
|
||||
for (;;) {
|
||||
ssize_t retval = 0;
|
||||
if ((sizeof(buffer) - 1 - len) > 0) {
|
||||
if (len < (ssize_t)(sizeof(buffer) - 1)) {
|
||||
retval =
|
||||
read(cli->getSocket(), buffer + len, sizeof(buffer) - 1 - len);
|
||||
}
|
||||
if ((retval == 0) && (len == 0)) {
|
||||
if ((retval == 0) && (len <= 0)) {
|
||||
break;
|
||||
}
|
||||
if (retval < 0) {
|
||||
|
@ -255,15 +245,16 @@ bool LogKlog::onDataAvailable(SocketClient* cli) {
|
|||
bool full = len == (sizeof(buffer) - 1);
|
||||
char* ep = buffer + len;
|
||||
*ep = '\0';
|
||||
size_t sublen;
|
||||
for (char *ptr = NULL, *tok = buffer;
|
||||
((tok = log_strntok_r(tok, &len, &ptr, &sublen))); tok = NULL) {
|
||||
ssize_t sublen;
|
||||
for (char *ptr = nullptr, *tok = buffer;
|
||||
!!(tok = android::log_strntok_r(tok, len, ptr, sublen));
|
||||
tok = nullptr) {
|
||||
if (((tok + sublen) >= ep) && (retval != 0) && full) {
|
||||
memmove(buffer, tok, sublen);
|
||||
if (sublen > 0) memmove(buffer, tok, sublen);
|
||||
len = sublen;
|
||||
break;
|
||||
}
|
||||
if (*tok) {
|
||||
if ((sublen > 0) && *tok) {
|
||||
log(tok, sublen);
|
||||
}
|
||||
}
|
||||
|
@ -273,9 +264,12 @@ bool LogKlog::onDataAvailable(SocketClient* cli) {
|
|||
}
|
||||
|
||||
void LogKlog::calculateCorrection(const log_time& monotonic,
|
||||
const char* real_string, size_t len) {
|
||||
const char* real_string, ssize_t len) {
|
||||
static const char real_format[] = "%Y-%m-%d %H:%M:%S.%09q UTC";
|
||||
if (len < (ssize_t)(strlen(real_format) + 5)) return;
|
||||
|
||||
log_time real;
|
||||
const char* ep = real.strptime(real_string, "%Y-%m-%d %H:%M:%S.%09q UTC");
|
||||
const char* ep = real.strptime(real_string, real_format);
|
||||
if (!ep || (ep > &real_string[len]) || (real > log_time(CLOCK_REALTIME))) {
|
||||
return;
|
||||
}
|
||||
|
@ -299,73 +293,50 @@ void LogKlog::calculateCorrection(const log_time& monotonic,
|
|||
}
|
||||
}
|
||||
|
||||
static const char suspendStr[] = "PM: suspend entry ";
|
||||
static const char resumeStr[] = "PM: suspend exit ";
|
||||
static const char suspendedStr[] = "Suspended for ";
|
||||
|
||||
const char* android::strnstr(const char* s, size_t len, const char* needle) {
|
||||
char c;
|
||||
|
||||
if (!len) return NULL;
|
||||
if ((c = *needle++) != 0) {
|
||||
size_t needleLen = strlen(needle);
|
||||
do {
|
||||
do {
|
||||
if (len <= needleLen) return NULL;
|
||||
--len;
|
||||
} while (*s++ != c);
|
||||
} while (fastcmp<memcmp>(s, needle, needleLen));
|
||||
s--;
|
||||
}
|
||||
return s;
|
||||
}
|
||||
|
||||
void LogKlog::sniffTime(log_time& now, const char** buf, size_t len,
|
||||
void LogKlog::sniffTime(log_time& now, const char*& buf, ssize_t len,
|
||||
bool reverse) {
|
||||
const char* cp = now.strptime(*buf, "[ %s.%q]");
|
||||
if (cp && (cp >= &(*buf)[len])) {
|
||||
cp = NULL;
|
||||
if (len <= 0) return;
|
||||
|
||||
const char* cp = nullptr;
|
||||
if ((len > 10) && (*buf == '[')) {
|
||||
cp = now.strptime(buf, "[ %s.%q]"); // can index beyond buffer bounds
|
||||
if (cp && (cp > &buf[len - 1])) cp = nullptr;
|
||||
}
|
||||
if (cp) {
|
||||
static const char healthd[] = "healthd";
|
||||
static const char battery[] = ": battery ";
|
||||
|
||||
len -= cp - *buf;
|
||||
if (len && isspace(*cp)) {
|
||||
len -= cp - buf;
|
||||
if ((len > 0) && isspace(*cp)) {
|
||||
++cp;
|
||||
--len;
|
||||
}
|
||||
*buf = cp;
|
||||
buf = cp;
|
||||
|
||||
if (isMonotonic()) {
|
||||
return;
|
||||
}
|
||||
if (isMonotonic()) return;
|
||||
|
||||
const char* b;
|
||||
if (((b = android::strnstr(cp, len, suspendStr))) &&
|
||||
((size_t)((b += sizeof(suspendStr) - 1) - cp) < len)) {
|
||||
(((b += strlen(suspendStr)) - cp) < len)) {
|
||||
len -= b - cp;
|
||||
calculateCorrection(now, b, len);
|
||||
} else if (((b = android::strnstr(cp, len, resumeStr))) &&
|
||||
((size_t)((b += sizeof(resumeStr) - 1) - cp) < len)) {
|
||||
(((b += strlen(resumeStr)) - cp) < len)) {
|
||||
len -= b - cp;
|
||||
calculateCorrection(now, b, len);
|
||||
} else if (((b = android::strnstr(cp, len, healthd))) &&
|
||||
((size_t)((b += sizeof(healthd) - 1) - cp) < len) &&
|
||||
((b = android::strnstr(b, len -= b - cp, battery))) &&
|
||||
((size_t)((b += sizeof(battery) - 1) - cp) < len)) {
|
||||
} else if (((b = android::strnstr(cp, len, healthdStr))) &&
|
||||
(((b += strlen(healthdStr)) - cp) < len) &&
|
||||
((b = android::strnstr(b, len -= b - cp, batteryStr))) &&
|
||||
(((b += strlen(batteryStr)) - cp) < len)) {
|
||||
// NB: healthd is roughly 150us late, so we use it instead to
|
||||
// trigger a check for ntp-induced or hardware clock drift.
|
||||
log_time real(CLOCK_REALTIME);
|
||||
log_time mono(CLOCK_MONOTONIC);
|
||||
correction = (real < mono) ? log_time::EPOCH : (real - mono);
|
||||
} else if (((b = android::strnstr(cp, len, suspendedStr))) &&
|
||||
((size_t)((b += sizeof(suspendStr) - 1) - cp) < len)) {
|
||||
(((b += strlen(suspendStr)) - cp) < len)) {
|
||||
len -= b - cp;
|
||||
log_time real;
|
||||
char* endp;
|
||||
real.tv_sec = strtol(b, &endp, 10);
|
||||
if ((*endp == '.') && ((size_t)(endp - b) < len)) {
|
||||
if ((*endp == '.') && ((endp - b) < len)) {
|
||||
unsigned long multiplier = NS_PER_SEC;
|
||||
real.tv_nsec = 0;
|
||||
len -= endp - b;
|
||||
|
@ -394,8 +365,15 @@ void LogKlog::sniffTime(log_time& now, const char** buf, size_t len,
|
|||
}
|
||||
}
|
||||
|
||||
pid_t LogKlog::sniffPid(const char** buf, size_t len) {
|
||||
const char* cp = *buf;
|
||||
pid_t LogKlog::sniffPid(const char*& buf, ssize_t len) {
|
||||
if (len <= 0) return 0;
|
||||
|
||||
const char* cp = buf;
|
||||
// sscanf does a strlen, let's check if the string is not nul terminated.
|
||||
// pseudo out-of-bounds access since we always have an extra char on buffer.
|
||||
if (((ssize_t)strnlen(cp, len) == len) && cp[len]) {
|
||||
return 0;
|
||||
}
|
||||
// HTC kernels with modified printk "c0 1648 "
|
||||
if ((len > 9) && (cp[0] == 'c') && isdigit(cp[1]) &&
|
||||
(isdigit(cp[2]) || (cp[2] == ' ')) && (cp[3] == ' ')) {
|
||||
|
@ -412,7 +390,7 @@ pid_t LogKlog::sniffPid(const char** buf, size_t len) {
|
|||
int pid = 0;
|
||||
char dummy;
|
||||
if (sscanf(cp + 4, "%d%c", &pid, &dummy) == 2) {
|
||||
*buf = cp + 10; // skip-it-all
|
||||
buf = cp + 10; // skip-it-all
|
||||
return pid;
|
||||
}
|
||||
}
|
||||
|
@ -434,28 +412,28 @@ pid_t LogKlog::sniffPid(const char** buf, size_t len) {
|
|||
}
|
||||
|
||||
// kernel log prefix, convert to a kernel log priority number
|
||||
static int parseKernelPrio(const char** buf, size_t len) {
|
||||
static int parseKernelPrio(const char*& buf, ssize_t len) {
|
||||
int pri = LOG_USER | LOG_INFO;
|
||||
const char* cp = *buf;
|
||||
if (len && (*cp == '<')) {
|
||||
const char* cp = buf;
|
||||
if ((len > 0) && (*cp == '<')) {
|
||||
pri = 0;
|
||||
while (--len && isdigit(*++cp)) {
|
||||
pri = (pri * 10) + *cp - '0';
|
||||
}
|
||||
if (len && (*cp == '>')) {
|
||||
if ((len > 0) && (*cp == '>')) {
|
||||
++cp;
|
||||
} else {
|
||||
cp = *buf;
|
||||
cp = buf;
|
||||
pri = LOG_USER | LOG_INFO;
|
||||
}
|
||||
*buf = cp;
|
||||
buf = cp;
|
||||
}
|
||||
return pri;
|
||||
}
|
||||
|
||||
// Passed the entire SYSLOG_ACTION_READ_ALL buffer and interpret a
|
||||
// compensated start time.
|
||||
void LogKlog::synchronize(const char* buf, size_t len) {
|
||||
void LogKlog::synchronize(const char* buf, ssize_t len) {
|
||||
const char* cp = android::strnstr(buf, len, suspendStr);
|
||||
if (!cp) {
|
||||
cp = android::strnstr(buf, len, resumeStr);
|
||||
|
@ -471,10 +449,10 @@ void LogKlog::synchronize(const char* buf, size_t len) {
|
|||
if (*cp == '\n') {
|
||||
++cp;
|
||||
}
|
||||
parseKernelPrio(&cp, len - (cp - buf));
|
||||
parseKernelPrio(cp, len - (cp - buf));
|
||||
|
||||
log_time now;
|
||||
sniffTime(now, &cp, len - (cp - buf), true);
|
||||
sniffTime(now, cp, len - (cp - buf), true);
|
||||
|
||||
const char* suspended = android::strnstr(buf, len, suspendedStr);
|
||||
if (!suspended || (suspended > cp)) {
|
||||
|
@ -488,9 +466,9 @@ void LogKlog::synchronize(const char* buf, size_t len) {
|
|||
if (*cp == '\n') {
|
||||
++cp;
|
||||
}
|
||||
parseKernelPrio(&cp, len - (cp - buf));
|
||||
parseKernelPrio(cp, len - (cp - buf));
|
||||
|
||||
sniffTime(now, &cp, len - (cp - buf), true);
|
||||
sniffTime(now, cp, len - (cp - buf), true);
|
||||
}
|
||||
|
||||
// Convert kernel log priority number into an Android Logger priority number
|
||||
|
@ -523,9 +501,9 @@ static int convertKernelPrioToAndroidPrio(int pri) {
|
|||
return ANDROID_LOG_INFO;
|
||||
}
|
||||
|
||||
static const char* strnrchr(const char* s, size_t len, char c) {
|
||||
const char* save = NULL;
|
||||
for (; len; ++s, len--) {
|
||||
static const char* strnrchr(const char* s, ssize_t len, char c) {
|
||||
const char* save = nullptr;
|
||||
for (; len > 0; ++s, len--) {
|
||||
if (*s == c) {
|
||||
save = s;
|
||||
}
|
||||
|
@ -566,22 +544,21 @@ static const char* strnrchr(const char* s, size_t len, char c) {
|
|||
// logd.klogd:
|
||||
// return -1 if message logd.klogd: <signature>
|
||||
//
|
||||
int LogKlog::log(const char* buf, size_t len) {
|
||||
if (auditd && android::strnstr(buf, len, " audit(")) {
|
||||
int LogKlog::log(const char* buf, ssize_t len) {
|
||||
if (auditd && android::strnstr(buf, len, auditStr)) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
const char* p = buf;
|
||||
int pri = parseKernelPrio(&p, len);
|
||||
int pri = parseKernelPrio(p, len);
|
||||
|
||||
log_time now;
|
||||
sniffTime(now, &p, len - (p - buf), false);
|
||||
sniffTime(now, p, len - (p - buf), false);
|
||||
|
||||
// sniff for start marker
|
||||
const char klogd_message[] = "logd.klogd: ";
|
||||
const char* start = android::strnstr(p, len - (p - buf), klogd_message);
|
||||
const char* start = android::strnstr(p, len - (p - buf), klogdStr);
|
||||
if (start) {
|
||||
uint64_t sig = strtoll(start + sizeof(klogd_message) - 1, NULL, 10);
|
||||
uint64_t sig = strtoll(start + strlen(klogdStr), nullptr, 10);
|
||||
if (sig == signature.nsec()) {
|
||||
if (initialized) {
|
||||
enableLogging = true;
|
||||
|
@ -598,7 +575,7 @@ int LogKlog::log(const char* buf, size_t len) {
|
|||
}
|
||||
|
||||
// Parse pid, tid and uid
|
||||
const pid_t pid = sniffPid(&p, len - (p - buf));
|
||||
const pid_t pid = sniffPid(p, len - (p - buf));
|
||||
const pid_t tid = pid;
|
||||
uid_t uid = AID_ROOT;
|
||||
if (pid) {
|
||||
|
@ -620,11 +597,11 @@ int LogKlog::log(const char* buf, size_t len) {
|
|||
start = p;
|
||||
const char* tag = "";
|
||||
const char* etag = tag;
|
||||
size_t taglen = len - (p - buf);
|
||||
ssize_t taglen = len - (p - buf);
|
||||
const char* bt = p;
|
||||
|
||||
static const char infoBrace[] = "[INFO]";
|
||||
static const size_t infoBraceLen = strlen(infoBrace);
|
||||
static const ssize_t infoBraceLen = strlen(infoBrace);
|
||||
if ((taglen >= infoBraceLen) &&
|
||||
!fastcmp<strncmp>(p, infoBrace, infoBraceLen)) {
|
||||
// <PRI>[<TIME>] "[INFO]"<tag> ":" message
|
||||
|
@ -633,26 +610,26 @@ int LogKlog::log(const char* buf, size_t len) {
|
|||
}
|
||||
|
||||
const char* et;
|
||||
for (et = bt; taglen && *et && (*et != ':') && !isspace(*et);
|
||||
for (et = bt; (taglen > 0) && *et && (*et != ':') && !isspace(*et);
|
||||
++et, --taglen) {
|
||||
// skip ':' within [ ... ]
|
||||
if (*et == '[') {
|
||||
while (taglen && *et && *et != ']') {
|
||||
while ((taglen > 0) && *et && *et != ']') {
|
||||
++et;
|
||||
--taglen;
|
||||
}
|
||||
if (!taglen) {
|
||||
if (taglen <= 0) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
const char* cp;
|
||||
for (cp = et; taglen && isspace(*cp); ++cp, --taglen) {
|
||||
for (cp = et; (taglen > 0) && isspace(*cp); ++cp, --taglen) {
|
||||
}
|
||||
|
||||
// Validate tag
|
||||
size_t size = et - bt;
|
||||
if (taglen && size) {
|
||||
ssize_t size = et - bt;
|
||||
if ((taglen > 0) && (size > 0)) {
|
||||
if (*cp == ':') {
|
||||
// ToDo: handle case insensitive colon separated logging stutter:
|
||||
// <tag> : <tag>: ...
|
||||
|
@ -672,12 +649,12 @@ int LogKlog::log(const char* buf, size_t len) {
|
|||
const char* b = cp;
|
||||
cp += size;
|
||||
taglen -= size;
|
||||
while (--taglen && !isspace(*++cp) && (*cp != ':')) {
|
||||
while ((--taglen > 0) && !isspace(*++cp) && (*cp != ':')) {
|
||||
}
|
||||
const char* e;
|
||||
for (e = cp; taglen && isspace(*cp); ++cp, --taglen) {
|
||||
for (e = cp; (taglen > 0) && isspace(*cp); ++cp, --taglen) {
|
||||
}
|
||||
if (taglen && (*cp == ':')) {
|
||||
if ((taglen > 0) && (*cp == ':')) {
|
||||
tag = b;
|
||||
etag = e;
|
||||
p = cp + 1;
|
||||
|
@ -685,7 +662,7 @@ int LogKlog::log(const char* buf, size_t len) {
|
|||
} else {
|
||||
// what about <PRI>[<TIME>] <tag>_host '<tag><stuff>' : message
|
||||
static const char host[] = "_host";
|
||||
static const size_t hostlen = strlen(host);
|
||||
static const ssize_t hostlen = strlen(host);
|
||||
if ((size > hostlen) &&
|
||||
!fastcmp<strncmp>(bt + size - hostlen, host, hostlen) &&
|
||||
!fastcmp<strncmp>(bt + 1, cp + 1, size - hostlen - 1)) {
|
||||
|
@ -693,12 +670,14 @@ int LogKlog::log(const char* buf, size_t len) {
|
|||
cp += size - hostlen;
|
||||
taglen -= size - hostlen;
|
||||
if (*cp == '.') {
|
||||
while (--taglen && !isspace(*++cp) && (*cp != ':')) {
|
||||
while ((--taglen > 0) && !isspace(*++cp) &&
|
||||
(*cp != ':')) {
|
||||
}
|
||||
const char* e;
|
||||
for (e = cp; taglen && isspace(*cp); ++cp, --taglen) {
|
||||
for (e = cp; (taglen > 0) && isspace(*cp);
|
||||
++cp, --taglen) {
|
||||
}
|
||||
if (taglen && (*cp == ':')) {
|
||||
if ((taglen > 0) && (*cp == ':')) {
|
||||
tag = b;
|
||||
etag = e;
|
||||
p = cp + 1;
|
||||
|
@ -711,13 +690,13 @@ int LogKlog::log(const char* buf, size_t len) {
|
|||
} else {
|
||||
// <PRI>[<TIME>] <tag> <stuff>' : message
|
||||
twoWord:
|
||||
while (--taglen && !isspace(*++cp) && (*cp != ':')) {
|
||||
while ((--taglen > 0) && !isspace(*++cp) && (*cp != ':')) {
|
||||
}
|
||||
const char* e;
|
||||
for (e = cp; taglen && isspace(*cp); ++cp, --taglen) {
|
||||
for (e = cp; (taglen > 0) && isspace(*cp); ++cp, --taglen) {
|
||||
}
|
||||
// Two words
|
||||
if (taglen && (*cp == ':')) {
|
||||
if ((taglen > 0) && (*cp == ':')) {
|
||||
tag = bt;
|
||||
etag = e;
|
||||
p = cp + 1;
|
||||
|
@ -726,13 +705,13 @@ int LogKlog::log(const char* buf, size_t len) {
|
|||
} // else no tag
|
||||
|
||||
static const char cpu[] = "CPU";
|
||||
static const size_t cpuLen = strlen(cpu);
|
||||
static const ssize_t cpuLen = strlen(cpu);
|
||||
static const char warning[] = "WARNING";
|
||||
static const size_t warningLen = strlen(warning);
|
||||
static const ssize_t warningLen = strlen(warning);
|
||||
static const char error[] = "ERROR";
|
||||
static const size_t errorLen = strlen(error);
|
||||
static const ssize_t errorLen = strlen(error);
|
||||
static const char info[] = "INFO";
|
||||
static const size_t infoLen = strlen(info);
|
||||
static const ssize_t infoLen = strlen(info);
|
||||
|
||||
size = etag - tag;
|
||||
if ((size <= 1) ||
|
||||
|
@ -756,13 +735,13 @@ int LogKlog::log(const char* buf, size_t len) {
|
|||
// Mediatek-special printk induced stutter
|
||||
const char* mp = strnrchr(tag, taglen, ']');
|
||||
if (mp && (++mp < etag)) {
|
||||
size_t s = etag - mp;
|
||||
ssize_t s = etag - mp;
|
||||
if (((s + s) < taglen) && !fastcmp<memcmp>(mp, mp - 1 - s, s)) {
|
||||
taglen = mp - tag;
|
||||
}
|
||||
}
|
||||
// Deal with sloppy and simplistic harmless p = cp + 1 etc above.
|
||||
if (len < (size_t)(p - buf)) {
|
||||
if (len < (p - buf)) {
|
||||
p = &buf[len];
|
||||
}
|
||||
// skip leading space
|
||||
|
@ -770,12 +749,12 @@ int LogKlog::log(const char* buf, size_t len) {
|
|||
++p;
|
||||
}
|
||||
// truncate trailing space or nuls
|
||||
size_t b = len - (p - buf);
|
||||
while (b && (isspace(p[b - 1]) || !p[b - 1])) {
|
||||
ssize_t b = len - (p - buf);
|
||||
while ((b > 0) && (isspace(p[b - 1]) || !p[b - 1])) {
|
||||
--b;
|
||||
}
|
||||
// trick ... allow tag with empty content to be logged. log() drops empty
|
||||
if (!b && taglen) {
|
||||
if ((b <= 0) && (taglen > 0)) {
|
||||
p = " ";
|
||||
b = 1;
|
||||
}
|
||||
|
@ -787,9 +766,9 @@ int LogKlog::log(const char* buf, size_t len) {
|
|||
taglen = LOGGER_ENTRY_MAX_PAYLOAD;
|
||||
}
|
||||
// calculate buffer copy requirements
|
||||
size_t n = 1 + taglen + 1 + b + 1;
|
||||
ssize_t n = 1 + taglen + 1 + b + 1;
|
||||
// paranoid sanity check, first two just can not happen ...
|
||||
if ((taglen > n) || (b > n) || (n > USHRT_MAX)) {
|
||||
if ((taglen > n) || (b > n) || (n > (ssize_t)USHRT_MAX) || (n <= 0)) {
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
|
|
|
@ -20,8 +20,6 @@
|
|||
#include <private/android_logger.h>
|
||||
#include <sysutils/SocketListener.h>
|
||||
|
||||
char* log_strntok_r(char* s, size_t* len, char** saveptr, size_t* sublen);
|
||||
|
||||
class LogBuffer;
|
||||
class LogReader;
|
||||
|
||||
|
@ -43,8 +41,8 @@ class LogKlog : public SocketListener {
|
|||
public:
|
||||
LogKlog(LogBuffer* buf, LogReader* reader, int fdWrite, int fdRead,
|
||||
bool auditd);
|
||||
int log(const char* buf, size_t len);
|
||||
void synchronize(const char* buf, size_t len);
|
||||
int log(const char* buf, ssize_t len);
|
||||
void synchronize(const char* buf, ssize_t len);
|
||||
|
||||
bool isMonotonic() {
|
||||
return logbuf->isMonotonic();
|
||||
|
@ -57,10 +55,10 @@ class LogKlog : public SocketListener {
|
|||
}
|
||||
|
||||
protected:
|
||||
void sniffTime(log_time& now, const char** buf, size_t len, bool reverse);
|
||||
pid_t sniffPid(const char** buf, size_t len);
|
||||
void sniffTime(log_time& now, const char*& buf, ssize_t len, bool reverse);
|
||||
pid_t sniffPid(const char*& buf, ssize_t len);
|
||||
void calculateCorrection(const log_time& monotonic, const char* real_string,
|
||||
size_t len);
|
||||
ssize_t len);
|
||||
virtual bool onDataAvailable(SocketClient* cli);
|
||||
};
|
||||
|
||||
|
|
|
@ -43,8 +43,25 @@ char* tidToName(pid_t tid);
|
|||
const char* tagToName(uint32_t tag);
|
||||
void ReReadEventLogTags();
|
||||
|
||||
// Furnished by LogKlog.cpp.
|
||||
const char* strnstr(const char* s, size_t len, const char* needle);
|
||||
// Furnished by LogKlog.cpp
|
||||
char* log_strntok_r(char* s, ssize_t& len, char*& saveptr, ssize_t& sublen);
|
||||
|
||||
// needle should reference a string longer than 1 character
|
||||
static inline const char* strnstr(const char* s, ssize_t len,
|
||||
const char* needle) {
|
||||
if (len <= 0) return nullptr;
|
||||
|
||||
const char c = *needle++;
|
||||
const size_t needleLen = strlen(needle);
|
||||
do {
|
||||
do {
|
||||
if (len <= (ssize_t)needleLen) return nullptr;
|
||||
--len;
|
||||
} while (*s++ != c);
|
||||
} while (fastcmp<memcmp>(s, needle, needleLen));
|
||||
s--;
|
||||
return s;
|
||||
}
|
||||
}
|
||||
|
||||
// Furnished in LogCommand.cpp
|
||||
|
|
|
@ -221,7 +221,7 @@ static char* name;
|
|||
|
||||
static sem_t reinit;
|
||||
static bool reinit_running = false;
|
||||
static LogBuffer* logBuf = NULL;
|
||||
static LogBuffer* logBuf = nullptr;
|
||||
|
||||
static bool package_list_parser_cb(pkg_info* info, void* /* userdata */) {
|
||||
bool rc = true;
|
||||
|
@ -254,9 +254,9 @@ static void* reinit_thread_start(void* /*obj*/) {
|
|||
while (reinit_running && !sem_wait(&reinit) && reinit_running) {
|
||||
// uidToName Privileged Worker
|
||||
if (uid) {
|
||||
name = NULL;
|
||||
name = nullptr;
|
||||
|
||||
packagelist_parse(package_list_parser_cb, NULL);
|
||||
packagelist_parse(package_list_parser_cb, nullptr);
|
||||
|
||||
uid = 0;
|
||||
sem_post(&uidName);
|
||||
|
@ -291,19 +291,19 @@ static void* reinit_thread_start(void* /*obj*/) {
|
|||
// Anything that reads persist.<property>
|
||||
if (logBuf) {
|
||||
logBuf->init();
|
||||
logBuf->initPrune(NULL);
|
||||
logBuf->initPrune(nullptr);
|
||||
}
|
||||
android::ReReadEventLogTags();
|
||||
}
|
||||
|
||||
return NULL;
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
static sem_t sem_name;
|
||||
|
||||
char* android::uidToName(uid_t u) {
|
||||
if (!u || !reinit_running) {
|
||||
return NULL;
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
sem_wait(&sem_name);
|
||||
|
@ -311,7 +311,7 @@ char* android::uidToName(uid_t u) {
|
|||
// Not multi-thread safe, we use sem_name to protect
|
||||
uid = u;
|
||||
|
||||
name = NULL;
|
||||
name = nullptr;
|
||||
sem_post(&reinit);
|
||||
sem_wait(&uidName);
|
||||
char* ret = name;
|
||||
|
@ -332,12 +332,13 @@ static void readDmesg(LogAudit* al, LogKlog* kl) {
|
|||
return;
|
||||
}
|
||||
|
||||
int rc = klogctl(KLOG_SIZE_BUFFER, NULL, 0);
|
||||
int rc = klogctl(KLOG_SIZE_BUFFER, nullptr, 0);
|
||||
if (rc <= 0) {
|
||||
return;
|
||||
}
|
||||
|
||||
size_t len = rc + 1024; // Margin for additional input race or trailing nul
|
||||
// Margin for additional input race or trailing nul
|
||||
ssize_t len = rc + 1024;
|
||||
std::unique_ptr<char[]> buf(new char[len]);
|
||||
|
||||
rc = klogctl(KLOG_READ_ALL, buf.get(), len);
|
||||
|
@ -345,7 +346,7 @@ static void readDmesg(LogAudit* al, LogKlog* kl) {
|
|||
return;
|
||||
}
|
||||
|
||||
if ((size_t)rc < len) {
|
||||
if (rc < len) {
|
||||
len = rc + 1;
|
||||
}
|
||||
buf[--len] = '\0';
|
||||
|
@ -354,10 +355,11 @@ static void readDmesg(LogAudit* al, LogKlog* kl) {
|
|||
kl->synchronize(buf.get(), len);
|
||||
}
|
||||
|
||||
size_t sublen;
|
||||
for (char *ptr = NULL, *tok = buf.get();
|
||||
(rc >= 0) && ((tok = log_strntok_r(tok, &len, &ptr, &sublen)));
|
||||
tok = NULL) {
|
||||
ssize_t sublen;
|
||||
for (char *ptr = nullptr, *tok = buf.get();
|
||||
(rc >= 0) && !!(tok = android::log_strntok_r(tok, len, ptr, sublen));
|
||||
tok = nullptr) {
|
||||
if ((sublen <= 0) || !*tok) continue;
|
||||
if (al) {
|
||||
rc = al->log(tok, sublen);
|
||||
}
|
||||
|
@ -444,7 +446,7 @@ int main(int argc, char* argv[]) {
|
|||
if (!pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED)) {
|
||||
pthread_t thread;
|
||||
reinit_running = true;
|
||||
if (pthread_create(&thread, &attr, reinit_thread_start, NULL)) {
|
||||
if (pthread_create(&thread, &attr, reinit_thread_start, nullptr)) {
|
||||
reinit_running = false;
|
||||
}
|
||||
}
|
||||
|
@ -507,7 +509,7 @@ int main(int argc, char* argv[]) {
|
|||
// initiated log messages. New log entries are added to LogBuffer
|
||||
// and LogReader is notified to send updates to connected clients.
|
||||
|
||||
LogAudit* al = NULL;
|
||||
LogAudit* al = nullptr;
|
||||
if (auditd) {
|
||||
al = new LogAudit(logBuf, reader,
|
||||
__android_logger_property_get_bool(
|
||||
|
@ -516,9 +518,9 @@ int main(int argc, char* argv[]) {
|
|||
: -1);
|
||||
}
|
||||
|
||||
LogKlog* kl = NULL;
|
||||
LogKlog* kl = nullptr;
|
||||
if (klogd) {
|
||||
kl = new LogKlog(logBuf, reader, fdDmesg, fdPmesg, al != NULL);
|
||||
kl = new LogKlog(logBuf, reader, fdDmesg, fdPmesg, al != nullptr);
|
||||
}
|
||||
|
||||
readDmesg(al, kl);
|
||||
|
|
Loading…
Reference in a new issue