1
0
mirror of https://git.tartarus.org/simon/putty.git synced 2025-01-10 01:48:00 +00:00

Build option to disable scrollback compression.

This was requested by a downstream of the code, who wanted to change
the time/space tradeoff in the terminal. I currently have no plans to
change this setting for upstream PuTTY, although there is a cmake
option for it just to make testing it easy.

To avoid sprinkling ifdefs over the whole terminal code, the strategy
is to keep the separate type 'compressed_scrollback_line', and turn it
into a typedef for a 'termline *'. So compressline() becomes almost
trivial, and decompressline() even more so.

Memory management is the fiddly part. To make this work sensibly on
both sides, I've broken up each of compressline() and decompressline()
into two versions, one of which takes ownership of (and logically
speaking frees) its input, and the other doesn't. So at call sites
where a function was followed by a free, it's now calling the
'and_free' version of the function, and where the input object was
reused afterwards, it's calling the 'no_free' version. This means that
in different branches of the #if, I can make one function call the
other or vice versa, and no call site is stuck with having to do
things in a more roundabout way than necessary.

The freeing of the _return_ value from decompressline() is handled for
us, because termlines already have a 'temporary' flag which is set
when they're returned from the decompressor, and anyone receiving a
termline from lineptr() calls unlineptr() when they're finished with
it, which will _conditionally_ free it, depending on that 'temporary'
flag. So in the new mode, 'temporary' is never set at all, and all
those unlineptr() calls do nothing.

However, we also still need to free compressed lines properly when
they're actually being thrown away (scrolled off the top of the
scrollback, or cleaned up in term_free), and for that, I've made a new
special-purpose free_compressed_line() function.
This commit is contained in:
Simon Tatham 2022-11-20 10:55:33 +00:00
parent fec6719a2b
commit 5f2eff2fea
3 changed files with 93 additions and 14 deletions

View File

@ -1,6 +1,7 @@
#cmakedefine NO_IPV6
#cmakedefine NO_GSSAPI
#cmakedefine STATIC_GSSAPI
#cmakedefine NO_SCROLLBACK_COMPRESSION
#cmakedefine NO_MULTIMON

View File

@ -15,6 +15,12 @@ set(PUTTY_FUZZING OFF
CACHE BOOL "Build PuTTY binaries suitable for fuzzing, NOT FOR REAL USE")
set(PUTTY_COVERAGE OFF
CACHE BOOL "Build PuTTY binaries suitable for code coverage analysis")
set(PUTTY_COMPRESS_SCROLLBACK ON
# This is always on in production versions of PuTTY, but downstreams
# of the code have been known to find it a better tradeoff to
# disable it. So there's a #ifdef in terminal.c, and a cmake option
# to enable that ifdef just in case it needs testing or debugging.
CACHE BOOL "Store terminal scrollback in compressed form")
set(STRICT OFF
CACHE BOOL "Enable extra compiler warnings and make them errors")
@ -108,6 +114,9 @@ endif()
if(PUTTY_FUZZING)
add_compile_definitions(FUZZING)
endif()
if(NOT PUTTY_COMPRESS_SCROLLBACK)
set(NO_SCROLLBACK_COMPRESSION ON)
endif()
if(PUTTY_COVERAGE)
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -fprofile-arcs -ftest-coverage -g ")
endif()

View File

@ -398,6 +398,8 @@ static void move_termchar(termline *line, termchar *dest, termchar *src)
#endif
}
#ifndef NO_SCROLLBACK_COMPRESSION
/*
* Compress and decompress a termline into an RLE-based format for
* storing in scrollback. (Since scrollback almost never needs to
@ -688,11 +690,12 @@ static void makeliteral_cc(strbuf *b, termchar *c, unsigned long *state)
typedef struct compressed_scrollback_line {
size_t len;
/* compressed data follows after this */
} compressed_scrollback_line;
static termline *decompressline(compressed_scrollback_line *line);
static termline *decompressline_no_free(compressed_scrollback_line *line);
static compressed_scrollback_line *compressline(termline *ldata)
static compressed_scrollback_line *compressline_no_free(termline *ldata)
{
strbuf *b = strbuf_new();
@ -768,7 +771,7 @@ static compressed_scrollback_line *compressline(termline *ldata)
printf("\n");
#endif
dcl = decompressline(line);
dcl = decompressline_no_free(line);
assert(ldata->cols == dcl->cols);
assert(ldata->lattr == dcl->lattr);
for (i = 0; i < ldata->cols; i++)
@ -788,6 +791,13 @@ static compressed_scrollback_line *compressline(termline *ldata)
return line;
}
static compressed_scrollback_line *compressline_and_free(termline *ldata)
{
compressed_scrollback_line *cline = compressline_no_free(ldata);
freetermline(ldata);
return cline;
}
static void readrle(BinarySource *bs, termline *ldata,
void (*readliteral)(BinarySource *bs, termchar *c,
termline *ldata, unsigned long *state))
@ -921,7 +931,7 @@ static void readliteral_cc(BinarySource *bs, termchar *c, termline *ldata,
}
}
static termline *decompressline(compressed_scrollback_line *line)
static termline *decompressline_no_free(compressed_scrollback_line *line)
{
int ncols, byte, shift;
BinarySource bs[1];
@ -988,6 +998,66 @@ static termline *decompressline(compressed_scrollback_line *line)
return ldata;
}
static inline void free_compressed_line(compressed_scrollback_line *cline)
{
sfree(cline);
}
static termline *decompressline_and_free(compressed_scrollback_line *cline)
{
termline *ldata = decompressline_no_free(cline);
free_compressed_line(cline);
return ldata;
}
#else /* NO_SCROLLBACK_COMPRESSION */
static termline *duptermline(termline *oldline)
{
termline *newline = snew(termline);
*newline = *oldline; /* copy the POD structure fields */
newline->chars = snewn(newline->size, termchar);
for (int j = 0; j < newline->size; j++)
newline->chars[j] = oldline->chars[j];
return newline;
}
typedef termline compressed_scrollback_line;
static inline compressed_scrollback_line *compressline_and_free(
termline *ldata)
{
return ldata;
}
static inline compressed_scrollback_line *compressline_no_free(termline *ldata)
{
return duptermline(ldata);
}
static inline termline *decompressline_no_free(
compressed_scrollback_line *line)
{
/* This will return a line without the 'temporary' flag, which
* means that unlineptr() is already set up to avoid freeing it */
return line;
}
static inline termline *decompressline_and_free(
compressed_scrollback_line *line)
{
/* Same as decompressline_no_free, because the caller will free
* our returned termline, and that does all the freeing necessary */
return line;
}
static inline void free_compressed_line(compressed_scrollback_line *line)
{
freetermline(line);
}
#endif /* NO_SCROLLBACK_COMPRESSION */
/*
* Resize a line to make it `cols' columns wide.
*/
@ -1134,7 +1204,7 @@ static termline *lineptr(Terminal *term, int y, int lineno, int screen)
compressed_scrollback_line *cline = index234(whichtree, treeindex);
if (!cline)
null_line_error(term, y, lineno, whichtree, treeindex, "cline");
line = decompressline(cline);
line = decompressline_no_free(cline);
} else {
line = index234(whichtree, treeindex);
}
@ -2065,12 +2135,13 @@ Terminal *term_init(Conf *myconf, struct unicode_data *ucsdata, TermWin *win)
void term_free(Terminal *term)
{
compressed_scrollback_line *cline;
termline *line;
struct beeptime *beep;
int i;
while ((line = delpos234(term->scrollback, 0)) != NULL)
sfree(line); /* compressed data, not a termline */
while ((cline = delpos234(term->scrollback, 0)) != NULL)
free_compressed_line(cline);
freetree234(term->scrollback);
while ((line = delpos234(term->screen, 0)) != NULL)
freetermline(line);
@ -2194,8 +2265,7 @@ void term_size(Terminal *term, int newrows, int newcols, int newsavelines)
/* Insert a line from the scrollback at the top of the screen. */
assert(sblen >= term->tempsblines);
cline = delpos234(term->scrollback, --sblen);
line = decompressline(cline);
sfree(cline);
line = decompressline_and_free(cline);
line->temporary = false; /* reconstituted line is now real */
term->tempsblines -= 1;
addpos234(term->screen, line, 0);
@ -2219,8 +2289,7 @@ void term_size(Terminal *term, int newrows, int newcols, int newsavelines)
} else {
/* push top row to scrollback */
line = delpos234(term->screen, 0);
addpos234(term->scrollback, compressline(line), sblen++);
freetermline(line);
addpos234(term->scrollback, compressline_and_free(line), sblen++);
term->tempsblines += 1;
term->curs.y -= 1;
term->savecurs.y -= 1;
@ -2596,15 +2665,15 @@ static void scroll(Terminal *term, int topline, int botline,
* the scrollback is full.
*/
if (sblen == term->savelines) {
unsigned char *cline;
compressed_scrollback_line *cline;
sblen--;
cline = delpos234(term->scrollback, 0);
sfree(cline);
free_compressed_line(cline);
} else
term->tempsblines += 1;
addpos234(term->scrollback, compressline(line), sblen);
addpos234(term->scrollback, compressline_no_free(line), sblen);
/* now `line' itself can be reused as the bottom line */