Linux Audio

Check our new training course

Loading...
v6.9.4
  1#include <errno.h>
  2#include <signal.h>
  3#include <stdbool.h>
  4#include <stdlib.h>
  5#include <termios.h>
  6#include <unistd.h>
  7#include <linux/kernel.h>
  8#ifdef HAVE_BACKTRACE_SUPPORT
  9#include <execinfo.h>
 10#endif
 11
 12#include "../../util/color.h"
 13#include "../../util/debug.h"
 14#include "../browser.h"
 15#include "../helpline.h"
 16#include "../ui.h"
 17#include "../util.h"
 18#include "../libslang.h"
 19#include "../keysyms.h"
 20#include "tui.h"
 21
 22static volatile int ui__need_resize;
 23
 24extern struct perf_error_ops perf_tui_eops;
 25extern bool tui_helpline__set;
 26
 27extern void hist_browser__init_hpp(void);
 28
 29void ui__refresh_dimensions(bool force)
 30{
 31	if (force || ui__need_resize) {
 32		ui__need_resize = 0;
 33		mutex_lock(&ui__lock);
 34		SLtt_get_screen_size();
 35		SLsmg_reinit_smg();
 36		mutex_unlock(&ui__lock);
 37	}
 38}
 39
 40static void ui__sigwinch(int sig __maybe_unused)
 41{
 42	ui__need_resize = 1;
 43}
 44
 45static void ui__setup_sigwinch(void)
 46{
 47	static bool done;
 48
 49	if (done)
 50		return;
 51
 52	done = true;
 53	pthread__unblock_sigwinch();
 54	signal(SIGWINCH, ui__sigwinch);
 55}
 56
 57int ui__getch(int delay_secs)
 58{
 59	struct timeval timeout, *ptimeout = delay_secs ? &timeout : NULL;
 60	fd_set read_set;
 61	int err, key;
 62
 63	ui__setup_sigwinch();
 64
 65	FD_ZERO(&read_set);
 66	FD_SET(0, &read_set);
 67
 68	if (delay_secs) {
 69		timeout.tv_sec = delay_secs;
 70		timeout.tv_usec = 0;
 71	}
 72
 73        err = select(1, &read_set, NULL, NULL, ptimeout);
 74
 75	if (err == 0)
 76		return K_TIMER;
 77
 78	if (err == -1) {
 79		if (errno == EINTR)
 80			return K_RESIZE;
 81		return K_ERROR;
 82	}
 83
 84	key = SLang_getkey();
 85	if (key != K_ESC)
 86		return key;
 87
 88	FD_ZERO(&read_set);
 89	FD_SET(0, &read_set);
 90	timeout.tv_sec = 0;
 91	timeout.tv_usec = 20;
 92        err = select(1, &read_set, NULL, NULL, &timeout);
 93	if (err == 0)
 94		return K_ESC;
 95
 96	SLang_ungetkey(key);
 97	return SLkp_getkey();
 98}
 99
100#ifdef HAVE_BACKTRACE_SUPPORT
101static void ui__signal_backtrace(int sig)
102{
103	void *stackdump[32];
104	size_t size;
105
106	ui__exit(false);
107	psignal(sig, "perf");
108
109	printf("-------- backtrace --------\n");
110	size = backtrace(stackdump, ARRAY_SIZE(stackdump));
111	backtrace_symbols_fd(stackdump, size, STDOUT_FILENO);
112
113	exit(0);
114}
115#else
116# define ui__signal_backtrace  ui__signal
117#endif
118
119static void ui__signal(int sig)
120{
121	ui__exit(false);
122	psignal(sig, "perf");
123	exit(0);
124}
125
126static void ui__sigcont(int sig)
127{
128	static struct termios tty;
129
130	if (sig == SIGTSTP) {
131		while (tcgetattr(SLang_TT_Read_FD, &tty) == -1 && errno == EINTR)
132			;
133		while (write(SLang_TT_Read_FD, PERF_COLOR_RESET, sizeof(PERF_COLOR_RESET) - 1) == -1 && errno == EINTR)
134			;
135		raise(SIGSTOP);
136	} else {
137		while (tcsetattr(SLang_TT_Read_FD, TCSADRAIN, &tty) == -1 && errno == EINTR)
138			;
139		raise(SIGWINCH);
140	}
141}
142
143int ui__init(void)
144{
145	int err;
146
147	SLutf8_enable(-1);
148	SLtt_get_terminfo();
149	SLtt_get_screen_size();
150
151	err = SLsmg_init_smg();
152	if (err < 0)
153		goto out;
154	err = SLang_init_tty(-1, 0, 0);
155	if (err < 0)
156		goto out;
157	SLtty_set_suspend_state(true);
158
159	err = SLkp_init();
160	if (err < 0) {
161		pr_err("TUI initialization failed.\n");
162		goto out;
163	}
164
165	SLkp_define_keysym("^(kB)", SL_KEY_UNTAB);
 
 
 
 
166
167	signal(SIGSEGV, ui__signal_backtrace);
168	signal(SIGFPE, ui__signal_backtrace);
169	signal(SIGINT, ui__signal);
170	signal(SIGQUIT, ui__signal);
171	signal(SIGTERM, ui__signal);
172	signal(SIGTSTP, ui__sigcont);
173	signal(SIGCONT, ui__sigcont);
174
175	perf_error__register(&perf_tui_eops);
176
177	ui_helpline__init();
178	ui_browser__init();
179	tui_progress__init();
180
181	hist_browser__init_hpp();
182out:
183	return err;
184}
185
186void ui__exit(bool wait_for_ok)
187{
188	if (wait_for_ok && tui_helpline__set)
189		ui__question_window("Fatal Error",
190				    ui_helpline__last_msg,
191				    "Press any key...", 0);
192
193	SLtt_set_cursor_visibility(1);
194	if (mutex_trylock(&ui__lock)) {
195		SLsmg_refresh();
196		SLsmg_reset_smg();
197		mutex_unlock(&ui__lock);
198	}
199	SLang_reset_tty();
 
200	perf_error__unregister(&perf_tui_eops);
201}
v3.15
 
  1#include <signal.h>
  2#include <stdbool.h>
 
 
 
 
 
 
 
  3
  4#include "../../util/cache.h"
  5#include "../../util/debug.h"
  6#include "../browser.h"
  7#include "../helpline.h"
  8#include "../ui.h"
  9#include "../util.h"
 10#include "../libslang.h"
 11#include "../keysyms.h"
 12#include "tui.h"
 13
 14static volatile int ui__need_resize;
 15
 16extern struct perf_error_ops perf_tui_eops;
 
 17
 18extern void hist_browser__init_hpp(void);
 19
 20void ui__refresh_dimensions(bool force)
 21{
 22	if (force || ui__need_resize) {
 23		ui__need_resize = 0;
 24		pthread_mutex_lock(&ui__lock);
 25		SLtt_get_screen_size();
 26		SLsmg_reinit_smg();
 27		pthread_mutex_unlock(&ui__lock);
 28	}
 29}
 30
 31static void ui__sigwinch(int sig __maybe_unused)
 32{
 33	ui__need_resize = 1;
 34}
 35
 36static void ui__setup_sigwinch(void)
 37{
 38	static bool done;
 39
 40	if (done)
 41		return;
 42
 43	done = true;
 44	pthread__unblock_sigwinch();
 45	signal(SIGWINCH, ui__sigwinch);
 46}
 47
 48int ui__getch(int delay_secs)
 49{
 50	struct timeval timeout, *ptimeout = delay_secs ? &timeout : NULL;
 51	fd_set read_set;
 52	int err, key;
 53
 54	ui__setup_sigwinch();
 55
 56	FD_ZERO(&read_set);
 57	FD_SET(0, &read_set);
 58
 59	if (delay_secs) {
 60		timeout.tv_sec = delay_secs;
 61		timeout.tv_usec = 0;
 62	}
 63
 64        err = select(1, &read_set, NULL, NULL, ptimeout);
 65
 66	if (err == 0)
 67		return K_TIMER;
 68
 69	if (err == -1) {
 70		if (errno == EINTR)
 71			return K_RESIZE;
 72		return K_ERROR;
 73	}
 74
 75	key = SLang_getkey();
 76	if (key != K_ESC)
 77		return key;
 78
 79	FD_ZERO(&read_set);
 80	FD_SET(0, &read_set);
 81	timeout.tv_sec = 0;
 82	timeout.tv_usec = 20;
 83        err = select(1, &read_set, NULL, NULL, &timeout);
 84	if (err == 0)
 85		return K_ESC;
 86
 87	SLang_ungetkey(key);
 88	return SLkp_getkey();
 89}
 90
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 91static void ui__signal(int sig)
 92{
 93	ui__exit(false);
 94	psignal(sig, "perf");
 95	exit(0);
 96}
 97
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 98int ui__init(void)
 99{
100	int err;
101
102	SLutf8_enable(-1);
103	SLtt_get_terminfo();
104	SLtt_get_screen_size();
105
106	err = SLsmg_init_smg();
107	if (err < 0)
108		goto out;
109	err = SLang_init_tty(0, 0, 0);
110	if (err < 0)
111		goto out;
 
112
113	err = SLkp_init();
114	if (err < 0) {
115		pr_err("TUI initialization failed.\n");
116		goto out;
117	}
118
119	SLkp_define_keysym((char *)"^(kB)", SL_KEY_UNTAB);
120
121	ui_helpline__init();
122	ui_browser__init();
123	tui_progress__init();
124
125	signal(SIGSEGV, ui__signal);
126	signal(SIGFPE, ui__signal);
127	signal(SIGINT, ui__signal);
128	signal(SIGQUIT, ui__signal);
129	signal(SIGTERM, ui__signal);
 
 
130
131	perf_error__register(&perf_tui_eops);
132
 
 
 
 
133	hist_browser__init_hpp();
134out:
135	return err;
136}
137
138void ui__exit(bool wait_for_ok)
139{
140	if (wait_for_ok)
141		ui__question_window("Fatal Error",
142				    ui_helpline__last_msg,
143				    "Press any key...", 0);
144
145	SLtt_set_cursor_visibility(1);
146	SLsmg_refresh();
147	SLsmg_reset_smg();
 
 
 
148	SLang_reset_tty();
149
150	perf_error__unregister(&perf_tui_eops);
151}