gdb/tui: convert if/error to an assert
[binutils-gdb.git] / gdb / tui / tui-win.c
1 /* TUI window generic functions.
2
3 Copyright (C) 1998-2023 Free Software Foundation, Inc.
4
5 Contributed by Hewlett-Packard Company.
6
7 This file is part of GDB.
8
9 This program is free software; you can redistribute it and/or modify
10 it under the terms of the GNU General Public License as published by
11 the Free Software Foundation; either version 3 of the License, or
12 (at your option) any later version.
13
14 This program is distributed in the hope that it will be useful,
15 but WITHOUT ANY WARRANTY; without even the implied warranty of
16 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 GNU General Public License for more details.
18
19 You should have received a copy of the GNU General Public License
20 along with this program. If not, see <http://www.gnu.org/licenses/>. */
21
22 /* This module contains procedures for handling tui window functions
23 like resize, scrolling, scrolling, changing focus, etc.
24
25 Author: Susan B. Macchia */
26
27 #include "defs.h"
28 #include "command.h"
29 #include "symtab.h"
30 #include "breakpoint.h"
31 #include "frame.h"
32 #include "cli/cli-cmds.h"
33 #include "cli/cli-style.h"
34 #include "top.h"
35 #include "source.h"
36 #include "gdbsupport/event-loop.h"
37 #include "gdbcmd.h"
38 #include "async-event.h"
39
40 #include "tui/tui.h"
41 #include "tui/tui-io.h"
42 #include "tui/tui-command.h"
43 #include "tui/tui-data.h"
44 #include "tui/tui-layout.h"
45 #include "tui/tui-wingeneral.h"
46 #include "tui/tui-stack.h"
47 #include "tui/tui-regs.h"
48 #include "tui/tui-disasm.h"
49 #include "tui/tui-source.h"
50 #include "tui/tui-winsource.h"
51 #include "tui/tui-win.h"
52
53 #include "gdb_curses.h"
54 #include <ctype.h>
55 #include "readline/readline.h"
56 #include "gdbsupport/gdb_string_view.h"
57
58 #include <signal.h>
59
60 static void tui_set_tab_width_command (const char *, int);
61 static void tui_refresh_all_command (const char *, int);
62 static void tui_all_windows_info (const char *, int);
63 static void tui_scroll_forward_command (const char *, int);
64 static void tui_scroll_backward_command (const char *, int);
65 static void tui_scroll_left_command (const char *, int);
66 static void tui_scroll_right_command (const char *, int);
67 static void parse_scrolling_args (const char *,
68 struct tui_win_info **,
69 int *);
70
71
72 #ifndef ACS_LRCORNER
73 # define ACS_LRCORNER '+'
74 #endif
75 #ifndef ACS_LLCORNER
76 # define ACS_LLCORNER '+'
77 #endif
78 #ifndef ACS_ULCORNER
79 # define ACS_ULCORNER '+'
80 #endif
81 #ifndef ACS_URCORNER
82 # define ACS_URCORNER '+'
83 #endif
84 #ifndef ACS_HLINE
85 # define ACS_HLINE '-'
86 #endif
87 #ifndef ACS_VLINE
88 # define ACS_VLINE '|'
89 #endif
90
91 /* Possible values for tui-border-kind variable. */
92 static const char *const tui_border_kind_enums[] = {
93 "space",
94 "ascii",
95 "acs",
96 NULL
97 };
98
99 /* Possible values for tui-border-mode and tui-active-border-mode. */
100 static const char *const tui_border_mode_enums[] = {
101 "normal",
102 "standout",
103 "reverse",
104 "half",
105 "half-standout",
106 "bold",
107 "bold-standout",
108 NULL
109 };
110
111 struct tui_translate
112 {
113 const char *name;
114 int value;
115 };
116
117 /* Translation table for border-mode variables.
118 The list of values must be terminated by a NULL.
119 After the NULL value, an entry defines the default. */
120 static struct tui_translate tui_border_mode_translate[] = {
121 { "normal", A_NORMAL },
122 { "standout", A_STANDOUT },
123 { "reverse", A_REVERSE },
124 { "half", A_DIM },
125 { "half-standout", A_DIM | A_STANDOUT },
126 { "bold", A_BOLD },
127 { "bold-standout", A_BOLD | A_STANDOUT },
128 { 0, 0 },
129 { "normal", A_NORMAL }
130 };
131
132 /* Translation tables for border-kind, one for each border
133 character (see wborder, border curses operations).
134 -1 is used to indicate the ACS because ACS characters
135 are determined at run time by curses (depends on terminal). */
136 static struct tui_translate tui_border_kind_translate_vline[] = {
137 { "space", ' ' },
138 { "ascii", '|' },
139 { "acs", -1 },
140 { 0, 0 },
141 { "ascii", '|' }
142 };
143
144 static struct tui_translate tui_border_kind_translate_hline[] = {
145 { "space", ' ' },
146 { "ascii", '-' },
147 { "acs", -1 },
148 { 0, 0 },
149 { "ascii", '-' }
150 };
151
152 static struct tui_translate tui_border_kind_translate_ulcorner[] = {
153 { "space", ' ' },
154 { "ascii", '+' },
155 { "acs", -1 },
156 { 0, 0 },
157 { "ascii", '+' }
158 };
159
160 static struct tui_translate tui_border_kind_translate_urcorner[] = {
161 { "space", ' ' },
162 { "ascii", '+' },
163 { "acs", -1 },
164 { 0, 0 },
165 { "ascii", '+' }
166 };
167
168 static struct tui_translate tui_border_kind_translate_llcorner[] = {
169 { "space", ' ' },
170 { "ascii", '+' },
171 { "acs", -1 },
172 { 0, 0 },
173 { "ascii", '+' }
174 };
175
176 static struct tui_translate tui_border_kind_translate_lrcorner[] = {
177 { "space", ' ' },
178 { "ascii", '+' },
179 { "acs", -1 },
180 { 0, 0 },
181 { "ascii", '+' }
182 };
183
184
185 /* Tui configuration variables controlled with set/show command. */
186 static const char *tui_active_border_mode = "bold-standout";
187 static void
188 show_tui_active_border_mode (struct ui_file *file,
189 int from_tty,
190 struct cmd_list_element *c,
191 const char *value)
192 {
193 gdb_printf (file, _("\
194 The attribute mode to use for the active TUI window border is \"%s\".\n"),
195 value);
196 }
197
198 static const char *tui_border_mode = "normal";
199 static void
200 show_tui_border_mode (struct ui_file *file,
201 int from_tty,
202 struct cmd_list_element *c,
203 const char *value)
204 {
205 gdb_printf (file, _("\
206 The attribute mode to use for the TUI window borders is \"%s\".\n"),
207 value);
208 }
209
210 static const char *tui_border_kind = "acs";
211 static void
212 show_tui_border_kind (struct ui_file *file,
213 int from_tty,
214 struct cmd_list_element *c,
215 const char *value)
216 {
217 gdb_printf (file, _("The kind of border for TUI windows is \"%s\".\n"),
218 value);
219 }
220
221 /* Implementation of the "set/show style tui-current-position" commands. */
222
223 bool style_tui_current_position = false;
224
225 static void
226 show_style_tui_current_position (ui_file *file,
227 int from_tty,
228 cmd_list_element *c,
229 const char *value)
230 {
231 gdb_printf (file, _("\
232 Styling the text highlighted by the TUI's current position indicator is %s.\n"),
233 value);
234 }
235
236 static void
237 set_style_tui_current_position (const char *ignore, int from_tty,
238 cmd_list_element *c)
239 {
240 if (TUI_SRC_WIN != nullptr)
241 TUI_SRC_WIN->refill ();
242 if (TUI_DISASM_WIN != nullptr)
243 TUI_DISASM_WIN->refill ();
244 }
245
246 /* Tui internal configuration variables. These variables are updated
247 by tui_update_variables to reflect the tui configuration
248 variables. */
249 chtype tui_border_vline;
250 chtype tui_border_hline;
251 chtype tui_border_ulcorner;
252 chtype tui_border_urcorner;
253 chtype tui_border_llcorner;
254 chtype tui_border_lrcorner;
255
256 int tui_border_attrs;
257 int tui_active_border_attrs;
258
259 /* Identify the item in the translation table.
260 When the item is not recognized, use the default entry. */
261 static struct tui_translate *
262 translate (const char *name, struct tui_translate *table)
263 {
264 while (table->name)
265 {
266 if (name && strcmp (table->name, name) == 0)
267 return table;
268 table++;
269 }
270
271 /* Not found, return default entry. */
272 table++;
273 return table;
274 }
275
276 /* Update the tui internal configuration according to gdb settings.
277 Returns 1 if the configuration has changed and the screen should
278 be redrawn. */
279 bool
280 tui_update_variables ()
281 {
282 bool need_redraw = false;
283 struct tui_translate *entry;
284
285 entry = translate (tui_border_mode, tui_border_mode_translate);
286 if (tui_border_attrs != entry->value)
287 {
288 tui_border_attrs = entry->value;
289 need_redraw = true;
290 }
291 entry = translate (tui_active_border_mode, tui_border_mode_translate);
292 if (tui_active_border_attrs != entry->value)
293 {
294 tui_active_border_attrs = entry->value;
295 need_redraw = true;
296 }
297
298 /* If one corner changes, all characters are changed.
299 Only check the first one. The ACS characters are determined at
300 run time by curses terminal management. */
301 entry = translate (tui_border_kind, tui_border_kind_translate_lrcorner);
302 if (tui_border_lrcorner != (chtype) entry->value)
303 {
304 tui_border_lrcorner = (entry->value < 0) ? ACS_LRCORNER : entry->value;
305 need_redraw = true;
306 }
307 entry = translate (tui_border_kind, tui_border_kind_translate_llcorner);
308 tui_border_llcorner = (entry->value < 0) ? ACS_LLCORNER : entry->value;
309
310 entry = translate (tui_border_kind, tui_border_kind_translate_ulcorner);
311 tui_border_ulcorner = (entry->value < 0) ? ACS_ULCORNER : entry->value;
312
313 entry = translate (tui_border_kind, tui_border_kind_translate_urcorner);
314 tui_border_urcorner = (entry->value < 0) ? ACS_URCORNER : entry->value;
315
316 entry = translate (tui_border_kind, tui_border_kind_translate_hline);
317 tui_border_hline = (entry->value < 0) ? ACS_HLINE : entry->value;
318
319 entry = translate (tui_border_kind, tui_border_kind_translate_vline);
320 tui_border_vline = (entry->value < 0) ? ACS_VLINE : entry->value;
321
322 return need_redraw;
323 }
324
325 static struct cmd_list_element *tuilist;
326
327 struct cmd_list_element **
328 tui_get_cmd_list (void)
329 {
330 if (tuilist == 0)
331 add_basic_prefix_cmd ("tui", class_tui,
332 _("Text User Interface commands."),
333 &tuilist, 0, &cmdlist);
334 return &tuilist;
335 }
336
337 /* The set_func hook of "set tui ..." commands that affect the window
338 borders on the TUI display. */
339
340 static void
341 tui_set_var_cmd (const char *null_args,
342 int from_tty, struct cmd_list_element *c)
343 {
344 if (tui_update_variables () && tui_active)
345 tui_rehighlight_all ();
346 }
347
348 \f
349
350 /* True if TUI resizes should print a message. This is used by the
351 test suite. */
352
353 static bool resize_message;
354
355 static void
356 show_tui_resize_message (struct ui_file *file, int from_tty,
357 struct cmd_list_element *c, const char *value)
358 {
359 gdb_printf (file, _("TUI resize messaging is %s.\n"), value);
360 }
361
362 \f
363
364 /* Generic window name completion function. Complete window name pointed
365 to by TEXT and WORD. If INCLUDE_NEXT_PREV_P is true then the special
366 window names 'next' and 'prev' will also be considered as possible
367 completions of the window name. */
368
369 static void
370 window_name_completer (completion_tracker &tracker,
371 int include_next_prev_p,
372 const char *text, const char *word)
373 {
374 std::vector<const char *> completion_name_vec;
375
376 for (tui_win_info *win_info : all_tui_windows ())
377 {
378 const char *completion_name = NULL;
379
380 /* We can't focus on an invisible window. */
381 if (!win_info->is_visible ())
382 continue;
383
384 completion_name = win_info->name ();
385 gdb_assert (completion_name != NULL);
386 completion_name_vec.push_back (completion_name);
387 }
388
389 /* If no windows are considered visible then the TUI has not yet been
390 initialized. But still "focus src" and "focus cmd" will work because
391 invoking the focus command will entail initializing the TUI which sets the
392 default layout to "src". */
393 if (completion_name_vec.empty ())
394 {
395 completion_name_vec.push_back (SRC_NAME);
396 completion_name_vec.push_back (CMD_NAME);
397 }
398
399 if (include_next_prev_p)
400 {
401 completion_name_vec.push_back ("next");
402 completion_name_vec.push_back ("prev");
403 }
404
405
406 completion_name_vec.push_back (NULL);
407 complete_on_enum (tracker, completion_name_vec.data (), text, word);
408 }
409
410 /* Complete possible window names to focus on. TEXT is the complete text
411 entered so far, WORD is the word currently being completed. */
412
413 static void
414 focus_completer (struct cmd_list_element *ignore,
415 completion_tracker &tracker,
416 const char *text, const char *word)
417 {
418 window_name_completer (tracker, 1, text, word);
419 }
420
421 /* Complete possible window names for winheight command. TEXT is the
422 complete text entered so far, WORD is the word currently being
423 completed. */
424
425 static void
426 winheight_completer (struct cmd_list_element *ignore,
427 completion_tracker &tracker,
428 const char *text, const char *word)
429 {
430 /* The first word is the window name. That we can complete. Subsequent
431 words can't be completed. */
432 if (word != text)
433 return;
434
435 window_name_completer (tracker, 0, text, word);
436 }
437
438 /* Update gdb's knowledge of the terminal size. */
439 void
440 tui_update_gdb_sizes (void)
441 {
442 int width, height;
443
444 if (tui_active)
445 {
446 width = TUI_CMD_WIN->width;
447 height = TUI_CMD_WIN->height;
448 }
449 else
450 {
451 width = tui_term_width ();
452 height = tui_term_height ();
453 }
454
455 set_screen_width_and_height (width, height);
456 }
457
458
459 void
460 tui_win_info::forward_scroll (int num_to_scroll)
461 {
462 if (num_to_scroll == 0)
463 num_to_scroll = height - 3;
464
465 do_scroll_vertical (num_to_scroll);
466 }
467
468 void
469 tui_win_info::backward_scroll (int num_to_scroll)
470 {
471 if (num_to_scroll == 0)
472 num_to_scroll = height - 3;
473
474 do_scroll_vertical (-num_to_scroll);
475 }
476
477
478 void
479 tui_win_info::left_scroll (int num_to_scroll)
480 {
481 if (num_to_scroll == 0)
482 num_to_scroll = 1;
483
484 do_scroll_horizontal (num_to_scroll);
485 }
486
487
488 void
489 tui_win_info::right_scroll (int num_to_scroll)
490 {
491 if (num_to_scroll == 0)
492 num_to_scroll = 1;
493
494 do_scroll_horizontal (-num_to_scroll);
495 }
496
497
498 void
499 tui_refresh_all_win (void)
500 {
501 clearok (curscr, TRUE);
502 tui_refresh_all ();
503 }
504
505 void
506 tui_rehighlight_all (void)
507 {
508 for (tui_win_info *win_info : all_tui_windows ())
509 win_info->check_and_display_highlight_if_needed ();
510 }
511
512 /* Resize all the windows based on the terminal size. This function
513 gets called from within the readline SIGWINCH handler. */
514 void
515 tui_resize_all (void)
516 {
517 int height_diff, width_diff;
518 int screenheight, screenwidth;
519
520 rl_get_screen_size (&screenheight, &screenwidth);
521 width_diff = screenwidth - tui_term_width ();
522 height_diff = screenheight - tui_term_height ();
523 if (height_diff || width_diff)
524 {
525 #ifdef HAVE_RESIZE_TERM
526 resize_term (screenheight, screenwidth);
527 #endif
528 /* Turn keypad off while we resize. */
529 keypad (TUI_CMD_WIN->handle.get (), FALSE);
530 tui_update_gdb_sizes ();
531 tui_set_term_height_to (screenheight);
532 tui_set_term_width_to (screenwidth);
533
534 /* erase + clearok are used instead of a straightforward clear as
535 AIX 5.3 does not define clear. */
536 erase ();
537 clearok (curscr, TRUE);
538 /* Apply the current layout. The 'false' here allows the command
539 window to resize proportionately with containing terminal, rather
540 than maintaining a fixed size. */
541 tui_apply_current_layout (false); /* Turn keypad back on. */
542 keypad (TUI_CMD_WIN->handle.get (), TRUE);
543 }
544 }
545
546 #ifdef SIGWINCH
547 /* Token for use by TUI's asynchronous SIGWINCH handler. */
548 static struct async_signal_handler *tui_sigwinch_token;
549
550 /* TUI's SIGWINCH signal handler. */
551 static void
552 tui_sigwinch_handler (int signal)
553 {
554 mark_async_signal_handler (tui_sigwinch_token);
555 tui_set_win_resized_to (true);
556 }
557
558 /* Callback for asynchronously resizing TUI following a SIGWINCH signal. */
559 static void
560 tui_async_resize_screen (gdb_client_data arg)
561 {
562 rl_resize_terminal ();
563
564 if (!tui_active)
565 {
566 int screen_height, screen_width;
567
568 rl_get_screen_size (&screen_height, &screen_width);
569 set_screen_width_and_height (screen_width, screen_height);
570
571 /* win_resized is left set so that the next call to tui_enable()
572 resizes the TUI windows. */
573 }
574 else
575 {
576 tui_set_win_resized_to (false);
577 tui_resize_all ();
578 tui_refresh_all_win ();
579 tui_update_gdb_sizes ();
580 if (resize_message)
581 {
582 static int count;
583 printf_unfiltered ("@@ resize done %d, size = %dx%d\n", count,
584 tui_term_width (), tui_term_height ());
585 ++count;
586 }
587 tui_redisplay_readline ();
588 }
589 }
590 #endif
591
592 /* Initialize TUI's SIGWINCH signal handler. Note that the handler is not
593 uninstalled when we exit TUI, so the handler should not assume that TUI is
594 always active. */
595 void
596 tui_initialize_win (void)
597 {
598 #ifdef SIGWINCH
599 tui_sigwinch_token
600 = create_async_signal_handler (tui_async_resize_screen, NULL,
601 "tui-sigwinch");
602
603 {
604 #ifdef HAVE_SIGACTION
605 struct sigaction old_winch;
606
607 memset (&old_winch, 0, sizeof (old_winch));
608 old_winch.sa_handler = &tui_sigwinch_handler;
609 #ifdef SA_RESTART
610 old_winch.sa_flags = SA_RESTART;
611 #endif
612 sigaction (SIGWINCH, &old_winch, NULL);
613 #else
614 signal (SIGWINCH, &tui_sigwinch_handler);
615 #endif
616 }
617 #endif
618 }
619
620
621 static void
622 tui_scroll_forward_command (const char *arg, int from_tty)
623 {
624 int num_to_scroll = 1;
625 struct tui_win_info *win_to_scroll;
626
627 /* Make sure the curses mode is enabled. */
628 tui_enable ();
629 if (arg == NULL)
630 parse_scrolling_args (arg, &win_to_scroll, NULL);
631 else
632 parse_scrolling_args (arg, &win_to_scroll, &num_to_scroll);
633 win_to_scroll->forward_scroll (num_to_scroll);
634 }
635
636
637 static void
638 tui_scroll_backward_command (const char *arg, int from_tty)
639 {
640 int num_to_scroll = 1;
641 struct tui_win_info *win_to_scroll;
642
643 /* Make sure the curses mode is enabled. */
644 tui_enable ();
645 if (arg == NULL)
646 parse_scrolling_args (arg, &win_to_scroll, NULL);
647 else
648 parse_scrolling_args (arg, &win_to_scroll, &num_to_scroll);
649 win_to_scroll->backward_scroll (num_to_scroll);
650 }
651
652
653 static void
654 tui_scroll_left_command (const char *arg, int from_tty)
655 {
656 int num_to_scroll;
657 struct tui_win_info *win_to_scroll;
658
659 /* Make sure the curses mode is enabled. */
660 tui_enable ();
661 parse_scrolling_args (arg, &win_to_scroll, &num_to_scroll);
662 win_to_scroll->left_scroll (num_to_scroll);
663 }
664
665
666 static void
667 tui_scroll_right_command (const char *arg, int from_tty)
668 {
669 int num_to_scroll;
670 struct tui_win_info *win_to_scroll;
671
672 /* Make sure the curses mode is enabled. */
673 tui_enable ();
674 parse_scrolling_args (arg, &win_to_scroll, &num_to_scroll);
675 win_to_scroll->right_scroll (num_to_scroll);
676 }
677
678
679 /* Answer the window represented by name. */
680 static struct tui_win_info *
681 tui_partial_win_by_name (gdb::string_view name)
682 {
683 struct tui_win_info *best = nullptr;
684
685 for (tui_win_info *item : all_tui_windows ())
686 {
687 const char *cur_name = item->name ();
688
689 if (name == cur_name)
690 return item;
691 if (startswith (cur_name, name))
692 {
693 if (best != nullptr)
694 error (_("Window name \"%*s\" is ambiguous"),
695 (int) name.size (), name.data ());
696 best = item;
697 }
698 }
699
700 return best;
701 }
702
703 /* Set focus to the window named by 'arg'. */
704 static void
705 tui_set_focus_command (const char *arg, int from_tty)
706 {
707 tui_enable ();
708
709 if (arg == NULL)
710 error_no_arg (_("name of window to focus"));
711
712 struct tui_win_info *win_info = NULL;
713
714 if (startswith ("next", arg))
715 win_info = tui_next_win (tui_win_with_focus ());
716 else if (startswith ("prev", arg))
717 win_info = tui_prev_win (tui_win_with_focus ());
718 else
719 win_info = tui_partial_win_by_name (arg);
720
721 if (win_info == NULL)
722 error (_("Unrecognized window name \"%s\""), arg);
723
724 /* If a window is part of the current layout then it will have a
725 tui_win_info associated with it and be visible, otherwise, there will
726 be no tui_win_info and the above error will have been raised. */
727 gdb_assert (win_info->is_visible ());
728
729 if (!win_info->can_focus ())
730 error (_("Window \"%s\" cannot be focused"), arg);
731
732 tui_set_win_focus_to (win_info);
733 gdb_printf (_("Focus set to %s window.\n"),
734 tui_win_with_focus ()->name ());
735 }
736
737 static void
738 tui_all_windows_info (const char *arg, int from_tty)
739 {
740 if (!tui_active)
741 {
742 gdb_printf (_("The TUI is not active.\n"));
743 return;
744 }
745
746 struct tui_win_info *win_with_focus = tui_win_with_focus ();
747 struct ui_out *uiout = current_uiout;
748
749 ui_out_emit_table table_emitter (uiout, 4, -1, "tui-windows");
750 uiout->table_header (10, ui_left, "name", "Name");
751 uiout->table_header (5, ui_right, "lines", "Lines");
752 uiout->table_header (7, ui_right, "columns", "Columns");
753 uiout->table_header (10, ui_left, "focus", "Focus");
754 uiout->table_body ();
755
756 for (tui_win_info *win_info : all_tui_windows ())
757 if (win_info->is_visible ())
758 {
759 ui_out_emit_tuple tuple_emitter (uiout, nullptr);
760
761 uiout->field_string ("name", win_info->name ());
762 uiout->field_signed ("lines", win_info->height);
763 uiout->field_signed ("columns", win_info->width);
764 if (win_with_focus == win_info)
765 uiout->field_string ("focus", _("(has focus)"));
766 else
767 uiout->field_skip ("focus");
768 uiout->text ("\n");
769 }
770 }
771
772
773 static void
774 tui_refresh_all_command (const char *arg, int from_tty)
775 {
776 /* Make sure the curses mode is enabled. */
777 tui_enable ();
778
779 tui_refresh_all_win ();
780 }
781
782 #define DEFAULT_TAB_LEN 8
783
784 /* The tab width that should be used by the TUI. */
785
786 unsigned int tui_tab_width = DEFAULT_TAB_LEN;
787
788 /* The tab width as set by the user. */
789
790 static unsigned int internal_tab_width = DEFAULT_TAB_LEN;
791
792 /* After the tab width is set, call this to update the relevant
793 windows. */
794
795 static void
796 update_tab_width ()
797 {
798 for (tui_win_info *win_info : all_tui_windows ())
799 {
800 if (win_info->is_visible ())
801 win_info->update_tab_width ();
802 }
803 }
804
805 /* Callback for "set tui tab-width". */
806
807 static void
808 tui_set_tab_width (const char *ignore,
809 int from_tty, struct cmd_list_element *c)
810 {
811 if (internal_tab_width == 0)
812 {
813 internal_tab_width = tui_tab_width;
814 error (_("Tab width must not be 0"));
815 }
816
817 tui_tab_width = internal_tab_width;
818 update_tab_width ();
819 }
820
821 /* Callback for "show tui tab-width". */
822
823 static void
824 tui_show_tab_width (struct ui_file *file, int from_tty,
825 struct cmd_list_element *c, const char *value)
826 {
827 gdb_printf (file, _("TUI tab width is %s spaces.\n"), value);
828
829 }
830
831 /* See tui-win.h. */
832
833 bool compact_source = false;
834
835 /* Callback for "set tui compact-source". */
836
837 static void
838 tui_set_compact_source (const char *ignore, int from_tty,
839 struct cmd_list_element *c)
840 {
841 if (TUI_SRC_WIN != nullptr)
842 TUI_SRC_WIN->refill ();
843 }
844
845 /* Callback for "show tui compact-source". */
846
847 static void
848 tui_show_compact_source (struct ui_file *file, int from_tty,
849 struct cmd_list_element *c, const char *value)
850 {
851 gdb_printf (file, _("TUI source window compactness is %s.\n"), value);
852 }
853
854 /* Set the tab width of the specified window. */
855 static void
856 tui_set_tab_width_command (const char *arg, int from_tty)
857 {
858 /* Make sure the curses mode is enabled. */
859 tui_enable ();
860 if (arg != NULL)
861 {
862 int ts;
863
864 ts = atoi (arg);
865 if (ts <= 0)
866 warning (_("Tab widths greater than 0 must be specified."));
867 else
868 {
869 internal_tab_width = ts;
870 tui_tab_width = ts;
871
872 update_tab_width ();
873 }
874 }
875 }
876
877 /* Helper function for the user commands to adjust a window's width or
878 height. The ARG string contains the command line arguments from the
879 user, which should give the name of a window, and how to adjust the
880 size.
881
882 When SET_WIDTH_P is true the width of the window is adjusted based on
883 ARG, and when SET_WIDTH_P is false, the height of the window is adjusted
884 based on ARG.
885
886 On invalid input, or if the size can't be adjusted as requested, then an
887 error is thrown, otherwise, the window sizes are adjusted, and the
888 windows redrawn. */
889
890 static void
891 tui_set_win_size (const char *arg, bool set_width_p)
892 {
893 /* Make sure the curses mode is enabled. */
894 tui_enable ();
895 if (arg == NULL)
896 error_no_arg (_("name of window"));
897
898 const char *buf = arg;
899 const char *buf_ptr = buf;
900 int new_size;
901 struct tui_win_info *win_info;
902
903 buf_ptr = skip_to_space (buf_ptr);
904
905 /* Validate the window name. */
906 gdb::string_view wname (buf, buf_ptr - buf);
907 win_info = tui_partial_win_by_name (wname);
908
909 if (win_info == NULL)
910 error (_("Unrecognized window name \"%s\""), arg);
911 if (!win_info->is_visible ())
912 error (_("Window \"%s\" is not visible"), arg);
913
914 /* Process the size. */
915 buf_ptr = skip_spaces (buf_ptr);
916
917 if (*buf_ptr != '\0')
918 {
919 bool negate = false;
920 bool fixed_size = true;
921 int input_no;;
922
923 if (*buf_ptr == '+' || *buf_ptr == '-')
924 {
925 if (*buf_ptr == '-')
926 negate = true;
927 fixed_size = false;
928 buf_ptr++;
929 }
930 input_no = atoi (buf_ptr);
931 if (input_no > 0)
932 {
933 if (negate)
934 input_no *= (-1);
935 if (fixed_size)
936 new_size = input_no;
937 else
938 {
939 int curr_size;
940 if (set_width_p)
941 curr_size = win_info->width;
942 else
943 curr_size = win_info->height;
944 new_size = curr_size + input_no;
945 }
946
947 /* Now change the window's height, and adjust
948 all other windows around it. */
949 if (set_width_p)
950 tui_adjust_window_width (win_info, new_size);
951 else
952 tui_adjust_window_height (win_info, new_size);
953 tui_update_gdb_sizes ();
954 }
955 else
956 {
957 if (set_width_p)
958 error (_("Invalid window width specified"));
959 else
960 error (_("Invalid window height specified"));
961 }
962 }
963 }
964
965 /* Implement the 'tui window height' command (alias 'winheight'). */
966
967 static void
968 tui_set_win_height_command (const char *arg, int from_tty)
969 {
970 /* Pass false as the final argument to set the height. */
971 tui_set_win_size (arg, false);
972 }
973
974 /* Implement the 'tui window width' command (alias 'winwidth'). */
975
976 static void
977 tui_set_win_width_command (const char *arg, int from_tty)
978 {
979 /* Pass true as the final argument to set the width. */
980 tui_set_win_size (arg, true);
981 }
982
983 /* See tui-data.h. */
984
985 int
986 tui_win_info::max_height () const
987 {
988 return tui_term_height ();
989 }
990
991 /* See tui-data.h. */
992
993 int
994 tui_win_info::max_width () const
995 {
996 return tui_term_width ();
997 }
998
999 static void
1000 parse_scrolling_args (const char *arg,
1001 struct tui_win_info **win_to_scroll,
1002 int *num_to_scroll)
1003 {
1004 if (num_to_scroll)
1005 *num_to_scroll = 0;
1006 *win_to_scroll = tui_win_with_focus ();
1007
1008 /* First set up the default window to scroll, in case there is no
1009 window name arg. */
1010 if (arg != NULL)
1011 {
1012 char *buf_ptr;
1013
1014 /* Process the number of lines to scroll. */
1015 std::string copy = arg;
1016 buf_ptr = &copy[0];
1017 if (isdigit (*buf_ptr))
1018 {
1019 char *num_str;
1020
1021 num_str = buf_ptr;
1022 buf_ptr = strchr (buf_ptr, ' ');
1023 if (buf_ptr != NULL)
1024 {
1025 *buf_ptr = '\0';
1026 if (num_to_scroll)
1027 *num_to_scroll = atoi (num_str);
1028 buf_ptr++;
1029 }
1030 else if (num_to_scroll)
1031 *num_to_scroll = atoi (num_str);
1032 }
1033
1034 /* Process the window name if one is specified. */
1035 if (buf_ptr != NULL)
1036 {
1037 const char *wname;
1038
1039 wname = skip_spaces (buf_ptr);
1040
1041 if (*wname != '\0')
1042 {
1043 *win_to_scroll = tui_partial_win_by_name (wname);
1044
1045 if (*win_to_scroll == NULL)
1046 error (_("Unrecognized window `%s'"), wname);
1047 if (!(*win_to_scroll)->is_visible ())
1048 error (_("Window is not visible"));
1049 else if (*win_to_scroll == TUI_CMD_WIN)
1050 *win_to_scroll = *(tui_source_windows ().begin ());
1051 }
1052 }
1053 }
1054 }
1055
1056 /* The list of 'tui window' sub-commands. */
1057
1058 static cmd_list_element *tui_window_cmds = nullptr;
1059
1060 /* Called to implement 'tui window'. */
1061
1062 static void
1063 tui_window_command (const char *args, int from_tty)
1064 {
1065 help_list (tui_window_cmds, "tui window ", all_commands, gdb_stdout);
1066 }
1067
1068 /* Function to initialize gdb commands, for tui window
1069 manipulation. */
1070
1071 void _initialize_tui_win ();
1072 void
1073 _initialize_tui_win ()
1074 {
1075 static struct cmd_list_element *tui_setlist;
1076 static struct cmd_list_element *tui_showlist;
1077
1078 /* Define the classes of commands.
1079 They will appear in the help list in the reverse of this order. */
1080 add_setshow_prefix_cmd ("tui", class_tui,
1081 _("TUI configuration variables."),
1082 _("TUI configuration variables."),
1083 &tui_setlist, &tui_showlist,
1084 &setlist, &showlist);
1085
1086 cmd_list_element *refresh_cmd
1087 = add_cmd ("refresh", class_tui, tui_refresh_all_command,
1088 _("Refresh the terminal display."),
1089 tui_get_cmd_list ());
1090 add_com_alias ("refresh", refresh_cmd, class_tui, 0);
1091
1092 cmd_list_element *tabset_cmd
1093 = add_com ("tabset", class_tui, tui_set_tab_width_command, _("\
1094 Set the width (in characters) of tab stops.\n\
1095 Usage: tabset N"));
1096 deprecate_cmd (tabset_cmd, "set tui tab-width");
1097
1098 /* Setup the 'tui window' list of command. */
1099 add_prefix_cmd ("window", class_tui, tui_window_command,
1100 _("Text User Interface window commands."),
1101 &tui_window_cmds, 1, tui_get_cmd_list ());
1102
1103 cmd_list_element *winheight_cmd
1104 = add_cmd ("height", class_tui, tui_set_win_height_command, _("\
1105 Set or modify the height of a specified window.\n\
1106 Usage: tui window height WINDOW-NAME [+ | -] NUM-LINES\n\
1107 Use \"info win\" to see the names of the windows currently being displayed."),
1108 &tui_window_cmds);
1109 add_com_alias ("winheight", winheight_cmd, class_tui, 0);
1110 add_com_alias ("wh", winheight_cmd, class_tui, 0);
1111 set_cmd_completer (winheight_cmd, winheight_completer);
1112
1113 cmd_list_element *winwidth_cmd
1114 = add_cmd ("width", class_tui, tui_set_win_width_command, _("\
1115 Set or modify the width of a specified window.\n\
1116 Usage: tui window width WINDOW-NAME [+ | -] NUM-LINES\n\
1117 Use \"info win\" to see the names of the windows currently being displayed."),
1118 &tui_window_cmds);
1119 add_com_alias ("winwidth", winwidth_cmd, class_tui, 0);
1120 set_cmd_completer (winwidth_cmd, winheight_completer);
1121
1122 add_info ("win", tui_all_windows_info,
1123 _("List of all displayed windows.\n\
1124 Usage: info win"));
1125 cmd_list_element *focus_cmd
1126 = add_cmd ("focus", class_tui, tui_set_focus_command, _("\
1127 Set focus to named window or next/prev window.\n\
1128 Usage: tui focus [WINDOW-NAME | next | prev]\n\
1129 Use \"info win\" to see the names of the windows currently being displayed."),
1130 tui_get_cmd_list ());
1131 add_com_alias ("focus", focus_cmd, class_tui, 0);
1132 add_com_alias ("fs", focus_cmd, class_tui, 0);
1133 set_cmd_completer (focus_cmd, focus_completer);
1134 add_com ("+", class_tui, tui_scroll_forward_command, _("\
1135 Scroll window forward.\n\
1136 Usage: + [N] [WIN]\n\
1137 Scroll window WIN N lines forwards. Both WIN and N are optional, N\n\
1138 defaults to 1, and WIN defaults to the currently focused window."));
1139 add_com ("-", class_tui, tui_scroll_backward_command, _("\
1140 Scroll window backward.\n\
1141 Usage: - [N] [WIN]\n\
1142 Scroll window WIN N lines backwards. Both WIN and N are optional, N\n\
1143 defaults to 1, and WIN defaults to the currently focused window."));
1144 add_com ("<", class_tui, tui_scroll_left_command, _("\
1145 Scroll window text to the left.\n\
1146 Usage: < [N] [WIN]\n\
1147 Scroll window WIN N characters left. Both WIN and N are optional, N\n\
1148 defaults to 1, and WIN defaults to the currently focused window."));
1149 add_com (">", class_tui, tui_scroll_right_command, _("\
1150 Scroll window text to the right.\n\
1151 Usage: > [N] [WIN]\n\
1152 Scroll window WIN N characters right. Both WIN and N are optional, N\n\
1153 defaults to 1, and WIN defaults to the currently focused window."));
1154
1155 /* Define the tui control variables. */
1156 add_setshow_enum_cmd ("border-kind", no_class, tui_border_kind_enums,
1157 &tui_border_kind, _("\
1158 Set the kind of border for TUI windows."), _("\
1159 Show the kind of border for TUI windows."), _("\
1160 This variable controls the border of TUI windows:\n\
1161 space use a white space\n\
1162 ascii use ascii characters + - | for the border\n\
1163 acs use the Alternate Character Set"),
1164 tui_set_var_cmd,
1165 show_tui_border_kind,
1166 &tui_setlist, &tui_showlist);
1167
1168 add_setshow_enum_cmd ("border-mode", no_class, tui_border_mode_enums,
1169 &tui_border_mode, _("\
1170 Set the attribute mode to use for the TUI window borders."), _("\
1171 Show the attribute mode to use for the TUI window borders."), _("\
1172 This variable controls the attributes to use for the window borders:\n\
1173 normal normal display\n\
1174 standout use highlight mode of terminal\n\
1175 reverse use reverse video mode\n\
1176 half use half bright\n\
1177 half-standout use half bright and standout mode\n\
1178 bold use extra bright or bold\n\
1179 bold-standout use extra bright or bold with standout mode"),
1180 tui_set_var_cmd,
1181 show_tui_border_mode,
1182 &tui_setlist, &tui_showlist);
1183
1184 add_setshow_enum_cmd ("active-border-mode", no_class, tui_border_mode_enums,
1185 &tui_active_border_mode, _("\
1186 Set the attribute mode to use for the active TUI window border."), _("\
1187 Show the attribute mode to use for the active TUI window border."), _("\
1188 This variable controls the attributes to use for the active window border:\n\
1189 normal normal display\n\
1190 standout use highlight mode of terminal\n\
1191 reverse use reverse video mode\n\
1192 half use half bright\n\
1193 half-standout use half bright and standout mode\n\
1194 bold use extra bright or bold\n\
1195 bold-standout use extra bright or bold with standout mode"),
1196 tui_set_var_cmd,
1197 show_tui_active_border_mode,
1198 &tui_setlist, &tui_showlist);
1199
1200 add_setshow_zuinteger_cmd ("tab-width", no_class,
1201 &internal_tab_width, _("\
1202 Set the tab width, in characters, for the TUI."), _("\
1203 Show the tab witdh, in characters, for the TUI."), _("\
1204 This variable controls how many spaces are used to display a tab character."),
1205 tui_set_tab_width, tui_show_tab_width,
1206 &tui_setlist, &tui_showlist);
1207
1208 add_setshow_boolean_cmd ("tui-resize-message", class_maintenance,
1209 &resize_message, _("\
1210 Set TUI resize messaging."), _("\
1211 Show TUI resize messaging."), _("\
1212 When enabled GDB will print a message when the terminal is resized."),
1213 nullptr,
1214 show_tui_resize_message,
1215 &maintenance_set_cmdlist,
1216 &maintenance_show_cmdlist);
1217
1218 add_setshow_boolean_cmd ("compact-source", class_tui,
1219 &compact_source, _("\
1220 Set whether the TUI source window is compact."), _("\
1221 Show whether the TUI source window is compact."), _("\
1222 This variable controls whether the TUI source window is shown\n\
1223 in a compact form. The compact form puts the source closer to\n\
1224 the line numbers and uses less horizontal space."),
1225 tui_set_compact_source, tui_show_compact_source,
1226 &tui_setlist, &tui_showlist);
1227
1228 add_setshow_boolean_cmd ("tui-current-position", class_maintenance,
1229 &style_tui_current_position, _("\
1230 Set whether to style text highlighted by the TUI's current position indicator."),
1231 _("\
1232 Show whether to style text highlighted by the TUI's current position indicator."),
1233 _("\
1234 When enabled, the source and assembly code highlighted by the TUI's current\n\
1235 position indicator is styled."),
1236 set_style_tui_current_position,
1237 show_style_tui_current_position,
1238 &style_set_list,
1239 &style_show_list);
1240
1241 tui_border_style.changed.attach (tui_rehighlight_all, "tui-win");
1242 tui_active_border_style.changed.attach (tui_rehighlight_all, "tui-win");
1243 }