2010-06-23 23:03:09 -05:00
|
|
|
/*
|
|
|
|
* Logging infrastructure that aims to support multi-threading, indentation
|
|
|
|
* and ansi colors.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "rust_internal.h"
|
2010-07-19 16:05:18 -05:00
|
|
|
#include "sync/spin_lock.h"
|
|
|
|
#include "util/array_list.h"
|
|
|
|
#include <stdarg.h>
|
2010-06-23 23:03:09 -05:00
|
|
|
|
2010-07-19 16:05:18 -05:00
|
|
|
static uint32_t
|
|
|
|
read_type_bit_mask() {
|
2010-07-20 20:38:06 -05:00
|
|
|
uint32_t bits = rust_log::ULOG | rust_log::ERR;
|
2010-06-23 23:03:09 -05:00
|
|
|
char *env_str = getenv("RUST_LOG");
|
|
|
|
if (env_str) {
|
|
|
|
bits = 0;
|
|
|
|
bits |= strstr(env_str, "err") ? rust_log::ERR : 0;
|
|
|
|
bits |= strstr(env_str, "mem") ? rust_log::MEM : 0;
|
|
|
|
bits |= strstr(env_str, "comm") ? rust_log::COMM : 0;
|
|
|
|
bits |= strstr(env_str, "task") ? rust_log::TASK : 0;
|
|
|
|
bits |= strstr(env_str, "up") ? rust_log::UPCALL : 0;
|
|
|
|
bits |= strstr(env_str, "dom") ? rust_log::DOM : 0;
|
|
|
|
bits |= strstr(env_str, "ulog") ? rust_log::ULOG : 0;
|
|
|
|
bits |= strstr(env_str, "trace") ? rust_log::TRACE : 0;
|
|
|
|
bits |= strstr(env_str, "dwarf") ? rust_log::DWARF : 0;
|
|
|
|
bits |= strstr(env_str, "cache") ? rust_log::CACHE : 0;
|
|
|
|
bits |= strstr(env_str, "timer") ? rust_log::TIMER : 0;
|
2010-06-28 20:53:16 -05:00
|
|
|
bits |= strstr(env_str, "gc") ? rust_log::GC : 0;
|
2010-06-23 23:03:09 -05:00
|
|
|
bits |= strstr(env_str, "all") ? rust_log::ALL : 0;
|
|
|
|
}
|
|
|
|
return bits;
|
|
|
|
}
|
|
|
|
|
2010-07-19 16:05:18 -05:00
|
|
|
rust_log::ansi_color
|
|
|
|
get_type_color(rust_log::log_type type) {
|
|
|
|
rust_log::ansi_color color = rust_log::WHITE;
|
|
|
|
if (type & rust_log::ERR)
|
|
|
|
color = rust_log::RED;
|
|
|
|
if (type & rust_log::MEM)
|
|
|
|
color = rust_log::YELLOW;
|
|
|
|
if (type & rust_log::UPCALL)
|
|
|
|
color = rust_log::GREEN;
|
|
|
|
if (type & rust_log::COMM)
|
|
|
|
color = rust_log::MAGENTA;
|
|
|
|
if (type & rust_log::DOM)
|
|
|
|
color = rust_log::LIGHTTEAL;
|
|
|
|
if (type & rust_log::TASK)
|
|
|
|
color = rust_log::LIGHTTEAL;
|
|
|
|
return color;
|
2010-06-23 23:03:09 -05:00
|
|
|
}
|
|
|
|
|
2010-07-19 16:05:18 -05:00
|
|
|
static const char * _foreground_colors[] = { "[37m",
|
|
|
|
"[31m", "[1;31m",
|
|
|
|
"[32m", "[1;32m",
|
|
|
|
"[33m", "[1;33m",
|
|
|
|
"[31m", "[1;31m",
|
|
|
|
"[35m", "[1;35m",
|
|
|
|
"[36m", "[1;36m" };
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Synchronizes access to the underlying logging mechanism.
|
|
|
|
*/
|
|
|
|
static spin_lock _log_lock;
|
2010-07-28 01:07:27 -05:00
|
|
|
static uint32_t _last_thread_id;
|
2010-07-19 16:05:18 -05:00
|
|
|
|
2010-06-23 23:03:09 -05:00
|
|
|
rust_log::rust_log(rust_srv *srv, rust_dom *dom) :
|
2010-07-19 16:05:18 -05:00
|
|
|
_srv(srv),
|
|
|
|
_dom(dom),
|
|
|
|
_type_bit_mask(read_type_bit_mask()),
|
|
|
|
_use_colors(getenv("RUST_COLOR_LOG")),
|
|
|
|
_indent(0) {
|
2010-06-23 23:03:09 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
rust_log::~rust_log() {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2010-07-19 16:05:18 -05:00
|
|
|
const uint16_t
|
|
|
|
hash(uintptr_t ptr) {
|
|
|
|
// Robert Jenkins' 32 bit integer hash function
|
|
|
|
ptr = (ptr + 0x7ed55d16) + (ptr << 12);
|
|
|
|
ptr = (ptr ^ 0xc761c23c) ^ (ptr >> 19);
|
|
|
|
ptr = (ptr + 0x165667b1) + (ptr << 5);
|
|
|
|
ptr = (ptr + 0xd3a2646c) ^ (ptr << 9);
|
|
|
|
ptr = (ptr + 0xfd7046c5) + (ptr << 3);
|
|
|
|
ptr = (ptr ^ 0xb55a4f09) ^ (ptr >> 16);
|
|
|
|
return (uint16_t) ptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *
|
|
|
|
get_color(uintptr_t ptr) {
|
|
|
|
return _foreground_colors[hash(ptr) % rust_log::LIGHTTEAL];
|
|
|
|
}
|
|
|
|
|
|
|
|
char *
|
|
|
|
copy_string(char *dst, const char *src, size_t length) {
|
|
|
|
return strncpy(dst, src, length) + length;
|
|
|
|
}
|
|
|
|
|
|
|
|
char *
|
|
|
|
append_string(char *buffer, const char *format, ...) {
|
|
|
|
if (buffer != NULL && format) {
|
|
|
|
va_list args;
|
|
|
|
va_start(args, format);
|
|
|
|
vsprintf(buffer + strlen(buffer), format, args);
|
|
|
|
va_end(args);
|
2010-06-23 23:03:09 -05:00
|
|
|
}
|
2010-07-19 16:05:18 -05:00
|
|
|
return buffer;
|
|
|
|
}
|
2010-06-23 23:03:09 -05:00
|
|
|
|
2010-07-19 16:05:18 -05:00
|
|
|
char *
|
|
|
|
append_string(char *buffer, rust_log::ansi_color color,
|
|
|
|
const char *format, ...) {
|
|
|
|
if (buffer != NULL && format) {
|
|
|
|
append_string(buffer, "\x1b%s", _foreground_colors[color]);
|
|
|
|
va_list args;
|
|
|
|
va_start(args, format);
|
|
|
|
vsprintf(buffer + strlen(buffer), format, args);
|
|
|
|
va_end(args);
|
|
|
|
append_string(buffer, "\x1b[0m");
|
|
|
|
}
|
|
|
|
return buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2010-07-28 01:07:27 -05:00
|
|
|
rust_log::trace_ln(uint32_t thread_id, char *prefix, char *message) {
|
2010-07-19 16:05:18 -05:00
|
|
|
char buffer[1024] = "";
|
|
|
|
_log_lock.lock();
|
|
|
|
append_string(buffer, "%-34s", prefix);
|
2010-06-23 23:03:09 -05:00
|
|
|
for (uint32_t i = 0; i < _indent; i++) {
|
2010-07-19 16:05:18 -05:00
|
|
|
append_string(buffer, " ");
|
2010-06-23 23:03:09 -05:00
|
|
|
}
|
2010-07-19 16:05:18 -05:00
|
|
|
append_string(buffer, "%s", message);
|
2010-07-28 01:07:27 -05:00
|
|
|
if (_last_thread_id != thread_id) {
|
|
|
|
_last_thread_id = thread_id;
|
|
|
|
_srv->log("---");
|
|
|
|
}
|
2010-06-23 23:03:09 -05:00
|
|
|
_srv->log(buffer);
|
2010-07-19 16:05:18 -05:00
|
|
|
_log_lock.unlock();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
rust_log::trace_ln(rust_task *task, char *message) {
|
|
|
|
#if defined(__WIN32__)
|
|
|
|
uint32_t thread_id = 0;
|
|
|
|
#else
|
|
|
|
uint32_t thread_id = (uint32_t) pthread_self();
|
|
|
|
#endif
|
|
|
|
char prefix[1024] = "";
|
|
|
|
append_string(prefix, "0x%08" PRIxPTR ":0x%08" PRIxPTR ":",
|
|
|
|
thread_id, (uintptr_t) _dom);
|
|
|
|
if (task) {
|
|
|
|
append_string(prefix, "0x%08" PRIxPTR ":", (uintptr_t) task);
|
|
|
|
}
|
2010-07-28 01:07:27 -05:00
|
|
|
trace_ln(thread_id, prefix, message);
|
2010-06-23 23:03:09 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Traces a log message if the specified logging type is not filtered.
|
|
|
|
*/
|
2010-07-19 16:05:18 -05:00
|
|
|
void
|
|
|
|
rust_log::trace_ln(rust_task *task, uint32_t type_bits, char *message) {
|
|
|
|
trace_ln(task, get_type_color((rust_log::log_type) type_bits),
|
|
|
|
type_bits, message);
|
2010-06-23 23:03:09 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Traces a log message using the specified ANSI color code.
|
|
|
|
*/
|
2010-07-19 16:05:18 -05:00
|
|
|
void
|
|
|
|
rust_log::trace_ln(rust_task *task, ansi_color color,
|
|
|
|
uint32_t type_bits, char *message) {
|
2010-06-23 23:03:09 -05:00
|
|
|
if (is_tracing(type_bits)) {
|
|
|
|
if (_use_colors) {
|
2010-07-19 16:05:18 -05:00
|
|
|
char buffer[512] = "";
|
|
|
|
append_string(buffer, color, "%s", message);
|
|
|
|
trace_ln(task, buffer);
|
2010-06-23 23:03:09 -05:00
|
|
|
} else {
|
2010-07-19 16:05:18 -05:00
|
|
|
trace_ln(task, message);
|
2010-06-23 23:03:09 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-07-19 16:05:18 -05:00
|
|
|
bool
|
|
|
|
rust_log::is_tracing(uint32_t type_bits) {
|
2010-06-23 23:03:09 -05:00
|
|
|
return type_bits & _type_bit_mask;
|
|
|
|
}
|
|
|
|
|
2010-07-19 16:05:18 -05:00
|
|
|
void
|
|
|
|
rust_log::indent() {
|
2010-06-23 23:03:09 -05:00
|
|
|
_indent++;
|
|
|
|
}
|
|
|
|
|
2010-07-19 16:05:18 -05:00
|
|
|
void
|
|
|
|
rust_log::outdent() {
|
2010-06-23 23:03:09 -05:00
|
|
|
_indent--;
|
|
|
|
}
|
|
|
|
|
2010-07-19 16:05:18 -05:00
|
|
|
void
|
|
|
|
rust_log::reset_indent(uint32_t indent) {
|
2010-06-23 23:03:09 -05:00
|
|
|
_indent = indent;
|
|
|
|
}
|