2010-06-23 23:03:09 -05:00
|
|
|
/*
|
2011-04-19 05:21:57 -05:00
|
|
|
* Logging infrastructure that aims to support multi-threading,
|
2010-06-23 23:03:09 -05:00
|
|
|
* and ansi colors.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "rust_internal.h"
|
2010-07-19 16:05:18 -05:00
|
|
|
#include "util/array_list.h"
|
|
|
|
#include <stdarg.h>
|
2010-10-11 18:40:18 -05:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
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.
|
|
|
|
*/
|
2010-09-08 21:13:49 -05:00
|
|
|
static lock_and_signal _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),
|
2011-04-19 05:21:57 -05:00
|
|
|
_use_colors(getenv("RUST_COLOR_LOG")) {
|
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);
|
2011-01-14 18:01:43 -06:00
|
|
|
size_t off = strlen(buffer);
|
|
|
|
vsnprintf(buffer + off, BUF_BYTES - off, format, args);
|
2010-07-19 16:05:18 -05:00
|
|
|
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);
|
2011-01-14 18:01:43 -06:00
|
|
|
size_t off = strlen(buffer);
|
|
|
|
vsnprintf(buffer + off, BUF_BYTES - off, format, args);
|
2010-07-19 16:05:18 -05:00
|
|
|
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) {
|
2011-01-14 15:41:39 -06:00
|
|
|
char buffer[BUF_BYTES] = "";
|
2010-07-19 16:05:18 -05:00
|
|
|
_log_lock.lock();
|
|
|
|
append_string(buffer, "%-34s", prefix);
|
|
|
|
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
|
2011-04-19 05:21:57 -05:00
|
|
|
rust_log::trace_ln(rust_task *task, uint32_t level, char *message) {
|
2010-07-19 16:05:18 -05:00
|
|
|
#if defined(__WIN32__)
|
|
|
|
uint32_t thread_id = 0;
|
|
|
|
#else
|
2010-08-11 16:04:36 -05:00
|
|
|
uint32_t thread_id = hash((uint32_t) pthread_self());
|
2010-07-19 16:05:18 -05:00
|
|
|
#endif
|
2011-01-14 15:41:39 -06:00
|
|
|
char prefix[BUF_BYTES] = "";
|
2010-08-27 20:26:36 -05:00
|
|
|
if (_dom && _dom->name) {
|
2010-08-11 16:04:36 -05:00
|
|
|
append_string(prefix, "%04" PRIxPTR ":%.10s:",
|
|
|
|
thread_id, _dom->name);
|
|
|
|
} else {
|
|
|
|
append_string(prefix, "%04" PRIxPTR ":0x%08" PRIxPTR ":",
|
|
|
|
thread_id, (uintptr_t) _dom);
|
|
|
|
}
|
2010-07-19 16:05:18 -05:00
|
|
|
if (task) {
|
2010-08-11 16:04:36 -05:00
|
|
|
if (task->name) {
|
|
|
|
append_string(prefix, "%.10s:", task->name);
|
|
|
|
} else {
|
|
|
|
append_string(prefix, "0x%08" PRIxPTR ":", (uintptr_t) task);
|
|
|
|
}
|
2010-07-19 16:05:18 -05:00
|
|
|
}
|
2010-07-28 01:07:27 -05:00
|
|
|
trace_ln(thread_id, prefix, message);
|
2010-06-23 23:03:09 -05:00
|
|
|
}
|
|
|
|
|
2011-04-19 05:21:57 -05:00
|
|
|
// Reading log directives and setting log level vars
|
2011-04-18 09:18:55 -05:00
|
|
|
|
|
|
|
struct mod_entry {
|
|
|
|
const char* name;
|
2011-04-19 05:21:57 -05:00
|
|
|
size_t* state;
|
2011-04-18 09:18:55 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
struct cratemap {
|
2011-04-19 05:21:57 -05:00
|
|
|
const mod_entry* entries;
|
|
|
|
const cratemap* children[1];
|
2011-04-18 09:18:55 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
struct log_directive {
|
|
|
|
char* name;
|
|
|
|
size_t level;
|
|
|
|
};
|
|
|
|
|
|
|
|
const size_t max_log_directives = 255;
|
2011-04-19 05:21:57 -05:00
|
|
|
const size_t max_log_level = 1;
|
|
|
|
const size_t default_log_level = 0;
|
2011-04-18 09:18:55 -05:00
|
|
|
|
|
|
|
// This is a rather ugly parser for strings in the form
|
2011-04-19 05:21:57 -05:00
|
|
|
// "crate1,crate2.mod3,crate3.x=1". Log levels are 0-1 for now,
|
|
|
|
// eventually we'll have 0-3.
|
2011-04-18 09:18:55 -05:00
|
|
|
size_t parse_logging_spec(char* spec, log_directive* dirs) {
|
|
|
|
size_t dir = 0;
|
|
|
|
while (dir < max_log_directives && *spec) {
|
|
|
|
char* start = spec;
|
|
|
|
char cur;
|
|
|
|
while (true) {
|
|
|
|
cur = *spec;
|
|
|
|
if (cur == ',' || cur == '=' || cur == '\0') {
|
|
|
|
if (start == spec) {spec++; break;}
|
|
|
|
*spec = '\0';
|
|
|
|
spec++;
|
2011-04-19 05:21:57 -05:00
|
|
|
size_t level = max_log_level;
|
2011-04-18 09:18:55 -05:00
|
|
|
if (cur == '=') {
|
|
|
|
level = *spec - '0';
|
2011-04-19 05:21:57 -05:00
|
|
|
if (level > max_log_level) level = max_log_level;
|
2011-04-18 09:18:55 -05:00
|
|
|
if (*spec) ++spec;
|
|
|
|
}
|
|
|
|
dirs[dir].name = start;
|
|
|
|
dirs[dir++].level = level;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
spec++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return dir;
|
|
|
|
}
|
|
|
|
|
2011-04-19 05:21:57 -05:00
|
|
|
void update_module_map(const mod_entry* map, log_directive* dirs,
|
|
|
|
size_t n_dirs) {
|
|
|
|
for (const mod_entry* cur = map; cur->name; cur++) {
|
|
|
|
size_t level = default_log_level, longest_match = 0;
|
2011-04-18 09:18:55 -05:00
|
|
|
for (size_t d = 0; d < n_dirs; d++) {
|
|
|
|
if (strstr(cur->name, dirs[d].name) == cur->name &&
|
|
|
|
strlen(dirs[d].name) > longest_match) {
|
|
|
|
longest_match = strlen(dirs[d].name);
|
|
|
|
level = dirs[d].level;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*cur->state = level;
|
|
|
|
}
|
2011-04-19 05:21:57 -05:00
|
|
|
}
|
2011-04-18 09:18:55 -05:00
|
|
|
|
2011-04-19 05:21:57 -05:00
|
|
|
void update_crate_map(const cratemap* map, log_directive* dirs,
|
|
|
|
size_t n_dirs) {
|
|
|
|
// First update log levels for this crate
|
|
|
|
update_module_map(map->entries, dirs, n_dirs);
|
2011-04-18 09:18:55 -05:00
|
|
|
// Then recurse on linked crates
|
2011-04-19 05:21:57 -05:00
|
|
|
// FIXME this does double work in diamond-shaped deps. could keep
|
|
|
|
// a set of visited addresses, if it turns out to be actually slow
|
2011-04-18 09:18:55 -05:00
|
|
|
for (size_t i = 0; map->children[i]; i++) {
|
|
|
|
update_crate_map(map->children[i], dirs, n_dirs);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-04-19 05:21:57 -05:00
|
|
|
// These are pseudo-modules used to control logging in the runtime.
|
|
|
|
|
|
|
|
size_t log_rt_mem;
|
|
|
|
size_t log_rt_comm;
|
|
|
|
size_t log_rt_task;
|
|
|
|
size_t log_rt_dom;
|
|
|
|
size_t log_rt_trace;
|
|
|
|
size_t log_rt_dwarf;
|
|
|
|
size_t log_rt_cache;
|
|
|
|
size_t log_rt_upcall;
|
|
|
|
size_t log_rt_timer;
|
|
|
|
size_t log_rt_gc;
|
|
|
|
size_t log_rt_stdlib;
|
|
|
|
size_t log_rt_kern;
|
|
|
|
size_t log_rt_backtrace;
|
|
|
|
// Used to turn logging for rustboot-compiled code on and off
|
|
|
|
size_t log_rustboot;
|
|
|
|
|
|
|
|
static const mod_entry _rt_module_map[] =
|
|
|
|
{{"rt.mem", &log_rt_mem},
|
|
|
|
{"rt.comm", &log_rt_comm},
|
|
|
|
{"rt.task", &log_rt_task},
|
|
|
|
{"rt.dom", &log_rt_dom},
|
|
|
|
{"rt.trace", &log_rt_trace},
|
|
|
|
{"rt.dwarf", &log_rt_dwarf},
|
|
|
|
{"rt.cache", &log_rt_cache},
|
|
|
|
{"rt.upcall", &log_rt_upcall},
|
|
|
|
{"rt.timer", &log_rt_timer},
|
|
|
|
{"rt.gc", &log_rt_gc},
|
|
|
|
{"rt.stdlib", &log_rt_stdlib},
|
|
|
|
{"rt.kern", &log_rt_kern},
|
|
|
|
{"rt.backtrace", &log_rt_backtrace},
|
|
|
|
{"rustboot", &log_rustboot},
|
|
|
|
{NULL, NULL}};
|
|
|
|
|
2011-04-18 09:18:55 -05:00
|
|
|
void update_log_settings(void* crate_map, char* settings) {
|
|
|
|
char* buffer = NULL;
|
|
|
|
log_directive dirs[256];
|
2011-04-19 05:21:57 -05:00
|
|
|
size_t n_dirs = 0;
|
2011-04-18 09:18:55 -05:00
|
|
|
if (settings) {
|
|
|
|
buffer = (char*)malloc(strlen(settings));
|
|
|
|
strcpy(buffer, settings);
|
2011-04-19 05:21:57 -05:00
|
|
|
n_dirs = parse_logging_spec(buffer, &dirs[0]);
|
2011-04-18 09:18:55 -05:00
|
|
|
}
|
|
|
|
|
2011-04-19 05:21:57 -05:00
|
|
|
update_module_map(_rt_module_map, &dirs[0], n_dirs);
|
|
|
|
// FIXME check can be dropped when rustboot is gone
|
|
|
|
if (crate_map)
|
|
|
|
update_crate_map((const cratemap*)crate_map, &dirs[0], n_dirs);
|
2011-04-18 09:18:55 -05:00
|
|
|
|
|
|
|
free(buffer);
|
|
|
|
}
|