diff options
Diffstat (limited to 'fs/autofs/root.c')
-rw-r--r-- | fs/autofs/root.c | 83 |
1 files changed, 42 insertions, 41 deletions
diff --git a/fs/autofs/root.c b/fs/autofs/root.c index f2597205939d..c1489533277a 100644 --- a/fs/autofs/root.c +++ b/fs/autofs/root.c | |||
@@ -67,8 +67,8 @@ static int autofs_root_readdir(struct file *filp, void *dirent, filldir_t filldi | |||
67 | filp->f_pos = ++nr; | 67 | filp->f_pos = ++nr; |
68 | /* fall through */ | 68 | /* fall through */ |
69 | default: | 69 | default: |
70 | while ( onr = nr, ent = autofs_hash_enum(dirhash,&nr,ent) ) { | 70 | while (onr = nr, ent = autofs_hash_enum(dirhash,&nr,ent)) { |
71 | if ( !ent->dentry || d_mountpoint(ent->dentry) ) { | 71 | if (!ent->dentry || d_mountpoint(ent->dentry)) { |
72 | if (filldir(dirent,ent->name,ent->len,onr,ent->ino,DT_UNKNOWN) < 0) | 72 | if (filldir(dirent,ent->name,ent->len,onr,ent->ino,DT_UNKNOWN) < 0) |
73 | goto out; | 73 | goto out; |
74 | filp->f_pos = nr; | 74 | filp->f_pos = nr; |
@@ -88,10 +88,10 @@ static int try_to_fill_dentry(struct dentry *dentry, struct super_block *sb, str | |||
88 | struct autofs_dir_ent *ent; | 88 | struct autofs_dir_ent *ent; |
89 | int status = 0; | 89 | int status = 0; |
90 | 90 | ||
91 | if ( !(ent = autofs_hash_lookup(&sbi->dirhash, &dentry->d_name)) ) { | 91 | if (!(ent = autofs_hash_lookup(&sbi->dirhash, &dentry->d_name))) { |
92 | do { | 92 | do { |
93 | if ( status && dentry->d_inode ) { | 93 | if (status && dentry->d_inode) { |
94 | if ( status != -ENOENT ) | 94 | if (status != -ENOENT) |
95 | printk("autofs warning: lookup failure on positive dentry, status = %d, name = %s\n", status, dentry->d_name.name); | 95 | printk("autofs warning: lookup failure on positive dentry, status = %d, name = %s\n", status, dentry->d_name.name); |
96 | return 0; /* Try to get the kernel to invalidate this dentry */ | 96 | return 0; /* Try to get the kernel to invalidate this dentry */ |
97 | } | 97 | } |
@@ -106,7 +106,7 @@ static int try_to_fill_dentry(struct dentry *dentry, struct super_block *sb, str | |||
106 | return 1; | 106 | return 1; |
107 | } | 107 | } |
108 | status = autofs_wait(sbi, &dentry->d_name); | 108 | status = autofs_wait(sbi, &dentry->d_name); |
109 | } while (!(ent = autofs_hash_lookup(&sbi->dirhash, &dentry->d_name)) ); | 109 | } while (!(ent = autofs_hash_lookup(&sbi->dirhash, &dentry->d_name))); |
110 | } | 110 | } |
111 | 111 | ||
112 | /* Abuse this field as a pointer to the directory entry, used to | 112 | /* Abuse this field as a pointer to the directory entry, used to |
@@ -124,13 +124,13 @@ static int try_to_fill_dentry(struct dentry *dentry, struct super_block *sb, str | |||
124 | 124 | ||
125 | /* If this is a directory that isn't a mount point, bitch at the | 125 | /* If this is a directory that isn't a mount point, bitch at the |
126 | daemon and fix it in user space */ | 126 | daemon and fix it in user space */ |
127 | if ( S_ISDIR(dentry->d_inode->i_mode) && !d_mountpoint(dentry) ) { | 127 | if (S_ISDIR(dentry->d_inode->i_mode) && !d_mountpoint(dentry)) { |
128 | return !autofs_wait(sbi, &dentry->d_name); | 128 | return !autofs_wait(sbi, &dentry->d_name); |
129 | } | 129 | } |
130 | 130 | ||
131 | /* We don't update the usages for the autofs daemon itself, this | 131 | /* We don't update the usages for the autofs daemon itself, this |
132 | is necessary for recursive autofs mounts */ | 132 | is necessary for recursive autofs mounts */ |
133 | if ( !autofs_oz_mode(sbi) ) { | 133 | if (!autofs_oz_mode(sbi)) { |
134 | autofs_update_usage(&sbi->dirhash,ent); | 134 | autofs_update_usage(&sbi->dirhash,ent); |
135 | } | 135 | } |
136 | 136 | ||
@@ -157,7 +157,7 @@ static int autofs_revalidate(struct dentry * dentry, struct nameidata *nd) | |||
157 | sbi = autofs_sbi(dir->i_sb); | 157 | sbi = autofs_sbi(dir->i_sb); |
158 | 158 | ||
159 | /* Pending dentry */ | 159 | /* Pending dentry */ |
160 | if ( dentry->d_flags & DCACHE_AUTOFS_PENDING ) { | 160 | if (dentry->d_flags & DCACHE_AUTOFS_PENDING) { |
161 | if (autofs_oz_mode(sbi)) | 161 | if (autofs_oz_mode(sbi)) |
162 | res = 1; | 162 | res = 1; |
163 | else | 163 | else |
@@ -173,7 +173,7 @@ static int autofs_revalidate(struct dentry * dentry, struct nameidata *nd) | |||
173 | } | 173 | } |
174 | 174 | ||
175 | /* Check for a non-mountpoint directory */ | 175 | /* Check for a non-mountpoint directory */ |
176 | if ( S_ISDIR(dentry->d_inode->i_mode) && !d_mountpoint(dentry) ) { | 176 | if (S_ISDIR(dentry->d_inode->i_mode) && !d_mountpoint(dentry)) { |
177 | if (autofs_oz_mode(sbi)) | 177 | if (autofs_oz_mode(sbi)) |
178 | res = 1; | 178 | res = 1; |
179 | else | 179 | else |
@@ -183,9 +183,9 @@ static int autofs_revalidate(struct dentry * dentry, struct nameidata *nd) | |||
183 | } | 183 | } |
184 | 184 | ||
185 | /* Update the usage list */ | 185 | /* Update the usage list */ |
186 | if ( !autofs_oz_mode(sbi) ) { | 186 | if (!autofs_oz_mode(sbi)) { |
187 | ent = (struct autofs_dir_ent *) dentry->d_time; | 187 | ent = (struct autofs_dir_ent *) dentry->d_time; |
188 | if ( ent ) | 188 | if (ent) |
189 | autofs_update_usage(&sbi->dirhash,ent); | 189 | autofs_update_usage(&sbi->dirhash,ent); |
190 | } | 190 | } |
191 | unlock_kernel(); | 191 | unlock_kernel(); |
@@ -213,8 +213,10 @@ static struct dentry *autofs_root_lookup(struct inode *dir, struct dentry *dentr | |||
213 | sbi = autofs_sbi(dir->i_sb); | 213 | sbi = autofs_sbi(dir->i_sb); |
214 | 214 | ||
215 | oz_mode = autofs_oz_mode(sbi); | 215 | oz_mode = autofs_oz_mode(sbi); |
216 | DPRINTK(("autofs_lookup: pid = %u, pgrp = %u, catatonic = %d, oz_mode = %d\n", | 216 | DPRINTK(("autofs_lookup: pid = %u, pgrp = %u, catatonic = %d, " |
217 | current->pid, process_group(current), sbi->catatonic, oz_mode)); | 217 | "oz_mode = %d\n", pid_nr(task_pid(current)), |
218 | process_group(current), sbi->catatonic, | ||
219 | oz_mode)); | ||
218 | 220 | ||
219 | /* | 221 | /* |
220 | * Mark the dentry incomplete, but add it. This is needed so | 222 | * Mark the dentry incomplete, but add it. This is needed so |
@@ -258,7 +260,7 @@ static struct dentry *autofs_root_lookup(struct inode *dir, struct dentry *dentr | |||
258 | * doesn't do the right thing for all system calls, but it should | 260 | * doesn't do the right thing for all system calls, but it should |
259 | * be OK for the operations we permit from an autofs. | 261 | * be OK for the operations we permit from an autofs. |
260 | */ | 262 | */ |
261 | if ( dentry->d_inode && d_unhashed(dentry) ) | 263 | if (dentry->d_inode && d_unhashed(dentry)) |
262 | return ERR_PTR(-ENOENT); | 264 | return ERR_PTR(-ENOENT); |
263 | 265 | ||
264 | return NULL; | 266 | return NULL; |
@@ -277,18 +279,18 @@ static int autofs_root_symlink(struct inode *dir, struct dentry *dentry, const c | |||
277 | autofs_say(dentry->d_name.name,dentry->d_name.len); | 279 | autofs_say(dentry->d_name.name,dentry->d_name.len); |
278 | 280 | ||
279 | lock_kernel(); | 281 | lock_kernel(); |
280 | if ( !autofs_oz_mode(sbi) ) { | 282 | if (!autofs_oz_mode(sbi)) { |
281 | unlock_kernel(); | 283 | unlock_kernel(); |
282 | return -EACCES; | 284 | return -EACCES; |
283 | } | 285 | } |
284 | 286 | ||
285 | if ( autofs_hash_lookup(dh, &dentry->d_name) ) { | 287 | if (autofs_hash_lookup(dh, &dentry->d_name)) { |
286 | unlock_kernel(); | 288 | unlock_kernel(); |
287 | return -EEXIST; | 289 | return -EEXIST; |
288 | } | 290 | } |
289 | 291 | ||
290 | n = find_first_zero_bit(sbi->symlink_bitmap,AUTOFS_MAX_SYMLINKS); | 292 | n = find_first_zero_bit(sbi->symlink_bitmap,AUTOFS_MAX_SYMLINKS); |
291 | if ( n >= AUTOFS_MAX_SYMLINKS ) { | 293 | if (n >= AUTOFS_MAX_SYMLINKS) { |
292 | unlock_kernel(); | 294 | unlock_kernel(); |
293 | return -ENOSPC; | 295 | return -ENOSPC; |
294 | } | 296 | } |
@@ -297,14 +299,14 @@ static int autofs_root_symlink(struct inode *dir, struct dentry *dentry, const c | |||
297 | sl = &sbi->symlink[n]; | 299 | sl = &sbi->symlink[n]; |
298 | sl->len = strlen(symname); | 300 | sl->len = strlen(symname); |
299 | sl->data = kmalloc(slsize = sl->len+1, GFP_KERNEL); | 301 | sl->data = kmalloc(slsize = sl->len+1, GFP_KERNEL); |
300 | if ( !sl->data ) { | 302 | if (!sl->data) { |
301 | clear_bit(n,sbi->symlink_bitmap); | 303 | clear_bit(n,sbi->symlink_bitmap); |
302 | unlock_kernel(); | 304 | unlock_kernel(); |
303 | return -ENOSPC; | 305 | return -ENOSPC; |
304 | } | 306 | } |
305 | 307 | ||
306 | ent = kmalloc(sizeof(struct autofs_dir_ent), GFP_KERNEL); | 308 | ent = kmalloc(sizeof(struct autofs_dir_ent), GFP_KERNEL); |
307 | if ( !ent ) { | 309 | if (!ent) { |
308 | kfree(sl->data); | 310 | kfree(sl->data); |
309 | clear_bit(n,sbi->symlink_bitmap); | 311 | clear_bit(n,sbi->symlink_bitmap); |
310 | unlock_kernel(); | 312 | unlock_kernel(); |
@@ -312,7 +314,7 @@ static int autofs_root_symlink(struct inode *dir, struct dentry *dentry, const c | |||
312 | } | 314 | } |
313 | 315 | ||
314 | ent->name = kmalloc(dentry->d_name.len+1, GFP_KERNEL); | 316 | ent->name = kmalloc(dentry->d_name.len+1, GFP_KERNEL); |
315 | if ( !ent->name ) { | 317 | if (!ent->name) { |
316 | kfree(sl->data); | 318 | kfree(sl->data); |
317 | kfree(ent); | 319 | kfree(ent); |
318 | clear_bit(n,sbi->symlink_bitmap); | 320 | clear_bit(n,sbi->symlink_bitmap); |
@@ -354,23 +356,23 @@ static int autofs_root_unlink(struct inode *dir, struct dentry *dentry) | |||
354 | 356 | ||
355 | /* This allows root to remove symlinks */ | 357 | /* This allows root to remove symlinks */ |
356 | lock_kernel(); | 358 | lock_kernel(); |
357 | if ( !autofs_oz_mode(sbi) && !capable(CAP_SYS_ADMIN) ) { | 359 | if (!autofs_oz_mode(sbi) && !capable(CAP_SYS_ADMIN)) { |
358 | unlock_kernel(); | 360 | unlock_kernel(); |
359 | return -EACCES; | 361 | return -EACCES; |
360 | } | 362 | } |
361 | 363 | ||
362 | ent = autofs_hash_lookup(dh, &dentry->d_name); | 364 | ent = autofs_hash_lookup(dh, &dentry->d_name); |
363 | if ( !ent ) { | 365 | if (!ent) { |
364 | unlock_kernel(); | 366 | unlock_kernel(); |
365 | return -ENOENT; | 367 | return -ENOENT; |
366 | } | 368 | } |
367 | 369 | ||
368 | n = ent->ino - AUTOFS_FIRST_SYMLINK; | 370 | n = ent->ino - AUTOFS_FIRST_SYMLINK; |
369 | if ( n >= AUTOFS_MAX_SYMLINKS ) { | 371 | if (n >= AUTOFS_MAX_SYMLINKS) { |
370 | unlock_kernel(); | 372 | unlock_kernel(); |
371 | return -EISDIR; /* It's a directory, dummy */ | 373 | return -EISDIR; /* It's a directory, dummy */ |
372 | } | 374 | } |
373 | if ( !test_bit(n,sbi->symlink_bitmap) ) { | 375 | if (!test_bit(n,sbi->symlink_bitmap)) { |
374 | unlock_kernel(); | 376 | unlock_kernel(); |
375 | return -EINVAL; /* Nonexistent symlink? Shouldn't happen */ | 377 | return -EINVAL; /* Nonexistent symlink? Shouldn't happen */ |
376 | } | 378 | } |
@@ -392,23 +394,23 @@ static int autofs_root_rmdir(struct inode *dir, struct dentry *dentry) | |||
392 | struct autofs_dir_ent *ent; | 394 | struct autofs_dir_ent *ent; |
393 | 395 | ||
394 | lock_kernel(); | 396 | lock_kernel(); |
395 | if ( !autofs_oz_mode(sbi) ) { | 397 | if (!autofs_oz_mode(sbi)) { |
396 | unlock_kernel(); | 398 | unlock_kernel(); |
397 | return -EACCES; | 399 | return -EACCES; |
398 | } | 400 | } |
399 | 401 | ||
400 | ent = autofs_hash_lookup(dh, &dentry->d_name); | 402 | ent = autofs_hash_lookup(dh, &dentry->d_name); |
401 | if ( !ent ) { | 403 | if (!ent) { |
402 | unlock_kernel(); | 404 | unlock_kernel(); |
403 | return -ENOENT; | 405 | return -ENOENT; |
404 | } | 406 | } |
405 | 407 | ||
406 | if ( (unsigned int)ent->ino < AUTOFS_FIRST_DIR_INO ) { | 408 | if ((unsigned int)ent->ino < AUTOFS_FIRST_DIR_INO) { |
407 | unlock_kernel(); | 409 | unlock_kernel(); |
408 | return -ENOTDIR; /* Not a directory */ | 410 | return -ENOTDIR; /* Not a directory */ |
409 | } | 411 | } |
410 | 412 | ||
411 | if ( ent->dentry != dentry ) { | 413 | if (ent->dentry != dentry) { |
412 | printk("autofs_rmdir: odentry != dentry for entry %s\n", dentry->d_name.name); | 414 | printk("autofs_rmdir: odentry != dentry for entry %s\n", dentry->d_name.name); |
413 | } | 415 | } |
414 | 416 | ||
@@ -429,18 +431,18 @@ static int autofs_root_mkdir(struct inode *dir, struct dentry *dentry, int mode) | |||
429 | ino_t ino; | 431 | ino_t ino; |
430 | 432 | ||
431 | lock_kernel(); | 433 | lock_kernel(); |
432 | if ( !autofs_oz_mode(sbi) ) { | 434 | if (!autofs_oz_mode(sbi)) { |
433 | unlock_kernel(); | 435 | unlock_kernel(); |
434 | return -EACCES; | 436 | return -EACCES; |
435 | } | 437 | } |
436 | 438 | ||
437 | ent = autofs_hash_lookup(dh, &dentry->d_name); | 439 | ent = autofs_hash_lookup(dh, &dentry->d_name); |
438 | if ( ent ) { | 440 | if (ent) { |
439 | unlock_kernel(); | 441 | unlock_kernel(); |
440 | return -EEXIST; | 442 | return -EEXIST; |
441 | } | 443 | } |
442 | 444 | ||
443 | if ( sbi->next_dir_ino < AUTOFS_FIRST_DIR_INO ) { | 445 | if (sbi->next_dir_ino < AUTOFS_FIRST_DIR_INO) { |
444 | printk("autofs: Out of inode numbers -- what the heck did you do??\n"); | 446 | printk("autofs: Out of inode numbers -- what the heck did you do??\n"); |
445 | unlock_kernel(); | 447 | unlock_kernel(); |
446 | return -ENOSPC; | 448 | return -ENOSPC; |
@@ -448,13 +450,13 @@ static int autofs_root_mkdir(struct inode *dir, struct dentry *dentry, int mode) | |||
448 | ino = sbi->next_dir_ino++; | 450 | ino = sbi->next_dir_ino++; |
449 | 451 | ||
450 | ent = kmalloc(sizeof(struct autofs_dir_ent), GFP_KERNEL); | 452 | ent = kmalloc(sizeof(struct autofs_dir_ent), GFP_KERNEL); |
451 | if ( !ent ) { | 453 | if (!ent) { |
452 | unlock_kernel(); | 454 | unlock_kernel(); |
453 | return -ENOSPC; | 455 | return -ENOSPC; |
454 | } | 456 | } |
455 | 457 | ||
456 | ent->name = kmalloc(dentry->d_name.len+1, GFP_KERNEL); | 458 | ent->name = kmalloc(dentry->d_name.len+1, GFP_KERNEL); |
457 | if ( !ent->name ) { | 459 | if (!ent->name) { |
458 | kfree(ent); | 460 | kfree(ent); |
459 | unlock_kernel(); | 461 | unlock_kernel(); |
460 | return -ENOSPC; | 462 | return -ENOSPC; |
@@ -483,7 +485,7 @@ static inline int autofs_get_set_timeout(struct autofs_sb_info *sbi, | |||
483 | put_user(sbi->exp_timeout / HZ, p)) | 485 | put_user(sbi->exp_timeout / HZ, p)) |
484 | return -EFAULT; | 486 | return -EFAULT; |
485 | 487 | ||
486 | if ( ntimeout > ULONG_MAX/HZ ) | 488 | if (ntimeout > ULONG_MAX/HZ) |
487 | sbi->exp_timeout = 0; | 489 | sbi->exp_timeout = 0; |
488 | else | 490 | else |
489 | sbi->exp_timeout = ntimeout * HZ; | 491 | sbi->exp_timeout = ntimeout * HZ; |
@@ -511,15 +513,14 @@ static inline int autofs_expire_run(struct super_block *sb, | |||
511 | pkt.hdr.proto_version = AUTOFS_PROTO_VERSION; | 513 | pkt.hdr.proto_version = AUTOFS_PROTO_VERSION; |
512 | pkt.hdr.type = autofs_ptype_expire; | 514 | pkt.hdr.type = autofs_ptype_expire; |
513 | 515 | ||
514 | if ( !sbi->exp_timeout || | 516 | if (!sbi->exp_timeout || !(ent = autofs_expire(sb,sbi,mnt))) |
515 | !(ent = autofs_expire(sb,sbi,mnt)) ) | ||
516 | return -EAGAIN; | 517 | return -EAGAIN; |
517 | 518 | ||
518 | pkt.len = ent->len; | 519 | pkt.len = ent->len; |
519 | memcpy(pkt.name, ent->name, pkt.len); | 520 | memcpy(pkt.name, ent->name, pkt.len); |
520 | pkt.name[pkt.len] = '\0'; | 521 | pkt.name[pkt.len] = '\0'; |
521 | 522 | ||
522 | if ( copy_to_user(pkt_p, &pkt, sizeof(struct autofs_packet_expire)) ) | 523 | if (copy_to_user(pkt_p, &pkt, sizeof(struct autofs_packet_expire))) |
523 | return -EFAULT; | 524 | return -EFAULT; |
524 | 525 | ||
525 | return 0; | 526 | return 0; |
@@ -537,11 +538,11 @@ static int autofs_root_ioctl(struct inode *inode, struct file *filp, | |||
537 | 538 | ||
538 | DPRINTK(("autofs_ioctl: cmd = 0x%08x, arg = 0x%08lx, sbi = %p, pgrp = %u\n",cmd,arg,sbi,process_group(current))); | 539 | DPRINTK(("autofs_ioctl: cmd = 0x%08x, arg = 0x%08lx, sbi = %p, pgrp = %u\n",cmd,arg,sbi,process_group(current))); |
539 | 540 | ||
540 | if ( _IOC_TYPE(cmd) != _IOC_TYPE(AUTOFS_IOC_FIRST) || | 541 | if (_IOC_TYPE(cmd) != _IOC_TYPE(AUTOFS_IOC_FIRST) || |
541 | _IOC_NR(cmd) - _IOC_NR(AUTOFS_IOC_FIRST) >= AUTOFS_IOC_COUNT ) | 542 | _IOC_NR(cmd) - _IOC_NR(AUTOFS_IOC_FIRST) >= AUTOFS_IOC_COUNT) |
542 | return -ENOTTY; | 543 | return -ENOTTY; |
543 | 544 | ||
544 | if ( !autofs_oz_mode(sbi) && !capable(CAP_SYS_ADMIN) ) | 545 | if (!autofs_oz_mode(sbi) && !capable(CAP_SYS_ADMIN)) |
545 | return -EPERM; | 546 | return -EPERM; |
546 | 547 | ||
547 | switch(cmd) { | 548 | switch(cmd) { |