bcachefs: thread_with_stdio: fix bch2_stdio_redirect_readline()

This fixes a bug where we'd return data without waiting for a newline,
if data was present but a newline was not.

Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
This commit is contained in:
Kent Overstreet 2024-02-04 22:56:16 -05:00
parent a6777ca4ff
commit f704f108af

View File

@ -277,25 +277,36 @@ int bch2_stdio_redirect_read(struct stdio_redirect *stdio, char *ubuf, size_t le
int bch2_stdio_redirect_readline(struct stdio_redirect *stdio, char *ubuf, size_t len) int bch2_stdio_redirect_readline(struct stdio_redirect *stdio, char *ubuf, size_t len)
{ {
struct stdio_buf *buf = &stdio->input; struct stdio_buf *buf = &stdio->input;
size_t copied = 0;
ssize_t ret = 0;
again:
wait_event(buf->wait, stdio_redirect_has_input(stdio)); wait_event(buf->wait, stdio_redirect_has_input(stdio));
if (stdio->done) if (stdio->done) {
return -1; ret = -1;
goto out;
}
spin_lock(&buf->lock); spin_lock(&buf->lock);
int ret = min(len, buf->buf.nr); size_t b = min(len, buf->buf.nr);
char *n = memchr(buf->buf.data, '\n', ret); char *n = memchr(buf->buf.data, '\n', b);
if (!n) if (n)
ret = min(ret, n + 1 - buf->buf.data); b = min_t(size_t, b, n + 1 - buf->buf.data);
buf->buf.nr -= ret; buf->buf.nr -= b;
memcpy(ubuf, buf->buf.data, ret); memcpy(ubuf, buf->buf.data, b);
memmove(buf->buf.data, memmove(buf->buf.data,
buf->buf.data + ret, buf->buf.data + b,
buf->buf.nr); buf->buf.nr);
ubuf += b;
len -= b;
copied += b;
spin_unlock(&buf->lock); spin_unlock(&buf->lock);
wake_up(&buf->wait); wake_up(&buf->wait);
return ret;
if (!n && len)
goto again;
out:
return copied ?: ret;
} }
__printf(3, 0) __printf(3, 0)