iinfo.c 5.33 KB
Newer Older
J. R. Okajima's avatar
J. R. Okajima committed
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
// SPDX-License-Identifier: GPL-2.0
/*
 * Copyright (C) 2005-2019 Junjiro R. Okajima
 */

/*
 * inode private data
 */

#include "aufs.h"

struct inode *au_h_iptr(struct inode *inode, aufs_bindex_t bindex)
{
	struct inode *h_inode;
	struct au_hinode *hinode;

J. R. Okajima's avatar
J. R. Okajima committed
17
18
	IiMustAnyLock(inode);

J. R. Okajima's avatar
J. R. Okajima committed
19
20
21
22
23
24
25
26
27
	hinode = au_hinode(au_ii(inode), bindex);
	h_inode = hinode->hi_inode;
	AuDebugOn(h_inode && atomic_read(&h_inode->i_count) <= 0);
	return h_inode;
}

/* todo: hard/soft set? */
void au_hiput(struct au_hinode *hinode)
{
J. R. Okajima's avatar
J. R. Okajima committed
28
	dput(hinode->hi_whdentry);
J. R. Okajima's avatar
J. R. Okajima committed
29
30
31
	iput(hinode->hi_inode);
}

J. R. Okajima's avatar
J. R. Okajima committed
32
33
34
35
36
37
38
39
40
41
42
unsigned int au_hi_flags(struct inode *inode, int isdir)
{
	unsigned int flags;
	const unsigned int mnt_flags = au_mntflags(inode->i_sb);

	flags = 0;
	if (au_opt_test(mnt_flags, XINO))
		au_fset_hi(flags, XINO);
	return flags;
}

J. R. Okajima's avatar
J. R. Okajima committed
43
44
45
46
47
48
49
void au_set_h_iptr(struct inode *inode, aufs_bindex_t bindex,
		   struct inode *h_inode, unsigned int flags)
{
	struct au_hinode *hinode;
	struct inode *hi;
	struct au_iinfo *iinfo = au_ii(inode);

J. R. Okajima's avatar
J. R. Okajima committed
50
51
	IiMustWriteLock(inode);

J. R. Okajima's avatar
J. R. Okajima committed
52
53
54
55
56
57
58
59
	hinode = au_hinode(iinfo, bindex);
	hi = hinode->hi_inode;
	AuDebugOn(h_inode && atomic_read(&h_inode->i_count) <= 0);

	if (hi)
		au_hiput(hinode);
	hinode->hi_inode = h_inode;
	if (h_inode) {
J. R. Okajima's avatar
J. R. Okajima committed
60
		int err;
61
62
63
		struct super_block *sb = inode->i_sb;
		struct au_branch *br;

J. R. Okajima's avatar
J. R. Okajima committed
64
65
66
		AuDebugOn(inode->i_mode
			  && (h_inode->i_mode & S_IFMT)
			  != (inode->i_mode & S_IFMT));
J. R. Okajima's avatar
J. R. Okajima committed
67
68
		if (bindex == iinfo->ii_btop)
			au_cpup_igen(inode, h_inode);
69
70
		br = au_sbr(sb, bindex);
		hinode->hi_id = br->br_id;
J. R. Okajima's avatar
J. R. Okajima committed
71
72
73
74
75
76
		if (au_ftest_hi(flags, XINO)) {
			err = au_xino_write(sb, bindex, h_inode->i_ino,
					    inode->i_ino);
			if (unlikely(err))
				AuIOErr1("failed au_xino_write() %d\n", err);
		}
J. R. Okajima's avatar
J. R. Okajima committed
77
78
79
	}
}

J. R. Okajima's avatar
J. R. Okajima committed
80
81
82
83
84
85
86
87
88
89
90
91
void au_set_hi_wh(struct inode *inode, aufs_bindex_t bindex,
		  struct dentry *h_wh)
{
	struct au_hinode *hinode;

	IiMustWriteLock(inode);

	hinode = au_hinode(au_ii(inode), bindex);
	AuDebugOn(hinode->hi_whdentry);
	hinode->hi_whdentry = h_wh;
}

J. R. Okajima's avatar
J. R. Okajima committed
92
93
94
95
96
97
98
99
100
101
102
103
104
105
void au_update_iigen(struct inode *inode, int half)
{
	struct au_iinfo *iinfo;
	struct au_iigen *iigen;
	unsigned int sigen;

	sigen = au_sigen(inode->i_sb);
	iinfo = au_ii(inode);
	iigen = &iinfo->ii_generation;
	spin_lock(&iigen->ig_spin);
	iigen->ig_generation = sigen;
	spin_unlock(&iigen->ig_spin);
}

J. R. Okajima's avatar
J. R. Okajima committed
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
/* it may be called at remount time, too */
void au_update_ibrange(struct inode *inode, int do_put_zero)
{
	struct au_iinfo *iinfo;
	aufs_bindex_t bindex, bbot;

	AuDebugOn(au_is_bad_inode(inode));
	IiMustWriteLock(inode);

	iinfo = au_ii(inode);
	if (do_put_zero && iinfo->ii_btop >= 0) {
		for (bindex = iinfo->ii_btop; bindex <= iinfo->ii_bbot;
		     bindex++) {
			struct inode *h_i;

			h_i = au_hinode(iinfo, bindex)->hi_inode;
			if (h_i
			    && !h_i->i_nlink
			    && !(h_i->i_state & I_LINKABLE))
				au_set_h_iptr(inode, bindex, NULL, 0);
		}
	}

	iinfo->ii_btop = -1;
	iinfo->ii_bbot = -1;
	bbot = au_sbbot(inode->i_sb);
	for (bindex = 0; bindex <= bbot; bindex++)
		if (au_hinode(iinfo, bindex)->hi_inode) {
			iinfo->ii_btop = bindex;
			break;
		}
	if (iinfo->ii_btop >= 0)
		for (bindex = bbot; bindex >= iinfo->ii_btop; bindex--)
			if (au_hinode(iinfo, bindex)->hi_inode) {
				iinfo->ii_bbot = bindex;
				break;
			}
	AuDebugOn(iinfo->ii_btop > iinfo->ii_bbot);
}

J. R. Okajima's avatar
J. R. Okajima committed
146
147
148
149
150
151
152
/* ---------------------------------------------------------------------- */

void au_icntnr_init_once(void *_c)
{
	struct au_icntnr *c = _c;
	struct au_iinfo *iinfo = &c->iinfo;

J. R. Okajima's avatar
J. R. Okajima committed
153
	spin_lock_init(&iinfo->ii_generation.ig_spin);
J. R. Okajima's avatar
J. R. Okajima committed
154
	au_rw_init(&iinfo->ii_rwsem);
J. R. Okajima's avatar
J. R. Okajima committed
155
156
157
158
159
160
	inode_init_once(&c->vfs_inode);
}

void au_hinode_init(struct au_hinode *hinode)
{
	hinode->hi_inode = NULL;
161
	hinode->hi_id = -1;
J. R. Okajima's avatar
J. R. Okajima committed
162
	hinode->hi_whdentry = NULL;
J. R. Okajima's avatar
J. R. Okajima committed
163
164
165
166
167
168
169
170
171
172
173
}

int au_iinfo_init(struct inode *inode)
{
	struct au_iinfo *iinfo;
	struct super_block *sb;
	struct au_hinode *hi;
	int nbr, i;

	sb = inode->i_sb;
	iinfo = &(container_of(inode, struct au_icntnr, vfs_inode)->iinfo);
174
175
176
	nbr = au_sbbot(sb) + 1;
	if (unlikely(nbr <= 0))
		nbr = 1;
J. R. Okajima's avatar
J. R. Okajima committed
177
178
179
180
181
182
	hi = kmalloc_array(nbr, sizeof(*iinfo->ii_hinode), GFP_NOFS);
	if (hi) {
		iinfo->ii_hinode = hi;
		for (i = 0; i < nbr; i++, hi++)
			au_hinode_init(hi);

J. R. Okajima's avatar
J. R. Okajima committed
183
		iinfo->ii_generation.ig_generation = au_sigen(sb);
J. R. Okajima's avatar
J. R. Okajima committed
184
185
186
187
188
189
190
		iinfo->ii_btop = -1;
		iinfo->ii_bbot = -1;
		return 0;
	}
	return -ENOMEM;
}

191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
int au_hinode_realloc(struct au_iinfo *iinfo, int nbr, int may_shrink)
{
	int err, i;
	struct au_hinode *hip;

	AuRwMustWriteLock(&iinfo->ii_rwsem);

	err = -ENOMEM;
	hip = au_krealloc(iinfo->ii_hinode, sizeof(*hip) * nbr, GFP_NOFS,
			  may_shrink);
	if (hip) {
		iinfo->ii_hinode = hip;
		i = iinfo->ii_bbot + 1;
		hip += i;
		for (; i < nbr; i++, hip++)
			au_hinode_init(hip);
		err = 0;
	}

	return err;
}

J. R. Okajima's avatar
J. R. Okajima committed
213
214
215
216
void au_iinfo_fin(struct inode *inode)
{
	struct au_iinfo *iinfo;
	struct au_hinode *hi;
217
	struct super_block *sb;
J. R. Okajima's avatar
J. R. Okajima committed
218
	aufs_bindex_t bindex, bbot;
J. R. Okajima's avatar
J. R. Okajima committed
219
	const unsigned char unlinked = !inode->i_nlink;
J. R. Okajima's avatar
J. R. Okajima committed
220
221
222

	AuDebugOn(au_is_bad_inode(inode));

223
224
225
226
227
228
229
230
231
232
233
234
235
236
	sb = inode->i_sb;
	if (si_pid_test(sb))
		au_xino_delete_inode(inode, unlinked);
	else {
		/*
		 * it is safe to hide the dependency between sbinfo and
		 * sb->s_umount.
		 */
		lockdep_off();
		si_noflush_read_lock(sb);
		au_xino_delete_inode(inode, unlinked);
		si_read_unlock(sb);
		lockdep_on();
	}
J. R. Okajima's avatar
J. R. Okajima committed
237

J. R. Okajima's avatar
J. R. Okajima committed
238
239
240
241
242
243
244
245
246
247
248
	iinfo = au_ii(inode);
	bindex = iinfo->ii_btop;
	if (bindex >= 0) {
		hi = au_hinode(iinfo, bindex);
		bbot = iinfo->ii_bbot;
		while (bindex++ <= bbot) {
			if (hi->hi_inode)
				au_hiput(hi);
			hi++;
		}
	}
249
	au_kfree_rcu(iinfo->ii_hinode);
J. R. Okajima's avatar
J. R. Okajima committed
250
	AuRwDestroy(&iinfo->ii_rwsem);
J. R. Okajima's avatar
J. R. Okajima committed
251
}