blob: e651b4fec667731f28534e41351696969e0302bf [file] [log] [blame]
Tom Cherry43f3f762020-05-14 22:28:09 -07001/*
Tom Cherrye18346d2020-05-21 12:13:20 -07002 * Copyright (C) 2020 The Android Open Source Project
Tom Cherry43f3f762020-05-14 22:28:09 -07003 *
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
Tom Cherrye18346d2020-05-21 12:13:20 -070017#include "LogBufferTest.h"
Tom Cherry43f3f762020-05-14 22:28:09 -070018
19#include <unistd.h>
20
Tom Cherryb398a7c2020-05-20 12:09:22 -070021#include <limits>
Tom Cherry43f3f762020-05-14 22:28:09 -070022#include <memory>
Tom Cherryb398a7c2020-05-20 12:09:22 -070023#include <regex>
Tom Cherry43f3f762020-05-14 22:28:09 -070024#include <vector>
25
26#include <android-base/stringprintf.h>
27#include <android-base/strings.h>
Tom Cherry43f3f762020-05-14 22:28:09 -070028
Tom Cherry43f3f762020-05-14 22:28:09 -070029#include "LogReaderThread.h"
Tom Cherry43f3f762020-05-14 22:28:09 -070030#include "LogWriter.h"
31
32using android::base::Join;
Tom Cherryb398a7c2020-05-20 12:09:22 -070033using android::base::Split;
Tom Cherry43f3f762020-05-14 22:28:09 -070034using android::base::StringPrintf;
35
Tom Cherrya5151972020-05-15 11:39:58 -070036#ifndef __ANDROID__
37unsigned long __android_logger_get_buffer_size(log_id_t) {
38 return 1024 * 1024;
39}
40
41bool __android_logger_valid_buffer_size(unsigned long) {
42 return true;
43}
44#endif
45
Tom Cherry43f3f762020-05-14 22:28:09 -070046char* android::uidToName(uid_t) {
47 return nullptr;
48}
49
Tom Cherrye18346d2020-05-21 12:13:20 -070050static std::vector<std::string> CompareLoggerEntries(const logger_entry& expected,
51 const logger_entry& result, bool ignore_len) {
Tom Cherryb398a7c2020-05-20 12:09:22 -070052 std::vector<std::string> errors;
53 if (!ignore_len && expected.len != result.len) {
54 errors.emplace_back(
55 StringPrintf("len: expected %" PRIu16 " vs %" PRIu16, expected.len, result.len));
56 }
57 if (expected.hdr_size != result.hdr_size) {
58 errors.emplace_back(StringPrintf("hdr_size: %" PRIu16 " vs %" PRIu16, expected.hdr_size,
59 result.hdr_size));
60 }
61 if (expected.pid != result.pid) {
62 errors.emplace_back(
63 StringPrintf("pid: expected %" PRIi32 " vs %" PRIi32, expected.pid, result.pid));
64 }
65 if (expected.tid != result.tid) {
66 errors.emplace_back(
67 StringPrintf("tid: expected %" PRIu32 " vs %" PRIu32, expected.tid, result.tid));
68 }
69 if (expected.sec != result.sec) {
70 errors.emplace_back(
71 StringPrintf("sec: expected %" PRIu32 " vs %" PRIu32, expected.sec, result.sec));
72 }
73 if (expected.nsec != result.nsec) {
74 errors.emplace_back(
75 StringPrintf("nsec: expected %" PRIu32 " vs %" PRIu32, expected.nsec, result.nsec));
76 }
77 if (expected.lid != result.lid) {
78 errors.emplace_back(
79 StringPrintf("lid: expected %" PRIu32 " vs %" PRIu32, expected.lid, result.lid));
80 }
81 if (expected.uid != result.uid) {
82 errors.emplace_back(
83 StringPrintf("uid: expected %" PRIu32 " vs %" PRIu32, expected.uid, result.uid));
84 }
85 return errors;
86}
87
Tom Cherrye18346d2020-05-21 12:13:20 -070088static std::string MakePrintable(std::string in) {
Tom Cherryb398a7c2020-05-20 12:09:22 -070089 if (in.size() > 80) {
90 in = in.substr(0, 80) + "...";
91 }
92 std::string result;
93 for (const char c : in) {
94 if (isprint(c)) {
95 result.push_back(c);
96 } else {
97 result.append(StringPrintf("\\%02x", static_cast<int>(c) & 0xFF));
98 }
99 }
100 return result;
101}
102
Tom Cherrye18346d2020-05-21 12:13:20 -0700103static std::string CompareMessages(const std::string& expected, const std::string& result) {
Tom Cherryb398a7c2020-05-20 12:09:22 -0700104 if (expected == result) {
105 return {};
106 }
107 size_t diff_index = 0;
108 for (; diff_index < std::min(expected.size(), result.size()); ++diff_index) {
109 if (expected[diff_index] != result[diff_index]) {
110 break;
111 }
112 }
113
Tom Cherryb3e16332020-05-28 20:02:42 -0700114 if (diff_index < 80) {
Tom Cherryb398a7c2020-05-20 12:09:22 -0700115 auto expected_short = MakePrintable(expected);
116 auto result_short = MakePrintable(result);
117 return StringPrintf("msg: expected '%s' vs '%s'", expected_short.c_str(),
118 result_short.c_str());
119 }
120
121 auto expected_short = MakePrintable(expected.substr(diff_index));
122 auto result_short = MakePrintable(result.substr(diff_index));
123 return StringPrintf("msg: index %zu: expected '%s' vs '%s'", diff_index, expected_short.c_str(),
124 result_short.c_str());
125}
126
Tom Cherrye18346d2020-05-21 12:13:20 -0700127static std::string CompareRegexMessages(const std::string& expected, const std::string& result) {
Tom Cherryb398a7c2020-05-20 12:09:22 -0700128 auto expected_pieces = Split(expected, std::string("\0", 1));
129 auto result_pieces = Split(result, std::string("\0", 1));
130
131 if (expected_pieces.size() != 3 || result_pieces.size() != 3) {
132 return StringPrintf(
133 "msg: should have 3 null delimited strings found %d in expected, %d in result: "
134 "'%s' vs '%s'",
135 static_cast<int>(expected_pieces.size()), static_cast<int>(result_pieces.size()),
136 MakePrintable(expected).c_str(), MakePrintable(result).c_str());
137 }
138 if (expected_pieces[0] != result_pieces[0]) {
139 return StringPrintf("msg: tag/priority mismatch expected '%s' vs '%s'",
140 MakePrintable(expected_pieces[0]).c_str(),
141 MakePrintable(result_pieces[0]).c_str());
142 }
143 std::regex expected_tag_regex(expected_pieces[1]);
144 if (!std::regex_search(result_pieces[1], expected_tag_regex)) {
145 return StringPrintf("msg: message regex mismatch expected '%s' vs '%s'",
146 MakePrintable(expected_pieces[1]).c_str(),
147 MakePrintable(result_pieces[1]).c_str());
148 }
149 if (expected_pieces[2] != result_pieces[2]) {
150 return StringPrintf("msg: nothing expected after final null character '%s' vs '%s'",
151 MakePrintable(expected_pieces[2]).c_str(),
152 MakePrintable(result_pieces[2]).c_str());
153 }
154 return {};
155}
156
157void CompareLogMessages(const std::vector<LogMessage>& expected,
158 const std::vector<LogMessage>& result) {
159 EXPECT_EQ(expected.size(), result.size());
160 size_t end = std::min(expected.size(), result.size());
161 size_t num_errors = 0;
162 for (size_t i = 0; i < end; ++i) {
163 auto errors =
164 CompareLoggerEntries(expected[i].entry, result[i].entry, expected[i].regex_compare);
165 auto msg_error = expected[i].regex_compare
166 ? CompareRegexMessages(expected[i].message, result[i].message)
167 : CompareMessages(expected[i].message, result[i].message);
168 if (!msg_error.empty()) {
169 errors.emplace_back(msg_error);
170 }
171 if (!errors.empty()) {
172 GTEST_LOG_(ERROR) << "Mismatch log message " << i << "\n" << Join(errors, "\n");
173 ++num_errors;
174 }
175 }
176 EXPECT_EQ(0U, num_errors);
177}
178
Tom Cherrye18346d2020-05-21 12:13:20 -0700179void FixupMessages(std::vector<LogMessage>* messages) {
180 for (auto& [entry, message, _] : *messages) {
181 entry.hdr_size = sizeof(logger_entry);
182 entry.len = message.size();
Tom Cherry43f3f762020-05-14 22:28:09 -0700183 }
Tom Cherrye18346d2020-05-21 12:13:20 -0700184}
Tom Cherry43f3f762020-05-14 22:28:09 -0700185
Tom Cherrye18346d2020-05-21 12:13:20 -0700186TEST_P(LogBufferTest, smoke) {
Tom Cherryb398a7c2020-05-20 12:09:22 -0700187 std::vector<LogMessage> log_messages = {
Tom Cherry43f3f762020-05-14 22:28:09 -0700188 {{
189 .pid = 1,
190 .tid = 1,
191 .sec = 1234,
192 .nsec = 323001,
193 .lid = LOG_ID_MAIN,
194 .uid = 0,
195 },
196 "smoke test"},
197 };
198 FixupMessages(&log_messages);
199 LogMessages(log_messages);
200
Tom Cherryb398a7c2020-05-20 12:09:22 -0700201 std::vector<LogMessage> read_log_messages;
Tom Cherry43f3f762020-05-14 22:28:09 -0700202 std::unique_ptr<LogWriter> test_writer(new TestWriter(&read_log_messages, nullptr));
Tom Cherry855c7c82020-05-28 12:38:21 -0700203 std::unique_ptr<FlushToState> flush_to_state = log_buffer_->CreateFlushToState(1, kLogMaskAll);
204 EXPECT_TRUE(log_buffer_->FlushTo(test_writer.get(), *flush_to_state, nullptr));
205 EXPECT_EQ(2ULL, flush_to_state->start());
Tom Cherry43f3f762020-05-14 22:28:09 -0700206 CompareLogMessages(log_messages, read_log_messages);
207}
208
Tom Cherrye18346d2020-05-21 12:13:20 -0700209TEST_P(LogBufferTest, smoke_with_reader_thread) {
Tom Cherryb398a7c2020-05-20 12:09:22 -0700210 std::vector<LogMessage> log_messages = {
Tom Cherry43f3f762020-05-14 22:28:09 -0700211 {{.pid = 1, .tid = 2, .sec = 10000, .nsec = 20001, .lid = LOG_ID_MAIN, .uid = 0},
212 "first"},
213 {{.pid = 10, .tid = 2, .sec = 10000, .nsec = 20002, .lid = LOG_ID_MAIN, .uid = 0},
214 "second"},
215 {{.pid = 100, .tid = 2, .sec = 10000, .nsec = 20003, .lid = LOG_ID_KERNEL, .uid = 0},
216 "third"},
217 {{.pid = 10, .tid = 2, .sec = 10000, .nsec = 20004, .lid = LOG_ID_MAIN, .uid = 0},
218 "fourth"},
219 {{.pid = 1, .tid = 2, .sec = 10000, .nsec = 20005, .lid = LOG_ID_RADIO, .uid = 0},
220 "fifth"},
221 {{.pid = 2, .tid = 2, .sec = 10000, .nsec = 20006, .lid = LOG_ID_RADIO, .uid = 0},
222 "sixth"},
223 {{.pid = 3, .tid = 2, .sec = 10000, .nsec = 20007, .lid = LOG_ID_RADIO, .uid = 0},
224 "seventh"},
225 {{.pid = 4, .tid = 2, .sec = 10000, .nsec = 20008, .lid = LOG_ID_MAIN, .uid = 0},
226 "eighth"},
227 {{.pid = 5, .tid = 2, .sec = 10000, .nsec = 20009, .lid = LOG_ID_CRASH, .uid = 0},
228 "nineth"},
229 {{.pid = 6, .tid = 2, .sec = 10000, .nsec = 20011, .lid = LOG_ID_MAIN, .uid = 0},
230 "tenth"},
231 };
232 FixupMessages(&log_messages);
233 LogMessages(log_messages);
234
Tom Cherryb398a7c2020-05-20 12:09:22 -0700235 std::vector<LogMessage> read_log_messages;
Tom Cherry43f3f762020-05-14 22:28:09 -0700236 bool released = false;
237
238 {
239 auto lock = std::unique_lock{reader_list_.reader_threads_lock()};
240 std::unique_ptr<LogWriter> test_writer(new TestWriter(&read_log_messages, &released));
241 std::unique_ptr<LogReaderThread> log_reader(
242 new LogReaderThread(log_buffer_.get(), &reader_list_, std::move(test_writer), true,
243 0, ~0, 0, {}, 1, {}));
244 reader_list_.reader_threads().emplace_back(std::move(log_reader));
245 }
246
247 while (!released) {
248 usleep(5000);
249 }
250 {
251 auto lock = std::unique_lock{reader_list_.reader_threads_lock()};
252 EXPECT_EQ(0U, reader_list_.reader_threads().size());
253 }
254 CompareLogMessages(log_messages, read_log_messages);
255}
256
257// Generate random messages, set the 'sec' parameter explicit though, to be able to track the
258// expected order of messages.
Tom Cherryb398a7c2020-05-20 12:09:22 -0700259LogMessage GenerateRandomLogMessage(uint32_t sec) {
Tom Cherry43f3f762020-05-14 22:28:09 -0700260 auto rand_uint32 = [](int max) -> uint32_t { return rand() % max; };
261 logger_entry entry = {
262 .hdr_size = sizeof(logger_entry),
263 .pid = rand() % 5000,
264 .tid = rand_uint32(5000),
265 .sec = sec,
266 .nsec = rand_uint32(NS_PER_SEC),
267 .lid = rand_uint32(LOG_ID_STATS),
268 .uid = rand_uint32(100000),
269 };
270
271 // See comment in ChattyLogBuffer::Log() for why this is disallowed.
272 if (entry.nsec % 1000 == 0) {
273 ++entry.nsec;
274 }
275
276 if (entry.lid == LOG_ID_EVENTS) {
277 entry.lid = LOG_ID_KERNEL;
278 }
279
280 std::string message;
281 char priority = ANDROID_LOG_INFO + rand() % 2;
282 message.push_back(priority);
283
284 int tag_length = 2 + rand() % 10;
285 for (int i = 0; i < tag_length; ++i) {
286 message.push_back('a' + rand() % 26);
287 }
288 message.push_back('\0');
289
290 int msg_length = 2 + rand() % 1000;
291 for (int i = 0; i < msg_length; ++i) {
292 message.push_back('a' + rand() % 26);
293 }
294 message.push_back('\0');
295
296 entry.len = message.size();
297
298 return {entry, message};
299}
300
Tom Cherrye18346d2020-05-21 12:13:20 -0700301TEST_P(LogBufferTest, random_messages) {
Tom Cherry43f3f762020-05-14 22:28:09 -0700302 srand(1);
Tom Cherryb398a7c2020-05-20 12:09:22 -0700303 std::vector<LogMessage> log_messages;
Tom Cherry43f3f762020-05-14 22:28:09 -0700304 for (size_t i = 0; i < 1000; ++i) {
305 log_messages.emplace_back(GenerateRandomLogMessage(i));
306 }
307 LogMessages(log_messages);
308
Tom Cherryb398a7c2020-05-20 12:09:22 -0700309 std::vector<LogMessage> read_log_messages;
Tom Cherry43f3f762020-05-14 22:28:09 -0700310 bool released = false;
311
312 {
313 auto lock = std::unique_lock{reader_list_.reader_threads_lock()};
314 std::unique_ptr<LogWriter> test_writer(new TestWriter(&read_log_messages, &released));
315 std::unique_ptr<LogReaderThread> log_reader(
316 new LogReaderThread(log_buffer_.get(), &reader_list_, std::move(test_writer), true,
317 0, ~0, 0, {}, 1, {}));
318 reader_list_.reader_threads().emplace_back(std::move(log_reader));
319 }
320
321 while (!released) {
322 usleep(5000);
323 }
324 {
325 auto lock = std::unique_lock{reader_list_.reader_threads_lock()};
326 EXPECT_EQ(0U, reader_list_.reader_threads().size());
327 }
328 CompareLogMessages(log_messages, read_log_messages);
329}
Tom Cherryb398a7c2020-05-20 12:09:22 -0700330
Tom Cherry90e9ce02020-06-01 13:43:50 -0700331TEST_P(LogBufferTest, read_last_sequence) {
332 std::vector<LogMessage> log_messages = {
333 {{.pid = 1, .tid = 2, .sec = 10000, .nsec = 20001, .lid = LOG_ID_MAIN, .uid = 0},
334 "first"},
335 {{.pid = 10, .tid = 2, .sec = 10000, .nsec = 20002, .lid = LOG_ID_MAIN, .uid = 0},
336 "second"},
337 {{.pid = 100, .tid = 2, .sec = 10000, .nsec = 20003, .lid = LOG_ID_MAIN, .uid = 0},
338 "third"},
339 };
340 FixupMessages(&log_messages);
341 LogMessages(log_messages);
342
343 std::vector<LogMessage> read_log_messages;
344 bool released = false;
345
346 {
347 auto lock = std::unique_lock{reader_list_.reader_threads_lock()};
348 std::unique_ptr<LogWriter> test_writer(new TestWriter(&read_log_messages, &released));
349 std::unique_ptr<LogReaderThread> log_reader(
350 new LogReaderThread(log_buffer_.get(), &reader_list_, std::move(test_writer), true,
351 0, ~0, 0, {}, 3, {}));
352 reader_list_.reader_threads().emplace_back(std::move(log_reader));
353 }
354
355 while (!released) {
356 usleep(5000);
357 }
358 {
359 auto lock = std::unique_lock{reader_list_.reader_threads_lock()};
360 EXPECT_EQ(0U, reader_list_.reader_threads().size());
361 }
362 std::vector<LogMessage> expected_log_messages = {log_messages.back()};
363 CompareLogMessages(expected_log_messages, read_log_messages);
364}
365
Tom Cherry8f613462020-05-12 12:46:43 -0700366INSTANTIATE_TEST_CASE_P(LogBufferTests, LogBufferTest, testing::Values("chatty", "simple"));