82aff6cc07
Switch to the use of mutex wrappers that provide better error checking. Signed-off-by: Ian Rogers <irogers@google.com> Cc: Adrian Hunter <adrian.hunter@intel.com> Cc: Alexander Shishkin <alexander.shishkin@linux.intel.com> Cc: Alexandre Truong <alexandre.truong@arm.com> Cc: Alexey Bayduraev <alexey.v.bayduraev@linux.intel.com> Cc: Andi Kleen <ak@linux.intel.com> Cc: Andres Freund <andres@anarazel.de> Cc: Andrii Nakryiko <andrii@kernel.org> Cc: André Almeida <andrealmeid@igalia.com> Cc: Athira Jajeev <atrajeev@linux.vnet.ibm.com> Cc: Christophe JAILLET <christophe.jaillet@wanadoo.fr> Cc: Colin Ian King <colin.king@intel.com> Cc: Dario Petrillo <dario.pk1@gmail.com> Cc: Darren Hart <dvhart@infradead.org> Cc: Dave Marchevsky <davemarchevsky@fb.com> Cc: Davidlohr Bueso <dave@stgolabs.net> Cc: Fangrui Song <maskray@google.com> Cc: Hewenliang <hewenliang4@huawei.com> Cc: Ingo Molnar <mingo@redhat.com> Cc: James Clark <james.clark@arm.com> Cc: Jason Wang <wangborong@cdjrlc.com> Cc: Jiri Olsa <jolsa@kernel.org> Cc: Kajol Jain <kjain@linux.ibm.com> Cc: Kim Phillips <kim.phillips@amd.com> Cc: Leo Yan <leo.yan@linaro.org> Cc: Mark Rutland <mark.rutland@arm.com> Cc: Martin Liška <mliska@suse.cz> Cc: Masami Hiramatsu <mhiramat@kernel.org> Cc: Nathan Chancellor <nathan@kernel.org> Cc: Nick Desaulniers <ndesaulniers@google.com> Cc: Pavithra Gurushankar <gpavithrasha@gmail.com> Cc: Peter Zijlstra <peterz@infradead.org> Cc: Quentin Monnet <quentin@isovalent.com> Cc: Ravi Bangoria <ravi.bangoria@amd.com> Cc: Remi Bernon <rbernon@codeweavers.com> Cc: Riccardo Mancini <rickyman7@gmail.com> Cc: Song Liu <songliubraving@fb.com> Cc: Stephane Eranian <eranian@google.com> Cc: Thomas Gleixner <tglx@linutronix.de> Cc: Thomas Richter <tmricht@linux.ibm.com> Cc: Tom Rix <trix@redhat.com> Cc: Weiguo Li <liwg06@foxmail.com> Cc: Wenyu Liu <liuwenyu7@huawei.com> Cc: William Cohen <wcohen@redhat.com> Cc: Zechuan Chen <chenzechuan1@huawei.com> Cc: bpf@vger.kernel.org Cc: llvm@lists.linux.dev Cc: yaowenbin <yaowenbin1@huawei.com> Link: https://lore.kernel.org/r/20220826164242.43412-10-irogers@google.com Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
275 lines
5.5 KiB
C
275 lines
5.5 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
#include <signal.h>
|
|
#include <stdbool.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#include <sys/ttydefaults.h>
|
|
|
|
#include "../browser.h"
|
|
#include "../keysyms.h"
|
|
#include "../helpline.h"
|
|
#include "../ui.h"
|
|
#include "../util.h"
|
|
#include "../libslang.h"
|
|
|
|
static void ui_browser__argv_write(struct ui_browser *browser,
|
|
void *entry, int row)
|
|
{
|
|
char **arg = entry;
|
|
bool current_entry = ui_browser__is_current_entry(browser, row);
|
|
|
|
ui_browser__set_color(browser, current_entry ? HE_COLORSET_SELECTED :
|
|
HE_COLORSET_NORMAL);
|
|
ui_browser__write_nstring(browser, *arg, browser->width);
|
|
}
|
|
|
|
static int popup_menu__run(struct ui_browser *menu, int *keyp)
|
|
{
|
|
int key;
|
|
|
|
if (ui_browser__show(menu, " ", "ESC: exit, ENTER|->: Select option") < 0)
|
|
return -1;
|
|
|
|
while (1) {
|
|
key = ui_browser__run(menu, 0);
|
|
|
|
switch (key) {
|
|
case K_RIGHT:
|
|
case K_ENTER:
|
|
key = menu->index;
|
|
break;
|
|
case K_LEFT:
|
|
case K_ESC:
|
|
case 'q':
|
|
case CTRL('c'):
|
|
key = -1;
|
|
break;
|
|
default:
|
|
if (keyp) {
|
|
*keyp = key;
|
|
key = menu->nr_entries;
|
|
break;
|
|
}
|
|
continue;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
ui_browser__hide(menu);
|
|
return key;
|
|
}
|
|
|
|
int ui__popup_menu(int argc, char * const argv[], int *keyp)
|
|
{
|
|
struct ui_browser menu = {
|
|
.entries = (void *)argv,
|
|
.refresh = ui_browser__argv_refresh,
|
|
.seek = ui_browser__argv_seek,
|
|
.write = ui_browser__argv_write,
|
|
.nr_entries = argc,
|
|
};
|
|
return popup_menu__run(&menu, keyp);
|
|
}
|
|
|
|
int ui_browser__input_window(const char *title, const char *text, char *input,
|
|
const char *exit_msg, int delay_secs)
|
|
{
|
|
int x, y, len, key;
|
|
int max_len = 60, nr_lines = 0;
|
|
static char buf[50];
|
|
const char *t;
|
|
|
|
t = text;
|
|
while (1) {
|
|
const char *sep = strchr(t, '\n');
|
|
|
|
if (sep == NULL)
|
|
sep = strchr(t, '\0');
|
|
len = sep - t;
|
|
if (max_len < len)
|
|
max_len = len;
|
|
++nr_lines;
|
|
if (*sep == '\0')
|
|
break;
|
|
t = sep + 1;
|
|
}
|
|
|
|
mutex_lock(&ui__lock);
|
|
|
|
max_len += 2;
|
|
nr_lines += 8;
|
|
y = SLtt_Screen_Rows / 2 - nr_lines / 2;
|
|
x = SLtt_Screen_Cols / 2 - max_len / 2;
|
|
|
|
SLsmg_set_color(0);
|
|
SLsmg_draw_box(y, x++, nr_lines, max_len);
|
|
if (title) {
|
|
SLsmg_gotorc(y, x + 1);
|
|
SLsmg_write_string((char *)title);
|
|
}
|
|
SLsmg_gotorc(++y, x);
|
|
nr_lines -= 7;
|
|
max_len -= 2;
|
|
SLsmg_write_wrapped_string((unsigned char *)text, y, x,
|
|
nr_lines, max_len, 1);
|
|
y += nr_lines;
|
|
len = 5;
|
|
while (len--) {
|
|
SLsmg_gotorc(y + len - 1, x);
|
|
SLsmg_write_nstring((char *)" ", max_len);
|
|
}
|
|
SLsmg_draw_box(y++, x + 1, 3, max_len - 2);
|
|
|
|
SLsmg_gotorc(y + 3, x);
|
|
SLsmg_write_nstring((char *)exit_msg, max_len);
|
|
SLsmg_refresh();
|
|
|
|
mutex_unlock(&ui__lock);
|
|
|
|
x += 2;
|
|
len = 0;
|
|
key = ui__getch(delay_secs);
|
|
while (key != K_TIMER && key != K_ENTER && key != K_ESC) {
|
|
mutex_lock(&ui__lock);
|
|
|
|
if (key == K_BKSPC) {
|
|
if (len == 0) {
|
|
mutex_unlock(&ui__lock);
|
|
goto next_key;
|
|
}
|
|
SLsmg_gotorc(y, x + --len);
|
|
SLsmg_write_char(' ');
|
|
} else {
|
|
buf[len] = key;
|
|
SLsmg_gotorc(y, x + len++);
|
|
SLsmg_write_char(key);
|
|
}
|
|
SLsmg_refresh();
|
|
|
|
mutex_unlock(&ui__lock);
|
|
|
|
/* XXX more graceful overflow handling needed */
|
|
if (len == sizeof(buf) - 1) {
|
|
ui_helpline__push("maximum size of symbol name reached!");
|
|
key = K_ENTER;
|
|
break;
|
|
}
|
|
next_key:
|
|
key = ui__getch(delay_secs);
|
|
}
|
|
|
|
buf[len] = '\0';
|
|
strncpy(input, buf, len+1);
|
|
return key;
|
|
}
|
|
|
|
void __ui__info_window(const char *title, const char *text, const char *exit_msg)
|
|
{
|
|
int x, y;
|
|
int max_len = 0, nr_lines = 0;
|
|
const char *t;
|
|
|
|
t = text;
|
|
while (1) {
|
|
const char *sep = strchr(t, '\n');
|
|
int len;
|
|
|
|
if (sep == NULL)
|
|
sep = strchr(t, '\0');
|
|
len = sep - t;
|
|
if (max_len < len)
|
|
max_len = len;
|
|
++nr_lines;
|
|
if (*sep == '\0')
|
|
break;
|
|
t = sep + 1;
|
|
}
|
|
|
|
max_len += 2;
|
|
nr_lines += 2;
|
|
if (exit_msg)
|
|
nr_lines += 2;
|
|
y = SLtt_Screen_Rows / 2 - nr_lines / 2,
|
|
x = SLtt_Screen_Cols / 2 - max_len / 2;
|
|
|
|
SLsmg_set_color(0);
|
|
SLsmg_draw_box(y, x++, nr_lines, max_len);
|
|
if (title) {
|
|
SLsmg_gotorc(y, x + 1);
|
|
SLsmg_write_string((char *)title);
|
|
}
|
|
SLsmg_gotorc(++y, x);
|
|
if (exit_msg)
|
|
nr_lines -= 2;
|
|
max_len -= 2;
|
|
SLsmg_write_wrapped_string((unsigned char *)text, y, x,
|
|
nr_lines, max_len, 1);
|
|
if (exit_msg) {
|
|
SLsmg_gotorc(y + nr_lines - 2, x);
|
|
SLsmg_write_nstring((char *)" ", max_len);
|
|
SLsmg_gotorc(y + nr_lines - 1, x);
|
|
SLsmg_write_nstring((char *)exit_msg, max_len);
|
|
}
|
|
}
|
|
|
|
void ui__info_window(const char *title, const char *text)
|
|
{
|
|
mutex_lock(&ui__lock);
|
|
__ui__info_window(title, text, NULL);
|
|
SLsmg_refresh();
|
|
mutex_unlock(&ui__lock);
|
|
}
|
|
|
|
int ui__question_window(const char *title, const char *text,
|
|
const char *exit_msg, int delay_secs)
|
|
{
|
|
mutex_lock(&ui__lock);
|
|
__ui__info_window(title, text, exit_msg);
|
|
SLsmg_refresh();
|
|
mutex_unlock(&ui__lock);
|
|
return ui__getch(delay_secs);
|
|
}
|
|
|
|
int ui__help_window(const char *text)
|
|
{
|
|
return ui__question_window("Help", text, "Press any key...", 0);
|
|
}
|
|
|
|
int ui__dialog_yesno(const char *msg)
|
|
{
|
|
return ui__question_window(NULL, msg, "Enter: Yes, ESC: No", 0);
|
|
}
|
|
|
|
static int __ui__warning(const char *title, const char *format, va_list args)
|
|
{
|
|
char *s;
|
|
|
|
if (vasprintf(&s, format, args) > 0) {
|
|
int key;
|
|
|
|
key = ui__question_window(title, s, "Press any key...", 0);
|
|
free(s);
|
|
return key;
|
|
}
|
|
|
|
fprintf(stderr, "%s\n", title);
|
|
vfprintf(stderr, format, args);
|
|
return K_ESC;
|
|
}
|
|
|
|
static int perf_tui__error(const char *format, va_list args)
|
|
{
|
|
return __ui__warning("Error:", format, args);
|
|
}
|
|
|
|
static int perf_tui__warning(const char *format, va_list args)
|
|
{
|
|
return __ui__warning("Warning:", format, args);
|
|
}
|
|
|
|
struct perf_error_ops perf_tui_eops = {
|
|
.error = perf_tui__error,
|
|
.warning = perf_tui__warning,
|
|
};
|