mirror of
https://github.com/JakeHillion/drgn.git
synced 2024-12-23 01:33:06 +00:00
c1a2792e6a
Exceptions aren't enough to debug complicated code paths like debug info discovery or stack unwinding. We really need logs for that, so let's add a small logging framework. By default, we log to stderr, but we also provide a way to direct logs to a different file, or even an arbitrary callback so that logs can be directed to the application's logging library of choice. Signed-off-by: Omar Sandoval <osandov@osandov.com>
284 lines
6.4 KiB
C
284 lines
6.4 KiB
C
// Copyright (c) Meta Platforms, Inc. and affiliates.
|
|
// SPDX-License-Identifier: LGPL-2.1-or-later
|
|
|
|
#include <elfutils/libdw.h>
|
|
#include <elfutils/libdwfl.h>
|
|
#include <errno.h>
|
|
#include <inttypes.h>
|
|
#include <libelf.h>
|
|
#include <stdarg.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include "drgn.h"
|
|
#include "error.h"
|
|
#include "string_builder.h"
|
|
#include "util.h"
|
|
|
|
LIBDRGN_PUBLIC struct drgn_error drgn_enomem = {
|
|
.code = DRGN_ERROR_NO_MEMORY,
|
|
.message = "cannot allocate memory",
|
|
};
|
|
|
|
LIBDRGN_PUBLIC struct drgn_error drgn_not_found = {
|
|
.code = DRGN_ERROR_LOOKUP,
|
|
.message = "not found",
|
|
};
|
|
|
|
struct drgn_error drgn_stop = {
|
|
.code = DRGN_ERROR_STOP,
|
|
.message = "stop iteration",
|
|
};
|
|
|
|
struct drgn_error drgn_error_object_absent = {
|
|
.code = DRGN_ERROR_OBJECT_ABSENT,
|
|
.message = "object absent",
|
|
};
|
|
|
|
static struct drgn_error *drgn_error_create_nodup(enum drgn_error_code code,
|
|
char *message)
|
|
{
|
|
struct drgn_error *err;
|
|
|
|
err = malloc(sizeof(*err));
|
|
if (!err) {
|
|
free(message);
|
|
return &drgn_enomem;
|
|
}
|
|
|
|
err->code = code;
|
|
err->needs_destroy = true;
|
|
err->errnum = 0;
|
|
err->path = NULL;
|
|
err->address = 0;
|
|
err->message = message;
|
|
return err;
|
|
}
|
|
|
|
LIBDRGN_PUBLIC struct drgn_error *drgn_error_create(enum drgn_error_code code,
|
|
const char *message)
|
|
{
|
|
char *message_copy;
|
|
|
|
message_copy = strdup(message);
|
|
if (!message_copy)
|
|
return &drgn_enomem;
|
|
return drgn_error_create_nodup(code, message_copy);
|
|
}
|
|
|
|
LIBDRGN_PUBLIC struct drgn_error *
|
|
drgn_error_format_os(const char *message, int errnum, const char *path_format,
|
|
...)
|
|
{
|
|
struct drgn_error *err;
|
|
va_list ap;
|
|
int ret;
|
|
|
|
err = malloc(sizeof(*err));
|
|
if (!err)
|
|
return &drgn_enomem;
|
|
|
|
err->code = DRGN_ERROR_OS;
|
|
err->needs_destroy = true;
|
|
err->errnum = errnum;
|
|
if (path_format) {
|
|
va_start(ap, path_format);
|
|
ret = vasprintf(&err->path, path_format, ap);
|
|
va_end(ap);
|
|
if (ret == -1) {
|
|
free(err);
|
|
return &drgn_enomem;
|
|
}
|
|
} else {
|
|
err->path = NULL;
|
|
}
|
|
err->address = 0;
|
|
err->message = strdup(message);
|
|
if (!err->message) {
|
|
free(err->path);
|
|
free(err);
|
|
return &drgn_enomem;
|
|
}
|
|
return err;
|
|
}
|
|
|
|
LIBDRGN_PUBLIC struct drgn_error *drgn_error_create_os(const char *message,
|
|
int errnum,
|
|
const char *path)
|
|
{
|
|
if (path)
|
|
return drgn_error_format_os(message, errnum, "%s", path);
|
|
else
|
|
return drgn_error_format_os(message, errnum, NULL);
|
|
}
|
|
|
|
LIBDRGN_PUBLIC struct drgn_error *drgn_error_format(enum drgn_error_code code,
|
|
const char *format, ...)
|
|
{
|
|
char *message;
|
|
va_list ap;
|
|
int ret;
|
|
|
|
va_start(ap, format);
|
|
ret = vasprintf(&message, format, ap);
|
|
va_end(ap);
|
|
if (ret == -1)
|
|
return &drgn_enomem;
|
|
return drgn_error_create_nodup(code, message);
|
|
}
|
|
|
|
LIBDRGN_PUBLIC struct drgn_error *drgn_error_create_fault(const char *message,
|
|
uint64_t address)
|
|
{
|
|
struct drgn_error *err;
|
|
|
|
err = drgn_error_create(DRGN_ERROR_FAULT, message);
|
|
if (err != &drgn_enomem)
|
|
err->address = address;
|
|
return err;
|
|
}
|
|
|
|
LIBDRGN_PUBLIC struct drgn_error *
|
|
drgn_error_format_fault(uint64_t address, const char *format, ...)
|
|
{
|
|
struct drgn_error *err;
|
|
va_list ap;
|
|
char *message;
|
|
int ret;
|
|
|
|
va_start(ap, format);
|
|
ret = vasprintf(&message, format, ap);
|
|
va_end(ap);
|
|
if (ret == -1)
|
|
return &drgn_enomem;
|
|
err = drgn_error_create_nodup(DRGN_ERROR_FAULT, message);
|
|
if (err != &drgn_enomem)
|
|
err->address = address;
|
|
return err;
|
|
}
|
|
|
|
struct drgn_error *drgn_error_from_string_builder(enum drgn_error_code code,
|
|
struct string_builder *sb)
|
|
{
|
|
char *message = string_builder_null_terminate(sb);
|
|
if (!message) {
|
|
free(sb->str);
|
|
return &drgn_enomem;
|
|
}
|
|
return drgn_error_create_nodup(code, message);
|
|
}
|
|
|
|
LIBDRGN_PUBLIC struct drgn_error *drgn_error_copy(struct drgn_error *src)
|
|
{
|
|
if (!src->needs_destroy)
|
|
return src;
|
|
struct drgn_error *dst = malloc(sizeof(*dst));
|
|
if (!dst)
|
|
return &drgn_enomem;
|
|
dst->code = src->code;
|
|
dst->needs_destroy = true;
|
|
dst->errnum = src->errnum;
|
|
if (src->path) {
|
|
dst->path = strdup(src->path);
|
|
if (!dst->path) {
|
|
free(dst);
|
|
return &drgn_enomem;
|
|
}
|
|
} else {
|
|
dst->path = NULL;
|
|
}
|
|
dst->address = src->address;
|
|
if (src->message) {
|
|
dst->message = strdup(src->message);
|
|
if (!dst->message) {
|
|
free(dst->path);
|
|
free(dst);
|
|
return &drgn_enomem;
|
|
}
|
|
} else {
|
|
dst->message = NULL;
|
|
}
|
|
return dst;
|
|
}
|
|
|
|
#define emit_error(err) ( \
|
|
err->code == DRGN_ERROR_OS ? \
|
|
/* This is easier than dealing with strerror_r(). */ \
|
|
(errno = err->errnum, \
|
|
err->path ? \
|
|
emit_error_format("%s: %s: %m", err->message, err->path) : \
|
|
emit_error_format("%s: %m", err->message)) \
|
|
: err->code == DRGN_ERROR_FAULT ? \
|
|
emit_error_format("%s: 0x%" PRIx64, err->message, err->address) \
|
|
: \
|
|
emit_error_string(err->message) \
|
|
)
|
|
|
|
bool string_builder_append_error(struct string_builder *sb,
|
|
struct drgn_error *err)
|
|
{
|
|
#define emit_error_format(...) string_builder_appendf(sb, __VA_ARGS__)
|
|
#define emit_error_string(s) string_builder_append(sb, s)
|
|
return emit_error(err);
|
|
#undef emit_error_string
|
|
#undef emit_error_format
|
|
}
|
|
|
|
LIBDRGN_PUBLIC char *drgn_error_string(struct drgn_error *err)
|
|
{
|
|
char *tmp;
|
|
#define emit_error_format(...) (asprintf(&tmp, __VA_ARGS__) < 0 ? NULL : tmp)
|
|
#define emit_error_string(s) strdup(s)
|
|
return emit_error(err);
|
|
#undef emit_error_string
|
|
#undef emit_error_format
|
|
}
|
|
|
|
LIBDRGN_PUBLIC int drgn_error_fwrite(FILE *file, struct drgn_error *err)
|
|
{
|
|
#define emit_error_format(format, ...) fprintf(file, format "\n", __VA_ARGS__)
|
|
#define emit_error_string(s) fprintf(file, "%s\n", s)
|
|
return emit_error(err);
|
|
#undef emit_error_string
|
|
#undef emit_error_format
|
|
}
|
|
|
|
LIBDRGN_PUBLIC int drgn_error_dwrite(int fd, struct drgn_error *err)
|
|
{
|
|
#define emit_error_format(format, ...) dprintf(fd, format "\n", __VA_ARGS__)
|
|
#define emit_error_string(s) dprintf(fd, "%s\n", s)
|
|
return emit_error(err);
|
|
#undef emit_error_string
|
|
#undef emit_error_format
|
|
}
|
|
|
|
#undef emit_error
|
|
|
|
LIBDRGN_PUBLIC void drgn_error_destroy(struct drgn_error *err)
|
|
{
|
|
if (err && err->needs_destroy) {
|
|
free(err->path);
|
|
free(err->message);
|
|
free(err);
|
|
}
|
|
}
|
|
|
|
struct drgn_error *drgn_error_libelf(void)
|
|
{
|
|
return drgn_error_format(DRGN_ERROR_OTHER, "libelf error: %s",
|
|
elf_errmsg(-1));
|
|
}
|
|
|
|
struct drgn_error *drgn_error_libdw(void)
|
|
{
|
|
return drgn_error_format(DRGN_ERROR_OTHER, "libdw error: %s",
|
|
dwarf_errmsg(-1));
|
|
}
|
|
|
|
struct drgn_error *drgn_error_libdwfl(void)
|
|
{
|
|
return drgn_error_format(DRGN_ERROR_OTHER, "libdwfl error: %s",
|
|
dwfl_errmsg(-1));
|
|
}
|