1 /*
2 * Copyright (C) 2012 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17 #define LOG_TAG "common_time"
18 #include <utils/Log.h>
19
20 #include "utils.h"
21
22 namespace android {
23
setTimeout(int msec)24 void Timeout::setTimeout(int msec) {
25 if (msec < 0) {
26 mSystemEndTime = 0;
27 return;
28 }
29
30 mSystemEndTime = systemTime() + (static_cast<nsecs_t>(msec) * 1000000);
31 }
32
msecTillTimeout(nsecs_t nowTime)33 int Timeout::msecTillTimeout(nsecs_t nowTime) {
34 if (!mSystemEndTime) {
35 return -1;
36 }
37
38 if (mSystemEndTime < nowTime) {
39 return 0;
40 }
41
42 nsecs_t delta = mSystemEndTime - nowTime;
43 delta += 999999;
44 delta /= 1000000;
45 if (delta > 0x7FFFFFFF) {
46 return 0x7FFFFFFF;
47 }
48
49 return static_cast<int>(delta);
50 }
51
LogRing(const char * header,size_t entries)52 LogRing::LogRing(const char* header, size_t entries)
53 : mSize(entries)
54 , mWr(0)
55 , mIsFull(false)
56 , mHeader(header) {
57 mRingBuffer = new Entry[mSize];
58 if (NULL == mRingBuffer)
59 ALOGE("Failed to allocate log ring with %zu entries.", mSize);
60 }
61
~LogRing()62 LogRing::~LogRing() {
63 if (NULL != mRingBuffer)
64 delete[] mRingBuffer;
65 }
66
log(int prio,const char * tag,const char * fmt,...)67 void LogRing::log(int prio, const char* tag, const char* fmt, ...) {
68 va_list argp;
69 va_start(argp, fmt);
70 internalLog(prio, tag, fmt, argp);
71 va_end(argp);
72 }
73
log(const char * fmt,...)74 void LogRing::log(const char* fmt, ...) {
75 va_list argp;
76 va_start(argp, fmt);
77 internalLog(0, NULL, fmt, argp);
78 va_end(argp);
79 }
80
internalLog(int prio,const char * tag,const char * fmt,va_list argp)81 void LogRing::internalLog(int prio,
82 const char* tag,
83 const char* fmt,
84 va_list argp) {
85 if (NULL != mRingBuffer) {
86 Mutex::Autolock lock(&mLock);
87 String8 s(String8::formatV(fmt, argp));
88 Entry* last = NULL;
89
90 if (mIsFull || mWr)
91 last = &(mRingBuffer[(mWr + mSize - 1) % mSize]);
92
93
94 if ((NULL != last) && !last->s.compare(s)) {
95 gettimeofday(&(last->last_ts), NULL);
96 ++last->count;
97 } else {
98 gettimeofday(&mRingBuffer[mWr].first_ts, NULL);
99 mRingBuffer[mWr].last_ts = mRingBuffer[mWr].first_ts;
100 mRingBuffer[mWr].count = 1;
101 mRingBuffer[mWr].s.setTo(s);
102
103 mWr = (mWr + 1) % mSize;
104 if (!mWr)
105 mIsFull = true;
106 }
107 }
108
109 if (NULL != tag)
110 LOG_PRI_VA(prio, tag, fmt, argp);
111 }
112
dumpLog(int fd)113 void LogRing::dumpLog(int fd) {
114 if (NULL == mRingBuffer)
115 return;
116
117 Mutex::Autolock lock(&mLock);
118
119 if (!mWr && !mIsFull)
120 return;
121
122 char buf[1024];
123 int res;
124 size_t start = mIsFull ? mWr : 0;
125 size_t count = mIsFull ? mSize : mWr;
126 static const char* kTimeFmt = "%a %b %d %Y %H:%M:%S";
127
128 res = snprintf(buf, sizeof(buf), "\n%s\n", mHeader);
129 if (res > 0)
130 write(fd, buf, res);
131
132 for (size_t i = 0; i < count; ++i) {
133 struct tm t;
134 char timebuf[64];
135 char repbuf[96];
136 size_t ndx = (start + i) % mSize;
137
138 if (1 != mRingBuffer[ndx].count) {
139 localtime_r(&mRingBuffer[ndx].last_ts.tv_sec, &t);
140 strftime(timebuf, sizeof(timebuf), kTimeFmt, &t);
141 snprintf(repbuf, sizeof(repbuf),
142 " (repeated %d times, last was %s.%03ld)",
143 mRingBuffer[ndx].count,
144 timebuf,
145 mRingBuffer[ndx].last_ts.tv_usec / 1000);
146 repbuf[sizeof(repbuf) - 1] = 0;
147 } else {
148 repbuf[0] = 0;
149 }
150
151 localtime_r(&mRingBuffer[ndx].first_ts.tv_sec, &t);
152 strftime(timebuf, sizeof(timebuf), kTimeFmt, &t);
153 res = snprintf(buf, sizeof(buf), "[%2zu] %s.%03ld :: %s%s\n",
154 i, timebuf,
155 mRingBuffer[ndx].first_ts.tv_usec / 1000,
156 mRingBuffer[ndx].s.string(),
157 repbuf);
158
159 if (res > 0)
160 write(fd, buf, res);
161 }
162 }
163
164 } // namespace android
165