From 1b8fb1d436ad5144dca32df9be846736c748c388 Mon Sep 17 00:00:00 2001 From: Simon Tatham Date: Sat, 4 Mar 2023 17:04:07 +0000 Subject: [PATCH] terminal: remove the 'screen' parameter from lineptr(). It wasn't used for anything except in an assert statement, which was triggered by the use of the scrlineptr() macro wrapper. Now moved that check into scrlineptr() itself, via a helper function that passes the line number of the scrlineptr() call site. (Yes, this is introducing another modalfatalbox in terminal.c, much like the dreaded line==NULL one that caused us so many headaches in past decades. But the check in question was being done _already_ by the assert in lineptr(), so this change shouldn't make it go off in any _more_ circumstances - and now, if it does, it will at least give us slightly more useful information about where the problem is!) --- terminal/terminal.c | 34 ++++++++++++++++++++++++++++------ 1 file changed, 28 insertions(+), 6 deletions(-) diff --git a/terminal/terminal.c b/terminal/terminal.c index a2e1a9e8..efde48b1 100644 --- a/terminal/terminal.c +++ b/terminal/terminal.c @@ -89,7 +89,7 @@ static void term_userpass_state_free(struct term_userpass_state *s); * Internal prototypes. */ static void resizeline(Terminal *, termline *, int); -static termline *lineptr(Terminal *, int, int, int); +static termline *lineptr(Terminal *, int, int); static void unlineptr(termline *); static void check_line_size(Terminal *, termline *); static void do_paint(Terminal *); @@ -1171,12 +1171,19 @@ static void null_line_error(Terminal *term, int y, int lineno, term->alt_sblines, whichtree, treeindex, commitid); } +static inline int checkscr(int y, int lineno) +{ + if (y < 0) + modalfatalbox("screen line %d < 0 in terminal.c:%d", y, lineno); + return y; +} + /* * Retrieve a line of the screen or of the scrollback, according to * whether the y coordinate is non-negative or negative * (respectively). */ -static termline *lineptr(Terminal *term, int y, int lineno, int screen) +static termline *lineptr(Terminal *term, int y, int lineno) { termline *line; tree234 *whichtree; @@ -1188,8 +1195,6 @@ static termline *lineptr(Terminal *term, int y, int lineno, int screen) } else { int altlines = 0; - assert(!screen); - if (term->erase_to_scrollback && term->alt_which && term->alt_screen) { altlines = term->alt_sblines; @@ -1240,8 +1245,25 @@ static termline *lineptr(Terminal *term, int y, int lineno, int screen) return line; } -#define lineptr(x) (lineptr)(term,x,__LINE__,0) -#define scrlineptr(x) (lineptr)(term,x,__LINE__,1) +/* + * Macro wrappers for lineptr. The distinction between lineptr and + * scrlineptr is that lineptr can retrieve any line, from the screen + * _or_ from scrollback, and in return, you have to call unlineptr + * when you're done with it, in case it was a dynamically allocated + * line decompressed from scrollback that needs freeing. But + * scrlineptr will only retrieve lines from the active screen (and + * enforces this by an assertion), which means it's always just + * returning a pointer to an existing unpacked termline, and you don't + * have to call unlineptr afterwards. So drawing code (which might + * need the scrollback) will have to call lineptr/unlineptr, but + * update code during term_out can call scrlineptr. + * + * The 'assertion' in scrlineptr is done using a helper function that + * returns the input column number, which allows this macro to avoid + * double-evaluating its argument. + */ +#define lineptr(x) (lineptr)(term,x,__LINE__) +#define scrlineptr(x) (lineptr)(term,checkscr(x,__LINE__),__LINE__) /* * Coerce a termline to the terminal's current width. Unlike the