statfs.c 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230
  1. #include <linux/syscalls.h>
  2. #include <linux/module.h>
  3. #include <linux/fs.h>
  4. #include <linux/file.h>
  5. #include <linux/mount.h>
  6. #include <linux/namei.h>
  7. #include <linux/statfs.h>
  8. #include <linux/security.h>
  9. #include <linux/uaccess.h>
  10. static int flags_by_mnt(int mnt_flags)
  11. {
  12. int flags = 0;
  13. if (mnt_flags & MNT_READONLY)
  14. flags |= ST_RDONLY;
  15. if (mnt_flags & MNT_NOSUID)
  16. flags |= ST_NOSUID;
  17. if (mnt_flags & MNT_NODEV)
  18. flags |= ST_NODEV;
  19. if (mnt_flags & MNT_NOEXEC)
  20. flags |= ST_NOEXEC;
  21. if (mnt_flags & MNT_NOATIME)
  22. flags |= ST_NOATIME;
  23. if (mnt_flags & MNT_NODIRATIME)
  24. flags |= ST_NODIRATIME;
  25. if (mnt_flags & MNT_RELATIME)
  26. flags |= ST_RELATIME;
  27. return flags;
  28. }
  29. static int flags_by_sb(int s_flags)
  30. {
  31. int flags = 0;
  32. if (s_flags & MS_SYNCHRONOUS)
  33. flags |= ST_SYNCHRONOUS;
  34. if (s_flags & MS_MANDLOCK)
  35. flags |= ST_MANDLOCK;
  36. return flags;
  37. }
  38. static int calculate_f_flags(struct vfsmount *mnt)
  39. {
  40. return ST_VALID | flags_by_mnt(mnt->mnt_flags) |
  41. flags_by_sb(mnt->mnt_sb->s_flags);
  42. }
  43. int statfs_by_dentry(struct dentry *dentry, struct kstatfs *buf)
  44. {
  45. int retval;
  46. if (!dentry->d_sb->s_op->statfs)
  47. return -ENOSYS;
  48. memset(buf, 0, sizeof(*buf));
  49. retval = security_sb_statfs(dentry);
  50. if (retval)
  51. return retval;
  52. retval = dentry->d_sb->s_op->statfs(dentry, buf);
  53. if (retval == 0 && buf->f_frsize == 0)
  54. buf->f_frsize = buf->f_bsize;
  55. return retval;
  56. }
  57. int vfs_statfs(struct path *path, struct kstatfs *buf)
  58. {
  59. int error;
  60. error = statfs_by_dentry(path->dentry, buf);
  61. if (!error)
  62. buf->f_flags = calculate_f_flags(path->mnt);
  63. return error;
  64. }
  65. EXPORT_SYMBOL(vfs_statfs);
  66. int user_statfs(const char __user *pathname, struct kstatfs *st)
  67. {
  68. struct path path;
  69. int error = user_path_at(AT_FDCWD, pathname, LOOKUP_FOLLOW|LOOKUP_AUTOMOUNT, &path);
  70. if (!error) {
  71. error = vfs_statfs(&path, st);
  72. path_put(&path);
  73. }
  74. return error;
  75. }
  76. int fd_statfs(int fd, struct kstatfs *st)
  77. {
  78. struct file *file = fget(fd);
  79. int error = -EBADF;
  80. if (file) {
  81. error = vfs_statfs(&file->f_path, st);
  82. fput(file);
  83. }
  84. return error;
  85. }
  86. static int do_statfs_native(struct kstatfs *st, struct statfs __user *p)
  87. {
  88. struct statfs buf;
  89. if (sizeof(buf) == sizeof(*st))
  90. memcpy(&buf, st, sizeof(*st));
  91. else {
  92. if (sizeof buf.f_blocks == 4) {
  93. if ((st->f_blocks | st->f_bfree | st->f_bavail |
  94. st->f_bsize | st->f_frsize) &
  95. 0xffffffff00000000ULL)
  96. return -EOVERFLOW;
  97. /*
  98. * f_files and f_ffree may be -1; it's okay to stuff
  99. * that into 32 bits
  100. */
  101. if (st->f_files != -1 &&
  102. (st->f_files & 0xffffffff00000000ULL))
  103. return -EOVERFLOW;
  104. if (st->f_ffree != -1 &&
  105. (st->f_ffree & 0xffffffff00000000ULL))
  106. return -EOVERFLOW;
  107. }
  108. buf.f_type = st->f_type;
  109. buf.f_bsize = st->f_bsize;
  110. buf.f_blocks = st->f_blocks;
  111. buf.f_bfree = st->f_bfree;
  112. buf.f_bavail = st->f_bavail;
  113. buf.f_files = st->f_files;
  114. buf.f_ffree = st->f_ffree;
  115. buf.f_fsid = st->f_fsid;
  116. buf.f_namelen = st->f_namelen;
  117. buf.f_frsize = st->f_frsize;
  118. buf.f_flags = st->f_flags;
  119. memset(buf.f_spare, 0, sizeof(buf.f_spare));
  120. }
  121. if (copy_to_user(p, &buf, sizeof(buf)))
  122. return -EFAULT;
  123. return 0;
  124. }
  125. static int do_statfs64(struct kstatfs *st, struct statfs64 __user *p)
  126. {
  127. struct statfs64 buf;
  128. if (sizeof(buf) == sizeof(*st))
  129. memcpy(&buf, st, sizeof(*st));
  130. else {
  131. buf.f_type = st->f_type;
  132. buf.f_bsize = st->f_bsize;
  133. buf.f_blocks = st->f_blocks;
  134. buf.f_bfree = st->f_bfree;
  135. buf.f_bavail = st->f_bavail;
  136. buf.f_files = st->f_files;
  137. buf.f_ffree = st->f_ffree;
  138. buf.f_fsid = st->f_fsid;
  139. buf.f_namelen = st->f_namelen;
  140. buf.f_frsize = st->f_frsize;
  141. buf.f_flags = st->f_flags;
  142. memset(buf.f_spare, 0, sizeof(buf.f_spare));
  143. }
  144. if (copy_to_user(p, &buf, sizeof(buf)))
  145. return -EFAULT;
  146. return 0;
  147. }
  148. SYSCALL_DEFINE2(statfs, const char __user *, pathname, struct statfs __user *, buf)
  149. {
  150. struct kstatfs st;
  151. int error = user_statfs(pathname, &st);
  152. if (!error)
  153. error = do_statfs_native(&st, buf);
  154. return error;
  155. }
  156. SYSCALL_DEFINE3(statfs64, const char __user *, pathname, size_t, sz, struct statfs64 __user *, buf)
  157. {
  158. struct kstatfs st;
  159. int error;
  160. if (sz != sizeof(*buf))
  161. return -EINVAL;
  162. error = user_statfs(pathname, &st);
  163. if (!error)
  164. error = do_statfs64(&st, buf);
  165. return error;
  166. }
  167. SYSCALL_DEFINE2(fstatfs, unsigned int, fd, struct statfs __user *, buf)
  168. {
  169. struct kstatfs st;
  170. int error = fd_statfs(fd, &st);
  171. if (!error)
  172. error = do_statfs_native(&st, buf);
  173. return error;
  174. }
  175. SYSCALL_DEFINE3(fstatfs64, unsigned int, fd, size_t, sz, struct statfs64 __user *, buf)
  176. {
  177. struct kstatfs st;
  178. int error;
  179. if (sz != sizeof(*buf))
  180. return -EINVAL;
  181. error = fd_statfs(fd, &st);
  182. if (!error)
  183. error = do_statfs64(&st, buf);
  184. return error;
  185. }
  186. SYSCALL_DEFINE2(ustat, unsigned, dev, struct ustat __user *, ubuf)
  187. {
  188. struct super_block *s;
  189. struct ustat tmp;
  190. struct kstatfs sbuf;
  191. int err;
  192. s = user_get_super(new_decode_dev(dev));
  193. if (!s)
  194. return -EINVAL;
  195. err = statfs_by_dentry(s->s_root, &sbuf);
  196. drop_super(s);
  197. if (err)
  198. return err;
  199. memset(&tmp,0,sizeof(struct ustat));
  200. tmp.f_tfree = sbuf.f_bfree;
  201. tmp.f_tinode = sbuf.f_ffree;
  202. return copy_to_user(ubuf, &tmp, sizeof(struct ustat)) ? -EFAULT : 0;
  203. }