summaryrefslogtreecommitdiff
path: root/usr.bin/tmux
diff options
context:
space:
mode:
authorNicholas Marriott <nicm@cvs.openbsd.org>2020-06-04 20:41:58 +0000
committerNicholas Marriott <nicm@cvs.openbsd.org>2020-06-04 20:41:58 +0000
commitb2475cae72e9dc499868484afe169cacb53302d5 (patch)
tree9d220fbdf886c63bf7c8806d83857b51fa1802e6 /usr.bin/tmux
parent267cd1813af8dc39ffcc1eb6fd7daeb35818371d (diff)
Some improvements to performance of searching:
- Do not allow searches to be repeated at intervals of less than 50 milliseconds, to prevent a huge queue of repeat key presses blocking up everything for ages. - If the search text hasn't changed, the match count can't have changed and there is no need to do a full search, so only search the visible text. This includes both scrolling and repeating the search. - Do not redraw twice when jumping to the search location. GitHub issue 2258.
Diffstat (limited to 'usr.bin/tmux')
-rw-r--r--usr.bin/tmux/window-copy.c140
1 files changed, 87 insertions, 53 deletions
diff --git a/usr.bin/tmux/window-copy.c b/usr.bin/tmux/window-copy.c
index 7c94410f904..25aae27a822 100644
--- a/usr.bin/tmux/window-copy.c
+++ b/usr.bin/tmux/window-copy.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: window-copy.c,v 1.295 2020/06/04 10:24:14 nicm Exp $ */
+/* $OpenBSD: window-copy.c,v 1.296 2020/06/04 20:41:57 nicm Exp $ */
/*
* Copyright (c) 2007 Nicholas Marriott <nicholas.marriott@gmail.com>
@@ -52,7 +52,8 @@ static void window_copy_write_line(struct window_mode_entry *,
static void window_copy_write_lines(struct window_mode_entry *,
struct screen_write_ctx *, u_int, u_int);
-static void window_copy_scroll_to(struct window_mode_entry *, u_int, u_int);
+static void window_copy_scroll_to(struct window_mode_entry *, u_int, u_int,
+ int);
static int window_copy_search_compare(struct grid *, u_int, u_int,
struct grid *, u_int, int);
static int window_copy_search_lr(struct grid *, struct grid *, u_int *,
@@ -67,7 +68,7 @@ static char *window_copy_stringify(struct grid *, u_int, u_int, u_int,
static void window_copy_cstrtocellpos(struct grid *, u_int, u_int *, u_int *,
const char *str);
static int window_copy_search_marks(struct window_mode_entry *,
- struct screen *, int);
+ struct screen *, int, int);
static void window_copy_clear_marks(struct window_mode_entry *);
static void window_copy_move_left(struct screen *, u_int *, u_int *, int);
static void window_copy_move_right(struct screen *, u_int *, u_int *, int);
@@ -265,6 +266,7 @@ struct window_copy_mode_data {
u_int my;
int showmark;
+ uint64_t searchtime;
int searchtype;
int searchregex;
char *searchstr;
@@ -280,6 +282,7 @@ struct window_copy_mode_data {
int timeout; /* search has timed out */
#define WINDOW_COPY_SEARCH_TIMEOUT 10000
#define WINDOW_COPY_SEARCH_ALL_TIMEOUT 200
+#define WINDOW_COPY_SEARCH_REPEAT 50
int jumptype;
char jumpchar;
@@ -588,7 +591,7 @@ window_copy_pageup1(struct window_mode_entry *wme, int half_page)
}
if (data->searchmark != NULL && !data->timeout)
- window_copy_search_marks(wme, NULL, data->searchregex);
+ window_copy_search_marks(wme, NULL, data->searchregex, 1);
window_copy_update_selection(wme, 1, 0);
window_copy_redraw_screen(wme);
}
@@ -638,7 +641,7 @@ window_copy_pagedown(struct window_mode_entry *wme, int half_page,
if (scroll_exit && data->oy == 0)
return (1);
if (data->searchmark != NULL && !data->timeout)
- window_copy_search_marks(wme, NULL, data->searchregex);
+ window_copy_search_marks(wme, NULL, data->searchregex, 1);
window_copy_update_selection(wme, 1, 0);
window_copy_redraw_screen(wme);
return (0);
@@ -658,7 +661,7 @@ window_copy_previous_paragraph(struct window_mode_entry *wme)
while (oy > 0 && window_copy_find_length(wme, oy) > 0)
oy--;
- window_copy_scroll_to(wme, 0, oy);
+ window_copy_scroll_to(wme, 0, oy, 0);
}
static void
@@ -678,7 +681,7 @@ window_copy_next_paragraph(struct window_mode_entry *wme)
oy++;
ox = window_copy_find_length(wme, oy);
- window_copy_scroll_to(wme, ox, oy);
+ window_copy_scroll_to(wme, ox, oy, 0);
}
char *
@@ -754,7 +757,7 @@ window_copy_size_changed(struct window_mode_entry *wme)
screen_write_stop(&ctx);
if (search && !data->timeout)
- window_copy_search_marks(wme, NULL, data->searchregex);
+ window_copy_search_marks(wme, NULL, data->searchregex, 0);
data->searchx = data->cx;
data->searchy = data->cy;
data->searcho = data->oy;
@@ -1150,7 +1153,7 @@ window_copy_cmd_history_bottom(struct window_copy_cmd_state *cs)
data->oy = 0;
if (data->searchmark != NULL && !data->timeout)
- window_copy_search_marks(wme, NULL, data->searchregex);
+ window_copy_search_marks(wme, NULL, data->searchregex, 1);
window_copy_update_selection(wme, 1, 0);
return (WINDOW_COPY_CMD_REDRAW);
}
@@ -1171,7 +1174,7 @@ window_copy_cmd_history_top(struct window_copy_cmd_state *cs)
data->oy = screen_hsize(data->backing);
if (data->searchmark != NULL && !data->timeout)
- window_copy_search_marks(wme, NULL, data->searchregex);
+ window_copy_search_marks(wme, NULL, data->searchregex, 1);
window_copy_update_selection(wme, 1, 0);
return (WINDOW_COPY_CMD_REDRAW);
}
@@ -1325,7 +1328,7 @@ window_copy_cmd_previous_matching_bracket(struct window_copy_cmd_state *cs)
/* Move the cursor to the found location if any. */
if (!failed)
- window_copy_scroll_to(wme, px, py);
+ window_copy_scroll_to(wme, px, py, 0);
}
return (WINDOW_COPY_CMD_NOTHING);
@@ -1376,7 +1379,7 @@ window_copy_cmd_next_matching_bracket(struct window_copy_cmd_state *cs)
sx = data->cx;
sy = screen_hsize(s) + data->cy - data->oy;
- window_copy_scroll_to(wme, px, py);
+ window_copy_scroll_to(wme, px, py, 0);
window_copy_cmd_previous_matching_bracket(cs);
px = data->cx;
@@ -1385,7 +1388,7 @@ window_copy_cmd_next_matching_bracket(struct window_copy_cmd_state *cs)
if (gc.data.size == 1 &&
(~gc.flags & GRID_FLAG_PADDING) &&
strchr(close, *gc.data.data) != NULL)
- window_copy_scroll_to(wme, sx, sy);
+ window_copy_scroll_to(wme, sx, sy, 0);
break;
}
@@ -1447,7 +1450,7 @@ window_copy_cmd_next_matching_bracket(struct window_copy_cmd_state *cs)
/* Move the cursor to the found location if any. */
if (!failed)
- window_copy_scroll_to(wme, px, py);
+ window_copy_scroll_to(wme, px, py, 0);
}
return (WINDOW_COPY_CMD_NOTHING);
@@ -1659,6 +1662,10 @@ window_copy_cmd_search_again(struct window_copy_cmd_state *cs)
struct window_copy_mode_data *data = wme->data;
u_int np = wme->prefix;
+ if (data->searchtime != 0 &&
+ get_timer() - data->searchtime < WINDOW_COPY_SEARCH_REPEAT)
+ return (WINDOW_COPY_CMD_NOTHING);
+
if (data->searchtype == WINDOW_COPY_SEARCHUP) {
for (; np != 0; np--)
window_copy_search_up(wme, data->searchregex);
@@ -1666,6 +1673,7 @@ window_copy_cmd_search_again(struct window_copy_cmd_state *cs)
for (; np != 0; np--)
window_copy_search_down(wme, data->searchregex);
}
+ data->searchtime = get_timer();
return (WINDOW_COPY_CMD_NOTHING);
}
@@ -1676,6 +1684,10 @@ window_copy_cmd_search_reverse(struct window_copy_cmd_state *cs)
struct window_copy_mode_data *data = wme->data;
u_int np = wme->prefix;
+ if (data->searchtime != 0 &&
+ get_timer() - data->searchtime < WINDOW_COPY_SEARCH_REPEAT)
+ return (WINDOW_COPY_CMD_NOTHING);
+
if (data->searchtype == WINDOW_COPY_SEARCHUP) {
for (; np != 0; np--)
window_copy_search_down(wme, data->searchregex);
@@ -1683,6 +1695,7 @@ window_copy_cmd_search_reverse(struct window_copy_cmd_state *cs)
for (; np != 0; np--)
window_copy_search_up(wme, data->searchregex);
}
+ data->searchtime = get_timer();
return (WINDOW_COPY_CMD_NOTHING);
}
@@ -2327,7 +2340,8 @@ window_copy_command(struct window_mode_entry *wme, struct client *c,
}
static void
-window_copy_scroll_to(struct window_mode_entry *wme, u_int px, u_int py)
+window_copy_scroll_to(struct window_mode_entry *wme, u_int px, u_int py,
+ int no_redraw)
{
struct window_copy_mode_data *data = wme->data;
struct grid *gd = data->backing->grid;
@@ -2352,10 +2366,11 @@ window_copy_scroll_to(struct window_mode_entry *wme, u_int px, u_int py)
data->oy = gd->hsize - offset;
}
- if (data->searchmark != NULL && !data->timeout)
- window_copy_search_marks(wme, NULL, data->searchregex);
+ if (!no_redraw && data->searchmark != NULL && !data->timeout)
+ window_copy_search_marks(wme, NULL, data->searchregex, 1);
window_copy_update_selection(wme, 1, 0);
- window_copy_redraw_screen(wme);
+ if (!no_redraw)
+ window_copy_redraw_screen(wme);
}
static int
@@ -2877,7 +2892,7 @@ window_copy_search_jump(struct window_mode_entry *wme, struct grid *gd,
}
if (found) {
- window_copy_scroll_to(wme, px, i);
+ window_copy_scroll_to(wme, px, i, 1);
return (1);
}
if (wrap) {
@@ -2903,7 +2918,7 @@ window_copy_search(struct window_mode_entry *wme, int direction, int regex)
struct grid *gd = s->grid;
const char *str = data->searchstr;
u_int fx, fy, endline;
- int wrapflag, cis, found;
+ int wrapflag, cis, found, visible_only;
if (regex && str[strcspn(str, "^$*+()?[].\\")] == '\0')
regex = 0;
@@ -2911,6 +2926,10 @@ window_copy_search(struct window_mode_entry *wme, int direction, int regex)
if (data->timeout)
return (0);
+ if (wp->searchstr == NULL || wp->searchregex != regex)
+ visible_only = 0;
+ else
+ visible_only = (strcmp(wp->searchstr, str) == 0);
free(wp->searchstr);
wp->searchstr = xstrdup(str);
wp->searchregex = regex;
@@ -2936,23 +2955,37 @@ window_copy_search(struct window_mode_entry *wme, int direction, int regex)
found = window_copy_search_jump(wme, gd, ss.grid, fx, fy, endline, cis,
wrapflag, direction, regex);
-
- if (window_copy_search_marks(wme, &ss, regex))
- window_copy_redraw_screen(wme);
+ if (found)
+ window_copy_search_marks(wme, &ss, regex, visible_only);
+ window_copy_redraw_screen(wme);
screen_free(&ss);
return (found);
}
+static void
+window_copy_visible_lines(struct window_copy_mode_data *data, u_int *start,
+ u_int *end)
+{
+ struct grid *gd = data->backing->grid;
+ const struct grid_line *gl;
+
+ for (*start = gd->hsize - data->oy; *start > 0; (*start)--) {
+ gl = grid_peek_line(gd, (*start) - 1);
+ if (~gl->flags & GRID_LINE_WRAPPED)
+ break;
+ }
+ *end = gd->hsize - data->oy + gd->sy;
+}
+
static int
window_copy_search_marks(struct window_mode_entry *wme, struct screen *ssp,
- int regex)
+ int regex, int visible_only)
{
struct window_copy_mode_data *data = wme->data;
struct screen *s = data->backing, ss;
struct screen_write_ctx ctx;
struct grid *gd = s->grid;
- const struct grid_line *gl;
int found, cis, which = -1, stopped = 0;
int cflags = REG_EXTENDED;
u_int px, py, i, b, nfound = 0, width;
@@ -2988,9 +3021,13 @@ window_copy_search_marks(struct window_mode_entry *wme, struct screen *ssp,
}
tstart = get_timer();
- start = 0;
- end = gd->hsize + gd->sy;
- stop = get_timer() + WINDOW_COPY_SEARCH_ALL_TIMEOUT;
+ if (visible_only)
+ window_copy_visible_lines(data, &start, &end);
+ else {
+ start = 0;
+ end = gd->hsize + gd->sy;
+ stop = get_timer() + WINDOW_COPY_SEARCH_ALL_TIMEOUT;
+ }
again:
free(data->searchmark);
@@ -3045,34 +3082,31 @@ again:
if (stopped && stop != 0) {
/* Try again but just the visible context. */
- for (start = gd->hsize - data->oy; start > 0; start--) {
- gl = grid_peek_line(gd, start - 1);
- if (~gl->flags & GRID_LINE_WRAPPED)
- break;
- }
- end = gd->hsize - data->oy + gd->sy;
+ window_copy_visible_lines(data, &start, &end);
stop = 0;
goto again;
}
- if (stopped) {
- data->searchthis = -1;
- if (nfound > 1000)
- data->searchcount = 1000;
- else if (nfound > 100)
- data->searchcount = 100;
- else if (nfound > 10)
- data->searchcount = 10;
- else
- data->searchcount = -1;
- data->searchmore = 1;
- } else {
- if (which != -1)
- data->searchthis = 1 + nfound - which;
- else
+ if (!visible_only) {
+ if (stopped) {
data->searchthis = -1;
- data->searchcount = nfound;
- data->searchmore = 0;
+ if (nfound > 1000)
+ data->searchcount = 1000;
+ else if (nfound > 100)
+ data->searchcount = 100;
+ else if (nfound > 10)
+ data->searchcount = 10;
+ else
+ data->searchcount = -1;
+ data->searchmore = 1;
+ } else {
+ if (which != -1)
+ data->searchthis = 1 + nfound - which;
+ else
+ data->searchthis = -1;
+ data->searchcount = nfound;
+ data->searchmore = 0;
+ }
}
out:
@@ -4563,7 +4597,7 @@ window_copy_scroll_up(struct window_mode_entry *wme, u_int ny)
data->oy -= ny;
if (data->searchmark != NULL && !data->timeout)
- window_copy_search_marks(wme, NULL, data->searchregex);
+ window_copy_search_marks(wme, NULL, data->searchregex, 1);
window_copy_update_selection(wme, 0, 0);
screen_write_start_pane(&ctx, wp, NULL);
@@ -4599,7 +4633,7 @@ window_copy_scroll_down(struct window_mode_entry *wme, u_int ny)
data->oy += ny;
if (data->searchmark != NULL && !data->timeout)
- window_copy_search_marks(wme, NULL, data->searchregex);
+ window_copy_search_marks(wme, NULL, data->searchregex, 1);
window_copy_update_selection(wme, 0, 0);
screen_write_start_pane(&ctx, wp, NULL);