summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--miscutils/less.c114
1 files changed, 64 insertions, 50 deletions
diff --git a/miscutils/less.c b/miscutils/less.c
index 8d4f7ec..3a9dea0 100644
--- a/miscutils/less.c
+++ b/miscutils/less.c
@@ -98,8 +98,8 @@ struct globals {
unsigned max_lineno; /* this one tracks linewrap */
unsigned width;
ssize_t eof_error; /* eof if 0, error if < 0 */
- size_t readpos;
- size_t readeof;
+ ssize_t readpos;
+ ssize_t readeof; /* must be signed */
const char **buffer;
const char **flines;
const char *empty_line_marker;
@@ -114,7 +114,8 @@ struct globals {
#if ENABLE_FEATURE_LESS_REGEXP
unsigned *match_lines;
int match_pos; /* signed! */
- unsigned num_matches;
+ int wanted_match; /* signed! */
+ int num_matches;
regex_t pattern;
smallint pattern_valid;
#endif
@@ -146,6 +147,7 @@ struct globals {
#define match_lines (G.match_lines )
#define match_pos (G.match_pos )
#define num_matches (G.num_matches )
+#define wanted_match (G.wanted_match )
#define pattern (G.pattern )
#define pattern_valid (G.pattern_valid )
#endif
@@ -160,6 +162,7 @@ struct globals {
current_file = 1; \
eof_error = 1; \
terminated = 1; \
+ USE_FEATURE_LESS_REGEXP(wanted_match = -1;) \
} while (0)
/* Reset terminal input to normal */
@@ -235,15 +238,20 @@ static void read_lines(void)
{
#define readbuf bb_common_bufsiz1
char *current_line, *p;
- USE_FEATURE_LESS_REGEXP(unsigned old_max_fline = max_fline;)
int w = width;
char last_terminated = terminated;
+#if ENABLE_FEATURE_LESS_REGEXP
+ unsigned old_max_fline = max_fline;
+ time_t last_time = 0;
+ unsigned seconds_p1 = 3; /* seconds_to_loop + 1 */
+#endif
if (option_mask32 & FLAG_N)
w -= 8;
- current_line = xmalloc(w);
- p = current_line;
+ USE_FEATURE_LESS_REGEXP(again0:)
+
+ p = current_line = xmalloc(w);
max_fline += last_terminated;
if (!last_terminated) {
const char *cp = flines[max_fline];
@@ -251,49 +259,26 @@ static void read_lines(void)
cp += 8;
strcpy(current_line, cp);
p += strlen(current_line);
+ free((char*)flines[max_fline]);
/* linepos is still valid from previous read_lines() */
} else {
linepos = 0;
}
- while (1) {
- again:
+ while (1) { /* read lines until we reach cur_fline or wanted_match */
*p = '\0';
terminated = 0;
- while (1) {
+ while (1) { /* read chars until we have a line */
char c;
/* if no unprocessed chars left, eat more */
if (readpos >= readeof) {
- smallint yielded = 0;
-
ndelay_on(0);
- read_again:
eof_error = safe_read(0, readbuf, sizeof(readbuf));
+ ndelay_off(0);
readpos = 0;
readeof = eof_error;
- if (eof_error < 0) {
- if (errno == EAGAIN && !yielded) {
- /* We can hit EAGAIN while searching for regexp match.
- * Yield is not 100% reliable solution in general,
- * but for less it should be good enough -
- * we give stdin supplier some CPU time to produce
- * more input. We do it just once.
- * Currently, we do not stop when we found the Nth
- * occurrence we were looking for. We read till end
- * (or double EAGAIN). TODO? */
- sched_yield();
- yielded = 1;
- goto read_again;
- }
- readeof = 0;
- if (errno != EAGAIN)
- print_statusline("read error");
- }
- ndelay_off(0);
-
- if (eof_error <= 0) {
+ if (eof_error <= 0)
goto reached_eof;
- }
}
c = readbuf[readpos];
/* backspace? [needed for manpages] */
@@ -327,13 +312,13 @@ static void read_lines(void)
if (c == '\0') c = '\n';
*p++ = c;
*p = '\0';
- }
+ } /* end of "read chars until we have a line" loop */
/* Corner case: linewrap with only "" wrapping to next line */
/* Looks ugly on screen, so we do not store this empty line */
if (!last_terminated && !current_line[0]) {
last_terminated = 1;
max_lineno++;
- goto again;
+ continue;
}
reached_eof:
last_terminated = terminated;
@@ -353,22 +338,51 @@ static void read_lines(void)
eof_error = 0; /* Pretend we saw EOF */
break;
}
- if (max_fline > cur_fline + max_displayed_line)
+ if (max_fline > cur_fline + max_displayed_line) {
+#if !ENABLE_FEATURE_LESS_REGEXP
break;
+#else
+ if (wanted_match >= num_matches) { /* goto_match called us */
+ fill_match_lines(old_max_fline);
+ old_max_fline = max_fline;
+ }
+ if (wanted_match < num_matches)
+ break;
+#endif
+ }
if (eof_error <= 0) {
- if (eof_error < 0 && errno == EAGAIN) {
- /* not yet eof or error, reset flag (or else
- * we will hog CPU - select() will return
- * immediately */
- eof_error = 1;
+ if (eof_error < 0) {
+ if (errno == EAGAIN) {
+ /* not yet eof or error, reset flag (or else
+ * we will hog CPU - select() will return
+ * immediately */
+ eof_error = 1;
+ } else {
+ print_statusline("read error");
+ }
}
+#if !ENABLE_FEATURE_LESS_REGEXP
break;
+#else
+ if (wanted_match < num_matches) {
+ break;
+ } else { /* goto_match called us */
+ time_t t = time(NULL);
+ if (t != last_time) {
+ last_time = t;
+ if (--seconds_p1 == 0)
+ break;
+ }
+ sched_yield();
+ goto again0; /* go loop again (max 2 seconds) */
+ }
+#endif
}
max_fline++;
current_line = xmalloc(w);
p = current_line;
linepos = 0;
- }
+ } /* end of "read lines until we reach cur_fline" loop */
fill_match_lines(old_max_fline);
#undef readbuf
}
@@ -884,24 +898,24 @@ static void normalize_match_pos(int match)
static void goto_match(int match)
{
- int sv;
-
if (!pattern_valid)
return;
if (match < 0)
match = 0;
- sv = cur_fline;
/* Try to find next match if eof isn't reached yet */
if (match >= num_matches && eof_error > 0) {
- cur_fline = MAXLINES; /* look as far as needed */
+ wanted_match = match;
read_lines();
+ if (wanted_match >= num_matches) {
+ /* We still failed to find it. Prevent future
+ * read_lines() from trying... */
+ wanted_match = num_matches - 1;
+ }
}
if (num_matches) {
- cap_cur_fline(cur_fline);
normalize_match_pos(match);
buffer_line(match_lines[match_pos]);
} else {
- cur_fline = sv;
print_statusline("No matches found");
}
}
@@ -1370,7 +1384,7 @@ int less_main(int argc, char **argv)
get_terminal_width_height(kbd_fd, &width, &max_displayed_line);
/* 20: two tabstops + 4 */
if (width < 20 || max_displayed_line < 3)
- bb_error_msg_and_die("too narrow here");
+ return bb_cat(argv);
max_displayed_line -= 2;
buffer = xmalloc((max_displayed_line+1) * sizeof(char *));