pioctl.c
上传用户:jlfgdled
上传日期:2013-04-10
资源大小:33168k
文件大小:3k
源码类别:

Linux/Unix编程

开发平台:

Unix_Linux

  1. /*
  2.  * Pioctl operations for Coda.
  3.  * Original version: (C) 1996 Peter Braam 
  4.  * Rewritten for Linux 2.1: (C) 1997 Carnegie Mellon University
  5.  *
  6.  * Carnegie Mellon encourages users of this code to contribute improvements
  7.  * to the Coda project. Contact Peter Braam <coda@cs.cmu.edu>.
  8.  */
  9. #include <linux/types.h>
  10. #include <linux/kernel.h>
  11. #include <linux/sched.h>
  12. #include <linux/fs.h>
  13. #include <linux/stat.h>
  14. #include <linux/errno.h>
  15. #include <linux/locks.h>
  16. #include <linux/string.h>
  17. #define __NO_VERSION__
  18. #include <linux/module.h>
  19. #include <asm/uaccess.h>
  20. #include <linux/coda.h>
  21. #include <linux/coda_linux.h>
  22. #include <linux/coda_fs_i.h>
  23. #include <linux/coda_psdev.h>
  24. /* pioctl ops */
  25. static int coda_ioctl_permission(struct inode *inode, int mask);
  26. static int coda_pioctl(struct inode * inode, struct file * filp, 
  27.                        unsigned int cmd, unsigned long user_data);
  28. /* exported from this file */
  29. struct inode_operations coda_ioctl_inode_operations =
  30. {
  31. permission: coda_ioctl_permission,
  32. setattr: coda_notify_change,
  33. };
  34. struct file_operations coda_ioctl_operations = {
  35. owner: THIS_MODULE,
  36. ioctl: coda_pioctl,
  37. };
  38. /* the coda pioctl inode ops */
  39. static int coda_ioctl_permission(struct inode *inode, int mask)
  40. {
  41.         return 0;
  42. }
  43. static int coda_pioctl(struct inode * inode, struct file * filp, 
  44.                        unsigned int cmd, unsigned long user_data)
  45. {
  46. struct nameidata nd;
  47.         int error;
  48. struct PioctlData data;
  49.         struct inode *target_inode = NULL;
  50.         struct coda_inode_info *cnp;
  51.         /* get the Pioctl data arguments from user space */
  52.         if (copy_from_user(&data, (int *)user_data, sizeof(data))) {
  53.     return -EINVAL;
  54. }
  55.        
  56.         /* 
  57.          * Look up the pathname. Note that the pathname is in 
  58.          * user memory, and namei takes care of this
  59.          */
  60. CDEBUG(D_PIOCTL, "namei, data.follow = %dn", 
  61.        data.follow);
  62.         if ( data.follow ) {
  63.                 error = user_path_walk(data.path, &nd);
  64. } else {
  65.         error = user_path_walk_link(data.path, &nd);
  66. }
  67. if ( error ) {
  68.                 CDEBUG(D_PIOCTL, "error: lookup fails.n");
  69. return error;
  70.         } else {
  71.         target_inode = nd.dentry->d_inode;
  72. }
  73. CDEBUG(D_PIOCTL, "target ino: 0x%ld, dev: 0x%xn",
  74.        target_inode->i_ino, target_inode->i_dev);
  75. /* return if it is not a Coda inode */
  76. if ( target_inode->i_sb != inode->i_sb ) {
  77. path_release(&nd);
  78.         return  -EINVAL;
  79. }
  80. /* now proceed to make the upcall */
  81.         cnp = ITOC(target_inode);
  82. error = venus_pioctl(inode->i_sb, &(cnp->c_fid), cmd, &data);
  83.         CDEBUG(D_PIOCTL, "ioctl on inode %ldn", target_inode->i_ino);
  84. CDEBUG(D_DOWNCALL, "dput on ino: %ld, icount %d, dcount %dn", target_inode->i_ino, 
  85.        atomic_read(&target_inode->i_count), atomic_read(&nd.dentry->d_count));
  86. path_release(&nd);
  87.         return error;
  88. }