mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-26 11:45:08 +00:00
implement chunked reads
Provide a more efficient method for reading lines (or null terminated sequences) when the input is seekable. Another partial fix for #2007
This commit is contained in:
parent
2e65e27189
commit
5809608bd2
3 changed files with 59 additions and 35 deletions
|
@ -1944,44 +1944,40 @@ static int read_interactive(wcstring &buff, int nchars, bool shell, const wchar_
|
||||||
return exit_res;
|
return exit_res;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Read from the fd in chunks until we've read the requested number of characters or a newline or
|
/// Bash uses 128 bytes for its chunk size. Very informal testing I did suggested that a smaller
|
||||||
/// null, as appropriate, is seen. This is should only be used when the fd is seekable.
|
/// chunk size performed better. However, we're going to use the bash value under the assumption
|
||||||
static int read_in_chunks(int fd, wcstring &buff, int nchars, bool split_null) {
|
/// they've done more extensive testing.
|
||||||
|
#define READ_CHUNK_SIZE 128
|
||||||
|
|
||||||
|
/// Read from the fd in chunks until we see newline or null, as appropriate, is seen. This is only
|
||||||
|
/// used when the fd is seekable (so not from a tty or pipe) and we're not reading a specific number
|
||||||
|
/// of chars.
|
||||||
|
static int read_in_chunks(int fd, wcstring &buff, bool split_null) {
|
||||||
int exit_res = STATUS_BUILTIN_OK;
|
int exit_res = STATUS_BUILTIN_OK;
|
||||||
|
std::string str;
|
||||||
bool eof = false;
|
bool eof = false;
|
||||||
|
bool finished = false;
|
||||||
|
|
||||||
while (true) {
|
while (!finished) {
|
||||||
bool finished = false;
|
char inbuf[READ_CHUNK_SIZE];
|
||||||
wchar_t res = 0;
|
int bytes_read = read_blocked(fd, inbuf, READ_CHUNK_SIZE);
|
||||||
mbstate_t state = {};
|
|
||||||
|
|
||||||
while (!finished) {
|
if (bytes_read <= 0) {
|
||||||
char b;
|
eof = true;
|
||||||
if (read_blocked(fd, &b, 1) <= 0) {
|
break;
|
||||||
eof = true;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (MB_CUR_MAX == 1) {
|
|
||||||
res = (unsigned char)b;
|
|
||||||
finished = true;
|
|
||||||
} else {
|
|
||||||
size_t sz = mbrtowc(&res, &b, 1, &state);
|
|
||||||
if (sz == (size_t)-1) {
|
|
||||||
memset(&state, 0, sizeof(state));
|
|
||||||
} else if (sz != (size_t)-2) {
|
|
||||||
finished = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (eof) break;
|
int i;
|
||||||
if (!split_null && res == L'\n') break;
|
for (i = 0; i < bytes_read && !finished; i++) {
|
||||||
if (split_null && res == L'\0') break;
|
if ((!split_null && inbuf[i] == L'\n') || (split_null && inbuf[i] == L'\0')) {
|
||||||
|
finished = true;
|
||||||
buff.push_back(res);
|
} else {
|
||||||
if (0 < nchars && (size_t)nchars <= buff.size()) {
|
str.push_back(inbuf[i]);
|
||||||
break;
|
}
|
||||||
|
}
|
||||||
|
buff += str2wcstring(str);
|
||||||
|
if (i < bytes_read) {
|
||||||
|
CHECK(lseek(fd, i - bytes_read, SEEK_CUR) != -1, STATUS_BUILTIN_ERROR)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2201,15 +2197,18 @@ static int builtin_read(parser_t &parser, io_streams_t &streams, wchar_t **argv)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isatty(0) && streams.stdin_fd == STDIN_FILENO && !split_null) {
|
// TODO: Determine if the original set of conditions for interactive reads should be reinstated:
|
||||||
|
// if (isatty(0) && streams.stdin_fd == STDIN_FILENO && !split_null) {
|
||||||
|
int stream_stdin_is_a_tty = isatty(streams.stdin_fd);
|
||||||
|
if (stream_stdin_is_a_tty && !split_null) {
|
||||||
// We should read interactively using reader_readline(). This does not support splitting on
|
// We should read interactively using reader_readline(). This does not support splitting on
|
||||||
// null. The call to reader_readline may change woptind, so we save and restore it.
|
// null. The call to reader_readline may change woptind, so we save and restore it.
|
||||||
int saved_woptind = w.woptind;
|
int saved_woptind = w.woptind;
|
||||||
exit_res =
|
exit_res =
|
||||||
read_interactive(buff, nchars, shell, mode_name, prompt, right_prompt, commandline);
|
read_interactive(buff, nchars, shell, mode_name, prompt, right_prompt, commandline);
|
||||||
w.woptind = saved_woptind;
|
w.woptind = saved_woptind;
|
||||||
} else if (lseek(streams.stdin_fd, 0, SEEK_CUR) != -1) {
|
} else if (!nchars && !stream_stdin_is_a_tty && lseek(streams.stdin_fd, 0, SEEK_CUR) != -1) {
|
||||||
exit_res = read_in_chunks(streams.stdin_fd, buff, nchars, split_null);
|
exit_res = read_in_chunks(streams.stdin_fd, buff, split_null);
|
||||||
} else {
|
} else {
|
||||||
exit_res = read_one_char_at_a_time(streams.stdin_fd, buff, nchars, split_null);
|
exit_res = read_one_char_at_a_time(streams.stdin_fd, buff, nchars, split_null);
|
||||||
}
|
}
|
||||||
|
|
|
@ -75,3 +75,23 @@ send_line -h "12`_marker 7"
|
||||||
expect_prompt
|
expect_prompt
|
||||||
expect_marker 7
|
expect_marker 7
|
||||||
print_var_contents foo
|
print_var_contents foo
|
||||||
|
|
||||||
|
# ==========
|
||||||
|
# The fix for issue #2007 initially introduced a problem when using a function
|
||||||
|
# to read from /dev/stdin when that is associated with the tty. These tests
|
||||||
|
# are to ensure we don't introduce a regression.
|
||||||
|
send "r2l\n"
|
||||||
|
expect_read_prompt
|
||||||
|
send "abc\n"
|
||||||
|
expect_read_prompt
|
||||||
|
send "def\n"
|
||||||
|
expect "abc then def\r\n"
|
||||||
|
expect_prompt
|
||||||
|
|
||||||
|
send "r2l </dev/stdin\n"
|
||||||
|
expect_read_prompt
|
||||||
|
send "ghi\n"
|
||||||
|
expect_read_prompt
|
||||||
|
send "jkl\n"
|
||||||
|
expect "ghi then jkl\r\n"
|
||||||
|
expect_prompt
|
||||||
|
|
5
tests/test_functions/r2l.fish
Normal file
5
tests/test_functions/r2l.fish
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
function r2l
|
||||||
|
read line1
|
||||||
|
read line2
|
||||||
|
echo $line1 then $line2
|
||||||
|
end
|
Loading…
Reference in a new issue