iinfo.c 5.63 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
	au_hn_free(hinode);
J. R. Okajima's avatar
J. R. Okajima committed
29
	dput(hinode->hi_whdentry);
J. R. Okajima's avatar
J. R. Okajima committed
30
31
32
	iput(hinode->hi_inode);
}

J. R. Okajima's avatar
J. R. Okajima committed
33
34
35
36
37
38
39
40
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);
J. R. Okajima's avatar
J. R. Okajima committed
41
42
	if (isdir && au_opt_test(mnt_flags, UDBA_HNOTIFY))
		au_fset_hi(flags, HNOTIFY);
J. R. Okajima's avatar
J. R. Okajima committed
43
44
45
	return flags;
}

J. R. Okajima's avatar
J. R. Okajima committed
46
47
48
49
50
51
52
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
53
54
	IiMustWriteLock(inode);

J. R. Okajima's avatar
J. R. Okajima committed
55
56
57
58
59
60
61
62
	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
63
		int err;
64
65
66
		struct super_block *sb = inode->i_sb;
		struct au_branch *br;

J. R. Okajima's avatar
J. R. Okajima committed
67
68
69
		AuDebugOn(inode->i_mode
			  && (h_inode->i_mode & S_IFMT)
			  != (inode->i_mode & S_IFMT));
J. R. Okajima's avatar
J. R. Okajima committed
70
71
		if (bindex == iinfo->ii_btop)
			au_cpup_igen(inode, h_inode);
72
73
		br = au_sbr(sb, bindex);
		hinode->hi_id = br->br_id;
J. R. Okajima's avatar
J. R. Okajima committed
74
75
76
77
78
79
		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
80
81
82
83
84
85
86

		if (au_ftest_hi(flags, HNOTIFY)
		    && au_br_hnotifyable(br->br_perm)) {
			err = au_hn_alloc(hinode, inode);
			if (unlikely(err))
				AuIOErr1("au_hn_alloc() %d\n", err);
		}
J. R. Okajima's avatar
J. R. Okajima committed
87
88
89
	}
}

J. R. Okajima's avatar
J. R. Okajima committed
90
91
92
93
94
95
96
97
98
99
100
101
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
102
103
104
105
106
107
108
109
110
111
112
113
114
115
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
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
146
147
148
149
150
151
152
153
154
155
/* 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
156
157
158
159
160
161
162
/* ---------------------------------------------------------------------- */

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
163
	spin_lock_init(&iinfo->ii_generation.ig_spin);
J. R. Okajima's avatar
J. R. Okajima committed
164
	au_rw_init(&iinfo->ii_rwsem);
J. R. Okajima's avatar
J. R. Okajima committed
165
166
167
168
169
170
	inode_init_once(&c->vfs_inode);
}

void au_hinode_init(struct au_hinode *hinode)
{
	hinode->hi_inode = NULL;
171
	hinode->hi_id = -1;
J. R. Okajima's avatar
J. R. Okajima committed
172
	au_hn_init(hinode);
J. R. Okajima's avatar
J. R. Okajima committed
173
	hinode->hi_whdentry = NULL;
J. R. Okajima's avatar
J. R. Okajima committed
174
175
176
177
178
179
180
181
182
183
184
}

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);
185
186
187
	nbr = au_sbbot(sb) + 1;
	if (unlikely(nbr <= 0))
		nbr = 1;
J. R. Okajima's avatar
J. R. Okajima committed
188
189
190
191
192
193
	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
194
		iinfo->ii_generation.ig_generation = au_sigen(sb);
J. R. Okajima's avatar
J. R. Okajima committed
195
196
197
198
199
200
201
		iinfo->ii_btop = -1;
		iinfo->ii_bbot = -1;
		return 0;
	}
	return -ENOMEM;
}

202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
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
224
225
226
227
void au_iinfo_fin(struct inode *inode)
{
	struct au_iinfo *iinfo;
	struct au_hinode *hi;
228
	struct super_block *sb;
J. R. Okajima's avatar
J. R. Okajima committed
229
	aufs_bindex_t bindex, bbot;
J. R. Okajima's avatar
J. R. Okajima committed
230
	const unsigned char unlinked = !inode->i_nlink;
J. R. Okajima's avatar
J. R. Okajima committed
231
232
233

	AuDebugOn(au_is_bad_inode(inode));

234
235
236
237
238
239
240
241
242
243
244
245
246
247
	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
248

J. R. Okajima's avatar
J. R. Okajima committed
249
250
251
252
253
254
255
256
257
258
259
	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++;
		}
	}
260
	au_kfree_rcu(iinfo->ii_hinode);
J. R. Okajima's avatar
J. R. Okajima committed
261
	AuRwDestroy(&iinfo->ii_rwsem);
J. R. Okajima's avatar
J. R. Okajima committed
262
}