]> git.ipfire.org Git - thirdparty/systemd.git/commitdiff
fileio: fix truncated read handling in read_virtual_file()
authorLennart Poettering <lennart@poettering.net>
Thu, 16 Sep 2021 10:20:09 +0000 (12:20 +0200)
committerLennart Poettering <lennart@poettering.net>
Thu, 16 Sep 2021 10:23:25 +0000 (12:23 +0200)
We mishandled the case where the size we read from the file actually
matched the maximum size fully. In that case we cannot really make a
determination whether the file was fully read or only partially. In that
case let's do another loop, so that we operate with a buffer, and
we can detect the EOF (which will be signalled to us via a short read).

src/basic/fileio.c

index e69b888315240e21cbd587e96f5fb17003d125de..60a98851082fe481c9e24a375b92958cec699494 100644 (file)
@@ -470,9 +470,14 @@ int read_virtual_file(const char *filename, size_t max_size, char **ret_contents
                 if (n <= size)
                         break;
 
-                /* If a maximum size is specified and we already read as much, no need to try again */
-                if (max_size != SIZE_MAX && n >= max_size) {
-                        n = max_size;
+                /* If a maximum size is specified and we already read more we know the file is larger, and
+                 * can handle this as truncation case. Note that if the size of what we read equals the
+                 * maximum size then this doesn't mean truncation, the file might or might not end on that
+                 * byte. We need to rerun the loop in that case, with a larger buffer size, so that we read
+                 * at least one more byte to be able to distinguish EOF from truncation. */
+                if (max_size != SIZE_MAX && n > max_size) {
+                        n = size; /* Make sure we never use more than what we sized the buffer for (so that
+                                   * we have one free byte in it for the trailing NUL we add below).*/
                         truncated = true;
                         break;
                 }