iinfo.c 5.03 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
28
29
30
	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)
{
	iput(hinode->hi_inode);
}

J. R. Okajima's avatar
J. R. Okajima committed
31
32
33
34
35
36
37
38
39
40
41
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
42
43
44
45
46
47
48
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
49
50
	IiMustWriteLock(inode);

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

J. R. Okajima's avatar
J. R. Okajima committed
63
64
65
		AuDebugOn(inode->i_mode
			  && (h_inode->i_mode & S_IFMT)
			  != (inode->i_mode & S_IFMT));
J. R. Okajima's avatar
J. R. Okajima committed
66
67
		if (bindex == iinfo->ii_btop)
			au_cpup_igen(inode, h_inode);
68
69
		br = au_sbr(sb, bindex);
		hinode->hi_id = br->br_id;
J. R. Okajima's avatar
J. R. Okajima committed
70
71
72
73
74
75
		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
76
77
78
	}
}

J. R. Okajima's avatar
J. R. Okajima committed
79
80
81
82
83
84
85
86
87
88
89
90
91
92
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
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
128
129
130
131
132
/* 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
133
134
135
136
137
138
139
/* ---------------------------------------------------------------------- */

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
140
	spin_lock_init(&iinfo->ii_generation.ig_spin);
J. R. Okajima's avatar
J. R. Okajima committed
141
	au_rw_init(&iinfo->ii_rwsem);
J. R. Okajima's avatar
J. R. Okajima committed
142
143
144
145
146
147
	inode_init_once(&c->vfs_inode);
}

void au_hinode_init(struct au_hinode *hinode)
{
	hinode->hi_inode = NULL;
148
	hinode->hi_id = -1;
J. R. Okajima's avatar
J. R. Okajima committed
149
150
151
152
153
154
155
156
157
158
159
}

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);
160
161
162
	nbr = au_sbbot(sb) + 1;
	if (unlikely(nbr <= 0))
		nbr = 1;
J. R. Okajima's avatar
J. R. Okajima committed
163
164
165
166
167
168
	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
169
		iinfo->ii_generation.ig_generation = au_sigen(sb);
J. R. Okajima's avatar
J. R. Okajima committed
170
171
172
173
174
175
176
		iinfo->ii_btop = -1;
		iinfo->ii_bbot = -1;
		return 0;
	}
	return -ENOMEM;
}

177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
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
199
200
201
202
void au_iinfo_fin(struct inode *inode)
{
	struct au_iinfo *iinfo;
	struct au_hinode *hi;
203
	struct super_block *sb;
J. R. Okajima's avatar
J. R. Okajima committed
204
	aufs_bindex_t bindex, bbot;
J. R. Okajima's avatar
J. R. Okajima committed
205
	const unsigned char unlinked = !inode->i_nlink;
J. R. Okajima's avatar
J. R. Okajima committed
206
207
208

	AuDebugOn(au_is_bad_inode(inode));

209
210
211
212
213
214
215
216
217
218
219
220
221
222
	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
223

J. R. Okajima's avatar
J. R. Okajima committed
224
225
226
227
228
229
230
231
232
233
234
	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++;
		}
	}
235
	au_kfree_rcu(iinfo->ii_hinode);
J. R. Okajima's avatar
J. R. Okajima committed
236
	AuRwDestroy(&iinfo->ii_rwsem);
J. R. Okajima's avatar
J. R. Okajima committed
237
}