Blame view

fs/minix/dir.c 10.9 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
12
13
14
15
   */
  
  #include "minix.h"
  #include <linux/highmem.h>
  #include <linux/smp_lock.h>
  
  typedef struct minix_dir_entry minix_dirent;
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
16
  typedef struct minix3_dir_entry minix3_dirent;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
17
18
  
  static int minix_readdir(struct file *, void *, filldir_t);
4b6f5d20b   Arjan van de Ven   [PATCH] Make most...
19
  const struct file_operations minix_dir_operations = {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
20
21
22
23
24
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
51
52
53
54
55
56
57
58
59
60
61
62
63
64
  	.read		= generic_read_dir,
  	.readdir	= minix_readdir,
  	.fsync		= minix_sync_file,
  };
  
  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;
  }
  
  static int dir_commit_chunk(struct page *page, unsigned from, unsigned to)
  {
  	struct inode *dir = (struct inode *)page->mapping->host;
  	int err = 0;
  	page->mapping->a_ops->commit_write(NULL, page, from, to);
  	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...
65
  	struct page *page = read_mapping_page(mapping, n, NULL);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
66
  	if (!IS_ERR(page)) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
  		kmap(page);
  		if (!PageUptodate(page))
  			goto fail;
  	}
  	return page;
  
  fail:
  	dir_put_page(page);
  	return ERR_PTR(-EIO);
  }
  
  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
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
  
  	lock_kernel();
  
  	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 ...
111
112
113
114
115
116
117
118
119
120
121
  		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
122
  				int over;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
123

939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
124
  				unsigned l = strnlen(name, sbi->s_namelen);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
125
  				offset = p - kaddr;
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
126
127
128
  				over = filldir(dirent, name, l,
  					(n << PAGE_CACHE_SHIFT) | offset,
  					inumber, DT_UNKNOWN);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
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
156
157
158
159
160
161
162
163
164
165
166
167
168
169
  				if (over) {
  					dir_put_page(page);
  					goto done;
  				}
  			}
  		}
  		dir_put_page(page);
  	}
  
  done:
  	filp->f_pos = (n << PAGE_CACHE_SHIFT) | offset;
  	unlock_kernel();
  	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 ...
170
  	char *p;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
171

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

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

939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
395
  			if (inumber != 0) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
396
  				/* check for . and .. */
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
397
  				if (name[0] != '.')
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
398
  					goto not_empty;
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
399
400
  				if (!name[1]) {
  					if (inumber != inode->i_ino)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
401
  						goto not_empty;
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
402
  				} else if (name[1] != '.')
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
403
  					goto not_empty;
939b00df0   Andries Brouwer   [PATCH] Minix V3 ...
404
  				else if (name[2])
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
405
406
  					goto not_empty;
  			}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
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
457
458
459
460
461
462
463
464
  		}
  		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)
  {
  	struct inode *dir = (struct inode*)page->mapping->host;
  	struct minix_sb_info *sbi = minix_sb(dir->i_sb);
  	unsigned from = (char *)de-(char*)page_address(page);
  	unsigned to = from + sbi->s_dirsize;
  	int err;
  
  	lock_page(page);
  	err = page->mapping->a_ops->prepare_write(NULL, page, from, to);
  	if (err == 0) {
  		de->inode = inode->i_ino;
  		err = dir_commit_chunk(page, from, to);
  	} 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) {
  		res = de->inode;
  		dir_put_page(page);
  	}
  	return res;
  }