Blame view

fs/minix/dir.c 11.2 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
19
  
  static int minix_readdir(struct file *, void *, filldir_t);
4b6f5d20b   Arjan van de Ven   [PATCH] Make most...
20
  const struct file_operations minix_dir_operations = {
cc46759a8   Al Viro   get rid of BKL in...
21
  	.llseek		= generic_file_llseek,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
22
23
  	.read		= generic_read_dir,
  	.readdir	= minix_readdir,
1b061d924   Christoph Hellwig   rename the generi...
24
  	.fsync		= generic_file_fsync,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
25
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
  };
  
  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 ...
51
  static int dir_commit_chunk(struct page *page, loff_t pos, unsigned len)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
52
  {
4a66af9ea   Nick Piggin   minixfs: convert ...
53
54
  	struct address_space *mapping = page->mapping;
  	struct inode *dir = mapping->host;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
55
  	int err = 0;
4a66af9ea   Nick Piggin   minixfs: convert ...
56
57
58
59
60
61
  	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
62
63
64
65
66
67
68
69
70
71
  	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...
72
  	struct page *page = read_mapping_page(mapping, n, NULL);
49837a80b   Al Viro   remove detritus l...
73
  	if (!IS_ERR(page))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
74
  		kmap(page);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
75
  	return page;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
76
77
78
79
80
81
82
83
84
85
  }
  
  static inline void *minix_next_entry(void *de, struct minix_sb_info *sbi)
  {
  	return (void*)((char*)de + sbi->s_dirsize);
  }
  
  static int minix_readdir(struct file * filp, void * dirent, filldir_t filldir)
  {
  	unsigned long pos = filp->f_pos;
dcf258ae6   Josef Sipek   [PATCH] struct pa...
86
  	struct inode *inode = filp->f_path.dentry->d_inode;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
87
88
89
90
91
92
  	struct super_block *sb = inode->i_sb;
  	unsigned offset = pos & ~PAGE_CACHE_MASK;
  	unsigned long n = pos >> PAGE_CACHE_SHIFT;
  	unsigned long npages = dir_pages(inode);
  	struct minix_sb_info *sbi = minix_sb(sb);
  	unsigned chunk_size = sbi->s_dirsize;
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
93
94
  	char *name;
  	__u32 inumber;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
95

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
96
97
98
99
100
101
102
103
104
105
106
107
108
  	pos = (pos + chunk_size-1) & ~(chunk_size-1);
  	if (pos >= inode->i_size)
  		goto done;
  
  	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 ...
109
110
111
112
113
114
115
116
117
118
119
  		for ( ; 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;
  			}
  			if (inumber) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
120
  				int over;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
121

939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
122
  				unsigned l = strnlen(name, sbi->s_namelen);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
123
  				offset = p - kaddr;
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
124
125
126
  				over = filldir(dirent, name, l,
  					(n << PAGE_CACHE_SHIFT) | offset,
  					inumber, DT_UNKNOWN);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
127
128
129
130
131
132
133
134
135
136
137
  				if (over) {
  					dir_put_page(page);
  					goto done;
  				}
  			}
  		}
  		dir_put_page(page);
  	}
  
  done:
  	filp->f_pos = (n << PAGE_CACHE_SHIFT) | offset;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
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
163
164
165
166
  	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 ...
167
  	char *p;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
168

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

939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
376
  		page = dir_get_page(inode, i);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
377
378
379
380
  		if (IS_ERR(page))
  			continue;
  
  		kaddr = (char *)page_address(page);
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
381
382
383
384
385
386
387
388
389
390
391
  		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
392

939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
393
  			if (inumber != 0) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
394
  				/* check for . and .. */
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
395
  				if (name[0] != '.')
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
396
  					goto not_empty;
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
397
398
  				if (!name[1]) {
  					if (inumber != inode->i_ino)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
399
  						goto not_empty;
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
400
  				} else if (name[1] != '.')
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
401
  					goto not_empty;
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
402
  				else if (name[2])
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
403
404
  					goto not_empty;
  			}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
405
406
407
408
409
410
411
412
413
414
415
416
417
418
  		}
  		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...
419
  	struct inode *dir = page->mapping->host;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
420
  	struct minix_sb_info *sbi = minix_sb(dir->i_sb);
4a66af9ea   Nick Piggin   minixfs: convert ...
421
422
  	loff_t pos = page_offset(page) +
  			(char *)de-(char*)page_address(page);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
423
424
425
  	int err;
  
  	lock_page(page);
4a66af9ea   Nick Piggin   minixfs: convert ...
426

f4e420dc4   Christoph Hellwig   clean up write_be...
427
  	err = minix_prepare_chunk(page, pos, sbi->s_dirsize);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
428
  	if (err == 0) {
9f6c13339   Doug Graham   V3 minixfs: add m...
429
430
431
432
  		if (sbi->s_version == MINIX_V3)
  			((minix3_dirent *) de)->inode = inode->i_ino;
  		else
  			de->inode = inode->i_ino;
4a66af9ea   Nick Piggin   minixfs: convert ...
433
  		err = dir_commit_chunk(page, pos, sbi->s_dirsize);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
  	} 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...
462
463
464
465
466
467
468
469
  		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
470
471
472
473
  		dir_put_page(page);
  	}
  	return res;
  }