mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-26 11:45:08 +00:00
make debug() output more useful
This change does several things. First, and most important, it allows dumping the "n" most recent stack frames on each debug() call. Second, it demangles the C++ symbols. Third, it prepends each debug() message with the debug level. Unrelated to the above I've replaced all `assert(!is_forked_child());` statements with `ASSERT_IS_NOT_FORKED_CHILD()` for consistency.
This commit is contained in:
parent
d55113b5b5
commit
73f2992a2e
7 changed files with 114 additions and 44 deletions
|
@ -362,6 +362,7 @@ AC_SEARCH_LIBS( pthread_create, pthread, , [AC_MSG_ERROR([Cannot find the pthrea
|
||||||
AC_SEARCH_LIBS( setupterm, [ncurses tinfo curses], , [AC_MSG_ERROR([Could not find a curses implementation, needed to build fish. If this is Linux, try running 'sudo apt-get install libncurses5-dev' or 'sudo yum install ncurses-devel'])] )
|
AC_SEARCH_LIBS( setupterm, [ncurses tinfo curses], , [AC_MSG_ERROR([Could not find a curses implementation, needed to build fish. If this is Linux, try running 'sudo apt-get install libncurses5-dev' or 'sudo yum install ncurses-devel'])] )
|
||||||
AC_SEARCH_LIBS( [nan], [m], [AC_DEFINE( [HAVE_NAN], [1], [Define to 1 if you have the nan function])] )
|
AC_SEARCH_LIBS( [nan], [m], [AC_DEFINE( [HAVE_NAN], [1], [Define to 1 if you have the nan function])] )
|
||||||
AC_SEARCH_LIBS( [backtrace_symbols_fd], [execinfo] )
|
AC_SEARCH_LIBS( [backtrace_symbols_fd], [execinfo] )
|
||||||
|
AC_SEARCH_LIBS( [dladdr], [dl] )
|
||||||
|
|
||||||
if test x$local_gettext != xno; then
|
if test x$local_gettext != xno; then
|
||||||
AC_SEARCH_LIBS( gettext, intl,,)
|
AC_SEARCH_LIBS( gettext, intl,,)
|
||||||
|
|
|
@ -25,4 +25,6 @@ The following options are available:
|
||||||
|
|
||||||
- `-v` or `--version` display version and exit
|
- `-v` or `--version` display version and exit
|
||||||
|
|
||||||
|
- `-D` or `--debug-stack-frames=DEBUG_LEVEL` specify how many stack frames to display when debug messages are written. The default is zero. A value of 3 or 4 is usually sufficient to gain insight into how a given debug call was reached but you can specify a value up to 128.
|
||||||
|
|
||||||
The fish exit status is generally the exit status of the last foreground command. If fish is exiting because of a parse error, the exit status is 127.
|
The fish exit status is generally the exit status of the last foreground command. If fish is exiting because of a parse error, the exit status is 127.
|
||||||
|
|
112
src/common.cpp
112
src/common.cpp
|
@ -2,6 +2,8 @@
|
||||||
#include "config.h"
|
#include "config.h"
|
||||||
|
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
|
#include <cxxabi.h>
|
||||||
|
#include <dlfcn.h>
|
||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
#include <limits.h>
|
#include <limits.h>
|
||||||
#include <locale.h>
|
#include <locale.h>
|
||||||
|
@ -49,12 +51,16 @@ static bool thread_assertions_configured_for_testing = false;
|
||||||
|
|
||||||
wchar_t ellipsis_char;
|
wchar_t ellipsis_char;
|
||||||
wchar_t omitted_newline_char;
|
wchar_t omitted_newline_char;
|
||||||
|
|
||||||
bool g_profiling_active = false;
|
bool g_profiling_active = false;
|
||||||
|
|
||||||
const wchar_t *program_name;
|
const wchar_t *program_name;
|
||||||
|
int debug_level = 1; // default maximum debug output level (errors and warnings)
|
||||||
|
int debug_stack_frames = 0; // default number of stack frames to show on debug() calls
|
||||||
|
|
||||||
int debug_level = 1;
|
/// This allows us to notice when we've forked.
|
||||||
|
static pid_t initial_pid = 0;
|
||||||
|
|
||||||
|
/// Be able to restore the term's foreground process group.
|
||||||
|
static pid_t initial_foreground_process_group = -1;
|
||||||
|
|
||||||
/// This struct maintains the current state of the terminal size. It is updated on demand after
|
/// This struct maintains the current state of the terminal size. It is updated on demand after
|
||||||
/// receiving a SIGWINCH. Do not touch this struct directly, it's managed with a rwlock. Use
|
/// receiving a SIGWINCH. Do not touch this struct directly, it's managed with a rwlock. Use
|
||||||
|
@ -65,18 +71,53 @@ static rwlock_t termsize_rwlock;
|
||||||
|
|
||||||
static char *wcs2str_internal(const wchar_t *in, char *out);
|
static char *wcs2str_internal(const wchar_t *in, char *out);
|
||||||
|
|
||||||
void show_stackframe() {
|
// This function produces a stack backtrace with demangled function & method names. It is based on
|
||||||
|
// https://gist.github.com/fmela/591333 but adapted to the style of the fish project.
|
||||||
|
static const wcstring_list_t __attribute__((noinline))
|
||||||
|
demangled_backtrace(int max_frames, int skip_levels) {
|
||||||
|
void *callstack[128];
|
||||||
|
const int n_max_frames = sizeof(callstack) / sizeof(callstack[0]);
|
||||||
|
int n_frames = backtrace(callstack, n_max_frames);
|
||||||
|
char **symbols = backtrace_symbols(callstack, n_frames);
|
||||||
|
wchar_t text[1024];
|
||||||
|
std::vector<wcstring> backtrace_text;
|
||||||
|
|
||||||
|
if (skip_levels + max_frames < n_frames) n_frames = skip_levels + max_frames;
|
||||||
|
|
||||||
|
for (int i = skip_levels; i < n_frames; i++) {
|
||||||
|
Dl_info info;
|
||||||
|
if (dladdr(callstack[i], &info) && info.dli_sname) {
|
||||||
|
char *demangled = NULL;
|
||||||
|
int status = -1;
|
||||||
|
if (info.dli_sname[0] == '_')
|
||||||
|
demangled = abi::__cxa_demangle(info.dli_sname, NULL, 0, &status);
|
||||||
|
swprintf(text, sizeof(text) / sizeof(wchar_t), L"%-3d %s + %td", i - skip_levels,
|
||||||
|
status == 0 ? demangled : info.dli_sname == 0 ? symbols[i] : info.dli_sname,
|
||||||
|
(char *)callstack[i] - (char *)info.dli_saddr);
|
||||||
|
free(demangled);
|
||||||
|
} else {
|
||||||
|
swprintf(text, sizeof(text) / sizeof(wchar_t), L"%-3d %s", i - skip_levels, symbols[i]);
|
||||||
|
}
|
||||||
|
backtrace_text.push_back(text);
|
||||||
|
}
|
||||||
|
free(symbols);
|
||||||
|
return backtrace_text;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void debug_shared(const wchar_t msg_level, const wcstring &msg);
|
||||||
|
void __attribute__((noinline)) show_stackframe(const wchar_t msg_level, int frame_count,
|
||||||
|
int skip_levels) {
|
||||||
ASSERT_IS_NOT_FORKED_CHILD();
|
ASSERT_IS_NOT_FORKED_CHILD();
|
||||||
|
|
||||||
// Hack to avoid showing backtraces in the tester.
|
// Hack to avoid showing backtraces in the tester.
|
||||||
if (program_name && !wcscmp(program_name, L"(ignore)")) return;
|
if (program_name && !wcscmp(program_name, L"(ignore)")) return;
|
||||||
|
|
||||||
void *trace[32];
|
if (frame_count < 1) frame_count = 999;
|
||||||
int trace_size = 0;
|
debug_shared(msg_level, L"Backtrace:");
|
||||||
|
std::vector<wcstring> bt = demangled_backtrace(frame_count, skip_levels + 2);
|
||||||
trace_size = backtrace(trace, 32);
|
for (int i = 0; i < bt.size(); i++) {
|
||||||
debug(0, L"Backtrace:");
|
debug_shared(msg_level, bt[i]);
|
||||||
backtrace_symbols_fd(trace, trace_size, STDERR_FILENO);
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int fgetws2(wcstring *s, FILE *f) {
|
int fgetws2(wcstring *s, FILE *f) {
|
||||||
|
@ -493,23 +534,34 @@ static bool should_debug(int level) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void debug_shared(const wcstring &msg) {
|
static void debug_shared(const wchar_t level, const wcstring &msg) {
|
||||||
const wcstring sb = wcstring(program_name) + L": " + msg;
|
pid_t current_pid = getpid();
|
||||||
fwprintf(stderr, L"%ls", reformat_for_screen(sb).c_str());
|
|
||||||
|
if (current_pid == initial_pid) {
|
||||||
|
fwprintf(stderr, L"<%lc> %ls: %ls\n", (unsigned long)level, program_name, msg.c_str());
|
||||||
|
} else {
|
||||||
|
fwprintf(stderr, L"<%lc> %ls: %d: %ls\n", (unsigned long)level, program_name, current_pid,
|
||||||
|
msg.c_str());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void debug(int level, const wchar_t *msg, ...) {
|
static wchar_t level_char[] = {L'E', L'W', L'2', L'3', L'4', L'5'};
|
||||||
|
void __attribute__((noinline)) debug(int level, const wchar_t *msg, ...) {
|
||||||
if (!should_debug(level)) return;
|
if (!should_debug(level)) return;
|
||||||
int errno_old = errno;
|
int errno_old = errno;
|
||||||
va_list va;
|
va_list va;
|
||||||
va_start(va, msg);
|
va_start(va, msg);
|
||||||
wcstring local_msg = vformat_string(msg, va);
|
wcstring local_msg = vformat_string(msg, va);
|
||||||
va_end(va);
|
va_end(va);
|
||||||
debug_shared(local_msg);
|
const wchar_t msg_level = level <= 5 ? level_char[level] : L'9';
|
||||||
|
debug_shared(msg_level, local_msg);
|
||||||
|
if (debug_stack_frames > 0) {
|
||||||
|
show_stackframe(msg_level, debug_stack_frames, 1);
|
||||||
|
}
|
||||||
errno = errno_old;
|
errno = errno_old;
|
||||||
}
|
}
|
||||||
|
|
||||||
void debug(int level, const char *msg, ...) {
|
void __attribute__((noinline)) debug(int level, const char *msg, ...) {
|
||||||
if (!should_debug(level)) return;
|
if (!should_debug(level)) return;
|
||||||
int errno_old = errno;
|
int errno_old = errno;
|
||||||
char local_msg[512];
|
char local_msg[512];
|
||||||
|
@ -517,7 +569,11 @@ void debug(int level, const char *msg, ...) {
|
||||||
va_start(va, msg);
|
va_start(va, msg);
|
||||||
vsnprintf(local_msg, sizeof local_msg, msg, va);
|
vsnprintf(local_msg, sizeof local_msg, msg, va);
|
||||||
va_end(va);
|
va_end(va);
|
||||||
debug_shared(str2wcstring(local_msg));
|
const wchar_t msg_level = level <= 5 ? level_char[level] : L'9';
|
||||||
|
debug_shared(msg_level, str2wcstring(local_msg));
|
||||||
|
if (debug_stack_frames > 0) {
|
||||||
|
show_stackframe(msg_level, debug_stack_frames, 1);
|
||||||
|
}
|
||||||
errno = errno_old;
|
errno = errno_old;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1470,7 +1526,7 @@ int create_directory(const wcstring &d) {
|
||||||
}
|
}
|
||||||
|
|
||||||
__attribute__((noinline)) void bugreport() {
|
__attribute__((noinline)) void bugreport() {
|
||||||
debug(1, _(L"This is a bug. Break on bugreport to debug."
|
debug(1, _(L"This is a bug. Break on bugreport to debug. "
|
||||||
L"If you can reproduce it, please send a bug report to %s."),
|
L"If you can reproduce it, please send a bug report to %s."),
|
||||||
PACKAGE_BUGREPORT);
|
PACKAGE_BUGREPORT);
|
||||||
}
|
}
|
||||||
|
@ -1630,12 +1686,6 @@ void configure_thread_assertions_for_testing(void) {
|
||||||
thread_assertions_configured_for_testing = true;
|
thread_assertions_configured_for_testing = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Notice when we've forked.
|
|
||||||
static pid_t initial_pid = 0;
|
|
||||||
|
|
||||||
/// Be able to restore the term's foreground process group.
|
|
||||||
static pid_t initial_foreground_process_group = -1;
|
|
||||||
|
|
||||||
bool is_forked_child(void) {
|
bool is_forked_child(void) {
|
||||||
// Just bail if nobody's called setup_fork_guards, e.g. some of our tools.
|
// Just bail if nobody's called setup_fork_guards, e.g. some of our tools.
|
||||||
if (!initial_pid) return false;
|
if (!initial_pid) return false;
|
||||||
|
@ -1712,14 +1762,14 @@ void assert_is_locked(void *vmutex, const char *who, const char *caller) {
|
||||||
|
|
||||||
void scoped_lock::lock(void) {
|
void scoped_lock::lock(void) {
|
||||||
assert(!locked);
|
assert(!locked);
|
||||||
assert(!is_forked_child());
|
ASSERT_IS_NOT_FORKED_CHILD();
|
||||||
VOMIT_ON_FAILURE_NO_ERRNO(pthread_mutex_lock(lock_obj));
|
VOMIT_ON_FAILURE_NO_ERRNO(pthread_mutex_lock(lock_obj));
|
||||||
locked = true;
|
locked = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void scoped_lock::unlock(void) {
|
void scoped_lock::unlock(void) {
|
||||||
assert(locked);
|
assert(locked);
|
||||||
assert(!is_forked_child());
|
ASSERT_IS_NOT_FORKED_CHILD();
|
||||||
VOMIT_ON_FAILURE_NO_ERRNO(pthread_mutex_unlock(lock_obj));
|
VOMIT_ON_FAILURE_NO_ERRNO(pthread_mutex_unlock(lock_obj));
|
||||||
locked = false;
|
locked = false;
|
||||||
}
|
}
|
||||||
|
@ -1736,35 +1786,35 @@ scoped_lock::~scoped_lock() {
|
||||||
|
|
||||||
void scoped_rwlock::lock(void) {
|
void scoped_rwlock::lock(void) {
|
||||||
assert(!(locked || locked_shared));
|
assert(!(locked || locked_shared));
|
||||||
assert(!is_forked_child());
|
ASSERT_IS_NOT_FORKED_CHILD();
|
||||||
VOMIT_ON_FAILURE_NO_ERRNO(pthread_rwlock_rdlock(rwlock_obj));
|
VOMIT_ON_FAILURE_NO_ERRNO(pthread_rwlock_rdlock(rwlock_obj));
|
||||||
locked = true;
|
locked = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void scoped_rwlock::unlock(void) {
|
void scoped_rwlock::unlock(void) {
|
||||||
assert(locked);
|
assert(locked);
|
||||||
assert(!is_forked_child());
|
ASSERT_IS_NOT_FORKED_CHILD();
|
||||||
VOMIT_ON_FAILURE_NO_ERRNO(pthread_rwlock_unlock(rwlock_obj));
|
VOMIT_ON_FAILURE_NO_ERRNO(pthread_rwlock_unlock(rwlock_obj));
|
||||||
locked = false;
|
locked = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
void scoped_rwlock::lock_shared(void) {
|
void scoped_rwlock::lock_shared(void) {
|
||||||
assert(!(locked || locked_shared));
|
assert(!(locked || locked_shared));
|
||||||
assert(!is_forked_child());
|
ASSERT_IS_NOT_FORKED_CHILD();
|
||||||
VOMIT_ON_FAILURE_NO_ERRNO(pthread_rwlock_wrlock(rwlock_obj));
|
VOMIT_ON_FAILURE_NO_ERRNO(pthread_rwlock_wrlock(rwlock_obj));
|
||||||
locked_shared = true;
|
locked_shared = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void scoped_rwlock::unlock_shared(void) {
|
void scoped_rwlock::unlock_shared(void) {
|
||||||
assert(locked_shared);
|
assert(locked_shared);
|
||||||
assert(!is_forked_child());
|
ASSERT_IS_NOT_FORKED_CHILD();
|
||||||
VOMIT_ON_FAILURE_NO_ERRNO(pthread_rwlock_unlock(rwlock_obj));
|
VOMIT_ON_FAILURE_NO_ERRNO(pthread_rwlock_unlock(rwlock_obj));
|
||||||
locked_shared = false;
|
locked_shared = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
void scoped_rwlock::upgrade(void) {
|
void scoped_rwlock::upgrade(void) {
|
||||||
assert(locked_shared);
|
assert(locked_shared);
|
||||||
assert(!is_forked_child());
|
ASSERT_IS_NOT_FORKED_CHILD();
|
||||||
VOMIT_ON_FAILURE_NO_ERRNO(pthread_rwlock_unlock(rwlock_obj));
|
VOMIT_ON_FAILURE_NO_ERRNO(pthread_rwlock_unlock(rwlock_obj));
|
||||||
locked = false;
|
locked = false;
|
||||||
VOMIT_ON_FAILURE_NO_ERRNO(pthread_rwlock_wrlock(rwlock_obj));
|
VOMIT_ON_FAILURE_NO_ERRNO(pthread_rwlock_wrlock(rwlock_obj));
|
||||||
|
|
17
src/common.h
17
src/common.h
|
@ -175,6 +175,9 @@ extern wchar_t omitted_newline_char;
|
||||||
/// it will not be printed.
|
/// it will not be printed.
|
||||||
extern int debug_level;
|
extern int debug_level;
|
||||||
|
|
||||||
|
/// How many stack frames to show when a debug() call is made.
|
||||||
|
extern int debug_stack_frames;
|
||||||
|
|
||||||
/// Profiling flag. True if commands should be profiled.
|
/// Profiling flag. True if commands should be profiled.
|
||||||
extern bool g_profiling_active;
|
extern bool g_profiling_active;
|
||||||
|
|
||||||
|
@ -192,7 +195,7 @@ void write_ignore(int fd, const void *buff, size_t count);
|
||||||
if (!(arg)) { \
|
if (!(arg)) { \
|
||||||
debug(0, "function %s called with null value for argument %s. ", __func__, #arg); \
|
debug(0, "function %s called with null value for argument %s. ", __func__, #arg); \
|
||||||
bugreport(); \
|
bugreport(); \
|
||||||
show_stackframe(); \
|
show_stackframe(L'E'); \
|
||||||
return retval; \
|
return retval; \
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -200,7 +203,7 @@ void write_ignore(int fd, const void *buff, size_t count);
|
||||||
#define FATAL_EXIT() \
|
#define FATAL_EXIT() \
|
||||||
{ \
|
{ \
|
||||||
char exit_read_buff; \
|
char exit_read_buff; \
|
||||||
show_stackframe(); \
|
show_stackframe(L'E'); \
|
||||||
read_ignore(0, &exit_read_buff, 1); \
|
read_ignore(0, &exit_read_buff, 1); \
|
||||||
exit_without_destructors(1); \
|
exit_without_destructors(1); \
|
||||||
}
|
}
|
||||||
|
@ -219,7 +222,7 @@ void write_ignore(int fd, const void *buff, size_t count);
|
||||||
if (signal_is_blocked()) { \
|
if (signal_is_blocked()) { \
|
||||||
debug(0, "function %s called while blocking signals. ", __func__); \
|
debug(0, "function %s called while blocking signals. ", __func__); \
|
||||||
bugreport(); \
|
bugreport(); \
|
||||||
show_stackframe(); \
|
show_stackframe(L'E'); \
|
||||||
return retval; \
|
return retval; \
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -234,7 +237,7 @@ void write_ignore(int fd, const void *buff, size_t count);
|
||||||
#define contains(str, ...) contains_internal(str, 0, __VA_ARGS__, NULL)
|
#define contains(str, ...) contains_internal(str, 0, __VA_ARGS__, NULL)
|
||||||
|
|
||||||
/// Print a stack trace to stderr.
|
/// Print a stack trace to stderr.
|
||||||
void show_stackframe();
|
void show_stackframe(const wchar_t msg_level, int frame_count = -1, int skip_levels = 0);
|
||||||
|
|
||||||
/// Read a line from the stream f into the string. Returns the number of bytes read or -1 on
|
/// Read a line from the stream f into the string. Returns the number of bytes read or -1 on
|
||||||
/// failure.
|
/// failure.
|
||||||
|
@ -485,8 +488,6 @@ class null_terminated_array_t {
|
||||||
void convert_wide_array_to_narrow(const null_terminated_array_t<wchar_t> &arr,
|
void convert_wide_array_to_narrow(const null_terminated_array_t<wchar_t> &arr,
|
||||||
null_terminated_array_t<char> *output);
|
null_terminated_array_t<char> *output);
|
||||||
|
|
||||||
bool is_forked_child();
|
|
||||||
|
|
||||||
class mutex_lock_t {
|
class mutex_lock_t {
|
||||||
public:
|
public:
|
||||||
pthread_mutex_t mutex;
|
pthread_mutex_t mutex;
|
||||||
|
@ -668,8 +669,8 @@ ssize_t read_loop(int fd, void *buff, size_t count);
|
||||||
///
|
///
|
||||||
/// will print the string 'fish: Pi = 3.141', given that debug_level is 1 or higher, and that
|
/// will print the string 'fish: Pi = 3.141', given that debug_level is 1 or higher, and that
|
||||||
/// program_name is 'fish'.
|
/// program_name is 'fish'.
|
||||||
void debug(int level, const char *msg, ...);
|
void __attribute__((noinline)) debug(int level, const char *msg, ...);
|
||||||
void debug(int level, const wchar_t *msg, ...);
|
void __attribute__((noinline)) debug(int level, const wchar_t *msg, ...);
|
||||||
|
|
||||||
/// Replace special characters with backslash escape sequences. Newline is replaced with \n, etc.
|
/// Replace special characters with backslash escape sequences. Newline is replaced with \n, etc.
|
||||||
///
|
///
|
||||||
|
|
|
@ -948,7 +948,7 @@ void exec_job(parser_t &parser, job_t *j) {
|
||||||
bool builtin_io_done = do_builtin_io(outbuff.data(), outbuff.size(),
|
bool builtin_io_done = do_builtin_io(outbuff.data(), outbuff.size(),
|
||||||
errbuff.data(), errbuff.size());
|
errbuff.data(), errbuff.size());
|
||||||
if (!builtin_io_done && errno != EPIPE) {
|
if (!builtin_io_done && errno != EPIPE) {
|
||||||
show_stackframe();
|
show_stackframe(L'E');
|
||||||
}
|
}
|
||||||
fork_was_skipped = true;
|
fork_was_skipped = true;
|
||||||
}
|
}
|
||||||
|
|
22
src/fish.cpp
22
src/fish.cpp
|
@ -309,11 +309,12 @@ static int read_init(const struct config_paths_t &paths) {
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Parse the argument list, return the index of the first non-switch arguments.
|
/// Parse the argument list, return the index of the first non-flag arguments.
|
||||||
static int fish_parse_opt(int argc, char **argv, std::vector<std::string> *cmds) {
|
static int fish_parse_opt(int argc, char **argv, std::vector<std::string> *cmds) {
|
||||||
const char *short_opts = "+hilnvc:p:d:";
|
const char *short_opts = "+hilnvc:p:d:D:";
|
||||||
const struct option long_opts[] = {{"command", required_argument, NULL, 'c'},
|
const struct option long_opts[] = {{"command", required_argument, NULL, 'c'},
|
||||||
{"debug-level", required_argument, NULL, 'd'},
|
{"debug-level", required_argument, NULL, 'd'},
|
||||||
|
{"debug-stack-frames", required_argument, NULL, 'D'},
|
||||||
{"interactive", no_argument, NULL, 'i'},
|
{"interactive", no_argument, NULL, 'i'},
|
||||||
{"login", no_argument, NULL, 'l'},
|
{"login", no_argument, NULL, 'l'},
|
||||||
{"no-execute", no_argument, NULL, 'n'},
|
{"no-execute", no_argument, NULL, 'n'},
|
||||||
|
@ -343,7 +344,7 @@ static int fish_parse_opt(int argc, char **argv, std::vector<std::string> *cmds)
|
||||||
if (tmp >= 0 && tmp <= 10 && !*end && !errno) {
|
if (tmp >= 0 && tmp <= 10 && !*end && !errno) {
|
||||||
debug_level = (int)tmp;
|
debug_level = (int)tmp;
|
||||||
} else {
|
} else {
|
||||||
fwprintf(stderr, _(L"Invalid value '%s' for debug level switch"), optarg);
|
fwprintf(stderr, _(L"Invalid value '%s' for debug-level flag"), optarg);
|
||||||
exit(1);
|
exit(1);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
@ -373,6 +374,21 @@ static int fish_parse_opt(int argc, char **argv, std::vector<std::string> *cmds)
|
||||||
fwprintf(stdout, _(L"%s, version %s\n"), PACKAGE_NAME, get_fish_version());
|
fwprintf(stdout, _(L"%s, version %s\n"), PACKAGE_NAME, get_fish_version());
|
||||||
exit(0);
|
exit(0);
|
||||||
}
|
}
|
||||||
|
case 'D': {
|
||||||
|
char *end;
|
||||||
|
long tmp;
|
||||||
|
|
||||||
|
errno = 0;
|
||||||
|
tmp = strtol(optarg, &end, 10);
|
||||||
|
|
||||||
|
if (tmp > 0 && tmp <= 128 && !*end && !errno) {
|
||||||
|
debug_stack_frames = (int)tmp;
|
||||||
|
} else {
|
||||||
|
fwprintf(stderr, _(L"Invalid value '%s' for debug-stack-frames flag"), optarg);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
default: {
|
default: {
|
||||||
// We assume getopt_long() has already emitted a diagnostic msg.
|
// We assume getopt_long() has already emitted a diagnostic msg.
|
||||||
exit(1);
|
exit(1);
|
||||||
|
|
|
@ -1,2 +1,2 @@
|
||||||
fish: An error occurred while redirecting file '/'
|
<W> fish: An error occurred while redirecting file '/'
|
||||||
open: Is a directory
|
open: Is a directory
|
||||||
|
|
Loading…
Reference in a new issue