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