From: Li Guifu <blucer....@foxmail.com>
Add support exlcude directory when build image with long option arg --exclude-path=dir1/dir2,dir11/dir22 or short option arg -e dir1/dir2,dir11/dir22 that seperated by ',' Signed-off-by: Li Guifu <blucer....@foxmail.com> --- include/erofs/exclude.h | 28 ++++++++ lib/Makefile.am | 2 +- lib/exclude.c | 149 ++++++++++++++++++++++++++++++++++++++++ lib/inode.c | 10 +++ mkfs/main.c | 13 +++- 5 files changed, 200 insertions(+), 2 deletions(-) create mode 100644 include/erofs/exclude.h create mode 100644 lib/exclude.c diff --git a/include/erofs/exclude.h b/include/erofs/exclude.h new file mode 100644 index 0000000..3f9fb4d --- /dev/null +++ b/include/erofs/exclude.h @@ -0,0 +1,28 @@ +// SPDX-License-Identifier: GPL-2.0+ +/* + * erofs-utils\include\erofs\exclude.h + * Created by Li Guifu <blucer....@foxmail.com> + */ + +#ifndef __EROFS_EXCLUDE_H +#define __EROFS_EXCLUDE_H + +struct rule_entry { + struct list_head list; + struct list_head hlist; + char *name; +}; + +void erofs_init_rules(void); +void erofs_free_rules(void); +int erofs_parse_exclude_path(const char *args); +struct rule_entry *erofs_pattern_matched(const char *s); +struct rule_entry *erofs_entry_matched(struct rule_entry *e, + const char *s, unsigned int len); + +static inline int erofs_is_pattern_end(struct rule_entry *e) +{ + return list_empty(&e->hlist); +} +#endif + diff --git a/lib/Makefile.am b/lib/Makefile.am index 1ff81f9..e4b51e6 100644 --- a/lib/Makefile.am +++ b/lib/Makefile.am @@ -3,7 +3,7 @@ noinst_LTLIBRARIES = liberofs.la liberofs_la_SOURCES = config.c io.c cache.c inode.c xattr.c \ - compress.c compressor.c + compress.c compressor.c exclude.c liberofs_la_CFLAGS = -Wall -Werror -I$(top_srcdir)/include if ENABLE_LZ4 liberofs_la_CFLAGS += ${LZ4_CFLAGS} diff --git a/lib/exclude.c b/lib/exclude.c new file mode 100644 index 0000000..25a9d32 --- /dev/null +++ b/lib/exclude.c @@ -0,0 +1,149 @@ +// SPDX-License-Identifier: GPL-2.0+ +/* + * erofs-utils\lib\exclude.c + * Created by Li Guifu <blucer....@foxmail.com> + */ + +#include <string.h> +#include <errno.h> +#include <stdlib.h> + +#include "erofs/defs.h" +#include "erofs/list.h" +#include "erofs/print.h" +#include "erofs/exclude.h" + +static struct rule_entry ext_rule; + +static struct rule_entry *new_entry(const char *s, unsigned int len) +{ + struct rule_entry *e = malloc(sizeof(struct rule_entry)); + + if (!e) + return NULL; + + e->name = strndup(s, len); + if (!e->name) + goto err_strdup; + + init_list_head(&e->hlist); + init_list_head(&e->list); + + return e; + +err_strdup: + free(e); + return NULL; +} + +struct rule_entry *erofs_entry_matched(struct rule_entry *e, + const char *s, unsigned int len) +{ + struct rule_entry *pos; + + while (*s == '/') { + s++; + len--; + } + list_for_each_entry(pos, &e->hlist, list) { + unsigned int l = strlen(pos->name); + + if (l == len && !strcmp(pos->name, s)) + return pos; + } + + return NULL; +} + +static int add_rules(struct rule_entry *e, const char *s) +{ + const char *ptr; + struct rule_entry *le; + + while (*s == '/') + s++; + ptr = s; + +forward: + while(*ptr != '/' && *ptr != '\0') + ptr++; + + le = erofs_entry_matched(e, s, ptr - s); + if (!le) { + struct rule_entry *me = new_entry(s, ptr - s); + + if (!me) + return -ENOMEM; + list_add_tail(&me->list, &e->hlist); + le = me; + } + e = le; + + if (*ptr++ != '\0') { + s = ptr; + goto forward; + } + + return 0; +} + +static void free_rules(struct list_head *h) +{ + struct rule_entry *e, *n; + + list_for_each_entry_safe(e, n, h, list) { + list_del(&e->list); + free_rules(&e->hlist); + free(e->name); + free(e); + } +} +void erofs_init_rules(void) +{ + init_list_head(&ext_rule.list); + init_list_head(&ext_rule.hlist); + ext_rule.name = "/"; +} + +void erofs_free_rules(void) +{ + free_rules(&ext_rule.hlist); +} + +int erofs_parse_exclude_path(const char *args) +{ + const char *str, *ptr = args; + const char sep = ','; + +forward: + while(*ptr != sep && *ptr != '\0') + ptr++; + + str = strndup(args, ptr - args); + if (!str) + goto err_free_paths; + + if (add_rules(&ext_rule, str)) + goto err_free_paths; + + if (*ptr++ != '\0') { + args = ptr; + goto forward; + } + + return 0; + +err_free_paths: + erofs_free_rules(); + return -ENOMEM; +} + +struct rule_entry *erofs_pattern_matched(const char *s) +{ + unsigned int len = strlen(s); + + if (!len) + return &ext_rule; + + return erofs_entry_matched(&ext_rule, s, len); +} diff --git a/lib/inode.c b/lib/inode.c index bd0652b..6278ff8 100644 --- a/lib/inode.c +++ b/lib/inode.c @@ -20,6 +20,7 @@ #include "erofs/io.h" #include "erofs/compress.h" #include "erofs/xattr.h" +#include "erofs/exclude.h" struct erofs_sb_info sbi; @@ -825,6 +826,7 @@ struct erofs_inode *erofs_mkfs_build_tree(struct erofs_inode *dir) DIR *_dir; struct dirent *dp; struct erofs_dentry *d; + struct rule_entry *e; ret = erofs_prepare_xattr_ibody(dir->i_srcpath, &dir->i_xattrs); if (ret < 0) @@ -863,6 +865,7 @@ struct erofs_inode *erofs_mkfs_build_tree(struct erofs_inode *dir) return ERR_PTR(-errno); } + e = erofs_pattern_matched(dir->i_srcpath + strlen(cfg.c_src_path)); while (1) { /* * set errno to 0 before calling readdir() in order to @@ -876,7 +879,14 @@ struct erofs_inode *erofs_mkfs_build_tree(struct erofs_inode *dir) if (is_dot_dotdot(dp->d_name) || !strncmp(dp->d_name, "lost+found", strlen("lost+found"))) continue; + if (e) { + struct rule_entry *le; + le = erofs_entry_matched(e, dp->d_name, + strlen(dp->d_name)); + if (le && erofs_is_pattern_end(le)) + continue; + } d = erofs_d_alloc(dir, dp->d_name); if (IS_ERR(d)) { ret = PTR_ERR(d); diff --git a/mkfs/main.c b/mkfs/main.c index 817a6c1..ab718cb 100644 --- a/mkfs/main.c +++ b/mkfs/main.c @@ -21,6 +21,7 @@ #include "erofs/io.h" #include "erofs/compress.h" #include "erofs/xattr.h" +#include "erofs/exclude.h" #ifdef HAVE_LIBUUID #include <uuid/uuid.h> @@ -30,6 +31,7 @@ static struct option long_options[] = { {"help", no_argument, 0, 1}, + {"exclude-path", required_argument, NULL, 'e'}, {0, 0, 0, 0}, }; @@ -127,7 +129,7 @@ static int mkfs_parse_options_cfg(int argc, char *argv[]) char *endptr; int opt, i; - while((opt = getopt_long(argc, argv, "d:x:z:E:T:", + while((opt = getopt_long(argc, argv, "d:x:z:E:T:e:", long_options, NULL)) != -1) { switch (opt) { case 'z': @@ -178,6 +180,13 @@ static int mkfs_parse_options_cfg(int argc, char *argv[]) } break; + case 'e': + if (erofs_parse_exclude_path(optarg)) { + usage(); + exit(0); + } + break; + case 1: usage(); exit(0); @@ -334,6 +343,7 @@ int main(int argc, char **argv) struct timeval t; erofs_init_configure(); + erofs_init_rules(); fprintf(stderr, "%s %s\n", basename(argv[0]), cfg.c_version); cfg.c_legacy_compress = false; @@ -429,6 +439,7 @@ exit: z_erofs_compress_exit(); dev_close(); erofs_exit_configure(); + erofs_free_rules(); if (err) { erofs_err("\tCould not format the device : %s\n", -- 2.17.1