diff --git a/src/input/TextInputStream.cxx b/src/input/TextInputStream.cxx index 177606d6d..0eb5bc1a8 100644 --- a/src/input/TextInputStream.cxx +++ b/src/input/TextInputStream.cxx @@ -27,52 +27,60 @@ #include #include +char * +TextInputStream::ReadBufferedLine() +{ + auto r = buffer.Read(); + char *newline = reinterpret_cast(memchr(r.data, '\n', r.size)); + if (newline == nullptr) + return nullptr; + + buffer.Consume(newline + 1 - r.data); + + char *end = StripRight(r.data, newline); + *end = 0; + return r.data; +} + char * TextInputStream::ReadLine() { - char *src, *p; + char *line = ReadBufferedLine(); + if (line != nullptr) + return line; - do { - size_t nbytes; + while (true) { auto dest = buffer.Write(); - if (dest.size >= 2) { - /* reserve one byte for the null terminator if - the last line is not terminated by a - newline character */ - --dest.size; - - Error error; - nbytes = is.LockRead(dest.data, dest.size, error); - if (nbytes > 0) - buffer.Append(nbytes); - else if (error.IsDefined()) { - LogError(error); - return nullptr; - } - } else - nbytes = 0; - - auto src_p = buffer.Read(); - if (src_p.IsEmpty()) - return nullptr; - - src = src_p.data; - - p = reinterpret_cast(memchr(src, '\n', src_p.size)); - if (p == nullptr && nbytes == 0) { + if (dest.size < 2) { /* end of file (or line too long): terminate the current line */ - dest = buffer.Write(); + assert(!dest.IsEmpty()); dest[0] = 0; + line = buffer.Read().data; buffer.Clear(); - return src; + return line; } - } while (p == nullptr); - buffer.Consume(p - src + 1); + /* reserve one byte for the null terminator if the + last line is not terminated by a newline + character */ + --dest.size; - char *end = StripRight(src, p); - *end = 0; - return src; + Error error; + size_t nbytes = is.LockRead(dest.data, dest.size, error); + if (nbytes > 0) + buffer.Append(nbytes); + else if (error.IsDefined()) { + LogError(error); + return nullptr; + } + + line = ReadBufferedLine(); + if (line != nullptr) + return line; + + if (nbytes == 0) + return nullptr; + } } diff --git a/src/input/TextInputStream.hxx b/src/input/TextInputStream.hxx index 6f39d22cf..ce1eaddff 100644 --- a/src/input/TextInputStream.hxx +++ b/src/input/TextInputStream.hxx @@ -47,6 +47,9 @@ public: * @return a pointer to the line, or nullptr on end-of-file or error */ char *ReadLine(); + +private: + char *ReadBufferedLine(); }; #endif