blob: 34e5e8d5fe60aad5ed3029264c82878df8c24315 [file] [log] [blame]
Elliott Hughes13f5a582011-09-06 13:39:14 -07001/*
2 * Copyright (C) 2011 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#include "logging.h"
18
19#include "runtime.h"
Elliott Hughes5fe594f2011-09-08 12:33:17 -070020#include "thread.h"
Elliott Hughes13f5a582011-09-06 13:39:14 -070021#include "utils.h"
22
Elliott Hughesf5a7a472011-10-07 14:31:02 -070023namespace art {
Elliott Hughes5fe594f2011-09-08 12:33:17 -070024
Elliott Hughes8daa0922011-09-11 13:46:25 -070025art::Mutex& GetLoggingLock() {
26 static art::Mutex lock("LogMessage lock");
Elliott Hughes5fe594f2011-09-08 12:33:17 -070027 return lock;
28}
29
Elliott Hughes13f5a582011-09-06 13:39:14 -070030LogMessage::~LogMessage() {
Elliott Hughes5fe594f2011-09-08 12:33:17 -070031 // Finish constructing the message.
Elliott Hughes3b6baaa2011-10-14 19:13:56 -070032 if (data_->error != -1) {
33 data_->buffer << ": " << strerror(data_->error);
Elliott Hughes13f5a582011-09-06 13:39:14 -070034 }
Elliott Hughes3b6baaa2011-10-14 19:13:56 -070035 std::string msg(data_->buffer.str());
Elliott Hughes5fe594f2011-09-08 12:33:17 -070036
37 // Do the actual logging with the lock held.
38 {
39 art::MutexLock mu(GetLoggingLock());
40 if (msg.find('\n') == std::string::npos) {
41 LogLine(msg.c_str());
42 } else {
43 msg += '\n';
44 size_t i = 0;
45 while (i < msg.size()) {
46 size_t nl = msg.find('\n', i);
47 msg[nl] = '\0';
48 LogLine(&msg[i]);
49 i = nl + 1;
50 }
Elliott Hughes13f5a582011-09-06 13:39:14 -070051 }
52 }
53
Elliott Hughes5fe594f2011-09-08 12:33:17 -070054 // Abort if necessary.
Elliott Hughes3b6baaa2011-10-14 19:13:56 -070055 if (data_->severity == FATAL) {
56 art::Runtime::Abort(data_->file, data_->line_number);
Elliott Hughes13f5a582011-09-06 13:39:14 -070057 }
Elliott Hughes3b6baaa2011-10-14 19:13:56 -070058
59 delete data_;
Elliott Hughes13f5a582011-09-06 13:39:14 -070060}
61
62std::ostream& LogMessage::stream() {
Elliott Hughes3b6baaa2011-10-14 19:13:56 -070063 return data_->buffer;
Elliott Hughes13f5a582011-09-06 13:39:14 -070064}
Elliott Hughesf5a7a472011-10-07 14:31:02 -070065
Elliott Hughes872d4ec2011-10-21 17:07:15 -070066/*
67 * Print a hex dump in this format:
68 *
Elliott Hughes21f32d72011-11-09 17:44:13 -080069 * 01234567: 00 11 22 33 44 55 66 77 88 99 aa bb cc dd ee ff 0123456789abcdef
Elliott Hughes872d4ec2011-10-21 17:07:15 -070070 *
71 * Does not use printf() or other string-formatting calls.
72 */
73void HexDump(const void* address, size_t byte_count, bool show_actual_address) {
74 static const char gHexDigit[] = "0123456789abcdef";
75 const unsigned char* addr = reinterpret_cast<const unsigned char*>(address);
Elliott Hughes21f32d72011-11-09 17:44:13 -080076 char out[76]; /* exact fit */
Elliott Hughes872d4ec2011-10-21 17:07:15 -070077 unsigned int offset; /* offset to show while printing */
78
79 if (show_actual_address) {
80 offset = (int) addr;
81 } else {
82 offset = 0;
83 }
84 memset(out, ' ', sizeof(out)-1);
85 out[8] = ':';
Elliott Hughes872d4ec2011-10-21 17:07:15 -070086 out[sizeof(out)-1] = '\0';
87
88 int gap = (int) offset & 0x0f;
89 while (byte_count) {
90 unsigned int lineOffset = offset & ~0x0f;
91 int i, count;
92
93 char* hex = out;
94 char* asc = out + 59;
95
96 for (i = 0; i < 8; i++) {
97 *hex++ = gHexDigit[lineOffset >> 28];
98 lineOffset <<= 4;
99 }
100 hex++;
101 hex++;
102
103 count = ((int)byte_count > 16-gap) ? 16-gap : (int)byte_count; /* cap length */
104 CHECK_NE(count, 0);
105 CHECK_LE(count + gap, 16);
106
107 if (gap) {
108 /* only on first line */
109 hex += gap * 3;
110 asc += gap;
111 }
112
113 for (i = gap ; i < count+gap; i++) {
114 *hex++ = gHexDigit[*addr >> 4];
115 *hex++ = gHexDigit[*addr & 0x0f];
116 hex++;
117 if (*addr >= 0x20 && *addr < 0x7f /*isprint(*addr)*/)
118 *asc++ = *addr;
119 else
120 *asc++ = '.';
121 addr++;
122 }
123 for ( ; i < 16; i++) {
124 /* erase extra stuff; only happens on last line */
125 *hex++ = ' ';
126 *hex++ = ' ';
127 hex++;
128 *asc++ = ' ';
129 }
130
131 LOG(INFO) << out;
132
133 gap = 0;
134 byte_count -= count;
135 offset += count;
136 }
137}
138
Elliott Hughesf5a7a472011-10-07 14:31:02 -0700139} // namespace art