Blame view

fs/minix/dir.c 11.1 KB
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1
2
3
4
5
6
  /*
   *  linux/fs/minix/dir.c
   *
   *  Copyright (C) 1991, 1992 Linus Torvalds
   *
   *  minix directory handling functions
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
7
8
   *
   *  Updated to filesystem version 3 by Daniel Aragones
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
9
10
11
   */
  
  #include "minix.h"
4a66af9ea   Nick Piggin   minixfs: convert ...
12
  #include <linux/buffer_head.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
13
  #include <linux/highmem.h>
4a66af9ea   Nick Piggin   minixfs: convert ...
14
  #include <linux/swap.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
15
16
  
  typedef struct minix_dir_entry minix_dirent;
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
17
  typedef struct minix3_dir_entry minix3_dirent;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
18

80886298c   Al Viro   [readdir] simple ...
19
  static int minix_readdir(struct file *, struct dir_context *);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
20

4b6f5d20b   Arjan van de Ven   [PATCH] Make most...
21
  const struct file_operations minix_dir_operations = {
cc46759a8   Al Viro   get rid of BKL in...
22
  	.llseek		= generic_file_llseek,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
23
  	.read		= generic_read_dir,
80886298c   Al Viro   [readdir] simple ...
24
  	.iterate	= minix_readdir,
1b061d924   Christoph Hellwig   rename the generi...
25
  	.fsync		= generic_file_fsync,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
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
  };
  
  static inline void dir_put_page(struct page *page)
  {
  	kunmap(page);
  	page_cache_release(page);
  }
  
  /*
   * Return the offset into page `page_nr' of the last valid
   * byte in that page, plus one.
   */
  static unsigned
  minix_last_byte(struct inode *inode, unsigned long page_nr)
  {
  	unsigned last_byte = PAGE_CACHE_SIZE;
  
  	if (page_nr == (inode->i_size >> PAGE_CACHE_SHIFT))
  		last_byte = inode->i_size & (PAGE_CACHE_SIZE - 1);
  	return last_byte;
  }
  
  static inline unsigned long dir_pages(struct inode *inode)
  {
  	return (inode->i_size+PAGE_CACHE_SIZE-1)>>PAGE_CACHE_SHIFT;
  }
4a66af9ea   Nick Piggin   minixfs: convert ...
52
  static int dir_commit_chunk(struct page *page, loff_t pos, unsigned len)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
53
  {
4a66af9ea   Nick Piggin   minixfs: convert ...
54
55
  	struct address_space *mapping = page->mapping;
  	struct inode *dir = mapping->host;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
56
  	int err = 0;
4a66af9ea   Nick Piggin   minixfs: convert ...
57
58
59
60
61
62
  	block_write_end(NULL, mapping, pos, len, len, page, NULL);
  
  	if (pos+len > dir->i_size) {
  		i_size_write(dir, pos+len);
  		mark_inode_dirty(dir);
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
63
64
65
66
67
68
69
70
71
72
  	if (IS_DIRSYNC(dir))
  		err = write_one_page(page, 1);
  	else
  		unlock_page(page);
  	return err;
  }
  
  static struct page * dir_get_page(struct inode *dir, unsigned long n)
  {
  	struct address_space *mapping = dir->i_mapping;
090d2b185   Pekka Enberg   [PATCH] read_mapp...
73
  	struct page *page = read_mapping_page(mapping, n, NULL);
49837a80b   Al Viro   remove detritus l...
74
  	if (!IS_ERR(page))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
75
  		kmap(page);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
76
  	return page;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
77
78
79
80
81
82
  }
  
  static inline void *minix_next_entry(void *de, struct minix_sb_info *sbi)
  {
  	return (void*)((char*)de + sbi->s_dirsize);
  }
80886298c   Al Viro   [readdir] simple ...
83
  static int minix_readdir(struct file *file, struct dir_context *ctx)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
84
  {
80886298c   Al Viro   [readdir] simple ...
85
  	struct inode *inode = file_inode(file);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
86
  	struct super_block *sb = inode->i_sb;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
87
88
  	struct minix_sb_info *sbi = minix_sb(sb);
  	unsigned chunk_size = sbi->s_dirsize;
80886298c   Al Viro   [readdir] simple ...
89
90
91
92
  	unsigned long npages = dir_pages(inode);
  	unsigned long pos = ctx->pos;
  	unsigned offset;
  	unsigned long n;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
93

642b704cd   Dan Carpenter   minix: bug wideni...
94
  	ctx->pos = pos = ALIGN(pos, chunk_size);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
95
  	if (pos >= inode->i_size)
80886298c   Al Viro   [readdir] simple ...
96
97
98
99
  		return 0;
  
  	offset = pos & ~PAGE_CACHE_MASK;
  	n = pos >> PAGE_CACHE_SHIFT;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
100
101
102
103
104
105
106
107
108
109
  
  	for ( ; n < npages; n++, offset = 0) {
  		char *p, *kaddr, *limit;
  		struct page *page = dir_get_page(inode, n);
  
  		if (IS_ERR(page))
  			continue;
  		kaddr = (char *)page_address(page);
  		p = kaddr+offset;
  		limit = kaddr + minix_last_byte(inode, n) - chunk_size;
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
110
  		for ( ; p <= limit; p = minix_next_entry(p, sbi)) {
80886298c   Al Viro   [readdir] simple ...
111
112
  			const char *name;
  			__u32 inumber;
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
113
114
115
116
117
118
119
120
121
122
  			if (sbi->s_version == MINIX_V3) {
  				minix3_dirent *de3 = (minix3_dirent *)p;
  				name = de3->name;
  				inumber = de3->inode;
  	 		} else {
  				minix_dirent *de = (minix_dirent *)p;
  				name = de->name;
  				inumber = de->inode;
  			}
  			if (inumber) {
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
123
  				unsigned l = strnlen(name, sbi->s_namelen);
80886298c   Al Viro   [readdir] simple ...
124
125
  				if (!dir_emit(ctx, name, l,
  					      inumber, DT_UNKNOWN)) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
126
  					dir_put_page(page);
80886298c   Al Viro   [readdir] simple ...
127
  					return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
128
129
  				}
  			}
80886298c   Al Viro   [readdir] simple ...
130
  			ctx->pos += chunk_size;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
131
132
133
  		}
  		dir_put_page(page);
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
  	return 0;
  }
  
  static inline int namecompare(int len, int maxlen,
  	const char * name, const char * buffer)
  {
  	if (len < maxlen && buffer[len])
  		return 0;
  	return !memcmp(name, buffer, len);
  }
  
  /*
   *	minix_find_entry()
   *
   * finds an entry in the specified directory with the wanted name. It
   * returns the cache buffer in which the entry was found, and the entry
   * itself (as a parameter - res_dir). It does NOT read the inode of the
   * entry - you'll have to do that yourself if you want to.
   */
  minix_dirent *minix_find_entry(struct dentry *dentry, struct page **res_page)
  {
  	const char * name = dentry->d_name.name;
  	int namelen = dentry->d_name.len;
  	struct inode * dir = dentry->d_parent->d_inode;
  	struct super_block * sb = dir->i_sb;
  	struct minix_sb_info * sbi = minix_sb(sb);
  	unsigned long n;
  	unsigned long npages = dir_pages(dir);
  	struct page *page = NULL;
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
163
  	char *p;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
164

939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
165
166
  	char *namx;
  	__u32 inumber;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
167
168
169
  	*res_page = NULL;
  
  	for (n = 0; n < npages; n++) {
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
170
  		char *kaddr, *limit;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
171
172
173
174
175
  		page = dir_get_page(dir, n);
  		if (IS_ERR(page))
  			continue;
  
  		kaddr = (char*)page_address(page);
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
176
177
178
179
180
181
182
183
184
185
186
187
  		limit = kaddr + minix_last_byte(dir, n) - sbi->s_dirsize;
  		for (p = kaddr; p <= limit; p = minix_next_entry(p, sbi)) {
  			if (sbi->s_version == MINIX_V3) {
  				minix3_dirent *de3 = (minix3_dirent *)p;
  				namx = de3->name;
  				inumber = de3->inode;
   			} else {
  				minix_dirent *de = (minix_dirent *)p;
  				namx = de->name;
  				inumber = de->inode;
  			}
  			if (!inumber)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
188
  				continue;
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
189
  			if (namecompare(namelen, sbi->s_namelen, name, namx))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
190
191
192
193
194
195
196
197
  				goto found;
  		}
  		dir_put_page(page);
  	}
  	return NULL;
  
  found:
  	*res_page = page;
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
198
  	return (minix_dirent *)p;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
199
200
201
202
203
204
205
206
207
208
  }
  
  int minix_add_link(struct dentry *dentry, struct inode *inode)
  {
  	struct inode *dir = dentry->d_parent->d_inode;
  	const char * name = dentry->d_name.name;
  	int namelen = dentry->d_name.len;
  	struct super_block * sb = dir->i_sb;
  	struct minix_sb_info * sbi = minix_sb(sb);
  	struct page *page = NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
209
210
  	unsigned long npages = dir_pages(dir);
  	unsigned long n;
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
211
212
213
  	char *kaddr, *p;
  	minix_dirent *de;
  	minix3_dirent *de3;
4a66af9ea   Nick Piggin   minixfs: convert ...
214
  	loff_t pos;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
215
  	int err;
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
216
217
  	char *namx = NULL;
  	__u32 inumber;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
218
219
220
221
222
223
224
  
  	/*
  	 * We take care of directory expansion in the same loop
  	 * This code plays outside i_size, so it locks the page
  	 * to protect that region.
  	 */
  	for (n = 0; n <= npages; n++) {
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
225
  		char *limit, *dir_end;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
226
227
228
229
230
231
232
233
  
  		page = dir_get_page(dir, n);
  		err = PTR_ERR(page);
  		if (IS_ERR(page))
  			goto out;
  		lock_page(page);
  		kaddr = (char*)page_address(page);
  		dir_end = kaddr + minix_last_byte(dir, n);
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
234
235
236
237
238
239
240
241
242
243
244
245
  		limit = kaddr + PAGE_CACHE_SIZE - sbi->s_dirsize;
  		for (p = kaddr; p <= limit; p = minix_next_entry(p, sbi)) {
  			de = (minix_dirent *)p;
  			de3 = (minix3_dirent *)p;
  			if (sbi->s_version == MINIX_V3) {
  				namx = de3->name;
  				inumber = de3->inode;
  		 	} else {
    				namx = de->name;
  				inumber = de->inode;
  			}
  			if (p == dir_end) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
246
  				/* We hit i_size */
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
247
248
249
250
  				if (sbi->s_version == MINIX_V3)
  					de3->inode = 0;
  		 		else
  					de->inode = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
251
252
  				goto got_it;
  			}
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
253
  			if (!inumber)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
254
255
  				goto got_it;
  			err = -EEXIST;
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
256
  			if (namecompare(namelen, sbi->s_namelen, name, namx))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
257
  				goto out_unlock;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
258
259
260
261
262
263
264
265
  		}
  		unlock_page(page);
  		dir_put_page(page);
  	}
  	BUG();
  	return -EINVAL;
  
  got_it:
d6b54841f   Evgeniy Dushistov   minix: fix add li...
266
  	pos = page_offset(page) + p - (char *)page_address(page);
f4e420dc4   Christoph Hellwig   clean up write_be...
267
  	err = minix_prepare_chunk(page, pos, sbi->s_dirsize);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
268
269
  	if (err)
  		goto out_unlock;
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
270
271
272
273
274
275
276
277
  	memcpy (namx, name, namelen);
  	if (sbi->s_version == MINIX_V3) {
  		memset (namx + namelen, 0, sbi->s_dirsize - namelen - 4);
  		de3->inode = inode->i_ino;
  	} else {
  		memset (namx + namelen, 0, sbi->s_dirsize - namelen - 2);
  		de->inode = inode->i_ino;
  	}
4a66af9ea   Nick Piggin   minixfs: convert ...
278
  	err = dir_commit_chunk(page, pos, sbi->s_dirsize);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
279
280
281
282
283
284
285
286
287
288
289
290
291
  	dir->i_mtime = dir->i_ctime = CURRENT_TIME_SEC;
  	mark_inode_dirty(dir);
  out_put:
  	dir_put_page(page);
  out:
  	return err;
  out_unlock:
  	unlock_page(page);
  	goto out_put;
  }
  
  int minix_delete_entry(struct minix_dir_entry *de, struct page *page)
  {
f4e420dc4   Christoph Hellwig   clean up write_be...
292
  	struct inode *inode = page->mapping->host;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
293
  	char *kaddr = page_address(page);
4a66af9ea   Nick Piggin   minixfs: convert ...
294
  	loff_t pos = page_offset(page) + (char*)de - kaddr;
9f6c13339   Doug Graham   V3 minixfs: add m...
295
296
  	struct minix_sb_info *sbi = minix_sb(inode->i_sb);
  	unsigned len = sbi->s_dirsize;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
297
298
299
  	int err;
  
  	lock_page(page);
f4e420dc4   Christoph Hellwig   clean up write_be...
300
  	err = minix_prepare_chunk(page, pos, len);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
301
  	if (err == 0) {
9f6c13339   Doug Graham   V3 minixfs: add m...
302
303
304
305
  		if (sbi->s_version == MINIX_V3)
  			((minix3_dirent *) de)->inode = 0;
  		else
  			de->inode = 0;
4a66af9ea   Nick Piggin   minixfs: convert ...
306
  		err = dir_commit_chunk(page, pos, len);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
307
308
309
310
311
312
313
314
315
316
317
  	} else {
  		unlock_page(page);
  	}
  	dir_put_page(page);
  	inode->i_ctime = inode->i_mtime = CURRENT_TIME_SEC;
  	mark_inode_dirty(inode);
  	return err;
  }
  
  int minix_make_empty(struct inode *inode, struct inode *dir)
  {
f4e420dc4   Christoph Hellwig   clean up write_be...
318
  	struct page *page = grab_cache_page(inode->i_mapping, 0);
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
319
  	struct minix_sb_info *sbi = minix_sb(inode->i_sb);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
320
321
322
323
324
  	char *kaddr;
  	int err;
  
  	if (!page)
  		return -ENOMEM;
f4e420dc4   Christoph Hellwig   clean up write_be...
325
  	err = minix_prepare_chunk(page, 0, 2 * sbi->s_dirsize);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
326
327
328
329
  	if (err) {
  		unlock_page(page);
  		goto fail;
  	}
27a6d5c74   Cong Wang   minix: remove the...
330
  	kaddr = kmap_atomic(page);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
331
  	memset(kaddr, 0, PAGE_CACHE_SIZE);
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
  	if (sbi->s_version == MINIX_V3) {
  		minix3_dirent *de3 = (minix3_dirent *)kaddr;
  
  		de3->inode = inode->i_ino;
  		strcpy(de3->name, ".");
  		de3 = minix_next_entry(de3, sbi);
  		de3->inode = dir->i_ino;
  		strcpy(de3->name, "..");
  	} else {
  		minix_dirent *de = (minix_dirent *)kaddr;
  
  		de->inode = inode->i_ino;
  		strcpy(de->name, ".");
  		de = minix_next_entry(de, sbi);
  		de->inode = dir->i_ino;
  		strcpy(de->name, "..");
  	}
27a6d5c74   Cong Wang   minix: remove the...
349
  	kunmap_atomic(kaddr);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
  
  	err = dir_commit_chunk(page, 0, 2 * sbi->s_dirsize);
  fail:
  	page_cache_release(page);
  	return err;
  }
  
  /*
   * routine to check that the specified directory is empty (for rmdir)
   */
  int minix_empty_dir(struct inode * inode)
  {
  	struct page *page = NULL;
  	unsigned long i, npages = dir_pages(inode);
  	struct minix_sb_info *sbi = minix_sb(inode->i_sb);
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
365
366
  	char *name;
  	__u32 inumber;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
367
368
  
  	for (i = 0; i < npages; i++) {
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
369
  		char *p, *kaddr, *limit;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
370

939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
371
  		page = dir_get_page(inode, i);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
372
373
374
375
  		if (IS_ERR(page))
  			continue;
  
  		kaddr = (char *)page_address(page);
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
376
377
378
379
380
381
382
383
384
385
386
  		limit = kaddr + minix_last_byte(inode, i) - sbi->s_dirsize;
  		for (p = kaddr; p <= limit; p = minix_next_entry(p, sbi)) {
  			if (sbi->s_version == MINIX_V3) {
  				minix3_dirent *de3 = (minix3_dirent *)p;
  				name = de3->name;
  				inumber = de3->inode;
  			} else {
  				minix_dirent *de = (minix_dirent *)p;
  				name = de->name;
  				inumber = de->inode;
  			}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
387

939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
388
  			if (inumber != 0) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
389
  				/* check for . and .. */
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
390
  				if (name[0] != '.')
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
391
  					goto not_empty;
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
392
393
  				if (!name[1]) {
  					if (inumber != inode->i_ino)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
394
  						goto not_empty;
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
395
  				} else if (name[1] != '.')
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
396
  					goto not_empty;
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
397
  				else if (name[2])
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
398
399
  					goto not_empty;
  			}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
400
401
402
403
404
405
406
407
408
409
410
411
412
413
  		}
  		dir_put_page(page);
  	}
  	return 1;
  
  not_empty:
  	dir_put_page(page);
  	return 0;
  }
  
  /* Releases the page */
  void minix_set_link(struct minix_dir_entry *de, struct page *page,
  	struct inode *inode)
  {
f4e420dc4   Christoph Hellwig   clean up write_be...
414
  	struct inode *dir = page->mapping->host;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
415
  	struct minix_sb_info *sbi = minix_sb(dir->i_sb);
4a66af9ea   Nick Piggin   minixfs: convert ...
416
417
  	loff_t pos = page_offset(page) +
  			(char *)de-(char*)page_address(page);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
418
419
420
  	int err;
  
  	lock_page(page);
4a66af9ea   Nick Piggin   minixfs: convert ...
421

f4e420dc4   Christoph Hellwig   clean up write_be...
422
  	err = minix_prepare_chunk(page, pos, sbi->s_dirsize);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
423
  	if (err == 0) {
9f6c13339   Doug Graham   V3 minixfs: add m...
424
425
426
427
  		if (sbi->s_version == MINIX_V3)
  			((minix3_dirent *) de)->inode = inode->i_ino;
  		else
  			de->inode = inode->i_ino;
4a66af9ea   Nick Piggin   minixfs: convert ...
428
  		err = dir_commit_chunk(page, pos, sbi->s_dirsize);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
  	} else {
  		unlock_page(page);
  	}
  	dir_put_page(page);
  	dir->i_mtime = dir->i_ctime = CURRENT_TIME_SEC;
  	mark_inode_dirty(dir);
  }
  
  struct minix_dir_entry * minix_dotdot (struct inode *dir, struct page **p)
  {
  	struct page *page = dir_get_page(dir, 0);
  	struct minix_sb_info *sbi = minix_sb(dir->i_sb);
  	struct minix_dir_entry *de = NULL;
  
  	if (!IS_ERR(page)) {
  		de = minix_next_entry(page_address(page), sbi);
  		*p = page;
  	}
  	return de;
  }
  
  ino_t minix_inode_by_name(struct dentry *dentry)
  {
  	struct page *page;
  	struct minix_dir_entry *de = minix_find_entry(dentry, &page);
  	ino_t res = 0;
  
  	if (de) {
9f6c13339   Doug Graham   V3 minixfs: add m...
457
458
459
460
461
462
463
464
  		struct address_space *mapping = page->mapping;
  		struct inode *inode = mapping->host;
  		struct minix_sb_info *sbi = minix_sb(inode->i_sb);
  
  		if (sbi->s_version == MINIX_V3)
  			res = ((minix3_dirent *) de)->inode;
  		else
  			res = de->inode;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
465
466
467
468
  		dir_put_page(page);
  	}
  	return res;
  }