diff options
-rw-r--r-- | unpack/Makemodule.am | 2 | ||||
-rw-r--r-- | unpack/options.c | 171 | ||||
-rw-r--r-- | unpack/rdsquashfs.c | 197 | ||||
-rw-r--r-- | unpack/rdsquashfs.h | 19 |
4 files changed, 209 insertions, 180 deletions
diff --git a/unpack/Makemodule.am b/unpack/Makemodule.am index c70026d..225263b 100644 --- a/unpack/Makemodule.am +++ b/unpack/Makemodule.am @@ -1,5 +1,5 @@ rdsquashfs_SOURCES = unpack/rdsquashfs.c unpack/rdsquashfs.h -rdsquashfs_SOURCES += unpack/list_files.c +rdsquashfs_SOURCES += unpack/list_files.c unpack/options.c rdsquashfs_SOURCES += unpack/restore_fstree.c unpack/describe.c rdsquashfs_LDADD = libsquashfs.a libfstree.a libcompress.a libutil.a diff --git a/unpack/options.c b/unpack/options.c new file mode 100644 index 0000000..ccfe7c8 --- /dev/null +++ b/unpack/options.c @@ -0,0 +1,171 @@ +/* SPDX-License-Identifier: GPL-3.0-or-later */ +#include "rdsquashfs.h" + +static struct option long_opts[] = { + { "list", required_argument, NULL, 'l' }, + { "cat", required_argument, NULL, 'c' }, + { "unpack-root", required_argument, NULL, 'p' }, + { "unpack-path", required_argument, NULL, 'u' }, + { "no-dev", no_argument, NULL, 'D' }, + { "no-sock", no_argument, NULL, 'S' }, + { "no-fifo", no_argument, NULL, 'F' }, + { "no-slink", no_argument, NULL, 'L' }, + { "no-empty-dir", no_argument, NULL, 'E' }, + { "describe", no_argument, NULL, 'd' }, + { "chmod", no_argument, NULL, 'C' }, + { "chown", no_argument, NULL, 'O' }, + { "quiet", no_argument, NULL, 'q' }, + { "help", no_argument, NULL, 'h' }, + { "version", no_argument, NULL, 'V' }, +}; + +static const char *short_opts = "l:c:u:p:DSFLCOEdqhV"; + +static const char *help_string = +"Usage: %s [OPTIONS] <squashfs-file>\n" +"\n" +"View or extract the contents of a squashfs image.\n" +"\n" +"Possible options:\n" +"\n" +" --list, -l <path> Produce a directory listing for a given path in\n" +" the squashfs image.\n" +" --cat, -c <path> If the specified path is a regular file in the,\n" +" image, dump its contents to stdout.\n" +" --unpack-path, -u <path> Unpack this sub directory from the image. To\n" +" unpack everything, simply specify /.\n" +" --describe, -d Produce a file listing from the image.\n" +"\n" +" --unpack-root, -p <path> If used with --unpack-path, this is where the\n" +" data unpacked to. If used with --describe, this\n" +" is used as a prefix for the input path of\n" +" regular files.\n" +"\n" +" --no-dev, -D Do not unpack device special files.\n" +" --no-sock, -S Do not unpack socket files.\n" +" --no-fifo, -F Do not unpack named pipes.\n" +" --no-slink, -L Do not unpack symbolic links.\n" +" --no-empty-dir, -E Do not unpack directories that would end up\n" +" empty after applying the above rules.\n" +" --chmod, -C Change permission flags of unpacked files to\n" +" those store in the squashfs image.\n" +" --chown, -O Change ownership of unpacked files to the\n" +" UID/GID set in the squashfs image.\n" +" --quiet, -q Do not print out progress while unpacking.\n" +"\n" +" --help, -h Print help text and exit.\n" +" --version, -V Print version information and exit.\n" +"\n"; + +extern const char *__progname; + +static char *get_path(char *old, const char *arg) +{ + char *path; + + free(old); + + path = strdup(arg); + if (path == NULL) { + perror("processing arguments"); + exit(EXIT_FAILURE); + } + + if (canonicalize_name(path)) { + fprintf(stderr, "Invalid path: %s\n", arg); + free(path); + exit(EXIT_FAILURE); + } + + return path; +} + +void process_command_line(options_t *opt, int argc, char **argv) +{ + int i; + + opt->op = OP_NONE; + opt->rdtree_flags = 0; + opt->flags = 0; + opt->cmdpath = NULL; + opt->unpack_root = NULL; + opt->image_name = NULL; + + for (;;) { + i = getopt_long(argc, argv, short_opts, long_opts, NULL); + if (i == -1) + break; + + switch (i) { + case 'D': + opt->rdtree_flags |= RDTREE_NO_DEVICES; + break; + case 'S': + opt->rdtree_flags |= RDTREE_NO_SOCKETS; + break; + case 'F': + opt->rdtree_flags |= RDTREE_NO_FIFO; + break; + case 'L': + opt->rdtree_flags |= RDTREE_NO_SLINKS; + break; + case 'E': + opt->rdtree_flags |= RDTREE_NO_EMPTY; + break; + case 'C': + opt->flags |= UNPACK_CHMOD; + break; + case 'O': + opt->flags |= UNPACK_CHOWN; + break; + case 'c': + opt->op = OP_CAT; + opt->cmdpath = get_path(opt->cmdpath, optarg); + break; + case 'd': + opt->op = OP_DESCRIBE; + break; + case 'l': + opt->op = OP_LS; + opt->cmdpath = get_path(opt->cmdpath, optarg); + break; + case 'p': + opt->unpack_root = optarg; + break; + case 'u': + opt->op = OP_UNPACK; + opt->cmdpath = get_path(opt->cmdpath, optarg); + break; + case 'q': + opt->flags |= UNPACK_QUIET; + break; + case 'h': + printf(help_string, __progname); + free(opt->cmdpath); + exit(EXIT_SUCCESS); + case 'V': + print_version(); + free(opt->cmdpath); + exit(EXIT_SUCCESS); + default: + goto fail_arg; + } + } + + if (opt->op == OP_NONE) { + fputs("No opteration specified\n", stderr); + goto fail_arg; + } + + if (optind >= argc) { + fputs("Missing image argument\n", stderr); + goto fail_arg; + } + + opt->image_name = argv[optind++]; + return; +fail_arg: + fprintf(stderr, "Try `%s --help' for more information.\n", __progname); + free(opt->cmdpath); + exit(EXIT_FAILURE); +} diff --git a/unpack/rdsquashfs.c b/unpack/rdsquashfs.c index a24d6c3..27a0cf5 100644 --- a/unpack/rdsquashfs.c +++ b/unpack/rdsquashfs.c @@ -1,72 +1,6 @@ /* SPDX-License-Identifier: GPL-3.0-or-later */ #include "rdsquashfs.h" -enum { - OP_NONE = 0, - OP_LS, - OP_CAT, - OP_UNPACK, - OP_DESCRIBE, -}; - -static struct option long_opts[] = { - { "list", required_argument, NULL, 'l' }, - { "cat", required_argument, NULL, 'c' }, - { "unpack-root", required_argument, NULL, 'p' }, - { "unpack-path", required_argument, NULL, 'u' }, - { "no-dev", no_argument, NULL, 'D' }, - { "no-sock", no_argument, NULL, 'S' }, - { "no-fifo", no_argument, NULL, 'F' }, - { "no-slink", no_argument, NULL, 'L' }, - { "no-empty-dir", no_argument, NULL, 'E' }, - { "describe", no_argument, NULL, 'd' }, - { "chmod", no_argument, NULL, 'C' }, - { "chown", no_argument, NULL, 'O' }, - { "quiet", no_argument, NULL, 'q' }, - { "help", no_argument, NULL, 'h' }, - { "version", no_argument, NULL, 'V' }, -}; - -static const char *short_opts = "l:c:u:p:DSFLCOEdqhV"; - -static const char *help_string = -"Usage: %s [OPTIONS] <squashfs-file>\n" -"\n" -"View or extract the contents of a squashfs image.\n" -"\n" -"Possible options:\n" -"\n" -" --list, -l <path> Produce a directory listing for a given path in\n" -" the squashfs image.\n" -" --cat, -c <path> If the specified path is a regular file in the,\n" -" image, dump its contents to stdout.\n" -" --unpack-path, -u <path> Unpack this sub directory from the image. To\n" -" unpack everything, simply specify /.\n" -" --describe, -d Produce a file listing from the image.\n" -"\n" -" --unpack-root, -p <path> If used with --unpack-path, this is where the\n" -" data unpacked to. If used with --describe, this\n" -" is used as a prefix for the input path of\n" -" regular files.\n" -"\n" -" --no-dev, -D Do not unpack device special files.\n" -" --no-sock, -S Do not unpack socket files.\n" -" --no-fifo, -F Do not unpack named pipes.\n" -" --no-slink, -L Do not unpack symbolic links.\n" -" --no-empty-dir, -E Do not unpack directories that would end up\n" -" empty after applying the above rules.\n" -" --chmod, -C Change permission flags of unpacked files to\n" -" those store in the squashfs image.\n" -" --chown, -O Change ownership of unpacked files to the\n" -" UID/GID set in the squashfs image.\n" -" --quiet, -q Do not print out progress while unpacking.\n" -"\n" -" --help, -h Print help text and exit.\n" -" --version, -V Print version information and exit.\n" -"\n"; - -extern const char *__progname; - static tree_node_t *find_node(tree_node_t *n, const char *path) { const char *end; @@ -100,114 +34,22 @@ static tree_node_t *find_node(tree_node_t *n, const char *path) return n; } -static char *get_path(char *old, const char *arg) -{ - char *path; - - free(old); - - path = strdup(arg); - if (path == NULL) { - perror("processing arguments"); - exit(EXIT_FAILURE); - } - - if (canonicalize_name(path)) { - fprintf(stderr, "Invalid path: %s\n", arg); - free(path); - exit(EXIT_FAILURE); - } - - return path; -} - int main(int argc, char **argv) { - int i, status = EXIT_FAILURE, op = OP_NONE; - const char *unpack_root = NULL; - int rdtree_flags = 0, flags = 0; + int status = EXIT_FAILURE; data_reader_t *data = NULL; - char *cmdpath = NULL; sqfs_super_t super; compressor_t *cmp; tree_node_t *n; + options_t opt; fstree_t fs; int sqfsfd; - for (;;) { - i = getopt_long(argc, argv, short_opts, long_opts, NULL); - if (i == -1) - break; - - switch (i) { - case 'D': - rdtree_flags |= RDTREE_NO_DEVICES; - break; - case 'S': - rdtree_flags |= RDTREE_NO_SOCKETS; - break; - case 'F': - rdtree_flags |= RDTREE_NO_FIFO; - break; - case 'L': - rdtree_flags |= RDTREE_NO_SLINKS; - break; - case 'C': - flags |= UNPACK_CHMOD; - break; - case 'O': - flags |= UNPACK_CHOWN; - break; - case 'E': - rdtree_flags |= RDTREE_NO_EMPTY; - break; - case 'c': - op = OP_CAT; - cmdpath = get_path(cmdpath, optarg); - break; - case 'd': - op = OP_DESCRIBE; - break; - case 'l': - op = OP_LS; - cmdpath = get_path(cmdpath, optarg); - break; - case 'p': - unpack_root = optarg; - break; - case 'u': - op = OP_UNPACK; - cmdpath = get_path(cmdpath, optarg); - break; - case 'q': - flags |= UNPACK_QUIET; - break; - case 'h': - printf(help_string, __progname); - status = EXIT_SUCCESS; - goto out_cmd; - case 'V': - print_version(); - status = EXIT_SUCCESS; - goto out_cmd; - default: - goto fail_arg; - } - } - - if (op == OP_NONE) { - fputs("No opteration specified\n", stderr); - goto fail_arg; - } - - if (optind >= argc) { - fputs("Missing image argument\n", stderr); - goto fail_arg; - } + process_command_line(&opt, argc, argv); - sqfsfd = open(argv[optind], O_RDONLY); + sqfsfd = open(opt.image_name, O_RDONLY); if (sqfsfd < 0) { - perror(argv[optind]); + perror(opt.image_name); goto out_cmd; } @@ -240,28 +82,29 @@ int main(int argc, char **argv) goto out_cmp; } - if (deserialize_fstree(&fs, &super, cmp, sqfsfd, rdtree_flags)) + if (deserialize_fstree(&fs, &super, cmp, sqfsfd, opt.rdtree_flags)) goto out_cmp; - switch (op) { + switch (opt.op) { case OP_LS: - n = find_node(fs.root, cmdpath); + n = find_node(fs.root, opt.cmdpath); if (n == NULL) { - perror(cmdpath); + perror(opt.cmdpath); goto out_fs; } list_files(n); break; case OP_CAT: - n = find_node(fs.root, cmdpath); + n = find_node(fs.root, opt.cmdpath); if (n == NULL) { - perror(cmdpath); + perror(opt.cmdpath); goto out_fs; } if (!S_ISREG(n->mode)) { - fprintf(stderr, "/%s: not a regular file\n", cmdpath); + fprintf(stderr, "/%s: not a regular file\n", + opt.cmdpath); goto out_fs; } @@ -273,9 +116,9 @@ int main(int argc, char **argv) goto out_fs; break; case OP_UNPACK: - n = find_node(fs.root, cmdpath); + n = find_node(fs.root, opt.cmdpath); if (n == NULL) { - perror(cmdpath); + perror(opt.cmdpath); goto out_fs; } @@ -283,11 +126,11 @@ int main(int argc, char **argv) if (data == NULL) goto out_fs; - if (restore_fstree(unpack_root, n, data, flags)) + if (restore_fstree(opt.unpack_root, n, data, opt.flags)) goto out_fs; break; case OP_DESCRIBE: - describe_tree(fs.root, unpack_root); + describe_tree(fs.root, opt.unpack_root); break; } @@ -301,10 +144,6 @@ out_cmp: out_fd: close(sqfsfd); out_cmd: - free(cmdpath); + free(opt.cmdpath); return status; -fail_arg: - fprintf(stderr, "Try `%s --help' for more information.\n", __progname); - free(cmdpath); - return EXIT_FAILURE; } diff --git a/unpack/rdsquashfs.h b/unpack/rdsquashfs.h index 959960a..9d1ab4f 100644 --- a/unpack/rdsquashfs.h +++ b/unpack/rdsquashfs.h @@ -26,6 +26,23 @@ enum UNPACK_FLAGS { UNPACK_QUIET = 0x04, }; +enum { + OP_NONE = 0, + OP_LS, + OP_CAT, + OP_UNPACK, + OP_DESCRIBE, +}; + +typedef struct { + int op; + int rdtree_flags; + int flags; + char *cmdpath; + const char *unpack_root; + const char *image_name; +} options_t; + void list_files(tree_node_t *node); int restore_fstree(const char *rootdir, tree_node_t *root, @@ -33,4 +50,6 @@ int restore_fstree(const char *rootdir, tree_node_t *root, void describe_tree(tree_node_t *root, const char *unpack_root); +void process_command_line(options_t *opt, int argc, char **argv); + #endif /* RDSQUASHFS_H */ |