lin
2025-04-25 6a7002bcc41716f11f4ca7eb68ebd06c18fdd5e8
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
/*
 * quota.c --- code for handling ext4 quota inodes
 *
 */
 
#include "config.h"
#ifdef HAVE_SYS_MOUNT_H
#include <sys/param.h>
#include <sys/mount.h>
#define MNT_FL (MS_MGC_VAL | MS_RDONLY)
#endif
#ifdef HAVE_SYS_STAT_H
#include <sys/stat.h>
#endif
 
#include "e2fsck.h"
#include "problem.h"
 
static errcode_t move_quota_inode(ext2_filsys fs, ext2_ino_t from_ino,
                 ext2_ino_t to_ino, enum quota_type qtype)
{
   struct ext2_inode    inode;
   errcode_t        retval;
   char            qf_name[QUOTA_NAME_LEN];
 
   /* We need the inode bitmap to be loaded */
   retval = ext2fs_read_bitmaps(fs);
   if (retval) {
       com_err("ext2fs_read_bitmaps", retval, "%s",
           _("in move_quota_inode"));
       return retval;
   }
 
   retval = ext2fs_read_inode(fs, from_ino, &inode);
   if (retval) {
       com_err("ext2fs_read_inode", retval, "%s",
           _("in move_quota_inode"));
       return retval;
   }
 
   inode.i_links_count = 1;
   inode.i_mode = LINUX_S_IFREG | 0600;
   inode.i_flags = EXT2_IMMUTABLE_FL;
   if (ext2fs_has_feature_extents(fs->super))
       inode.i_flags |= EXT4_EXTENTS_FL;
 
   retval = ext2fs_write_new_inode(fs, to_ino, &inode);
   if (retval) {
       com_err("ext2fs_write_new_inode", retval, "%s",
           _("in move_quota_inode"));
       return retval;
   }
 
   /* unlink the old inode */
   quota_get_qf_name(qtype, QFMT_VFS_V1, qf_name);
   retval = ext2fs_unlink(fs, EXT2_ROOT_INO, qf_name, from_ino, 0);
   if (retval) {
       com_err("ext2fs_unlink", retval, "%s",
           _("in move_quota_inode"));
       return retval;
   }
   ext2fs_inode_alloc_stats(fs, from_ino, -1);
   /* Clear out the original inode in the inode-table block. */
   memset(&inode, 0, sizeof(struct ext2_inode));
   ext2fs_write_inode(fs, from_ino, &inode);
   return 0;
}
 
void e2fsck_hide_quota(e2fsck_t ctx)
{
   struct ext2_super_block *sb = ctx->fs->super;
   struct problem_context    pctx;
   ext2_filsys        fs = ctx->fs;
   enum quota_type qtype;
   ext2_ino_t quota_ino;
 
   clear_problem_context(&pctx);
 
   if ((ctx->options & E2F_OPT_READONLY) ||
       !ext2fs_has_feature_quota(sb))
       return;
 
   for (qtype = 0; qtype < MAXQUOTAS; qtype++) {
       pctx.dir = 2;    /* This is a guess, but it's a good one */
       pctx.ino = *quota_sb_inump(sb, qtype);
       pctx.num = qtype;
       quota_ino = quota_type2inum(qtype, fs->super);
       if (pctx.ino && (pctx.ino != quota_ino) &&
           fix_problem(ctx, PR_0_HIDE_QUOTA, &pctx)) {
           if (move_quota_inode(fs, pctx.ino, quota_ino, qtype))
               continue;
           *quota_sb_inump(sb, qtype) = quota_ino;
           ext2fs_mark_super_dirty(fs);
       }
   }
 
   return;
}
 
void e2fsck_validate_quota_inodes(e2fsck_t ctx)
{
   struct ext2_super_block *sb = ctx->fs->super;
   struct problem_context    pctx;
   ext2_filsys        fs = ctx->fs;
   enum quota_type qtype;
 
   clear_problem_context(&pctx);
 
   for (qtype = 0; qtype < MAXQUOTAS; qtype++) {
       pctx.ino = *quota_sb_inump(sb, qtype);
       pctx.num = qtype;
       if (pctx.ino &&
           ((pctx.ino == EXT2_BAD_INO) ||
            (pctx.ino == EXT2_ROOT_INO) ||
            (pctx.ino == EXT2_BOOT_LOADER_INO) ||
            (pctx.ino == EXT2_UNDEL_DIR_INO) ||
            (pctx.ino == EXT2_RESIZE_INO) ||
            (pctx.ino == EXT2_JOURNAL_INO) ||
            (pctx.ino == EXT2_EXCLUDE_INO) ||
            (pctx.ino == EXT4_REPLICA_INO) ||
            (pctx.ino > fs->super->s_inodes_count)) &&
           fix_problem(ctx, PR_0_INVALID_QUOTA_INO, &pctx)) {
           *quota_sb_inump(sb, qtype) = 0;
           ext2fs_mark_super_dirty(fs);
       }
   }
}