]>
Commit | Line | Data |
---|---|---|
a1d312de | 1 | // SPDX-License-Identifier: GPL-2.0-or-later |
1da177e4 LT |
2 | /* |
3 | * quota.c - NTFS kernel quota ($Quota) handling. Part of the Linux-NTFS | |
4 | * project. | |
5 | * | |
6 | * Copyright (c) 2004 Anton Altaparmakov | |
1da177e4 LT |
7 | */ |
8 | ||
9 | #ifdef NTFS_RW | |
10 | ||
11 | #include "index.h" | |
12 | #include "quota.h" | |
13 | #include "debug.h" | |
14 | #include "ntfs.h" | |
15 | ||
16 | /** | |
17 | * ntfs_mark_quotas_out_of_date - mark the quotas out of date on an ntfs volume | |
18 | * @vol: ntfs volume on which to mark the quotas out of date | |
19 | * | |
c49c3111 RK |
20 | * Mark the quotas out of date on the ntfs volume @vol and return 'true' on |
21 | * success and 'false' on error. | |
1da177e4 | 22 | */ |
c49c3111 | 23 | bool ntfs_mark_quotas_out_of_date(ntfs_volume *vol) |
1da177e4 LT |
24 | { |
25 | ntfs_index_context *ictx; | |
26 | QUOTA_CONTROL_ENTRY *qce; | |
27 | const le32 qid = QUOTA_DEFAULTS_ID; | |
28 | int err; | |
29 | ||
30 | ntfs_debug("Entering."); | |
31 | if (NVolQuotaOutOfDate(vol)) | |
32 | goto done; | |
33 | if (!vol->quota_ino || !vol->quota_q_ino) { | |
34 | ntfs_error(vol->sb, "Quota inodes are not open."); | |
c49c3111 | 35 | return false; |
1da177e4 | 36 | } |
5955102c | 37 | inode_lock(vol->quota_q_ino); |
1da177e4 LT |
38 | ictx = ntfs_index_ctx_get(NTFS_I(vol->quota_q_ino)); |
39 | if (!ictx) { | |
40 | ntfs_error(vol->sb, "Failed to get index context."); | |
41 | goto err_out; | |
42 | } | |
43 | err = ntfs_index_lookup(&qid, sizeof(qid), ictx); | |
44 | if (err) { | |
45 | if (err == -ENOENT) | |
46 | ntfs_error(vol->sb, "Quota defaults entry is not " | |
47 | "present."); | |
48 | else | |
49 | ntfs_error(vol->sb, "Lookup of quota defaults entry " | |
50 | "failed."); | |
51 | goto err_out; | |
52 | } | |
53 | if (ictx->data_len < offsetof(QUOTA_CONTROL_ENTRY, sid)) { | |
54 | ntfs_error(vol->sb, "Quota defaults entry size is invalid. " | |
55 | "Run chkdsk."); | |
56 | goto err_out; | |
57 | } | |
58 | qce = (QUOTA_CONTROL_ENTRY*)ictx->data; | |
59 | if (le32_to_cpu(qce->version) != QUOTA_VERSION) { | |
60 | ntfs_error(vol->sb, "Quota defaults entry version 0x%x is not " | |
61 | "supported.", le32_to_cpu(qce->version)); | |
62 | goto err_out; | |
63 | } | |
64 | ntfs_debug("Quota defaults flags = 0x%x.", le32_to_cpu(qce->flags)); | |
65 | /* If quotas are already marked out of date, no need to do anything. */ | |
66 | if (qce->flags & QUOTA_FLAG_OUT_OF_DATE) | |
67 | goto set_done; | |
68 | /* | |
69 | * If quota tracking is neither requested, nor enabled and there are no | |
70 | * pending deletes, no need to mark the quotas out of date. | |
71 | */ | |
72 | if (!(qce->flags & (QUOTA_FLAG_TRACKING_ENABLED | | |
73 | QUOTA_FLAG_TRACKING_REQUESTED | | |
74 | QUOTA_FLAG_PENDING_DELETES))) | |
75 | goto set_done; | |
76 | /* | |
77 | * Set the QUOTA_FLAG_OUT_OF_DATE bit thus marking quotas out of date. | |
78 | * This is verified on WinXP to be sufficient to cause windows to | |
79 | * rescan the volume on boot and update all quota entries. | |
80 | */ | |
81 | qce->flags |= QUOTA_FLAG_OUT_OF_DATE; | |
82 | /* Ensure the modified flags are written to disk. */ | |
83 | ntfs_index_entry_flush_dcache_page(ictx); | |
84 | ntfs_index_entry_mark_dirty(ictx); | |
85 | set_done: | |
86 | ntfs_index_ctx_put(ictx); | |
5955102c | 87 | inode_unlock(vol->quota_q_ino); |
1da177e4 LT |
88 | /* |
89 | * We set the flag so we do not try to mark the quotas out of date | |
90 | * again on remount. | |
91 | */ | |
92 | NVolSetQuotaOutOfDate(vol); | |
93 | done: | |
94 | ntfs_debug("Done."); | |
c49c3111 | 95 | return true; |
1da177e4 LT |
96 | err_out: |
97 | if (ictx) | |
98 | ntfs_index_ctx_put(ictx); | |
5955102c | 99 | inode_unlock(vol->quota_q_ino); |
c49c3111 | 100 | return false; |
1da177e4 LT |
101 | } |
102 | ||
103 | #endif /* NTFS_RW */ |