monado/src/xrt/auxiliary/util/u_debug.c

269 lines
5.3 KiB
C
Raw Normal View History

2020-07-03 13:28:55 +00:00
// Copyright 2019-2020, Collabora, Ltd.
2019-03-18 05:52:32 +00:00
// SPDX-License-Identifier: BSL-1.0
/*!
* @file
* @brief Small debug helpers.
* @author Jakob Bornecrantz <jakob@collabora.com>
* @ingroup aux_util
2019-03-18 05:52:32 +00:00
*
* Debug get option helpers heavily inspired from mesa ones.
*/
2020-07-16 02:39:11 +00:00
#if defined(_MSC_VER) && !defined(_CRT_SECURE_NO_WARNINGS)
#define _CRT_SECURE_NO_WARNINGS
#endif
2020-07-03 13:28:55 +00:00
#include "util/u_debug.h"
#include "util/u_logging.h"
#include <ctype.h>
2019-03-18 05:52:32 +00:00
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
DEBUG_GET_ONCE_BOOL_OPTION(print, "XRT_PRINT_OPTIONS", false)
static const char *
os_getenv(const char *name)
{
return getenv(name);
}
const char *
debug_get_option(const char *name, const char *_default)
{
const char *raw = getenv(name);
const char *ret;
if (raw == NULL) {
ret = _default;
} else {
ret = raw;
}
if (debug_get_bool_option_print()) {
2020-07-03 13:28:55 +00:00
U_LOG_RAW("%s=%s (%s)", name, ret, raw == NULL ? "nil" : raw);
2019-03-18 05:52:32 +00:00
}
return ret;
}
bool
debug_string_to_bool(const char *raw)
2019-03-18 05:52:32 +00:00
{
bool ret;
if (raw == NULL) {
ret = false;
2019-03-18 05:52:32 +00:00
} else if (!strcmp(raw, "false")) {
ret = false;
} else if (!strcmp(raw, "FALSE")) {
ret = false;
} else if (!strcmp(raw, "off")) {
ret = false;
} else if (!strcmp(raw, "OFF")) {
ret = false;
} else if (!strcmp(raw, "no")) {
ret = false;
} else if (!strcmp(raw, "NO")) {
ret = false;
} else if (!strcmp(raw, "n")) {
ret = false;
} else if (!strcmp(raw, "N")) {
ret = false;
} else if (!strcmp(raw, "f")) {
ret = false;
} else if (!strcmp(raw, "F")) {
ret = false;
} else if (!strcmp(raw, "0")) {
ret = false;
} else {
ret = true;
}
return ret;
}
bool
debug_get_bool_option(const char *name, bool _default)
{
const char *raw = os_getenv(name);
bool ret = raw == NULL ? _default : debug_string_to_bool(raw);
2019-03-18 05:52:32 +00:00
if (debug_get_bool_option_print()) {
2021-01-14 14:13:48 +00:00
U_LOG_RAW("%s=%s (%s)", name, ret ? "TRUE" : "FALSE", raw == NULL ? "nil" : raw);
2019-03-18 05:52:32 +00:00
}
return ret;
}
enum debug_tristate_option
debug_string_to_tristate(const char *raw)
{
enum debug_tristate_option ret;
if (raw == NULL) {
ret = DEBUG_TRISTATE_AUTO;
} else if (!strcmp(raw, "AUTO")) {
ret = DEBUG_TRISTATE_AUTO;
} else if (!strcmp(raw, "auto")) {
ret = DEBUG_TRISTATE_AUTO;
} else if (!strcmp(raw, "a")) {
ret = DEBUG_TRISTATE_AUTO;
} else if (!strcmp(raw, "A")) {
ret = DEBUG_TRISTATE_AUTO;
} else {
bool bool_ret = debug_string_to_bool(raw);
if (bool_ret) {
ret = DEBUG_TRISTATE_ON;
} else {
ret = DEBUG_TRISTATE_OFF;
}
}
return ret;
}
enum debug_tristate_option
debug_get_tristate_option(const char *name)
{
const char *raw = os_getenv(name);
enum debug_tristate_option ret = debug_string_to_tristate(raw);
if (debug_get_bool_option_print()) {
const char *pretty_val;
switch (ret) {
case DEBUG_TRISTATE_OFF: {
pretty_val = "OFF";
break;
}
case DEBUG_TRISTATE_AUTO: {
pretty_val = "AUTO";
break;
}
case DEBUG_TRISTATE_ON: {
pretty_val = "ON";
break;
}
default: pretty_val = "invalid";
}
U_LOG_RAW("%s=%s (%s)", name, pretty_val, raw == NULL ? "nil" : raw);
}
return ret;
}
2019-03-18 05:52:32 +00:00
long
debug_get_num_option(const char *name, long _default)
{
const char *raw = os_getenv(name);
long ret;
if (raw == NULL) {
ret = _default;
} else {
char *endptr;
ret = strtol(raw, &endptr, 0);
// Restore the default value when no digits were found.
if (raw == endptr) {
ret = _default;
}
}
if (debug_get_bool_option_print()) {
2020-07-03 13:28:55 +00:00
U_LOG_RAW("%s=%li (%s)", name, ret, raw == NULL ? "nil" : raw);
2019-03-18 05:52:32 +00:00
}
return ret;
}
float
debug_get_float_option(const char *name, float _default)
{
const char *raw = os_getenv(name);
float ret;
if (raw == NULL) {
ret = _default;
} else {
char *endptr;
ret = strtof(raw, &endptr);
// Restore the default value when no digits were found.
if (raw == endptr) {
ret = _default;
}
}
if (debug_get_bool_option_print()) {
2020-07-03 13:28:55 +00:00
U_LOG_RAW("%s=%f (%s)", name, ret, raw == NULL ? "nil" : raw);
2019-03-18 05:52:32 +00:00
}
return ret;
}
/*!
* This function checks @p str if it matches @p matches, it returns true as long
* as the complete @p str is in the starts of @p matches. Empty string does not
* match.
*/
static bool
is_str_in_start_of(const char *str, const char *matches)
{
if (str[0] == '\0') {
return false;
}
for (int i = 0; str[i] != '\0'; i++) {
if (matches[i] == '\0') {
return false;
}
if (matches[i] != tolower(str[i])) {
return false;
}
}
return true;
}
static const char *
level_to_str(enum u_logging_level level)
{
switch (level) {
case U_LOGGING_TRACE: return "trace";
case U_LOGGING_DEBUG: return "debug";
case U_LOGGING_INFO: return "info";
case U_LOGGING_WARN: return "warn";
case U_LOGGING_ERROR: return "error";
default: return "???";
}
}
enum u_logging_level
debug_get_log_option(const char *name, enum u_logging_level _default)
{
const char *raw = os_getenv(name);
enum u_logging_level ret;
if (raw == NULL) {
ret = _default;
} else if (is_str_in_start_of(raw, "trace")) {
ret = U_LOGGING_TRACE;
} else if (is_str_in_start_of(raw, "debug")) {
ret = U_LOGGING_DEBUG;
} else if (is_str_in_start_of(raw, "info")) {
ret = U_LOGGING_INFO;
} else if (is_str_in_start_of(raw, "warn")) {
ret = U_LOGGING_WARN;
} else if (is_str_in_start_of(raw, "error")) {
ret = U_LOGGING_ERROR;
} else {
ret = _default;
}
if (debug_get_bool_option_print()) {
2021-01-14 14:13:48 +00:00
U_LOG_RAW("%s=%s (%s)", name, level_to_str(ret), raw == NULL ? "nil" : raw);
}
return ret;
}