16c00db4bb
Pull AFS fixes from David Howells: "Here's a set of patches that fix a number of bugs in the in-kernel AFS client, including: - Fix directory locking to not use individual page locks for directory reading/scanning but rather to use a semaphore on the afs_vnode struct as the directory contents must be read in a single blob and data from different reads must not be mixed as the entire contents may be shuffled about between reads. - Fix address list parsing to handle port specifiers correctly. - Only give up callback records on a server if we actually talked to that server (we might not be able to access a server). - Fix some callback handling bugs, including refcounting, whole-volume callbacks and when callbacks actually get broken in response to a CB.CallBack op. - Fix some server/address rotation bugs, including giving up if we can't probe a server; giving up if a server says it doesn't have a volume, but there are more servers to try. - Fix the decoding of fetched statuses to be OpenAFS compatible. - Fix the handling of server lookups in Cache Manager ops (such as CB.InitCallBackState3) to use a UUID if possible and to handle no server being found. - Fix a bug in server lookup where not all addresses are compared. - Fix the non-encryption of calls that prevents some servers from being accessed (this also requires an AF_RXRPC patch that has already gone in through the net tree). There's also a patch that adds tracepoints to log Cache Manager ops that don't find a matching server, either by UUID or by address" * tag 'afs-fixes-20180514' of git://git.kernel.org/pub/scm/linux/kernel/git/dhowells/linux-fs: afs: Fix the non-encryption of calls afs: Fix CB.CallBack handling afs: Fix whole-volume callback handling afs: Fix afs_find_server search loop afs: Fix the handling of an unfound server in CM operations afs: Add a tracepoint to record callbacks from unlisted servers afs: Fix the handling of CB.InitCallBackState3 to find the server by UUID afs: Fix VNOVOL handling in address rotation afs: Fix AFSFetchStatus decoder to provide OpenAFS compatibility afs: Fix server rotation's handling of fileserver probe failure afs: Fix refcounting in callback registration afs: Fix giving up callbacks on server destruction afs: Fix address list parsing afs: Fix directory page locking
178 lines
3.2 KiB
C
178 lines
3.2 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
#include <errno.h>
|
|
#include <signal.h>
|
|
#include <stdbool.h>
|
|
#include <linux/kernel.h>
|
|
#ifdef HAVE_BACKTRACE_SUPPORT
|
|
#include <execinfo.h>
|
|
#endif
|
|
|
|
#include "../../util/cache.h"
|
|
#include "../../util/debug.h"
|
|
#include "../../util/util.h"
|
|
#include "../browser.h"
|
|
#include "../helpline.h"
|
|
#include "../ui.h"
|
|
#include "../util.h"
|
|
#include "../libslang.h"
|
|
#include "../keysyms.h"
|
|
#include "tui.h"
|
|
|
|
static volatile int ui__need_resize;
|
|
|
|
extern struct perf_error_ops perf_tui_eops;
|
|
extern bool tui_helpline__set;
|
|
|
|
extern void hist_browser__init_hpp(void);
|
|
|
|
void ui__refresh_dimensions(bool force)
|
|
{
|
|
if (force || ui__need_resize) {
|
|
ui__need_resize = 0;
|
|
pthread_mutex_lock(&ui__lock);
|
|
SLtt_get_screen_size();
|
|
SLsmg_reinit_smg();
|
|
pthread_mutex_unlock(&ui__lock);
|
|
}
|
|
}
|
|
|
|
static void ui__sigwinch(int sig __maybe_unused)
|
|
{
|
|
ui__need_resize = 1;
|
|
}
|
|
|
|
static void ui__setup_sigwinch(void)
|
|
{
|
|
static bool done;
|
|
|
|
if (done)
|
|
return;
|
|
|
|
done = true;
|
|
pthread__unblock_sigwinch();
|
|
signal(SIGWINCH, ui__sigwinch);
|
|
}
|
|
|
|
int ui__getch(int delay_secs)
|
|
{
|
|
struct timeval timeout, *ptimeout = delay_secs ? &timeout : NULL;
|
|
fd_set read_set;
|
|
int err, key;
|
|
|
|
ui__setup_sigwinch();
|
|
|
|
FD_ZERO(&read_set);
|
|
FD_SET(0, &read_set);
|
|
|
|
if (delay_secs) {
|
|
timeout.tv_sec = delay_secs;
|
|
timeout.tv_usec = 0;
|
|
}
|
|
|
|
err = select(1, &read_set, NULL, NULL, ptimeout);
|
|
|
|
if (err == 0)
|
|
return K_TIMER;
|
|
|
|
if (err == -1) {
|
|
if (errno == EINTR)
|
|
return K_RESIZE;
|
|
return K_ERROR;
|
|
}
|
|
|
|
key = SLang_getkey();
|
|
if (key != K_ESC)
|
|
return key;
|
|
|
|
FD_ZERO(&read_set);
|
|
FD_SET(0, &read_set);
|
|
timeout.tv_sec = 0;
|
|
timeout.tv_usec = 20;
|
|
err = select(1, &read_set, NULL, NULL, &timeout);
|
|
if (err == 0)
|
|
return K_ESC;
|
|
|
|
SLang_ungetkey(key);
|
|
return SLkp_getkey();
|
|
}
|
|
|
|
#ifdef HAVE_BACKTRACE_SUPPORT
|
|
static void ui__signal_backtrace(int sig)
|
|
{
|
|
void *stackdump[32];
|
|
size_t size;
|
|
|
|
ui__exit(false);
|
|
psignal(sig, "perf");
|
|
|
|
printf("-------- backtrace --------\n");
|
|
size = backtrace(stackdump, ARRAY_SIZE(stackdump));
|
|
backtrace_symbols_fd(stackdump, size, STDOUT_FILENO);
|
|
|
|
exit(0);
|
|
}
|
|
#else
|
|
# define ui__signal_backtrace ui__signal
|
|
#endif
|
|
|
|
static void ui__signal(int sig)
|
|
{
|
|
ui__exit(false);
|
|
psignal(sig, "perf");
|
|
exit(0);
|
|
}
|
|
|
|
int ui__init(void)
|
|
{
|
|
int err;
|
|
|
|
SLutf8_enable(-1);
|
|
SLtt_get_terminfo();
|
|
SLtt_get_screen_size();
|
|
|
|
err = SLsmg_init_smg();
|
|
if (err < 0)
|
|
goto out;
|
|
err = SLang_init_tty(-1, 0, 0);
|
|
if (err < 0)
|
|
goto out;
|
|
|
|
err = SLkp_init();
|
|
if (err < 0) {
|
|
pr_err("TUI initialization failed.\n");
|
|
goto out;
|
|
}
|
|
|
|
SLkp_define_keysym((char *)"^(kB)", SL_KEY_UNTAB);
|
|
|
|
signal(SIGSEGV, ui__signal_backtrace);
|
|
signal(SIGFPE, ui__signal_backtrace);
|
|
signal(SIGINT, ui__signal);
|
|
signal(SIGQUIT, ui__signal);
|
|
signal(SIGTERM, ui__signal);
|
|
|
|
perf_error__register(&perf_tui_eops);
|
|
|
|
ui_helpline__init();
|
|
ui_browser__init();
|
|
tui_progress__init();
|
|
|
|
hist_browser__init_hpp();
|
|
out:
|
|
return err;
|
|
}
|
|
|
|
void ui__exit(bool wait_for_ok)
|
|
{
|
|
if (wait_for_ok && tui_helpline__set)
|
|
ui__question_window("Fatal Error",
|
|
ui_helpline__last_msg,
|
|
"Press any key...", 0);
|
|
|
|
SLtt_set_cursor_visibility(1);
|
|
SLsmg_refresh();
|
|
SLsmg_reset_smg();
|
|
SLang_reset_tty();
|
|
|
|
perf_error__unregister(&perf_tui_eops);
|
|
}
|