aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorIgor Ostapenko <igoreisberg@gmail.com>2022-01-31 21:33:04 +0000
committerAutomerger Merge Worker <android-build-automerger-merge-worker@system.gserviceaccount.com>2022-01-31 21:33:04 +0000
commit4877916653ce51a967b6f8bddba21c98f21beea0 (patch)
tree08c292dce32000f7d771bdd29122d256356c48ba
parentb8f553912ab2fed57dcaf450baa0d83c6127b32d (diff)
parente60ad49783fcdae8e6b862f56b2f88c8c597e106 (diff)
downloaderofs-utils-4877916653ce51a967b6f8bddba21c98f21beea0.tar.gz
UPSTREAM: erofs-utils: fsck: add --extract=X support to extract to path X am: 3d56e12f95 am: 22891dc3d2 am: 8beee7eb12 am: e60ad49783
Original change: https://android-review.googlesource.com/c/platform/external/erofs-utils/+/1966842 Change-Id: I051fee0c912f5f4e900ff8093348a63c7168dbb3
-rw-r--r--configure.ac3
-rw-r--r--fsck/main.c577
-rw-r--r--mkfs/main.c2
3 files changed, 426 insertions, 156 deletions
diff --git a/configure.ac b/configure.ac
index 6fdb0e4..6f7e271 100644
--- a/configure.ac
+++ b/configure.ac
@@ -183,7 +183,8 @@ AC_CHECK_FUNCS(m4_flatten([
strerror
strrchr
strtoull
- tmpfile64]))
+ tmpfile64
+ utimensat]))
# Configure debug mode
AS_IF([test "x$enable_debug" != "xno"], [], [
diff --git a/fsck/main.c b/fsck/main.c
index 30d0a1b..14534b9 100644
--- a/fsck/main.c
+++ b/fsck/main.c
@@ -6,9 +6,12 @@
#include <stdlib.h>
#include <getopt.h>
#include <time.h>
+#include <utime.h>
+#include <unistd.h>
#include <sys/stat.h>
#include "erofs/print.h"
#include "erofs/io.h"
+#include "erofs/compress.h"
#include "erofs/decompress.h"
#include "erofs/dir.h"
@@ -18,6 +21,10 @@ struct erofsfsck_cfg {
bool corrupted;
bool print_comp_ratio;
bool check_decomp;
+ char *extract_path;
+ size_t extract_pos;
+ bool overwrite, preserve_owner, preserve_perms;
+ mode_t umask;
u64 physical_blocks;
u64 logical_blocks;
};
@@ -25,22 +32,47 @@ static struct erofsfsck_cfg fsckcfg;
static struct option long_options[] = {
{"help", no_argument, 0, 1},
- {"extract", no_argument, 0, 2},
+ {"extract", optional_argument, 0, 2},
{"device", required_argument, 0, 3},
+ {"no-same-owner", no_argument, 0, 4},
+ {"no-same-permissions", no_argument, 0, 5},
+ {"same-owner", no_argument, 0, 6},
+ {"same-permissions", no_argument, 0, 7},
+ {"overwrite", no_argument, 0, 8},
{0, 0, 0, 0},
};
+static void print_available_decompressors(FILE *f, const char *delim)
+{
+ unsigned int i = 0;
+ const char *s;
+
+ while ((s = z_erofs_list_available_compressors(i)) != NULL) {
+ if (i++)
+ fputs(delim, f);
+ fputs(s, f);
+ }
+ fputc('\n', f);
+}
+
static void usage(void)
{
fputs("usage: [options] IMAGE\n\n"
- "Check erofs filesystem integrity of IMAGE, and [options] are:\n"
- " -V print the version number of fsck.erofs and exit.\n"
- " -d# set output message level to # (maximum 9)\n"
- " -p print total compression ratio of all files\n"
- " --device=X specify an extra device to be used together\n"
- " --extract check if all files are well encoded\n"
- " --help display this help and exit.\n",
- stderr);
+ "Check erofs filesystem compatibility and integrity of IMAGE, and [options] are:\n"
+ " -V print the version number of fsck.erofs and exit\n"
+ " -d# set output message level to # (maximum 9)\n"
+ " -p print total compression ratio of all files\n"
+ " --device=X specify an extra device to be used together\n"
+ " --extract[=X] check if all files are well encoded, optionally extract to X\n"
+ " --help display this help and exit\n"
+ "\nExtraction options (--extract=X is required):\n"
+ " --no-same-owner extract files as yourself\n"
+ " --no-same-permissions apply the user's umask when extracting permission\n"
+ " --same-permissions extract information about file permissions\n"
+ " --same-owner extract files about the file ownerships\n"
+ " --overwrite if file already exists then overwrite\n"
+ "\nSupported algorithms are: ", stderr);
+ print_available_decompressors(stderr, ", ");
}
static void erofsfsck_print_version(void)
@@ -74,6 +106,23 @@ static int erofsfsck_parse_options_cfg(int argc, char **argv)
exit(0);
case 2:
fsckcfg.check_decomp = true;
+ if (optarg) {
+ size_t len = strlen(optarg);
+
+ if (len == 0)
+ return -EINVAL;
+
+ /* remove trailing slashes except root */
+ while (len > 1 && optarg[len - 1] == '/')
+ len--;
+
+ fsckcfg.extract_path = malloc(PATH_MAX);
+ if (!fsckcfg.extract_path)
+ return -ENOMEM;
+ strncpy(fsckcfg.extract_path, optarg, len);
+ fsckcfg.extract_path[len] = '\0';
+ fsckcfg.extract_pos = len;
+ }
break;
case 3:
ret = blob_open_ro(optarg);
@@ -81,6 +130,21 @@ static int erofsfsck_parse_options_cfg(int argc, char **argv)
return ret;
++sbi.extra_devices;
break;
+ case 4:
+ fsckcfg.preserve_owner = false;
+ break;
+ case 5:
+ fsckcfg.preserve_perms = false;
+ break;
+ case 6:
+ fsckcfg.preserve_owner = true;
+ break;
+ case 7:
+ fsckcfg.preserve_perms = true;
+ break;
+ case 8:
+ fsckcfg.overwrite = true;
+ break;
default:
return -EINVAL;
}
@@ -89,6 +153,11 @@ static int erofsfsck_parse_options_cfg(int argc, char **argv)
if (optind >= argc)
return -EINVAL;
+ if (!fsckcfg.extract_path)
+ if (fsckcfg.overwrite || fsckcfg.preserve_owner ||
+ fsckcfg.preserve_perms)
+ return -EINVAL;
+
cfg.c_img_path = strdup(argv[optind++]);
if (!cfg.c_img_path)
return -ENOMEM;
@@ -100,6 +169,41 @@ static int erofsfsck_parse_options_cfg(int argc, char **argv)
return 0;
}
+static void erofsfsck_set_attributes(struct erofs_inode *inode, char *path)
+{
+ int ret;
+
+#ifdef HAVE_UTIMENSAT
+ const struct timespec times[2] = {
+ [0] = { .tv_sec = inode->i_ctime,
+ .tv_nsec = inode->i_ctime_nsec },
+ [1] = { .tv_sec = inode->i_ctime,
+ .tv_nsec = inode->i_ctime_nsec },
+ };
+
+ if (utimensat(AT_FDCWD, path, times, AT_SYMLINK_NOFOLLOW) < 0)
+#else
+ if (utime(path, &((struct utimbuf){.actime = inode->i_ctime,
+ .modtime = inode->i_ctime})) < 0)
+#endif
+ erofs_warn("failed to set times: %s", path);
+
+ if (!S_ISLNK(inode->i_mode)) {
+ if (fsckcfg.preserve_perms)
+ ret = chmod(path, inode->i_mode);
+ else
+ ret = chmod(path, inode->i_mode & ~fsckcfg.umask);
+ if (ret < 0)
+ erofs_warn("failed to set permissions: %s", path);
+ }
+
+ if (fsckcfg.preserve_owner) {
+ ret = lchown(path, inode->i_uid, inode->i_gid);
+ if (ret < 0)
+ erofs_warn("failed to change ownership: %s", path);
+ }
+}
+
static int erofs_check_sb_chksum(void)
{
int ret;
@@ -127,137 +231,6 @@ static int erofs_check_sb_chksum(void)
return 0;
}
-static int verify_uncompressed_inode(struct erofs_inode *inode)
-{
- struct erofs_map_blocks map = {
- .index = UINT_MAX,
- };
- int ret;
- erofs_off_t ptr = 0;
- u64 i_blocks = DIV_ROUND_UP(inode->i_size, EROFS_BLKSIZ);
-
- while (ptr < inode->i_size) {
- map.m_la = ptr;
- ret = erofs_map_blocks(inode, &map, 0);
- if (ret)
- return ret;
-
- if (map.m_plen != map.m_llen || ptr != map.m_la) {
- erofs_err("broken data chunk layout m_la %" PRIu64 " ptr %" PRIu64 " m_llen %" PRIu64 " m_plen %" PRIu64,
- map.m_la, ptr, map.m_llen, map.m_plen);
- return -EFSCORRUPTED;
- }
-
- if (!(map.m_flags & EROFS_MAP_MAPPED) && !map.m_llen) {
- /* reached EOF */
- ptr = inode->i_size;
- continue;
- }
-
- ptr += map.m_llen;
- }
-
- if (fsckcfg.print_comp_ratio) {
- fsckcfg.logical_blocks += i_blocks;
- fsckcfg.physical_blocks += i_blocks;
- }
-
- return 0;
-}
-
-static int verify_compressed_inode(struct erofs_inode *inode)
-{
- struct erofs_map_blocks map = {
- .index = UINT_MAX,
- };
- struct erofs_map_dev mdev;
- int ret = 0;
- u64 pchunk_len = 0;
- erofs_off_t end = inode->i_size;
- unsigned int raw_size = 0, buffer_size = 0;
- char *raw = NULL, *buffer = NULL;
-
- while (end > 0) {
- map.m_la = end - 1;
-
- ret = z_erofs_map_blocks_iter(inode, &map, 0);
- if (ret)
- goto out;
-
- if (end > map.m_la + map.m_llen) {
- erofs_err("broken compressed chunk layout m_la %" PRIu64 " m_llen %" PRIu64 " end %" PRIu64,
- map.m_la, map.m_llen, end);
- ret = -EFSCORRUPTED;
- goto out;
- }
-
- pchunk_len += map.m_plen;
- end = map.m_la;
-
- if (!fsckcfg.check_decomp || !(map.m_flags & EROFS_MAP_MAPPED))
- continue;
-
- if (map.m_plen > raw_size) {
- raw_size = map.m_plen;
- raw = realloc(raw, raw_size);
- BUG_ON(!raw);
- }
-
- if (map.m_llen > buffer_size) {
- buffer_size = map.m_llen;
- buffer = realloc(buffer, buffer_size);
- BUG_ON(!buffer);
- }
-
- mdev = (struct erofs_map_dev) {
- .m_deviceid = map.m_deviceid,
- .m_pa = map.m_pa,
- };
- ret = erofs_map_dev(&sbi, &mdev);
- if (ret) {
- erofs_err("failed to map device of m_pa %" PRIu64 ", m_deviceid %u @ nid %llu: %d",
- map.m_pa, map.m_deviceid, inode->nid | 0ULL, ret);
- goto out;
- }
-
- ret = dev_read(mdev.m_deviceid, raw, mdev.m_pa, map.m_plen);
- if (ret < 0) {
- erofs_err("failed to read compressed data of m_pa %" PRIu64 ", m_plen %" PRIu64 " @ nid %llu: %d",
- mdev.m_pa, map.m_plen, inode->nid | 0ULL, ret);
- goto out;
- }
-
- ret = z_erofs_decompress(&(struct z_erofs_decompress_req) {
- .in = raw,
- .out = buffer,
- .decodedskip = 0,
- .inputsize = map.m_plen,
- .decodedlength = map.m_llen,
- .alg = map.m_algorithmformat,
- .partial_decoding = 0
- });
-
- if (ret < 0) {
- erofs_err("failed to decompress data of m_pa %" PRIu64 ", m_plen %" PRIu64 " @ nid %llu: %d",
- mdev.m_pa, map.m_plen, inode->nid | 0ULL, ret);
- goto out;
- }
- }
-
- if (fsckcfg.print_comp_ratio) {
- fsckcfg.logical_blocks +=
- DIV_ROUND_UP(inode->i_size, EROFS_BLKSIZ);
- fsckcfg.physical_blocks +=
- DIV_ROUND_UP(pchunk_len, EROFS_BLKSIZ);
- }
-out:
- if (raw)
- free(raw);
- if (buffer)
- free(buffer);
- return ret < 0 ? ret : 0;
-}
-
static int erofs_verify_xattr(struct erofs_inode *inode)
{
unsigned int xattr_hdr_size = sizeof(struct erofs_xattr_ibody_header);
@@ -336,9 +309,18 @@ out:
return ret;
}
-static int erofs_verify_inode_data(struct erofs_inode *inode)
+static int erofs_verify_inode_data(struct erofs_inode *inode, int outfd)
{
- int ret;
+ struct erofs_map_blocks map = {
+ .index = UINT_MAX,
+ };
+ struct erofs_map_dev mdev;
+ int ret = 0;
+ bool compressed;
+ erofs_off_t pos = 0;
+ u64 pchunk_len = 0;
+ unsigned int raw_size = 0, buffer_size = 0;
+ char *raw = NULL, *buffer = NULL;
erofs_dbg("verify data chunk of nid(%llu): type(%d)",
inode->nid | 0ULL, inode->datalayout);
@@ -347,30 +329,284 @@ static int erofs_verify_inode_data(struct erofs_inode *inode)
case EROFS_INODE_FLAT_PLAIN:
case EROFS_INODE_FLAT_INLINE:
case EROFS_INODE_CHUNK_BASED:
- ret = verify_uncompressed_inode(inode);
+ compressed = false;
break;
case EROFS_INODE_FLAT_COMPRESSION_LEGACY:
case EROFS_INODE_FLAT_COMPRESSION:
- ret = verify_compressed_inode(inode);
+ compressed = true;
break;
default:
- ret = -EINVAL;
- break;
+ erofs_err("unknown datalayout");
+ return -EINVAL;
}
- if (ret == -EIO)
- erofs_err("I/O error occurred when verifying data chunk of nid(%llu)",
- inode->nid | 0ULL);
+ while (pos < inode->i_size) {
+ map.m_la = pos;
+ if (compressed)
+ ret = z_erofs_map_blocks_iter(inode, &map,
+ EROFS_GET_BLOCKS_FIEMAP);
+ else
+ ret = erofs_map_blocks(inode, &map,
+ EROFS_GET_BLOCKS_FIEMAP);
+ if (ret)
+ goto out;
+
+ if (!compressed && map.m_llen != map.m_plen) {
+ erofs_err("broken chunk length m_la %" PRIu64 " m_llen %" PRIu64 " m_plen %" PRIu64,
+ map.m_la, map.m_llen, map.m_plen);
+ ret = -EFSCORRUPTED;
+ goto out;
+ }
+
+ /* the last lcluster can be divided into 3 parts */
+ if (map.m_la + map.m_llen > inode->i_size)
+ map.m_llen = inode->i_size - map.m_la;
+
+ pchunk_len += map.m_plen;
+ pos += map.m_llen;
+
+ /* should skip decomp? */
+ if (!(map.m_flags & EROFS_MAP_MAPPED) || !fsckcfg.check_decomp)
+ continue;
+
+ if (map.m_plen > raw_size) {
+ raw_size = map.m_plen;
+ raw = realloc(raw, raw_size);
+ BUG_ON(!raw);
+ }
+
+ mdev = (struct erofs_map_dev) {
+ .m_deviceid = map.m_deviceid,
+ .m_pa = map.m_pa,
+ };
+ ret = erofs_map_dev(&sbi, &mdev);
+ if (ret) {
+ erofs_err("failed to map device of m_pa %" PRIu64 ", m_deviceid %u @ nid %llu: %d",
+ map.m_pa, map.m_deviceid, inode->nid | 0ULL,
+ ret);
+ goto out;
+ }
+
+ if (compressed && map.m_llen > buffer_size) {
+ buffer_size = map.m_llen;
+ buffer = realloc(buffer, buffer_size);
+ BUG_ON(!buffer);
+ }
+
+ ret = dev_read(mdev.m_deviceid, raw, mdev.m_pa, map.m_plen);
+ if (ret < 0) {
+ erofs_err("failed to read data of m_pa %" PRIu64 ", m_plen %" PRIu64 " @ nid %llu: %d",
+ mdev.m_pa, map.m_plen, inode->nid | 0ULL,
+ ret);
+ goto out;
+ }
+
+ if (compressed) {
+ struct z_erofs_decompress_req rq = {
+ .in = raw,
+ .out = buffer,
+ .decodedskip = 0,
+ .inputsize = map.m_plen,
+ .decodedlength = map.m_llen,
+ .alg = map.m_algorithmformat,
+ .partial_decoding = 0
+ };
+
+ ret = z_erofs_decompress(&rq);
+ if (ret < 0) {
+ erofs_err("failed to decompress data of m_pa %" PRIu64 ", m_plen %" PRIu64 " @ nid %llu: %s",
+ mdev.m_pa, map.m_plen,
+ inode->nid | 0ULL, strerror(-ret));
+ goto out;
+ }
+ }
+
+ if (outfd >= 0 && write(outfd, compressed ? buffer : raw,
+ map.m_llen) < 0) {
+ erofs_err("I/O error occurred when verifying data chunk @ nid %llu",
+ inode->nid | 0ULL);
+ ret = -EIO;
+ goto out;
+ }
+ }
+
+ if (fsckcfg.print_comp_ratio) {
+ fsckcfg.logical_blocks +=
+ DIV_ROUND_UP(inode->i_size, EROFS_BLKSIZ);
+ fsckcfg.physical_blocks +=
+ DIV_ROUND_UP(pchunk_len, EROFS_BLKSIZ);
+ }
+out:
+ if (raw)
+ free(raw);
+ if (buffer)
+ free(buffer);
+ return ret < 0 ? ret : 0;
+}
+
+static inline int erofs_extract_dir(struct erofs_inode *inode)
+{
+ int ret;
+ erofs_dbg("create directory %s", fsckcfg.extract_path);
+
+ /* verify data chunk layout */
+ ret = erofs_verify_inode_data(inode, -1);
+ if (ret)
+ return ret;
+
+ /*
+ * Make directory with default user rwx permissions rather than
+ * the permissions from the filesystem, as these may not have
+ * write/execute permission. These are fixed up later in
+ * erofsfsck_set_attributes().
+ */
+ if (mkdir(fsckcfg.extract_path, 0700) < 0) {
+ struct stat st;
+
+ if (errno != EEXIST) {
+ erofs_err("failed to create directory %s: %s",
+ fsckcfg.extract_path, strerror(errno));
+ return -errno;
+ }
+
+ if (lstat(fsckcfg.extract_path, &st) ||
+ !S_ISDIR(st.st_mode)) {
+ erofs_err("path is not a directory: %s",
+ fsckcfg.extract_path);
+ return -ENOTDIR;
+ }
+
+ /*
+ * Try to change permissions of existing directory so
+ * that we can write to it
+ */
+ if (chmod(fsckcfg.extract_path, 0700) < 0)
+ return -errno;
+ }
+ return 0;
+}
+
+static inline int erofs_extract_file(struct erofs_inode *inode)
+{
+ bool tryagain = true;
+ int ret, fd;
+
+ erofs_dbg("extract file to path: %s", fsckcfg.extract_path);
+
+again:
+ fd = open(fsckcfg.extract_path,
+ O_WRONLY | O_CREAT | (fsckcfg.overwrite ? O_TRUNC : O_EXCL),
+ 0700);
+ if (fd < 0) {
+ if (fsckcfg.overwrite && tryagain) {
+ if (errno == EISDIR) {
+ erofs_warn("try to forcely remove directory %s",
+ fsckcfg.extract_path);
+ if (rmdir(fsckcfg.extract_path) < 0) {
+ erofs_err("failed to remove: %s",
+ fsckcfg.extract_path);
+ return -EISDIR;
+ }
+ } else if (errno == EACCES &&
+ chmod(fsckcfg.extract_path, 0700) < 0) {
+ return -errno;
+ }
+ tryagain = false;
+ goto again;
+ }
+ erofs_err("failed to open %s: %s", fsckcfg.extract_path,
+ strerror(errno));
+ return -errno;
+ }
+
+ /* verify data chunk layout */
+ ret = erofs_verify_inode_data(inode, fd);
+ if (ret)
+ return ret;
+
+ if (close(fd))
+ return -errno;
+ return ret;
+}
+
+static inline int erofs_extract_symlink(struct erofs_inode *inode)
+{
+ bool tryagain = true;
+ int ret;
+ char *buf = NULL;
+
+ erofs_dbg("extract symlink to path: %s", fsckcfg.extract_path);
+
+ /* verify data chunk layout */
+ ret = erofs_verify_inode_data(inode, -1);
+ if (ret)
+ return ret;
+
+ buf = malloc(inode->i_size + 1);
+ if (!buf) {
+ ret = -ENOMEM;
+ goto out;
+ }
+
+ ret = erofs_pread(inode, buf, inode->i_size, 0);
+ if (ret) {
+ erofs_err("I/O error occurred when reading symlink @ nid %llu: %d",
+ inode->nid | 0ULL, ret);
+ goto out;
+ }
+
+ buf[inode->i_size] = '\0';
+again:
+ if (symlink(buf, fsckcfg.extract_path) < 0) {
+ if (errno == EEXIST && fsckcfg.overwrite && tryagain) {
+ erofs_warn("try to forcely remove file %s",
+ fsckcfg.extract_path);
+ if (unlink(fsckcfg.extract_path) < 0) {
+ erofs_err("failed to remove: %s",
+ fsckcfg.extract_path);
+ ret = -errno;
+ goto out;
+ }
+ tryagain = false;
+ goto again;
+ }
+ erofs_err("failed to create symlink: %s",
+ fsckcfg.extract_path);
+ ret = -errno;
+ }
+out:
+ if (buf)
+ free(buf);
return ret;
}
static int erofsfsck_dirent_iter(struct erofs_dir_context *ctx)
{
+ int ret;
+ size_t prev_pos = fsckcfg.extract_pos;
+
if (ctx->dot_dotdot)
return 0;
- return erofsfsck_check_inode(ctx->dir->nid, ctx->de_nid);
+ if (fsckcfg.extract_path) {
+ size_t curr_pos = prev_pos;
+
+ fsckcfg.extract_path[curr_pos++] = '/';
+ strncpy(fsckcfg.extract_path + curr_pos, ctx->dname,
+ ctx->de_namelen);
+ curr_pos += ctx->de_namelen;
+ fsckcfg.extract_path[curr_pos] = '\0';
+ fsckcfg.extract_pos = curr_pos;
+ }
+
+ ret = erofsfsck_check_inode(ctx->dir->nid, ctx->de_nid);
+
+ if (fsckcfg.extract_path) {
+ fsckcfg.extract_path[prev_pos] = '\0';
+ fsckcfg.extract_pos = prev_pos;
+ }
+ return ret;
}
static int erofsfsck_check_inode(erofs_nid_t pnid, erofs_nid_t nid)
@@ -394,8 +630,26 @@ static int erofsfsck_check_inode(erofs_nid_t pnid, erofs_nid_t nid)
if (ret)
goto out;
- /* verify data chunk layout */
- ret = erofs_verify_inode_data(&inode);
+ if (fsckcfg.extract_path) {
+ switch (inode.i_mode & S_IFMT) {
+ case S_IFDIR:
+ ret = erofs_extract_dir(&inode);
+ break;
+ case S_IFREG:
+ ret = erofs_extract_file(&inode);
+ break;
+ case S_IFLNK:
+ ret = erofs_extract_symlink(&inode);
+ break;
+ default:
+ /* TODO */
+ goto verify;
+ }
+ } else {
+verify:
+ /* verify data chunk layout */
+ ret = erofs_verify_inode_data(&inode, -1);
+ }
if (ret)
goto out;
@@ -410,6 +664,10 @@ static int erofsfsck_check_inode(erofs_nid_t pnid, erofs_nid_t nid)
ret = erofs_iterate_dir(&ctx, true);
}
+
+ if (!ret)
+ erofsfsck_set_attributes(&inode, fsckcfg.extract_path);
+
out:
if (ret && ret != -EIO)
fsckcfg.corrupted = true;
@@ -425,6 +683,8 @@ int main(int argc, char **argv)
fsckcfg.corrupted = false;
fsckcfg.print_comp_ratio = false;
fsckcfg.check_decomp = false;
+ fsckcfg.extract_path = NULL;
+ fsckcfg.extract_pos = 0;
fsckcfg.logical_blocks = 0;
fsckcfg.physical_blocks = 0;
@@ -435,6 +695,8 @@ int main(int argc, char **argv)
goto exit;
}
+ fsckcfg.umask = umask(0);
+
err = dev_open_ro(cfg.c_img_path);
if (err) {
erofs_err("failed to open image file");
@@ -454,10 +716,17 @@ int main(int argc, char **argv)
err = erofsfsck_check_inode(sbi.root_nid, sbi.root_nid);
if (fsckcfg.corrupted) {
- erofs_err("Found some filesystem corruption");
+ if (!fsckcfg.extract_path)
+ erofs_err("Found some filesystem corruption");
+ else
+ erofs_err("Failed to extract filesystem");
err = -EFSCORRUPTED;
} else if (!err) {
- erofs_info("No error found");
+ if (!fsckcfg.extract_path)
+ erofs_info("No error found");
+ else
+ erofs_info("Extract data successfully");
+
if (fsckcfg.print_comp_ratio) {
double comp_ratio =
(double)fsckcfg.physical_blocks * 100 /
diff --git a/mkfs/main.c b/mkfs/main.c
index 90cedde..1787b2c 100644
--- a/mkfs/main.c
+++ b/mkfs/main.c
@@ -589,7 +589,7 @@ int main(int argc, char **argv)
err = lstat64(cfg.c_src_path, &st);
if (err)
return 1;
- if ((st.st_mode & S_IFMT) != S_IFDIR) {
+ if (!S_ISDIR(st.st_mode)) {
erofs_err("root of the filesystem is not a directory - %s",
cfg.c_src_path);
usage();