cregit-Linux how code gets into the kernel

Release 4.15 fs/xfs/scrub/quota.c

Directory: fs/xfs/scrub
/*
 * Copyright (C) 2017 Oracle.  All Rights Reserved.
 *
 * Author: Darrick J. Wong <darrick.wong@oracle.com>
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License
 * as published by the Free Software Foundation; either version 2
 * of the License, or (at your option) any later version.
 *
 * This program is distributed in the hope that it would be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write the Free Software Foundation,
 * Inc.,  51 Franklin St, Fifth Floor, Boston, MA  02110-1301, USA.
 */
#include "xfs.h"
#include "xfs_fs.h"
#include "xfs_shared.h"
#include "xfs_format.h"
#include "xfs_trans_resv.h"
#include "xfs_mount.h"
#include "xfs_defer.h"
#include "xfs_btree.h"
#include "xfs_bit.h"
#include "xfs_log_format.h"
#include "xfs_trans.h"
#include "xfs_sb.h"
#include "xfs_inode.h"
#include "xfs_inode_fork.h"
#include "xfs_alloc.h"
#include "xfs_bmap.h"
#include "xfs_quota.h"
#include "xfs_qm.h"
#include "xfs_dquot.h"
#include "xfs_dquot_item.h"
#include "scrub/xfs_scrub.h"
#include "scrub/scrub.h"
#include "scrub/common.h"
#include "scrub/trace.h"

/* Convert a scrub type code to a DQ flag, or return 0 if error. */

static inline uint xfs_scrub_quota_to_dqtype( struct xfs_scrub_context *sc) { switch (sc->sm->sm_type) { case XFS_SCRUB_TYPE_UQUOTA: return XFS_DQ_USER; case XFS_SCRUB_TYPE_GQUOTA: return XFS_DQ_GROUP; case XFS_SCRUB_TYPE_PQUOTA: return XFS_DQ_PROJ; default: return 0; } }

Contributors

PersonTokensPropCommitsCommitProp
Darrick J. Wong44100.00%1100.00%
Total44100.00%1100.00%

/* Set us up to scrub a quota. */
int xfs_scrub_setup_quota( struct xfs_scrub_context *sc, struct xfs_inode *ip) { uint dqtype; /* * If userspace gave us an AG number or inode data, they don't * know what they're doing. Get out. */ if (sc->sm->sm_agno || sc->sm->sm_ino || sc->sm->sm_gen) return -EINVAL; dqtype = xfs_scrub_quota_to_dqtype(sc); if (dqtype == 0) return -EINVAL; if (!xfs_this_quota_on(sc->mp, dqtype)) return -ENOENT; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Darrick J. Wong79100.00%1100.00%
Total79100.00%1100.00%

/* Quotas. */ /* Scrub the fields in an individual quota item. */
STATIC void xfs_scrub_quota_item( struct xfs_scrub_context *sc, uint dqtype, struct xfs_dquot *dq, xfs_dqid_t id) { struct xfs_mount *mp = sc->mp; struct xfs_disk_dquot *d = &dq->q_core; struct xfs_quotainfo *qi = mp->m_quotainfo; xfs_fileoff_t offset; unsigned long long bsoft; unsigned long long isoft; unsigned long long rsoft; unsigned long long bhard; unsigned long long ihard; unsigned long long rhard; unsigned long long bcount; unsigned long long icount; unsigned long long rcount; xfs_ino_t fs_icount; offset = id / qi->qi_dqperchunk; /* * We fed $id and DQNEXT into the xfs_qm_dqget call, which means * that the actual dquot we got must either have the same id or * the next higher id. */ if (id > be32_to_cpu(d->d_id)) xfs_scrub_fblock_set_corrupt(sc, XFS_DATA_FORK, offset); /* Did we get the dquot type we wanted? */ if (dqtype != (d->d_flags & XFS_DQ_ALLTYPES)) xfs_scrub_fblock_set_corrupt(sc, XFS_DATA_FORK, offset); if (d->d_pad0 != cpu_to_be32(0) || d->d_pad != cpu_to_be16(0)) xfs_scrub_fblock_set_corrupt(sc, XFS_DATA_FORK, offset); /* Check the limits. */ bhard = be64_to_cpu(d->d_blk_hardlimit); ihard = be64_to_cpu(d->d_ino_hardlimit); rhard = be64_to_cpu(d->d_rtb_hardlimit); bsoft = be64_to_cpu(d->d_blk_softlimit); isoft = be64_to_cpu(d->d_ino_softlimit); rsoft = be64_to_cpu(d->d_rtb_softlimit); /* * Warn if the hard limits are larger than the fs. * Administrators can do this, though in production this seems * suspect, which is why we flag it for review. * * Complain about corruption if the soft limit is greater than * the hard limit. */ if (bhard > mp->m_sb.sb_dblocks) xfs_scrub_fblock_set_warning(sc, XFS_DATA_FORK, offset); if (bsoft > bhard) xfs_scrub_fblock_set_corrupt(sc, XFS_DATA_FORK, offset); if (ihard > mp->m_maxicount) xfs_scrub_fblock_set_warning(sc, XFS_DATA_FORK, offset); if (isoft > ihard) xfs_scrub_fblock_set_corrupt(sc, XFS_DATA_FORK, offset); if (rhard > mp->m_sb.sb_rblocks) xfs_scrub_fblock_set_warning(sc, XFS_DATA_FORK, offset); if (rsoft > rhard) xfs_scrub_fblock_set_corrupt(sc, XFS_DATA_FORK, offset); /* Check the resource counts. */ bcount = be64_to_cpu(d->d_bcount); icount = be64_to_cpu(d->d_icount); rcount = be64_to_cpu(d->d_rtbcount); fs_icount = percpu_counter_sum(&mp->m_icount); /* * Check that usage doesn't exceed physical limits. However, on * a reflink filesystem we're allowed to exceed physical space * if there are no quota limits. */ if (xfs_sb_version_hasreflink(&mp->m_sb)) { if (mp->m_sb.sb_dblocks < bcount) xfs_scrub_fblock_set_warning(sc, XFS_DATA_FORK, offset); } else { if (mp->m_sb.sb_dblocks < bcount) xfs_scrub_fblock_set_corrupt(sc, XFS_DATA_FORK, offset); } if (icount > fs_icount || rcount > mp->m_sb.sb_rblocks) xfs_scrub_fblock_set_corrupt(sc, XFS_DATA_FORK, offset); /* * We can violate the hard limits if the admin suddenly sets a * lower limit than the actual usage. However, we flag it for * admin review. */ if (id != 0 && bhard != 0 && bcount > bhard) xfs_scrub_fblock_set_warning(sc, XFS_DATA_FORK, offset); if (id != 0 && ihard != 0 && icount > ihard) xfs_scrub_fblock_set_warning(sc, XFS_DATA_FORK, offset); if (id != 0 && rhard != 0 && rcount > rhard) xfs_scrub_fblock_set_warning(sc, XFS_DATA_FORK, offset); }

Contributors

PersonTokensPropCommitsCommitProp
Darrick J. Wong52199.81%150.00%
Eric Sandeen10.19%150.00%
Total522100.00%2100.00%

/* Scrub all of a quota type's items. */
int xfs_scrub_quota( struct xfs_scrub_context *sc) { struct xfs_bmbt_irec irec = { 0 }; struct xfs_mount *mp = sc->mp; struct xfs_inode *ip; struct xfs_quotainfo *qi = mp->m_quotainfo; struct xfs_dquot *dq; xfs_fileoff_t max_dqid_off; xfs_fileoff_t off = 0; xfs_dqid_t id = 0; uint dqtype; int nimaps; int error = 0; if (!XFS_IS_QUOTA_RUNNING(mp) || !XFS_IS_QUOTA_ON(mp)) return -ENOENT; mutex_lock(&qi->qi_quotaofflock); dqtype = xfs_scrub_quota_to_dqtype(sc); if (!xfs_this_quota_on(sc->mp, dqtype)) { error = -ENOENT; goto out_unlock_quota; } /* Attach to the quota inode and set sc->ip so that reporting works. */ ip = xfs_quota_inode(sc->mp, dqtype); sc->ip = ip; /* Look for problem extents. */ xfs_ilock(ip, XFS_ILOCK_EXCL); if (ip->i_d.di_flags & XFS_DIFLAG_REALTIME) { xfs_scrub_ino_set_corrupt(sc, sc->ip->i_ino, NULL); goto out_unlock_inode; } max_dqid_off = ((xfs_dqid_t)-1) / qi->qi_dqperchunk; while (1) { if (xfs_scrub_should_terminate(sc, &error)) break; off = irec.br_startoff + irec.br_blockcount; nimaps = 1; error = xfs_bmapi_read(ip, off, -1, &irec, &nimaps, XFS_BMAPI_ENTIRE); if (!xfs_scrub_fblock_process_error(sc, XFS_DATA_FORK, off, &error)) goto out_unlock_inode; if (!nimaps) break; if (irec.br_startblock == HOLESTARTBLOCK) continue; /* Check the extent record doesn't point to crap. */ if (irec.br_startblock + irec.br_blockcount <= irec.br_startblock) xfs_scrub_fblock_set_corrupt(sc, XFS_DATA_FORK, irec.br_startoff); if (!xfs_verify_fsbno(mp, irec.br_startblock) || !xfs_verify_fsbno(mp, irec.br_startblock + irec.br_blockcount - 1)) xfs_scrub_fblock_set_corrupt(sc, XFS_DATA_FORK, irec.br_startoff); /* * Unwritten extents or blocks mapped above the highest * quota id shouldn't happen. */ if (isnullstartblock(irec.br_startblock) || irec.br_startoff > max_dqid_off || irec.br_startoff + irec.br_blockcount > max_dqid_off + 1) xfs_scrub_fblock_set_corrupt(sc, XFS_DATA_FORK, off); } xfs_iunlock(ip, XFS_ILOCK_EXCL); if (sc->sm->sm_flags & XFS_SCRUB_OFLAG_CORRUPT) goto out; /* Check all the quota items. */ while (id < ((xfs_dqid_t)-1ULL)) { if (xfs_scrub_should_terminate(sc, &error)) break; error = xfs_qm_dqget(mp, NULL, id, dqtype, XFS_QMOPT_DQNEXT, &dq); if (error == -ENOENT) break; if (!xfs_scrub_fblock_process_error(sc, XFS_DATA_FORK, id * qi->qi_dqperchunk, &error)) break; xfs_scrub_quota_item(sc, dqtype, dq, id); id = be32_to_cpu(dq->q_core.d_id) + 1; xfs_qm_dqput(dq); if (!id) break; } out: /* We set sc->ip earlier, so make sure we clear it now. */ sc->ip = NULL; out_unlock_quota: mutex_unlock(&qi->qi_quotaofflock); return error; out_unlock_inode: xfs_iunlock(ip, XFS_ILOCK_EXCL); goto out; }

Contributors

PersonTokensPropCommitsCommitProp
Darrick J. Wong53899.63%150.00%
Eric Sandeen20.37%150.00%
Total540100.00%2100.00%


Overall Contributors

PersonTokensPropCommitsCommitProp
Darrick J. Wong126099.76%133.33%
Eric Sandeen30.24%266.67%
Total1263100.00%3100.00%
Directory: fs/xfs/scrub
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.