diff options
Diffstat (limited to 'sutils')
-rw-r--r-- | sutils/fsck.c | 121 |
1 files changed, 82 insertions, 39 deletions
diff --git a/sutils/fsck.c b/sutils/fsck.c index 34f0078c..fea3ccbd 100644 --- a/sutils/fsck.c +++ b/sutils/fsck.c @@ -54,6 +54,16 @@ #include "fstab.h" +/* for debugging */ +static int _debug = 0; +#define debug(fmt, args...) \ + do { if (_debug) { \ + fprintf (stderr, "[%s: ", __FUNCTION__); \ + fprintf (stderr, fmt , ##args); \ + fprintf (stderr, "]\n"); } } while (0) +#define fs_debug(fs, fmt, args...) \ + debug ("%s: " fmt, (fs)->mntent.mnt_dir , ##args) + #define FSCK_SEARCH_FMTS "/sbin/fsck.%s" /* Exit codes we return. */ @@ -139,16 +149,6 @@ fs_start_fsck (struct fs *fs, int flags) *argp++ = fs->mntent.mnt_fsname; *argp = 0; - if (flags & FSCK_F_VERBOSE) - { - char *argz; - size_t argz_len; - argz_create (argv, &argz, &argz_len); - argz_stringify (argz, argz_len, ' '); - puts (argz); - free (argz); - } - pid = fork (); if (pid < 0) { @@ -163,6 +163,18 @@ fs_start_fsck (struct fs *fs, int flags) exit (FSCK_EX_EXEC); /* Exec failed. */ } + if ((flags & FSCK_F_VERBOSE) || _debug) + { + char *argz; + size_t argz_len; + argz_create (argv, &argz, &argz_len); + argz_stringify (argz, argz_len, ' '); + fs_debug (fs, "Spawned pid %d: %s", pid, argz); + if (flags & FSCK_F_VERBOSE) + puts (argz); + free (argz); + } + return pid; } @@ -180,20 +192,26 @@ fscks_start_fsck (struct fscks *fscks, struct fs *fs) if (got_sigint) /* We got SIGINT, so we pretend that all fscks got a signal without even attempting to run them. */ - return FSCK_EX_SIGNAL; + { + fs_debug (fs, "Forcing signal"); + return FSCK_EX_SIGNAL; + } #define CK(err, fmt, args...) \ do { if (err) { error (0, err, fmt , ##args); return FSCK_EX_ERROR; } } while (0) + fs_debug (fs, "Checking mounted state"); err = fs_mounted (fs, &mounted); CK (err, "%s: Cannot check mounted state", fs->mntent.mnt_dir); if (mounted) { + fs_debug (fs, "Checking readonly state"); err = fs_readonly (fs, &was_readonly); CK (err, "%s: Cannot check readonly state", fs->mntent.mnt_dir); if (! was_readonly) { + fs_debug (fs, "Making readonly"); err = fs_set_readonly (fs, 1); CK (err, "%s: Cannot make readonly", fs->mntent.mnt_dir); } @@ -238,18 +256,23 @@ fsck_cleanup (struct fsck *fsck, int remount, int restore_writable) *fsck->self = fsck->next; /* Remove from chain. */ + fs_debug (fs, "Cleaning up after fsck (remount = %d, restore_writable = %d)", + remount, restore_writable); + if (fs->mounted > 0) /* It's currently mounted; if the fsck modified the device, tell the running filesystem to remount it. Also we may make it writable. */ { if (remount) { + fs_debug (fs, "Remounting"); err = fs_remount (fs); if (err) error (0, err, "%s: Cannot remount", fs->mntent.mnt_dir); } if (!err && !fsck->was_readonly && restore_writable) { + fs_debug (fs, "Making writable"); err = fs_set_readonly (fs, 0); if (err) error (0, err, "%s: Cannot make writable", fs->mntent.mnt_dir); @@ -273,9 +296,14 @@ fscks_wait (struct fscks *fscks) { next = fsck->next; if (fsck->pid == 0) - fsck_cleanup (fsck, 0, 1); + { + fs_debug (fsck->fs, "Pruning failed fsck"); + fsck_cleanup (fsck, 0, 1); + } } + debug ("Waiting..."); + do pid = wait (&wstatus); while (pid < 0 && errno == EINTR); @@ -294,6 +322,7 @@ fscks_wait (struct fscks *fscks) { int remount = (status != 0); int restore_writable = (status == 0 || FSCK_EX_IS_FIXED (status)); + fs_debug (fsck->fs, "Fsck finished (status = %d)", status); fsck_cleanup (fsck, remount, restore_writable); fscks->free_slots++; break; @@ -349,28 +378,34 @@ fsck (struct fstab *fstab, int flags, int max_parallel) for (pass = 1; pass >= 0; pass = fstab_next_pass (fstab, pass)) /* Submit all filesystems in the given pass, up to MAX_PARALLEL at a time. */ - for (fs = fstab->entries; fs; fs = fs->next) - if (fs->mntent.mnt_passno == pass) - /* FS is applicable for this pass. */ - { - struct fstype *type; - error_t err = fs_type (fs, &type); - - if (err) - { - error (0, err, "%s: Cannot find fsck program (type %s)", - fs->mntent.mnt_dir, fs->mntent.mnt_type); - merge_status (FSCK_EX_ERROR); - } - else if (type->program) - /* This is a fsckable filesystem. */ - { - while (fscks->free_slots == 0) - /* No room; wait for another fsck to finish. */ - merge_status (fscks_wait (fscks)); - merge_status (fscks_start_fsck (fscks, fs)); - } - } + { + debug ("Pass %d", pass); + for (fs = fstab->entries; fs; fs = fs->next) + if (fs->mntent.mnt_passno == pass) + /* FS is applicable for this pass. */ + { + struct fstype *type; + error_t err = fs_type (fs, &type); + + if (err) + { + error (0, err, "%s: Cannot find fsck program (type %s)", + fs->mntent.mnt_dir, fs->mntent.mnt_type); + merge_status (FSCK_EX_ERROR); + } + else if (type->program) + /* This is a fsckable filesystem. */ + { + fs_debug (fs, "Fsckable; free_slots = %d", fscks->free_slots); + while (fscks->free_slots == 0) + /* No room; wait for another fsck to finish. */ + merge_status (fscks_wait (fscks)); + merge_status (fscks_start_fsck (fscks, fs)); + } + else + fs_debug (fs, "Not fsckable"); + } + } free (fscks); @@ -386,14 +421,16 @@ options[] = {"fstab", 't', "FILE", 0, "File to use instead of " _PATH_MNTTAB}, {"parallel", 'l', "NUM", 0, "Limit the number of parallel checks to NUM"}, {"verbose", 'v', 0, 0, "Print informational messages"}, + {"debug", 'D', 0, OPTION_HIDDEN }, {"search-fmts",'S', "FMTS", 0, "`:' separated list of formats to use for finding fsck programs"}, {0, 0, 0, 0, "In --preen mode, the following also apply:", 2}, {"force", 'f', 0, 0, "Check even if clean"}, {"silent", 's', 0, 0, "Only print diagostic messages"}, + {"quiet", 'q', 0, OPTION_ALIAS | OPTION_HIDDEN }, {0, 0} }; -static const char *args_doc = "DEVICE"; +static const char *args_doc = "[ DEVICE|FSYS... ]"; static const char *doc = 0; int @@ -402,13 +439,14 @@ main (int argc, char **argv) error_t err; struct fstab *fstab, *check; struct fstypes *types; + int status; /* exit status */ int flags = 0; char *names = 0; size_t names_len = 0; char *search_fmts = FSCK_SEARCH_FMTS; size_t search_fmts_len = sizeof FSCK_SEARCH_FMTS; char *fstab_path = _PATH_MNTTAB; - int max_parallel = -1; + int max_parallel = -1; /* -1 => use default */ error_t parse_opt (int key, char *arg, struct argp_state *state) { @@ -421,6 +459,7 @@ main (int argc, char **argv) case 's': flags |= FSCK_F_SILENT; break; case 'v': flags |= FSCK_F_VERBOSE; break; case 't': fstab_path = arg; break; + case 'D': _debug = 1; break; case 'l': max_parallel = atoi (arg); if (! max_parallel) @@ -451,17 +490,19 @@ main (int argc, char **argv) if (err) error (101, err, "fstab_create"); - err = fstab_read (fstab, _PATH_MNTTAB); + debug ("Reading %s...", fstab_path); + err = fstab_read (fstab, fstab_path); if (err) - error (103, err, "%s", _PATH_MNTTAB); + error (103, err, "%s", fstab_path); if (names) /* Fsck specified filesystems; also look at /var/run/mtab. */ { char *name; + debug ("Reading %s...", _PATH_MOUNTED); err = fstab_read (fstab, _PATH_MOUNTED); - if (err) + if (err && err != ENOENT) error (104, err, "%s", _PATH_MOUNTED); err = fstab_create (types, &check); @@ -473,6 +514,7 @@ main (int argc, char **argv) struct fs *fs = fstab_find (fstab, name); if (! fs) error (106, 0, "%s: Unknown device or filesystem", name); + fs_debug (fs, "Adding to checked filesystems"); fstab_add_fs (check, fs, 0); } } @@ -496,6 +538,7 @@ main (int argc, char **argv) got a signal. */ signal (SIGINT, sigint); + debug ("Fscking..."); status = fsck (check, flags, max_parallel); if (got_sigquit && status < FSCK_EX_QUIT) status = FSCK_EX_QUIT; |