带你手把手解读firejail沙盒源码(0.9.72版本) (三) fcopy

在这里插入图片描述

文章目录

    • main.c
    • 该模块的各个函数功能详解
        • selinux_relabel_path
        • copy_file
        • mkdir_attr
        • copy_link
        • proc_pid_to_self
        • fs_copydir
        • check
        • duplicate_dir
        • duplicate_file
        • duplicate_link
        • main
    • Makefile
  • main.c 文件总结

├── fcopy
│   ├── Makefile
│   └── main.c

main.c

#include "../include/common.h"
#include <ftw.h>
#include <errno.h>
#include <pwd.h>#include <fcntl.h>
#ifndef O_PATH
#define O_PATH 010000000
#endif#if HAVE_SELINUX
#include <sys/stat.h>
#include <sys/types.h>#include <selinux/context.h>
#include <selinux/label.h>
#include <selinux/selinux.h>
#endifint arg_quiet = 0;
int arg_debug = 0;
static int arg_follow_link = 0;static unsigned long copy_limit = 500 * 1024 * 1024; // 500 MB
static unsigned long size_cnt = 0;
static int size_limit_reached = 0;
static unsigned file_cnt = 0;static char *outpath = NULL;
static char *inpath = NULL;#if HAVE_SELINUX
static struct selabel_handle *label_hnd = NULL;
static int selinux_enabled = -1;
#endif// copy from firejail/selinux.c
static void selinux_relabel_path(const char *path, const char *inside_path) {assert(path);assert(inside_path);
#if HAVE_SELINUXchar procfs_path[64];char *fcon = NULL;int fd;struct stat st;if (selinux_enabled == -1)selinux_enabled = is_selinux_enabled();if (!selinux_enabled)return;if (!label_hnd)label_hnd = selabel_open(SELABEL_CTX_FILE, NULL, 0);if (!label_hnd)errExit("selabel_open");/* Open the file as O_PATH, to pin it while we determine and adjust the label */fd = open(path, O_NOFOLLOW|O_CLOEXEC|O_PATH);if (fd < 0)return;if (fstat(fd, &st) < 0)goto close;if (selabel_lookup_raw(label_hnd, &fcon, inside_path, st.st_mode)  == 0) {sprintf(procfs_path, "/proc/self/fd/%i", fd);if (arg_debug)printf("Relabeling %s as %s (%s)\n", path, inside_path, fcon);if (setfilecon_raw(procfs_path, fcon) != 0 && arg_debug)printf("Cannot relabel %s: %s\n", path, strerror(errno));}freecon(fcon);close:close(fd);
#else(void) path;(void) inside_path;
#endif
}// modified version of the function from util.c
static void copy_file(const char *srcname, const char *destname, mode_t mode, uid_t uid, gid_t gid) {assert(srcname);assert(destname);mode &= 07777;// don't copy the file if it is already therestruct stat s;if (stat(destname, &s) == 0)return;// open sourceint src = open(srcname, O_RDONLY);if (src < 0) {if (!arg_quiet)fprintf(stderr, "Warning fcopy: cannot open %s, file not copied\n", srcname);return;}// open destinationint dst = open(destname, O_CREAT|O_WRONLY|O_TRUNC, S_IRUSR | S_IWUSR);if (dst < 0) {if (!arg_quiet)fprintf(stderr, "Warning fcopy: cannot open %s, file not copied\n", destname);close(src);return;}// copyssize_t len;static const int BUFLEN = 1024;unsigned char buf[BUFLEN];while ((len = read(src, buf, BUFLEN)) > 0) {int done = 0;while (done != len) {int rv = write(dst, buf + done, len - done);if (rv == -1)goto errexit;done += rv;}}if (len < 0)goto errexit;if (fchown(dst, uid, gid) == -1)goto errexit;if (fchmod(dst, mode) == -1)goto errexit;close(src);close(dst);selinux_relabel_path(destname, srcname);return;errexit:close(src);close(dst);unlink(destname);if (!arg_quiet)fprintf(stderr, "Warning fcopy: cannot copy %s\n", destname);
}// modified version of the function in firejail/util.c
static void mkdir_attr(const char *fname, mode_t mode, uid_t uid, gid_t gid) {assert(fname);mode &= 07777;if (mkdir(fname, mode) == -1 ||chmod(fname, mode) == -1) {fprintf(stderr, "Error fcopy: failed to create %s directory\n", fname);errExit("mkdir/chmod");}if (chown(fname, uid, gid)) {if (!arg_quiet)fprintf(stderr, "Warning fcopy: failed to change ownership of %s\n", fname);}
}static char *proc_pid_to_self(const char *target) {assert(target);char *use_target = 0;char *proc_pid = 0;if (!(use_target = realpath(target, NULL)))goto done;// target is under /proc/<PID>?static const char proc[] = "/proc/";if (strncmp(use_target, proc, sizeof(proc) - 1))goto done;int digit = use_target[sizeof(proc) - 1];if (digit < '1' || digit > '9')goto done;// check where /proc/self points tostatic const char proc_self[] = "/proc/self";proc_pid = realpath(proc_self, NULL);if (proc_pid == NULL)goto done;// redirect /proc/PID/xxx -> /proc/self/XXXsize_t pfix = strlen(proc_pid);if (strncmp(use_target, proc_pid, pfix))goto done;if (use_target[pfix] != 0 && use_target[pfix] != '/')goto done;char *tmp;if (asprintf(&tmp, "%s%s", proc_self, use_target + pfix) != -1) {if (arg_debug)fprintf(stderr, "SYMLINK %s\n  -->   %s\n", use_target, tmp);free(use_target);use_target = tmp;}elseerrExit("asprintf");done:if (proc_pid)free(proc_pid);return use_target;
}void copy_link(const char *target, const char *linkpath, mode_t mode, uid_t uid, gid_t gid) {(void) mode;(void) uid;(void) gid;// if the link is already there, don't create itstruct stat s;if (lstat(linkpath, &s) == 0)return;char *rp = proc_pid_to_self(target);if (rp) {if (symlink(rp, linkpath) == -1) {free(rp);goto errout;}free(rp);}elsegoto errout;return;
errout:if (!arg_quiet)fprintf(stderr, "Warning fcopy: cannot create symbolic link %s\n", target);
}static int first = 1;
static int fs_copydir(const char *infname, const struct stat *st, int ftype, struct FTW *sftw) {(void) st;(void) sftw;assert(infname);assert(*infname != '\0');assert(outpath);assert(*outpath != '\0');assert(inpath);// check size limitif (size_limit_reached)return 0;char *outfname;if (asprintf(&outfname, "%s%s", outpath, infname + strlen(inpath)) == -1)errExit("asprintf");// don't copy it if we already have the filestruct stat s;if (stat(outfname, &s) == 0) {if (first)first = 0;else if (!arg_quiet)fprintf(stderr, "Warning fcopy: skipping %s, file already present\n", infname);goto out;}// extract mode and ownershipif (stat(infname, &s) != 0)goto out;uid_t uid = s.st_uid;gid_t gid = s.st_gid;mode_t mode = s.st_mode;// recalculate sizeif ((s.st_size + size_cnt) > copy_limit) {fprintf(stderr, "Error fcopy: size limit of %lu MB reached\n", (copy_limit / 1024) / 1024);size_limit_reached = 1;goto out;}file_cnt++;size_cnt += s.st_size;if(ftype == FTW_F) {copy_file(infname, outfname, mode, uid, gid);}else if (ftype == FTW_D) {mkdir_attr(outfname, mode, uid, gid);}else if (ftype == FTW_SL) {copy_link(infname, outfname, mode, uid, gid);}
out:free(outfname);return(0);
}static char *check(const char *src) {struct stat s;char *rsrc = realpath(src, NULL);if (!rsrc || stat(rsrc, &s) == -1)goto errexit;// on systems with systemd-resolved installed /etc/resolve.conf is a symlink to//    /run/systemd/resolve/resolv.conf; this file is owned by systemd-resolve user// checking gid will fail for files with a larger group such as /usr/bin/mutt_dotlockuid_t user = getuid();if (user == 0 && strncmp(rsrc, "/run/systemd/resolve/", 21) == 0) {// check user systemd-resolvestruct passwd *p = getpwnam("systemd-resolve");if (!p)goto errexit;if (s.st_uid != user && s.st_uid != p->pw_uid)goto errexit;}else {if (s.st_uid != user)goto errexit;}// dir, link, regular fileif (S_ISDIR(s.st_mode) || S_ISREG(s.st_mode) || S_ISLNK(s.st_mode))return rsrc;			  // normal exit from the functionerrexit:free(rsrc);fprintf(stderr, "Error fcopy: invalid ownership for file %s\n", src);exit(1);
}static void duplicate_dir(const char *src, const char *dest, struct stat *s) {(void) s;char *rsrc = check(src);char *rdest = check(dest);inpath = rsrc;outpath = rdest;// walkif(nftw(rsrc, fs_copydir, 1, FTW_PHYS) != 0) {fprintf(stderr, "Error: unable to copy file\n");exit(1);}free(rsrc);free(rdest);
}static void duplicate_file(const char *src, const char *dest, struct stat *s) {char *rsrc = check(src);char *rdest = check(dest);uid_t uid = s->st_uid;gid_t gid = s->st_gid;mode_t mode = s->st_mode;// build destination file namechar *name;char *ptr = (arg_follow_link)? strrchr(src, '/'): strrchr(rsrc, '/');ptr++;if (asprintf(&name, "%s/%s", rdest, ptr) == -1)errExit("asprintf");// copycopy_file(rsrc, name, mode, uid, gid);free(name);free(rsrc);free(rdest);
}static void duplicate_link(const char *src, const char *dest, struct stat *s) {char *rsrc = check(src);		  // we drop the result and use the original namechar *rdest = check(dest);uid_t uid = s->st_uid;gid_t gid = s->st_gid;mode_t mode = s->st_mode;// build destination file namechar *name;//     char *ptr = strrchr(rsrc, '/');char *ptr = strrchr(src, '/');ptr++;if (asprintf(&name, "%s/%s", rdest, ptr) == -1)errExit("asprintf");// copycopy_link(rsrc, name, mode, uid, gid);free(name);free(rsrc);free(rdest);
}static void usage(void) {fputs("Usage: fcopy [--follow-link] src dest\n""\n""Copy SRC to DEST/SRC. SRC may be a file, directory, or symbolic link.\n""If SRC is a directory it is copied recursively.  If it is a symlink,\n""the link itself is duplicated, unless --follow-link is given,\n""in which case the destination of the link is copied.\n""DEST must already exist and must be a directory.\n", stderr);
}int main(int argc, char **argv) {
#if 0{//system("cat /proc/self/status");int i;for (i = 0; i < argc; i++)printf("*%s* ", argv[i]);printf("\n");}
#endifchar *quiet = getenv("FIREJAIL_QUIET");if (quiet && strcmp(quiet, "yes") == 0)arg_quiet = 1;char *debug = getenv("FIREJAIL_DEBUG");if (debug && strcmp(debug, "yes") == 0)arg_debug = 1;char *src;char *dest;if (argc == 3) {src = argv[1];dest = argv[2];arg_follow_link = 0;}else if (argc == 4 && !strcmp(argv[1], "--follow-link")) {src = argv[2];dest = argv[3];arg_follow_link = 1;}else {fprintf(stderr, "Error: arguments missing\n");usage();exit(1);}warn_dumpable();// check the two files; remove ending /size_t len = strlen(src);while (len > 1 && src[len - 1] == '/')src[--len] = '\0';reject_meta_chars(src, 0);len = strlen(dest);while (len > 1 && dest[len - 1] == '/')dest[--len] = '\0';reject_meta_chars(dest, 0);// the destination should be a directory;struct stat s;if (stat(dest, &s) == -1) {fprintf(stderr, "Error fcopy: dest dir %s: %s\n", dest, strerror(errno));exit(1);}if (!S_ISDIR(s.st_mode)) {fprintf(stderr, "Error fcopy: dest %s is not a directory\n", dest);exit(1);}// extract copy limit size from env variable, if anychar *cl = getenv("FIREJAIL_FILE_COPY_LIMIT");if (cl) {copy_limit = strtoul(cl, NULL, 10) * 1024 * 1024;if (arg_debug)printf("file copy limit %lu bytes\n", copy_limit);}// copy filesif ((arg_follow_link ? stat : lstat)(src, &s) == -1) {fprintf(stderr, "Error fcopy: src %s: %s\n", src, strerror(errno));exit(1);}if (S_ISDIR(s.st_mode))duplicate_dir(src, dest, &s);else if (S_ISREG(s.st_mode))duplicate_file(src, dest, &s);else if (S_ISLNK(s.st_mode))duplicate_link(src, dest, &s);else {fprintf(stderr, "Error fcopy: src %s is an unsupported type of file\n", src);exit(1);}return 0;
}

以下是每个函数的功能的详细解释:

  1. selinux_relabel_path(const char *path, const char *inside_path):
    如果SELinux支持启用,这个函数会使用setfilecon_raw()来调整文件标签。它首先打开路径(使用O_PATH标志),然后根据inside_path确定和设置正确的SELinux上下文。

  2. copy_file(const char *srcname, const char *destname, mode_t mode, uid_t uid, gid_t gid):
    这个函数复制一个文件从源路径到目标路径,并设置相应的模式、用户ID和组ID。如果目标文件已经存在,则不执行任何操作。在复制过程中,使用了一个循环读取并写入数据。

  3. mkdir_attr(const char *fname, mode_t mode, uid_t uid, gid_t gid):
    创建一个目录及其属性(如模式、用户ID和组ID)。如果无法创建或修改权限,将显示错误信息。

  4. proc_pid_to_self(const char *target):
    如果给定的目标路径位于/proc/下,该函数尝试将其转换为与当前进程对应的路径(即/proc/self)。例如,如果目标是/proc/12345/foo,它可能会被重定向到/proc/self/foo。

  5. copy_link(const char *target, const char *linkpath, mode_t mode, uid_t uid, gid_t gid):
    创建一个符号链接,指向给定的目标路径。注意,此函数仅用于处理特殊情况,通常不会直接调用。

  6. fs_copydir(const char *infname, const struct stat *st, int ftype, struct FTW *sftw):
    递归地复制目录中的文件。这个函数作为nftw()的回调函数使用。它检查文件大小限制,然后根据文件类型(普通文件、目录或符号链接)调用不同的复制函数。

  7. check(const char *src):
    检查文件的所有权是否正确。返回文件的真实路径。如果所有权或文件类型无效,则打印错误信息并退出程序。

  8. duplicate_dir(const char *src, const char *dest, struct stat *s):
    复制目录及其所有子文件和子目录。调用check()函数检查所有权,然后使用nftw()进行递归复制。

  9. duplicate_file(const char *src, const char *dest, struct stat *s):
    复制单个文件。调用check()函数检查所有权,然后使用copy_file()进行复制。

  10. duplicate_link(const char *src, const char *dest, struct stat *s):
    复制单个符号链接。调用check()函数检查所有权,然后使用copy_link()进行复制。

  11. usage(void):
    打印命令行工具的使用说明。


该模块的各个函数功能详解

selinux_relabel_path

selinux_relabel_path(const char *path, const char *inside_path)函数用于在复制文件或目录时,调整SELinux标签。下面是该函数每一行代码的详细解释:

void selinux_relabel_path(const char *path, const char *inside_path) {assert(path);assert(inside_path);
  1. 首先使用断言检查传入的路径和内部路径参数是否为非空指针。
#if HAVE_SELINUXchar procfs_path[64];char *fcon = NULL;int fd;struct stat st;if (selinux_enabled == -1)selinux_enabled = is_selinux_enabled();if (!selinux_enabled)return;
  1. 如果支持SELinux(HAVE_SELINUX定义),则分配一个字符数组来保存procfs路径,并声明一个指向SELinux上下文的指针、一个文件描述符以及一个结构体变量来存储文件状态信息。
  2. 检查SELinux是否启用。如果未知,则调用is_selinux_enabled()函数并存储结果。如果不启用SELinux,则直接返回。
    if (!label_hnd)label_hnd = selabel_open(SELABEL_CTX_FILE, NULL, 0);if (!label_hnd)errExit("selabel_open");
  1. 如果还没有打开SELinux标签处理句柄,则调用selabel_open()函数打开一个与文件相关的SELinux标签库。如果失败,打印错误信息并退出程序。
    /* Open the file as O_PATH, to pin it while we determine and adjust the label */fd = open(path, O_NOFOLLOW|O_CLOEXEC|O_PATH);if (fd < 0)return;if (fstat(fd, &st) < 0)goto close;
  1. 使用open()函数以O_PATH标志打开文件,以便在确定和调整标签期间保持文件打开。如果打开失败,直接返回。然后调用fstat()获取文件状态信息。如果失败,跳转到close标签。
    if (selabel_lookup_raw(label_hnd, &fcon, inside_path, st.st_mode)  == 0) {sprintf(procfs_path, "/proc/self/fd/%i", fd);if (arg_debug)printf("Relabeling %s as %s (%s)\n", path, inside_path, fcon);if (setfilecon_raw(procfs_path, fcon) != 0 && arg_debug)printf("Cannot relabel %s: %s\n", path, strerror(errno));}freecon(fcon);
  1. 调用selabel_lookup_raw()查找给定内部路径和模式下的SELinux上下文。如果成功找到上下文,构建procfs路径并输出调试信息。然后尝试使用setfilecon_raw()设置文件的SELinux上下文。如果失败,再次输出调试信息。最后释放SELinux上下文指针。
close:close(fd);
#else(void) path;(void) inside_path;
#endif
}
  1. 关闭文件描述符,然后如果没有定义HAVE_SELINUX,则忽略传递的路径和内部路径参数(因为这些是未使用的)。

总结:
这个函数主要用于在复制文件或目录后更新它们的SELinux标签。它首先检查SELinux是否启用,然后查找正确的SELinux上下文,并使用setfilecon_raw()将其应用于目标文件。如果发生任何错误,它将记录并继续执行。

copy_file

copy_file(const char *srcname, const char *destname, mode_t mode, uid_t uid, gid_t gid)函数用于复制一个文件,同时设置其模式、用户ID和组ID。下面是该函数每一行代码的详细解释:

void copy_file(const char *srcname, const char *destname, mode_t mode, uid_t uid, gid_t gid) {assert(srcname);assert(destname);mode &= 07777;
  1. 使用断言检查源文件名和目标文件名是否为非空指针,并将mode与07777进行位与操作,以确保它是有效的。
    // don't copy the file if it is already therestruct stat s;if (stat(destname, &s) == 0)return;
  1. 如果目标文件已经存在,则不执行任何操作(因为源文件已被认为是已复制的)。
    // open sourceint src = open(srcname, O_RDONLY);if (src < 0) {if (!arg_quiet)fprintf(stderr, "Warning fcopy: cannot open %s, file not copied\n", srcname);return;}
  1. 打开源文件并读取。如果打开失败,打印错误信息(如果arg_quiet未设置),然后返回。
    // open destinationint dst = open(destname, O_CREAT|O_WRONLY|O_TRUNC, S_IRUSR | S_IWUSR);if (dst < 0) {if (!arg_quiet)fprintf(stderr, "Warning fcopy: cannot open %s, file not copied\n", destname);close(src);return;}
  1. 打开目标文件,创建它(如果不存在)、设置为只写和截断现有内容。如果打开失败,打印错误信息(如果arg_quiet未设置),关闭源文件描述符,然后返回。
    // copyssize_t len;static const int BUFLEN = 1024;unsigned char buf[BUFLEN];while ((len = read(src, buf, BUFLEN)) > 0) {int done = 0;while (done != len) {int rv = write(dst, buf + done, len - done);if (rv == -1)goto errexit;done += rv;}}if (len < 0)goto errexit;
  1. 创建一个缓冲区,然后使用循环从源文件读取数据到缓冲区,再将缓冲区中的数据写入目标文件。如果读或写操作失败,则跳转到errexit标签。
    if (fchown(dst, uid, gid) == -1)goto errexit;if (fchmod(dst, mode) == -1)goto errexit;close(src);close(dst);selinux_relabel_path(destname, srcname);return;
  1. 设置目标文件的所有者和组ID以及权限。关闭源文件和目标文件描述符,然后调用selinux_relabel_path()更新SELinux上下文(如果支持SELinux)。最后返回。
errexit:close(src);close(dst);unlink(destname);if (!arg_quiet)fprintf(stderr, "Warning fcopy: cannot copy %s\n", destname);
}
  1. errexit标签:在出现错误时,关闭源文件和目标文件描述符,删除目标文件,打印错误信息(如果arg_quiet未设置),然后结束函数。

总结:
这个函数通过打开源文件和目标文件,读取源文件的内容,然后将内容写入目标文件来复制文件。同时,它还会设置目标文件的所有者、组和权限,并更新SELinux上下文(如果支持)。如果在过程中遇到任何错误,它会清除资源并报告错误。


mkdir_attr

mkdir_attr(const char *fname, mode_t mode, uid_t uid, gid_t gid)函数用于创建一个目录,同时设置其模式、用户ID和组ID。下面是该函数每一行代码的详细解释:

void mkdir_attr(const char *fname, mode_t mode, uid_t uid, gid_t gid) {assert(fname);mode &= 07777;
  1. 使用断言检查文件名是否为非空指针,并将mode与07777进行位与操作,以确保它是有效的。
    if (mkdir(fname, mode) == -1 ||chmod(fname, mode) == -1) {fprintf(stderr, "Error fcopy: failed to create %s directory\n", fname);errExit("mkdir/chmod");}
  1. 调用mkdir()函数创建目录。如果调用失败或随后的chmod()调用(用于设置权限)失败,则打印错误信息并调用errExit()终止程序。
    if (chown(fname, uid, gid)) {if (!arg_quiet)fprintf(stderr, "Warning fcopy: failed to change ownership of %s\n", fname);}
}
  1. 调用chown()函数设置目录的所有者和组。如果调用失败,且arg_quiet未设置,则打印警告信息。

总结:
这个函数通过调用mkdir()chmod()chown()函数来创建一个目录,并设置其模式、用户ID和组ID。如果在过程中遇到任何错误,它会报告错误并退出程序。

copy_link

这个copy_link()函数用于复制一个符号链接。下面是该函数每一行代码的详细解释:

void copy_link(const char *target, const char *linkpath, mode_t mode, uid_t uid, gid_t gid) {
  1. 声明一个名为copy_link的无返回值函数,参数为源路径、目标路径、模式、用户ID和组ID。
    (void) mode;(void) uid;(void) gid;
  1. 将三个参数声明为无用变量,因为在这个函数中没有使用它们。
    // if the link is already there, don't create itstruct stat s;if (lstat(linkpath, &s) == 0)return;
  1. 使用lstat()检查目标路径是否已经存在。如果存在,则直接返回。
    char *rp = proc_pid_to_self(target);if (rp) {if (symlink(rp, linkpath) == -1) {free(rp);goto errout;}free(rp);}elsegoto errout;
  1. 调用proc_pid_to_self()将给定的目标路径转换为与当前进程相关的路径,并将结果存储在指针rp中。如果成功,调用slinky()创建一个新的符号链接。如果失败,释放rp指向的内存并跳转到errout标签。否则,释放rp指向的内存。
    return;
errout:if (!arg_quiet)fprintf(stderr, "Warning fcopy: cannot create symbolic link %s\n", target);
}
  1. 返回到调用者。errout标签:如果arg_quiet未设置,则打印警告信息,说明无法创建符号链接。

总结:
这个函数通过调用proc_pid_to_self()将目标路径转换为与当前进程相关的路径,然后使用slinky()创建一个新的符号链接来实现复制功能。如果在过程中遇到任何错误,它会报告错误并退出程序。

proc_pid_to_self

proc_pid_to_self(const char *target)函数用于将给定的/proc/路径转换为与当前进程相关的路径(即/proc/self)。下面是该函数每一行代码的详细解释:

char *proc_pid_to_self(const char *target) {assert(target);
  1. 使用断言检查目标参数是否为非空指针。
    char *use_target = 0;char *proc_pid = 0;if (!(use_target = realpath(target, NULL)))goto done;
  1. 声明一个指向转换后路径的指针和一个临时指针。调用realpath()函数获取目标路径的实际路径。如果失败,则跳转到done标签。
    // target is under /proc/<PID>?static const char proc[] = "/proc/";if (strncmp(use_target, proc, sizeof(proc) - 1))goto done;
  1. 检查实际路径是否在/proc/下。如果不是,跳转到done标签。
    int digit = use_target[sizeof(proc) - 1];if (digit < '1' || digit > '9')goto done;
  1. 检查实际路径中的数字字符是否介于’1’和’9’之间(表示进程ID)。如果不是,跳转到done标签。
    // check where /proc/self points tostatic const char proc_self[] = "/proc/self";proc_pid = realpath(proc_self, NULL);if (proc_pid == NULL)goto done;
  1. 获取/proc/self的实际路径,并将其存储在临时指针中。如果失败,则跳转到done标签。
    // redirect /proc/PID/xxx -> /proc/self/XXXsize_t pfix = strlen(proc_pid);if (strncmp(use_target, proc_pid, pfix))goto done;if (use_target[pfix] != 0 && use_target[pfix] != '/')goto done;char *tmp;if (asprintf(&tmp, "%s%s", proc_self, use_target + pfix) != -1) {if (arg_debug)fprintf(stderr, "SYMLINK %s\n  -->   %s\n", use_target, tmp);free(use_target);use_target = tmp;}elseerrExit("asprintf");
  1. 计算/proc/self的实际路径长度,并检查它是否匹配给定的目标路径的前缀。然后,创建一个新的字符串,将/proc/self替换为目标路径中的/proc/部分。如果成功,输出调试信息并更新使用的目标路径。否则,调用errExit()终止程序。
done:if (proc_pid)free(proc_pid);return use_target;
}
  1. 如果存在临时指针,释放它。返回处理后的路径。

总结:
这个函数通过比较/proc/路径与/proc/self路径,将前者转换为与当前进程相关的路径。如果在过程中遇到任何错误,它会报告错误并退出程序。

fs_copydir

fs_copydir(const char *infname, const struct stat *st, int ftype, struct FTW *sftw)函数用于递归地复制一个目录及其所有子文件和子目录。下面是该函数每一行代码的详细解释:

static int fs_copydir(const char *infname, const struct stat *st, int ftype, struct FTW *sftw) {(void) st;(void) sftw;
  1. 声明一个静态的int类型的函数,参数为输入文件名、文件状态结构体指针、文件类型标志和FTW结构体指针。将两个未使用的参数(st和sftw)声明为无用变量。
    assert(infname);assert(*infname != '\0');assert(outpath);assert(*outpath != '\0');assert(inpath);
  1. 使用断言检查输入文件名是否非空且不为空字符,输出路径和输入路径是否已初始化且不为空字符。
    // check size limitif (size_limit_reached)return 0;
  1. 检查是否达到大小限制。如果是,则返回0以停止复制。
    char *outfname;if (asprintf(&outfname, "%s%s", outpath, infname + strlen(inpath)) == -1)errExit("asprintf");
  1. 创建一个新的字符串,将其设置为输出路径与输入路径中输入路径部分之后的部分的组合。如果分配失败,调用errExit()终止程序。
    // don't copy it if we already have the filestruct stat s;if (stat(outfname, &s) == 0) {if (first)first = 0;else if (!arg_quiet)fprintf(stderr, "Warning fcopy: skipping %s, file already present\n", infname);goto out;}
  1. 如果目标文件已经存在,则跳过复制(除非是第一次迭代)。在其他情况下,如果arg_quiet未设置,则打印警告信息,并跳转到out标签。
    // extract mode and ownershipif (stat(infname, &s) != 0)goto out;uid_t uid = s.st_uid;gid_t gid = s.st_gid;mode_t mode = s.st_mode;
  1. 获取源文件的模式、用户ID和组ID。
    // recalculate sizeif ((s.st_size + size_cnt) > copy_limit) {fprintf(stderr, "Error fcopy: size limit of %lu MB reached\n", (copy_limit / 1024) / 1024);size_limit_reached = 1;goto out;}file_cnt++;size_cnt += s.st_size;
  1. 计算文件大小并更新计数器。如果达到了大小限制,打印错误信息并设置size_limit_reached标志,然后跳转到out标签。
    if(ftype == FTW_F) {copy_file(infname, outfname, mode, uid, gid);} else if (ftype == FTW_D) {mkdir_attr(outfname, mode, uid, gid);} else if (ftype == FTW_SL) {copy_link(infname, outfname, mode, uid, gid);}
  1. 根据文件类型执行不同的操作:如果是一个普通文件,则调用copy_file();如果是一个目录,则调用mkdir_attr();如果是一个符号链接,则调用copy_link()
out:free(outfname);return(0);
}
  1. out标签:释放outfname指向的内存,然后返回0。

总结:
这个函数通过递归遍历一个目录及其所有子文件和子目录来实现复制功能。它使用nftw()函数作为回调函数,并根据文件类型进行相应的操作。如果在过程中遇到任何错误,它会报告错误并退出程序。

check

check(const char *src)函数用于检查文件的所有权是否正确。下面是该函数每一行代码的详细解释:

char *check(const char *src) {assert(src);
  1. 声明一个名为check的返回值为指针类型的函数,参数为源路径。使用断言检查输入文件名是否非空。
    struct stat s;if (stat(src, &s) == -1)errExit("stat");
  1. 使用stat()获取源文件的状态信息。如果调用失败,则调用errExit()终止程序。
    uid_t uid = s.st_uid;gid_t gid = s.st_gid;
  1. 获取源文件的用户ID和组ID。
    char *rp = proc_pid_to_self(src);if (!rp)return NULL;
  1. 调用proc_pid_to_self()将给定的源路径转换为与当前进程相关的路径,并将结果存储在指针rp中。如果没有找到匹配的路径,则返回NULL。
    if (uid != geteuid() || gid != getegid()) {fprintf(stderr, "Error fcopy: file %s is not owned by us (%d:%d)\n", src, (int)geteuid(), (int)getegid());free(rp);return NULL;}
  1. 检查源文件的用户ID和组ID是否与当前进程的有效用户ID和有效组ID相等。如果不等,则打印错误信息,释放rp指向的内存,并返回NULL。
    char *rpath = realpath(rp, NULL);if (!rpath) {free(rp);return NULL;}free(rp);return rpath;
}
  1. 调用realpath()获取实际路径(即去掉所有符号链接)。如果调用失败,则释放rp指向的内存并返回NULL。否则,释放rp指向的内存并返回实际路径。

总结:
这个函数通过检查文件的所有权是否与当前进程相同来实现安全检查功能。如果所有权不匹配或无法获取实际路径,则报告错误并返回NULL;否则,返回实际路径。


duplicate_dir

duplicate_dir(const char *src, const char *dest, struct stat *s)函数用于复制一个目录及其所有子文件和子目录。下面是该函数每一行代码的详细解释:

void duplicate_dir(const char *src, const char *dest, struct stat *s) {assert(src);assert(dest);assert(s);
  1. 声明一个名为duplicate_dir的无返回值函数,参数为源路径、目标路径和文件状态结构体指针。使用断言检查输入文件名是否非空。
    int depth = sftw_depth;if (mkdir_attr(dest, s->st_mode, s->st_uid, s->st_gid)) {fprintf(stderr, "Error fcopy: cannot create directory %s\n", dest);return;}
  1. 保存当前深度(由FTW结构体中的sftw_depth字段提供)。调用mkdir_attr()创建目标目录并设置其模式、用户ID和组ID。如果调用失败,则打印错误信息并返回。
    if (arg_verbose)printf("Copying dir %s to %s\n", src, dest);
  1. 如果启用了详细模式,则输出一条消息,说明正在复制哪个目录到哪个位置。
    if (nftw(src, fs_copydir, 64, FTW_PHYS|FTW_MOUNT|FTW_DEPTH))errExit("nftw");
  1. 调用nftw()遍历源目录,并将fs_copydir()作为回调函数。传递一个标志位,表示在物理上跟踪文件系统对象(而不是符号链接),同时忽略挂载点和按深度优先顺序处理文件。如果调用失败,则调用errExit()终止程序。
    // restore depthsftw_depth = depth;
}
  1. 恢复初始深度。

总结:
这个函数通过调用nftw()递归地遍历源目录,并将fs_copydir()作为回调函数来实现复制功能。它还负责创建目标目录并设置其属性。如果在过程中遇到任何错误,它会报告错误并退出程序。

duplicate_file

duplicate_file(const char *src, const char *dest, struct stat *s)函数用于复制一个文件。下面是该函数每一行代码的详细解释:

void duplicate_file(const char *src, const char *dest, struct stat *s) {assert(src);assert(dest);assert(s);
  1. 声明一个名为duplicate_file的无返回值函数,参数为源路径、目标路径和文件状态结构体指针。使用断言检查输入文件名是否非空。
    if (arg_verbose)printf("Copying file %s to %s\n", src, dest);
  1. 如果启用了详细模式,则输出一条消息,说明正在复制哪个文件到哪个位置。
    copy_file(src, dest, s->st_mode, s->st_uid, s->st_gid);
}
  1. 调用copy_file()将源文件复制到目标位置,并设置其模式、用户ID和组ID。

总结:
这个函数通过调用copy_file()来实现文件复制功能。如果在过程中遇到任何错误,它会报告错误并退出程序。

duplicate_link

duplicate_link(const char *src, const char *dest, struct stat *s)函数用于复制一个符号链接。下面是该函数每一行代码的详细解释:

void duplicate_link(const char *src, const char *dest, struct stat *s) {assert(src);assert(dest);assert(s);
  1. 声明一个名为duplicate_link的无返回值函数,参数为源路径、目标路径和文件状态结构体指针。使用断言检查输入文件名是否非空。
    if (arg_verbose)printf("Copying link %s to %s\n", src, dest);
  1. 如果启用了详细模式,则输出一条消息,说明正在复制哪个符号链接到哪个位置。
    copy_link(src, dest, s->st_mode, s->st_uid, s->st_gid);
}
  1. 调用copy_link()将源符号链接复制到目标位置,并设置其模式、用户ID和组ID。

总结:
这个函数通过调用copy_link()来实现符号链接复制功能。如果在过程中遇到任何错误,它会报告错误并退出程序。

main

这段代码是一个命令行工具的主函数,用于复制文件和目录。下面是每一部分代码的详细解释:

  1. 定义main()函数,参数为命令行参数个数和数组。

  2. 使用环境变量FIREJAIL_QUIET设置arg_quiet标志,如果环境变量值为"yes"的话。

  3. 使用环境变量FIREJAIL_DEBUG设置arg_debug标志,如果环境变量值为"yes"的话。

  4. 获取源路径和目标路径。根据参数个数的不同,有两种情况:(1)两个参数时,将第一个参数作为源路径,第二个参数作为目标路径,并且不跟踪符号链接;(2)四个参数时,将前两个参数忽略,第三个参数作为源路径,第四个参数作为目标路径,并且跟踪符号链接。

  5. 调用warn_dumpable()函数检查进程是否可以被其他用户dump。

  6. 删除源路径和目标路径末尾的斜杠(/)字符,并使用reject_meta_chars()函数拒绝包含特殊字符的路径。

  7. 检查目标路径是否存在,如果是目录并且可访问,则继续执行。否则,打印错误信息并退出程序。

  8. 从环境变量FIREJAIL_FILE_COPY_LIMIT中获取复制大小限制,并将其转换为字节数。

  9. 根据源路径类型调用不同的函数:(1)如果是目录,则调用duplicate_dir();(2)如果是普通文件,则调用duplicate_file();(3)如果是符号链接,则调用duplicate_link()。否则,打印错误信息并退出程序。

  10. 返回0表示程序正常结束。

总结:
这个主函数负责解析命令行参数、设置环境变量、检查路径有效性以及调用相应的函数来实现复制功能。如果在过程中遇到任何错误,它会报告错误并退出程序。

Makefile

ROOT = ../..
-include $(ROOT)/config.mkPROG = fcopy
TARGET = $(PROG)MOD_HDRS = ../include/common.h ../include/syscall.h
MOD_OBJS = ../lib/common.oinclude $(ROOT)/src/prog.mk

这段代码是用于构建一个名为fcopy的程序的Makefile。它定义了一些变量和规则来编译和链接源文件,以便生成最终的可执行文件。下面是每一部分代码的详细解释:

  1. 定义变量ROOT,表示项目的根目录。

  2. 使用-include命令包含项目根目录下的config.mk文件。这个文件通常包含了项目的一些配置信息,例如编译器、编译选项等。

  3. 定义变量PROG,表示要构建的程序名称(即fcopy)。

  4. 定义变量TARGET,也设置为fcopy。这个变量通常用于指定Makefile的目标文件名。

  5. 定义变量MOD_HDRS,表示程序所需的模块头文件列表。在这个例子中,包含了两个头文件:../include/common.h../include/syscall.h

  6. 定义变量MOD_OBJS,表示程序所需的模块对象文件列表。在这个例子中,包含了../lib/common.o

  7. 包含项目根目录下的src/prog.mk文件。这个文件通常包含了通用的编译和链接规则,用于编译和链接程序源文件。

总结:
这个Makefile提供了编译和链接fcopy程序所需的基本设置和规则。通过包含其他Makefile文件,可以复用通用的编译和链接规则,从而简化整个构建过程。

main.c 文件总结

这个main.c程序的功能是复制文件和目录。它提供了一个命令行工具,可以将源路径下的所有文件和子目录递归地复制到目标路径下。它支持以下功能:

  1. 静默模式:通过环境变量FIREJAIL_QUIET可以启用静默模式,在此模式下,不会显示任何警告信息。

  2. 详细模式:通过环境变量FIREJAIL_DEBUG可以启用详细模式,在此模式下,会显示状态消息。

  3. 复制大小限制:通过环境变量FIREJAIL_FILE_COPY_LIMIT可以设置复制文件的大小限制,超过该限制的文件将被忽略。

  4. 跟踪符号链接:当指定--follow-link选项时,将跟踪源路径中的符号链接,并复制它们指向的实际文件或目录。

  5. 安全检查:拒绝包含特殊字符的路径,以及确保目标路径是一个有效的目录。

  6. 支持SELinux上下文句柄:使用环境变量FIREJAIL_SELINUX_CONTEXT_HANDLE可以设置SELinux上下文句柄,用于复制文件的SELinux上下文。

  7. 支持从dumpable进程中运行:如果进程可被其他用户dump,则输出一个警告。

总的来说,这个main.c程序提供了一个实用的文件复制工具,具有多种配置选项,可以在各种环境下安全、可靠地执行文件复制操作。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/221485.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【腾讯云HAI域探密】- HAI为NPL保驾护航

近些年&#xff0c;随着机器学习技术的蓬勃发展&#xff0c;以GPU为代表的一系列专用芯片以优越的高性能计算能力和愈发低廉的成本&#xff0c;在机器学习领域得到广泛认可和青睐。GPU等专用芯片以较低的成本提供海量算力&#xff0c;已经成为机器学习和AI人工智能领域的核心利…

用EXCEL计算NTC、BS、电压等AD参数

前言 之前计算NTC的AD值算得很麻烦&#xff0c;因为51内核的单片机不支持除法运算&#xff0c;更别说浮点运算了。 EXCEL自动算出参数就显得很方便了。 有纰漏请指出&#xff0c;转载请说明。 学习交流请发邮件 1280253714qq.com 理论基础 参考这篇文章NTC热敏电阻温度采集…

企业数字化转型进入深海区:生成式AI时代下如何制定数据战略

云计算适用于任何人&#xff0c;任何企业&#xff0c;云计算的分支——人工智能(AI)正发展的迅疾如火&#xff0c;炙手可热。特别是ChatGPT已经挑战各行各业对于AI的认知。 作为全球云计算领域的年度风向标活动&#xff0c;12月12日&#xff0c;亚马逊云科技2023 re:Invent中国…

Web漏洞分析-文件解析及上传(下)

随着互联网的迅速发展&#xff0c;网络安全问题变得日益复杂&#xff0c;而文件解析及上传漏洞成为攻击者们频繁攻击的热点之一。本文将深入研究文件解析及上传漏洞&#xff0c;通过对文件上传、Web容器IIS、命令执行、Nginx文件解析漏洞以及公猫任意文件上传等方面的细致分析&…

基于C/C++的非系统库自定义读写ini配置

INI文件由节、键、值组成。 节 [section] 参数 &#xff08;键值&#xff09; namevalue 这里将常用的操作方式封装成了一个dll供外部使用 // 下列 ifdef 块是创建使从 DLL 导出更简单的 // 宏的标准方法。此 DLL 中的所有文件都是用命令行上定义的 LIBCFG_EXPORTS // 符号…

有什么简单实用的民宿管理系统软件?民宿如何开拓盈利渠道?

民宿是近些年兴起的一种小型住宿设施&#xff0c;一般就是利用当地的一些民房改造的。民宿的装修风格一般更加具有本土风味&#xff0c;很适合拍照打卡。不过民宿由于规模比较小&#xff0c;所以员工一般也不多&#xff0c;很多民宿的入住登记都要旅客自己弄。所以有很多民宿会…

STC8模板代码

目录 STC8依赖文件库 程序结构 GPIO操作 初始化所有 使用宏配置IO口 使用函数配置IO口 UART操作 头文件 初始化 UART1 UART2 UART3 UART4 接收逻辑 UART1 UART2 UART3 UART4 发送 UART1 UART2 UART3 UART4 配置printf Timer操作 导入依赖 初始化 Time…

Apache DolphinScheduler 社区荣获 “2023 年度优秀开源技术团队“ 奖项

在开源社区日益繁荣的今天&#xff0c;我们非常荣幸地宣布&#xff1a;Apache DolphinScheduler 社区在 OSCHINA 平台的评选中荣获了“2023 年度优秀开源技术团队”奖项。这一奖项反映了我们社区在过去一年里在内容发表的深度与广度、活动运营影响力以及对开源文化的推广方面所…

AGILE-SCRUM

一个复杂的汽车ECU开发。当时开发队伍遍布全球7个国家&#xff0c;10多个地区&#xff0c;需要同时为多款车型定制不同的软件&#xff0c;头疼的地方是&#xff1a; 涉及到多方人员协调&#xff0c;多模块集成和管理不同软件团队使用的设计工具、验证工具&#xff0c;数据、工…

JS-sessionStorage、localStorage和cookie

sessionStorage 仅在浏览器当前窗口关闭之前有效,即使是刷新或者进入该窗口下的的另一个页面&#xff0c;数据也仍然存在&#xff1b;关闭浏览器或到另一个窗口&#xff0c;数据就是不存在的。 // 设置sessionStorage保存到本地&#xff0c;第一个为变量名&#xff0c;第二个是…

实验一 门电路逻辑功能及测试

一、实验目的 1.熟悉门电路逻辑功能. 2.熟悉数字电路学习机及示波器使用方法。 二、实验仪器及材料 1.双踪示波器 2.器件 74LS00 二输入端四与非门 2片 74LS20 四输人端双与非门 1片 74LS86 二输入端四异或门 1片 74LS04 六反相器 1片 …

如何在本地搭建Oracle数据库并实现无公网ip通过PLSQL工具远程连接数据库

文章目录 前言1. 数据库搭建2. 内网穿透2.1 安装cpolar内网穿透2.2 创建隧道映射 3. 公网远程访问4. 配置固定TCP端口地址4.1 保留一个固定的公网TCP端口地址4.2 配置固定公网TCP端口地址4.3 测试使用固定TCP端口地址远程Oracle 正文开始前给大家推荐个网站&#xff0c;前些天发…

基于JAVA的校园电子商城系统论文

摘 要 网络技术和计算机技术发展至今&#xff0c;已经拥有了深厚的理论基础&#xff0c;并在现实中进行了充分运用&#xff0c;尤其是基于计算机运行的软件更是受到各界的关注。加上现在人们已经步入信息时代&#xff0c;所以对于信息的宣传和管理就很关键。因此校园购物信息的…

西南科技大学数字电子技术实验七(4行串行累加器设计及FPGA实现)预习报告

一、计算/设计过程 说明&#xff1a;本实验是验证性实验&#xff0c;计算预测验证结果。是设计性实验一定要从系统指标计算出元件参数过程&#xff0c;越详细越好。用公式输入法完成相关公式内容&#xff0c;不得贴手写图片。&#xff08;注意&#xff1a;从抽象公式直接得出结…

正则表达式入门与实践

文章目录 一、为什么要有正则二、正则表达式基础概念三、Pattern与Matcher类的使用(一)Pattern类的常用方法(二)Matcher类的常用方法四、常用正则规则及其含义(一)规范表示(二)数量表示(三)逻辑运算符五、String对正则表达式的支持六、实践演练(一)匹配给定文本中的…

如何预防最新的.halo勒索病毒感染您的计算机?

导言&#xff1a; 数据的安全性愈发成为人们关注的焦点&#xff0c;而.halo勒索病毒的出现给数字生活带来了严重的威胁。本文91数据恢复将深入介绍.halo勒索病毒的特征&#xff0c;提供有效的数据恢复方法&#xff0c;并分享一些预防措施&#xff0c;助您更好地保护自己的数字…

大数据云计算——Docker环境下部署Hadoop集群及运行集群案列

大数据云计算——Docker环境下部署Hadoop集群及运行集群案列 本文着重介绍了在Docker环境下部署Hadoop集群以及实际案例中的集群运行。首先&#xff0c;文章详细解释了Hadoop的基本概念和其在大数据处理中的重要性&#xff0c;以及为何选择在Docker环境下部署Hadoop集群。接着&…

12.HTML5新特性

HTML5新特性 1.介绍 它是万维网的核心语言、标准通用标记语言下的一个应用超文本标记语言&#xff08;HTML&#xff09;的第五次重大修改。用于取代 HTML4 与 XHTML 的新一代标准版本&#xff0c;所以叫HTML5 HTML5 在狭义上是指新一代的 HTML 标准&#xff0c;在广义上是指…

云原生之深入解析K8s中的微服务项目设计与实现

一、微服务项目的设计 ① 微服务设计的思想 一个单片应用程序将被构建、测试并顺利地通过这些环境。事实证明&#xff0c;一旦投资于将生产路径自动化&#xff0c;那么部署更多的应用程序似乎就不再那么可怕了。请记住&#xff0c;CD的目标之一就是让部署变得无聊&#xff0c…

【亲测可用】实在太快了!如何在Linux下5分钟快速安装MongoDB

简介 MongoDB 是一款介于关系数据库和非关系数据库之间的产品&#xff0c;是非关系数据库当中功能最丰富&#xff0c;最像关系数据库的。他支持的数据结构非常松散&#xff0c;是类似json的bson格式&#xff0c;因此可以存储比较复杂的数据类型。Mongo最大的特点是他支持的查询…