// Copyright 2019-2020, Collabora, Ltd. // SPDX-License-Identifier: BSL-1.0 /*! * @file * @brief Small debug helpers. * @author Jakob Bornecrantz * @ingroup aux_util * * Debug get option helpers heavily inspired from mesa ones. */ #if defined(_MSC_VER) && !defined(_CRT_SECURE_NO_WARNINGS) #define _CRT_SECURE_NO_WARNINGS #endif #include "util/u_debug.h" #include "util/u_logging.h" #include #include #include #include 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()) { U_LOG_RAW("%s=%s (%s)", name, ret, raw == NULL ? "nil" : raw); } return ret; } bool debug_string_to_bool(const char *raw) { bool ret; if (raw == NULL) { ret = false; } 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); if (debug_get_bool_option_print()) { U_LOG_RAW("%s=%s (%s)", name, ret ? "TRUE" : "FALSE", raw == NULL ? "nil" : raw); } 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; } long debug_string_to_num(const char *raw, long _default) { if (raw == NULL) { return _default; } char *endptr; long ret = strtol(raw, &endptr, 0); // Restore the default value when no digits were found. if (raw == endptr) { ret = _default; } return ret; } long debug_get_num_option(const char *name, long _default) { const char *raw = os_getenv(name); long ret = debug_string_to_num(raw, _default); if (debug_get_bool_option_print()) { U_LOG_RAW("%s=%li (%s)", name, ret, raw == NULL ? "nil" : raw); } 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()) { U_LOG_RAW("%s=%f (%s)", name, ret, raw == NULL ? "nil" : raw); } 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()) { U_LOG_RAW("%s=%s (%s)", name, level_to_str(ret), raw == NULL ? "nil" : raw); } return ret; }