[lxc-devel] [lxc/master] parse: protect against config updates during parse
brauner on Github
lxc-bot at linuxcontainers.org
Wed Oct 17 16:37:29 UTC 2018
A non-text attachment was scrubbed...
Name: not available
Type: text/x-mailbox
Size: 364 bytes
Desc: not available
URL: <http://lists.linuxcontainers.org/pipermail/lxc-devel/attachments/20181017/5890a9c8/attachment.bin>
-------------- next part --------------
From 24b97297056b3d61b53aa467bbdb32037ff7e0d7 Mon Sep 17 00:00:00 2001
From: Christian Brauner <christian.brauner at ubuntu.com>
Date: Wed, 17 Oct 2018 18:36:28 +0200
Subject: [PATCH] parse: protect against config updates during parse
Signed-off-by: Christian Brauner <christian.brauner at ubuntu.com>
---
src/lxc/parse.c | 81 +++++++++++++++++++++++++++++++++++++------------
1 file changed, 62 insertions(+), 19 deletions(-)
diff --git a/src/lxc/parse.c b/src/lxc/parse.c
index 2fdb18ec8..439cc29c3 100644
--- a/src/lxc/parse.c
+++ b/src/lxc/parse.c
@@ -30,6 +30,7 @@
#include <stdlib.h>
#include <string.h>
#include <sys/mman.h>
+#include <sys/sendfile.h>
#include "config.h"
#include "log.h"
@@ -67,32 +68,71 @@ int lxc_strmunmap(void *addr, size_t length)
int lxc_file_for_each_line_mmap(const char *file, lxc_file_cb callback, void *data)
{
- int fd, saved_errno;
- char *buf, *line;
+ int saved_errno;
+ char *line;
struct stat st;
- int ret = 0;
+ ssize_t ret = 0;
+ int fd = -1, memfd = -1;
+ char *buf = NULL;
+
+ memfd = memfd_create(".lxc_mount_file", MFD_CLOEXEC);
+ if (memfd < 0) {
+ char template[] = P_tmpdir "/.lxc_mountinfo_XXXXXX";
+
+ if (errno != ENOSYS) {
+ SYSERROR("Failed to create memory file");
+ goto on_error;
+ }
+
+ TRACE("Failed to create in-memory file. Falling back to "
+ "temporary file");
+ memfd = lxc_make_tmpfile(template, true);
+ if (memfd < 0) {
+ SYSERROR("Failed to create temporary file \"%s\"", template);
+ goto on_error;
+ }
+ }
fd = open(file, O_RDONLY | O_CLOEXEC);
if (fd < 0) {
- SYSERROR("Failed to open config file \"%s\"", file);
+ SYSERROR("Failed to open file \"%s\"", file);
return -1;
}
ret = fstat(fd, &st);
if (ret < 0) {
- SYSERROR("Failed to stat config file \"%s\"", file);
- goto on_error_fstat;
+ SYSERROR("Failed to stat file \"%s\"", file);
+ goto on_error;
}
ret = 0;
if (st.st_size == 0)
- goto on_error_fstat;
+ goto on_error;
+
+ /* sendfile() handles up to 2GB. No config file should be that big. */
+ ret = sendfile(memfd, fd, NULL, st.st_size);
+ if (ret < 0) {
+ SYSERROR("Failed to sendfile \"%s\"", file);
+ goto on_error;
+ }
- ret = -1;
- buf = lxc_strmmap(NULL, st.st_size, PROT_READ | PROT_WRITE,
- MAP_PRIVATE | MAP_POPULATE, fd, 0);
+ ret = write(memfd, "\0", 1);
+ if (ret < 0) {
+ SYSERROR("Failed to append zero byte");
+ goto on_error;
+ }
+
+ ret = lseek(memfd, 0, SEEK_SET);
+ if (ret < 0) {
+ SYSERROR("Failed to lseek");
+ goto on_error;
+ }
+
+ buf = mmap(NULL, st.st_size + 1, PROT_READ | PROT_WRITE,
+ MAP_SHARED | MAP_POPULATE, memfd, 0);
if (buf == MAP_FAILED) {
- SYSERROR("Failed to map config file \"%s\"", file);
+ buf = NULL;
+ SYSERROR("Failed to mmap");
goto on_error;
}
@@ -111,15 +151,18 @@ int lxc_file_for_each_line_mmap(const char *file, lxc_file_cb callback, void *da
}
on_error:
- if (lxc_strmunmap(buf, st.st_size) < 0) {
- SYSERROR("Failed to unmap config file \"%s\"", file);
- if (ret == 0)
- ret = -1;
- }
-
-on_error_fstat:
saved_errno = errno;
- close(fd);
+ if (fd >= 0)
+ close(fd);
+ if (memfd >= 0)
+ close(memfd);
+ if (buf) {
+ if (munmap(buf, st.st_size + 1) < 0) {
+ SYSERROR("Failed to unmap");
+ if (ret == 0)
+ ret = -1;
+ }
+ }
errno = saved_errno;
return ret;
More information about the lxc-devel
mailing list