Blame gfs2/fsck/pass3.c

Packit 6ef888
#include "clusterautoconfig.h"
Packit 6ef888
Packit 6ef888
#include <stdio.h>
Packit 6ef888
#include <stdlib.h>
Packit 6ef888
#include <inttypes.h>
Packit 6ef888
#include <string.h>
Packit 6ef888
#include <dirent.h>
Packit 6ef888
#include <libintl.h>
Packit 6ef888
#define _(String) gettext(String)
Packit 6ef888
Packit 6ef888
#include <logging.h>
Packit 6ef888
#include "libgfs2.h"
Packit 6ef888
#include "osi_list.h"
Packit 6ef888
#include "fsck.h"
Packit 6ef888
#include "lost_n_found.h"
Packit 6ef888
#include "link.h"
Packit 6ef888
#include "metawalk.h"
Packit 6ef888
#include "util.h"
Packit 6ef888
#include "afterpass1_common.h"
Packit 6ef888
Packit 6ef888
static int attach_dotdot_to(struct gfs2_sbd *sdp, uint64_t newdotdot,
Packit 6ef888
			    uint64_t olddotdot, uint64_t block)
Packit 6ef888
{
Packit 6ef888
	const char *filename = "..";
Packit 6ef888
	int filename_len = 2;
Packit 6ef888
	int err;
Packit 6ef888
	struct gfs2_inode *ip, *pip;
Packit 6ef888
Packit 6ef888
	ip = fsck_load_inode(sdp, block);
Packit 6ef888
	pip = fsck_load_inode(sdp, newdotdot);
Packit 6ef888
	/* FIXME: Need to add some interactive
Packit 6ef888
	 * options here and come up with a
Packit 6ef888
	 * good default for non-interactive */
Packit 6ef888
	/* FIXME: do i need to correct the
Packit 6ef888
	 * '..' entry for this directory in
Packit 6ef888
	 * this case? */
Packit 6ef888
Packit 6ef888
	if (gfs2_dirent_del(ip, filename, filename_len))
Packit 6ef888
		log_warn( _("Unable to remove \"..\" directory entry.\n"));
Packit 6ef888
	else
Packit 6ef888
		decr_link_count(olddotdot, block, sdp->gfs1, _("old \"..\""));
Packit 6ef888
	err = dir_add(ip, filename, filename_len, &pip->i_di.di_num,
Packit 6ef888
		      (sdp->gfs1 ? GFS_FILE_DIR : DT_DIR));
Packit 6ef888
	if (err) {
Packit 6ef888
		log_err(_("Error adding directory %s: %s\n"),
Packit 6ef888
		        filename, strerror(errno));
Packit 6ef888
		exit(FSCK_ERROR);
Packit 6ef888
	}
Packit 6ef888
	incr_link_count(pip->i_di.di_num, ip, _("new \"..\""));
Packit 6ef888
	fsck_inode_put(&ip);
Packit 6ef888
	fsck_inode_put(&pip;;
Packit 6ef888
	return 0;
Packit 6ef888
}
Packit 6ef888
Packit 6ef888
static struct dir_info *mark_and_return_parent(struct gfs2_sbd *sdp,
Packit 6ef888
					       struct dir_info *di)
Packit 6ef888
{
Packit 6ef888
	struct dir_info *pdi;
Packit 6ef888
	int q_dotdot, q_treewalk;
Packit 6ef888
	int error = 0;
Packit 6ef888
	struct dir_info *dt_dotdot, *dt_treewalk;
Packit 6ef888
Packit 6ef888
	di->checked = 1;
Packit 6ef888
Packit 6ef888
	if (!di->treewalk_parent)
Packit 6ef888
		return NULL;
Packit 6ef888
Packit 6ef888
	if (di->dotdot_parent.no_addr == di->treewalk_parent) {
Packit 6ef888
		q_dotdot = bitmap_type(sdp, di->dotdot_parent.no_addr);
Packit 6ef888
		if (q_dotdot != GFS2_BLKST_DINODE) {
Packit 6ef888
			log_err( _("Orphaned directory at block %llu (0x%llx) "
Packit 6ef888
				   "moved to lost+found\n"),
Packit 6ef888
				 (unsigned long long)di->dinode.no_addr,
Packit 6ef888
				 (unsigned long long)di->dinode.no_addr);
Packit 6ef888
			return NULL;
Packit 6ef888
		}
Packit 6ef888
		goto out;
Packit 6ef888
	}
Packit 6ef888
Packit 6ef888
	log_warn( _("Directory '..' and treewalk connections disagree for "
Packit 6ef888
		    "inode %llu (0x%llx)\n"),
Packit 6ef888
		  (unsigned long long)di->dinode.no_addr,
Packit 6ef888
		  (unsigned long long)di->dinode.no_addr);
Packit 6ef888
	log_notice( _("'..' has %llu (0x%llx), treewalk has %llu (0x%llx)\n"),
Packit 6ef888
		    (unsigned long long)di->dotdot_parent.no_addr,
Packit 6ef888
		    (unsigned long long)di->dotdot_parent.no_addr,
Packit 6ef888
		    (unsigned long long)di->treewalk_parent,
Packit 6ef888
		    (unsigned long long)di->treewalk_parent);
Packit 6ef888
	q_dotdot = bitmap_type(sdp, di->dotdot_parent.no_addr);
Packit 6ef888
	dt_dotdot = dirtree_find(di->dotdot_parent.no_addr);
Packit 6ef888
	q_treewalk = bitmap_type(sdp, di->treewalk_parent);
Packit 6ef888
	dt_treewalk = dirtree_find(di->treewalk_parent);
Packit 6ef888
	/* if the dotdot entry isn't a directory, but the
Packit 6ef888
	 * treewalk is, treewalk is correct - if the treewalk
Packit 6ef888
	 * entry isn't a directory, but the dotdot is, dotdot
Packit 6ef888
	 * is correct - if both are directories, which do we
Packit 6ef888
	 * choose? if neither are directories, we have a
Packit 6ef888
	 * problem - need to move this directory into lost+found
Packit 6ef888
	 */
Packit 6ef888
	if (q_dotdot != GFS2_BLKST_DINODE || dt_dotdot == NULL) {
Packit 6ef888
		if (q_treewalk != GFS2_BLKST_DINODE) {
Packit 6ef888
			log_err( _("Orphaned directory, move to "
Packit 6ef888
				   "lost+found\n"));
Packit 6ef888
			return NULL;
Packit 6ef888
		} else {
Packit 6ef888
			log_warn( _("Treewalk parent is correct, fixing "
Packit 6ef888
				    "dotdot -> %llu (0x%llx)\n"),
Packit 6ef888
				  (unsigned long long)di->treewalk_parent,
Packit 6ef888
				  (unsigned long long)di->treewalk_parent);
Packit 6ef888
			attach_dotdot_to(sdp, di->treewalk_parent,
Packit 6ef888
					 di->dotdot_parent.no_addr,
Packit 6ef888
					 di->dinode.no_addr);
Packit 6ef888
			di->dotdot_parent.no_addr = di->treewalk_parent;
Packit 6ef888
		}
Packit 6ef888
		goto out;
Packit 6ef888
	}
Packit 6ef888
	if (dt_treewalk) {
Packit 6ef888
		log_err( _("Both .. and treewalk parents are directories, "
Packit 6ef888
			   "going with treewalk...\n"));
Packit 6ef888
		attach_dotdot_to(sdp, di->treewalk_parent,
Packit 6ef888
				 di->dotdot_parent.no_addr,
Packit 6ef888
				 di->dinode.no_addr);
Packit 6ef888
		di->dotdot_parent.no_addr = di->treewalk_parent;
Packit 6ef888
		goto out;
Packit 6ef888
	}
Packit 6ef888
	log_warn( _(".. parent is valid, but treewalk is bad - reattaching to "
Packit 6ef888
		    "lost+found"));
Packit 6ef888
Packit 6ef888
	/* FIXME: add a dinode for this entry instead? */
Packit 6ef888
Packit 6ef888
	if (!query( _("Remove directory entry for bad inode %llu (0x%llx) in "
Packit 6ef888
		      "%llu (0x%llx)? (y/n)"),
Packit 6ef888
		    (unsigned long long)di->dinode.no_addr,
Packit 6ef888
		    (unsigned long long)di->dinode.no_addr,
Packit 6ef888
		    (unsigned long long)di->treewalk_parent,
Packit 6ef888
		    (unsigned long long)di->treewalk_parent)) {
Packit 6ef888
		log_err( _("Directory entry to invalid inode remains\n"));
Packit 6ef888
		return NULL;
Packit 6ef888
	}
Packit 6ef888
	error = remove_dentry_from_dir(sdp, di->treewalk_parent,
Packit 6ef888
				       di->dinode.no_addr);
Packit 6ef888
	if (error < 0) {
Packit 6ef888
		stack;
Packit 6ef888
		return NULL;
Packit 6ef888
	}
Packit 6ef888
	if (error > 0)
Packit 6ef888
		log_warn( _("Unable to find dentry for block %llu"
Packit 6ef888
			    " (0x%llx) in %llu (0x%llx)\n"),
Packit 6ef888
			  (unsigned long long)di->dinode.no_addr,
Packit 6ef888
			  (unsigned long long)di->dinode.no_addr,
Packit 6ef888
			  (unsigned long long)di->treewalk_parent,
Packit 6ef888
			  (unsigned long long)di->treewalk_parent);
Packit 6ef888
	log_warn( _("Directory entry removed\n"));
Packit 6ef888
	log_info( _("Marking directory unlinked\n"));
Packit 6ef888
Packit 6ef888
	return NULL;
Packit 6ef888
Packit 6ef888
out:
Packit 6ef888
	pdi = dirtree_find(di->dotdot_parent.no_addr);
Packit 6ef888
Packit 6ef888
	return pdi;
Packit 6ef888
}
Packit 6ef888
Packit 6ef888
/**
Packit 6ef888
 * pass3 - check connectivity of directories
Packit 6ef888
 *
Packit 6ef888
 * handle disconnected directories
Packit 6ef888
 * handle lost+found directory errors (missing, not a directory, no space)
Packit 6ef888
 */
Packit 6ef888
int pass3(struct gfs2_sbd *sdp)
Packit 6ef888
{
Packit 6ef888
	struct osi_node *tmp, *next = NULL;
Packit 6ef888
	struct dir_info *di, *tdi;
Packit 6ef888
	struct gfs2_inode *ip;
Packit 6ef888
	int q;
Packit 6ef888
Packit 6ef888
	di = dirtree_find(sdp->md.rooti->i_di.di_num.no_addr);
Packit 6ef888
	if (di) {
Packit 6ef888
		log_info( _("Marking root inode connected\n"));
Packit 6ef888
		di->checked = 1;
Packit 6ef888
	}
Packit 6ef888
	if (sdp->gfs1) {
Packit 6ef888
		di = dirtree_find(sdp->md.statfs->i_di.di_num.no_addr);
Packit 6ef888
		if (di) {
Packit 6ef888
			log_info( _("Marking GFS1 statfs file inode "
Packit 6ef888
				    "connected\n"));
Packit 6ef888
			di->checked = 1;
Packit 6ef888
		}
Packit 6ef888
		di = dirtree_find(sdp->md.jiinode->i_di.di_num.no_addr);
Packit 6ef888
		if (di) {
Packit 6ef888
			log_info( _("Marking GFS1 jindex file inode "
Packit 6ef888
				    "connected\n"));
Packit 6ef888
			di->checked = 1;
Packit 6ef888
		}
Packit 6ef888
		di = dirtree_find(sdp->md.riinode->i_di.di_num.no_addr);
Packit 6ef888
		if (di) {
Packit 6ef888
			log_info( _("Marking GFS1 rindex file inode "
Packit 6ef888
				    "connected\n"));
Packit 6ef888
			di->checked = 1;
Packit 6ef888
		}
Packit 6ef888
		di = dirtree_find(sdp->md.qinode->i_di.di_num.no_addr);
Packit 6ef888
		if (di) {
Packit 6ef888
			log_info( _("Marking GFS1 quota file inode "
Packit 6ef888
				    "connected\n"));
Packit 6ef888
			di->checked = 1;
Packit 6ef888
		}
Packit 6ef888
	} else {
Packit 6ef888
		di = dirtree_find(sdp->master_dir->i_di.di_num.no_addr);
Packit 6ef888
		if (di) {
Packit 6ef888
			log_info( _("Marking master directory inode "
Packit 6ef888
				    "connected\n"));
Packit 6ef888
			di->checked = 1;
Packit 6ef888
		}
Packit 6ef888
	}
Packit 6ef888
Packit 6ef888
	/* Go through the directory list, working up through the parents
Packit 6ef888
	 * until we find one that's been checked already.  If we don't
Packit 6ef888
	 * find a parent, put in lost+found.
Packit 6ef888
	 */
Packit 6ef888
	log_info( _("Checking directory linkage.\n"));
Packit 6ef888
	for (tmp = osi_first(&dirtree); tmp; tmp = next) {
Packit 6ef888
		next = osi_next(tmp);
Packit 6ef888
		di = (struct dir_info *)tmp;
Packit 6ef888
		while (!di->checked) {
Packit 6ef888
			/* FIXME: Change this so it returns success or
Packit 6ef888
			 * failure and put the parent inode in a
Packit 6ef888
			 * param */
Packit 6ef888
			if (skip_this_pass || fsck_abort) /* if asked to skip the rest */
Packit 6ef888
				return FSCK_OK;
Packit 6ef888
			tdi = mark_and_return_parent(sdp, di);
Packit 6ef888
Packit 6ef888
			if (tdi) {
Packit 6ef888
				log_debug( _("Directory at block %llu "
Packit 6ef888
					     "(0x%llx) connected\n"),
Packit 6ef888
					   (unsigned long long)di->dinode.no_addr,
Packit 6ef888
					   (unsigned long long)di->dinode.no_addr);
Packit 6ef888
				di = tdi;
Packit 6ef888
				continue;
Packit 6ef888
			}
Packit 6ef888
			q = bitmap_type(sdp, di->dinode.no_addr);
Packit 6ef888
			ip = fsck_load_inode(sdp, di->dinode.no_addr);
Packit 6ef888
			if (q == GFS2_BLKST_FREE) {
Packit 6ef888
				log_err( _("Found unlinked directory "
Packit 6ef888
					   "containing bad block at block %llu"
Packit 6ef888
					   " (0x%llx)\n"),
Packit 6ef888
					(unsigned long long)di->dinode.no_addr,
Packit 6ef888
					(unsigned long long)di->dinode.no_addr);
Packit 6ef888
				if (query(_("Clear unlinked directory "
Packit 6ef888
					   "with bad blocks? (y/n) "))) {
Packit 6ef888
					log_warn( _("inode %lld (0x%llx) is "
Packit 6ef888
						    "now marked as free\n"),
Packit 6ef888
						  (unsigned long long)
Packit 6ef888
						  di->dinode.no_addr,
Packit 6ef888
						  (unsigned long long)
Packit 6ef888
						  di->dinode.no_addr);
Packit 6ef888
					check_n_fix_bitmap(sdp, ip->i_rgd,
Packit 6ef888
							   di->dinode.no_addr,
Packit 6ef888
							   0, GFS2_BLKST_FREE);
Packit 6ef888
					fsck_inode_put(&ip);
Packit 6ef888
					break;
Packit 6ef888
				} else
Packit 6ef888
					log_err( _("Unlinked directory with bad block remains\n"));
Packit 6ef888
			}
Packit 6ef888
			if (q != GFS2_BLKST_DINODE) {
Packit 6ef888
				log_err( _("Unlinked block marked as an inode "
Packit 6ef888
					   "is not an inode\n"));
Packit 6ef888
				if (!query(_("Clear the unlinked block?"
Packit 6ef888
					    " (y/n) "))) {
Packit 6ef888
					log_err( _("The block was not "
Packit 6ef888
						   "cleared\n"));
Packit 6ef888
					fsck_inode_put(&ip);
Packit 6ef888
					break;
Packit 6ef888
				}
Packit 6ef888
				log_warn( _("inode %lld (0x%llx) is now "
Packit 6ef888
					    "marked as free\n"),
Packit 6ef888
					  (unsigned long long)di->dinode.no_addr,
Packit 6ef888
					  (unsigned long long)di->dinode.no_addr);
Packit 6ef888
				check_n_fix_bitmap(sdp, ip->i_rgd,
Packit 6ef888
						   di->dinode.no_addr, 0,
Packit 6ef888
						   GFS2_BLKST_FREE);
Packit 6ef888
				log_err( _("The block was cleared\n"));
Packit 6ef888
				fsck_inode_put(&ip);
Packit 6ef888
				break;
Packit 6ef888
			}
Packit 6ef888
Packit 6ef888
			log_err( _("Found unlinked directory at block %llu"
Packit 6ef888
				   " (0x%llx)\n"),
Packit 6ef888
				 (unsigned long long)di->dinode.no_addr,
Packit 6ef888
				 (unsigned long long)di->dinode.no_addr);
Packit 6ef888
			/* Don't skip zero size directories with eattrs */
Packit 6ef888
			if (!ip->i_di.di_size && !ip->i_di.di_eattr){
Packit 6ef888
				log_err( _("Unlinked directory has zero "
Packit 6ef888
					   "size.\n"));
Packit 6ef888
				if (query( _("Remove zero-size unlinked "
Packit 6ef888
					    "directory? (y/n) "))) {
Packit 6ef888
					fsck_bitmap_set(ip, di->dinode.no_addr,
Packit 6ef888
						_("zero-sized unlinked inode"),
Packit 6ef888
							GFS2_BLKST_FREE);
Packit 6ef888
					fsck_inode_put(&ip);
Packit 6ef888
					break;
Packit 6ef888
				} else {
Packit 6ef888
					log_err( _("Zero-size unlinked "
Packit 6ef888
						   "directory remains\n"));
Packit 6ef888
				}
Packit 6ef888
			}
Packit 6ef888
			if (query( _("Add unlinked directory to "
Packit 6ef888
				    "lost+found? (y/n) "))) {
Packit 6ef888
				if (add_inode_to_lf(ip)) {
Packit 6ef888
					fsck_inode_put(&ip);
Packit 6ef888
					stack;
Packit 6ef888
					return FSCK_ERROR;
Packit 6ef888
				}
Packit 6ef888
				log_warn( _("Directory relinked to lost+found\n"));
Packit 6ef888
			} else {
Packit 6ef888
				log_err( _("Unlinked directory remains unlinked\n"));
Packit 6ef888
			}
Packit 6ef888
			fsck_inode_put(&ip);
Packit 6ef888
			break;
Packit 6ef888
		}
Packit 6ef888
	}
Packit 6ef888
	if (lf_dip) {
Packit 6ef888
		log_debug( _("At end of pass3, lost+found entries is %u\n"),
Packit 6ef888
				  lf_dip->i_di.di_entries);
Packit 6ef888
	}
Packit 6ef888
	return FSCK_OK;
Packit 6ef888
}