dir.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486
  1. // SPDX-License-Identifier: MIT
  2. /*
  3. * VirtualBox Guest Shared Folders support: Directory inode and file operations
  4. *
  5. * Copyright (C) 2006-2018 Oracle Corporation
  6. */
  7. #include <linux/namei.h>
  8. #include <linux/vbox_utils.h>
  9. #include "vfsmod.h"
  10. static int vboxsf_dir_open(struct inode *inode, struct file *file)
  11. {
  12. struct vboxsf_sbi *sbi = VBOXSF_SBI(inode->i_sb);
  13. struct shfl_createparms params = {};
  14. struct vboxsf_dir_info *sf_d;
  15. int err;
  16. sf_d = vboxsf_dir_info_alloc();
  17. if (!sf_d)
  18. return -ENOMEM;
  19. params.handle = SHFL_HANDLE_NIL;
  20. params.create_flags = SHFL_CF_DIRECTORY | SHFL_CF_ACT_OPEN_IF_EXISTS |
  21. SHFL_CF_ACT_FAIL_IF_NEW | SHFL_CF_ACCESS_READ;
  22. err = vboxsf_create_at_dentry(file_dentry(file), &params);
  23. if (err)
  24. goto err_free_dir_info;
  25. if (params.result != SHFL_FILE_EXISTS) {
  26. err = -ENOENT;
  27. goto err_close;
  28. }
  29. err = vboxsf_dir_read_all(sbi, sf_d, params.handle);
  30. if (err)
  31. goto err_close;
  32. vboxsf_close(sbi->root, params.handle);
  33. file->private_data = sf_d;
  34. return 0;
  35. err_close:
  36. vboxsf_close(sbi->root, params.handle);
  37. err_free_dir_info:
  38. vboxsf_dir_info_free(sf_d);
  39. return err;
  40. }
  41. static int vboxsf_dir_release(struct inode *inode, struct file *file)
  42. {
  43. if (file->private_data)
  44. vboxsf_dir_info_free(file->private_data);
  45. return 0;
  46. }
  47. static unsigned int vboxsf_get_d_type(u32 mode)
  48. {
  49. unsigned int d_type;
  50. switch (mode & SHFL_TYPE_MASK) {
  51. case SHFL_TYPE_FIFO:
  52. d_type = DT_FIFO;
  53. break;
  54. case SHFL_TYPE_DEV_CHAR:
  55. d_type = DT_CHR;
  56. break;
  57. case SHFL_TYPE_DIRECTORY:
  58. d_type = DT_DIR;
  59. break;
  60. case SHFL_TYPE_DEV_BLOCK:
  61. d_type = DT_BLK;
  62. break;
  63. case SHFL_TYPE_FILE:
  64. d_type = DT_REG;
  65. break;
  66. case SHFL_TYPE_SYMLINK:
  67. d_type = DT_LNK;
  68. break;
  69. case SHFL_TYPE_SOCKET:
  70. d_type = DT_SOCK;
  71. break;
  72. case SHFL_TYPE_WHITEOUT:
  73. d_type = DT_WHT;
  74. break;
  75. default:
  76. d_type = DT_UNKNOWN;
  77. break;
  78. }
  79. return d_type;
  80. }
  81. static bool vboxsf_dir_emit(struct file *dir, struct dir_context *ctx)
  82. {
  83. struct vboxsf_sbi *sbi = VBOXSF_SBI(file_inode(dir)->i_sb);
  84. struct vboxsf_dir_info *sf_d = dir->private_data;
  85. struct shfl_dirinfo *info;
  86. struct vboxsf_dir_buf *b;
  87. unsigned int d_type;
  88. loff_t i, cur = 0;
  89. ino_t fake_ino;
  90. void *end;
  91. int err;
  92. list_for_each_entry(b, &sf_d->info_list, head) {
  93. try_next_entry:
  94. if (ctx->pos >= cur + b->entries) {
  95. cur += b->entries;
  96. continue;
  97. }
  98. /*
  99. * Note the vboxsf_dir_info objects we are iterating over here
  100. * are variable sized, so the info pointer may end up being
  101. * unaligned. This is how we get the data from the host.
  102. * Since vboxsf is only supported on x86 machines this is not
  103. * a problem.
  104. */
  105. for (i = 0, info = b->buf; i < ctx->pos - cur; i++) {
  106. end = &info->name.string.utf8[info->name.size];
  107. /* Only happens if the host gives us corrupt data */
  108. if (WARN_ON(end > (b->buf + b->used)))
  109. return false;
  110. info = end;
  111. }
  112. end = &info->name.string.utf8[info->name.size];
  113. if (WARN_ON(end > (b->buf + b->used)))
  114. return false;
  115. /* Info now points to the right entry, emit it. */
  116. d_type = vboxsf_get_d_type(info->info.attr.mode);
  117. /*
  118. * On 32-bit systems pos is 64-bit signed, while ino is 32-bit
  119. * unsigned so fake_ino may overflow, check for this.
  120. */
  121. if ((ino_t)(ctx->pos + 1) != (u64)(ctx->pos + 1)) {
  122. vbg_err("vboxsf: fake ino overflow, truncating dir\n");
  123. return false;
  124. }
  125. fake_ino = ctx->pos + 1;
  126. if (sbi->nls) {
  127. char d_name[NAME_MAX];
  128. err = vboxsf_nlscpy(sbi, d_name, NAME_MAX,
  129. info->name.string.utf8,
  130. info->name.length);
  131. if (err) {
  132. /* skip erroneous entry and proceed */
  133. ctx->pos += 1;
  134. goto try_next_entry;
  135. }
  136. return dir_emit(ctx, d_name, strlen(d_name),
  137. fake_ino, d_type);
  138. }
  139. return dir_emit(ctx, info->name.string.utf8, info->name.length,
  140. fake_ino, d_type);
  141. }
  142. return false;
  143. }
  144. static int vboxsf_dir_iterate(struct file *dir, struct dir_context *ctx)
  145. {
  146. bool emitted;
  147. do {
  148. emitted = vboxsf_dir_emit(dir, ctx);
  149. if (emitted)
  150. ctx->pos += 1;
  151. } while (emitted);
  152. return 0;
  153. }
  154. WRAP_DIR_ITER(vboxsf_dir_iterate) // FIXME!
  155. const struct file_operations vboxsf_dir_fops = {
  156. .open = vboxsf_dir_open,
  157. .iterate_shared = shared_vboxsf_dir_iterate,
  158. .release = vboxsf_dir_release,
  159. .read = generic_read_dir,
  160. .llseek = generic_file_llseek,
  161. };
  162. /*
  163. * This is called during name resolution/lookup to check if the @dentry in
  164. * the cache is still valid. the job is handled by vboxsf_inode_revalidate.
  165. */
  166. static int vboxsf_dentry_revalidate(struct dentry *dentry, unsigned int flags)
  167. {
  168. if (flags & LOOKUP_RCU)
  169. return -ECHILD;
  170. if (d_really_is_positive(dentry))
  171. return vboxsf_inode_revalidate(dentry) == 0;
  172. else
  173. return vboxsf_stat_dentry(dentry, NULL) == -ENOENT;
  174. }
  175. const struct dentry_operations vboxsf_dentry_ops = {
  176. .d_revalidate = vboxsf_dentry_revalidate
  177. };
  178. /* iops */
  179. static struct dentry *vboxsf_dir_lookup(struct inode *parent,
  180. struct dentry *dentry,
  181. unsigned int flags)
  182. {
  183. struct vboxsf_sbi *sbi = VBOXSF_SBI(parent->i_sb);
  184. struct shfl_fsobjinfo fsinfo;
  185. struct inode *inode;
  186. int err;
  187. dentry->d_time = jiffies;
  188. err = vboxsf_stat_dentry(dentry, &fsinfo);
  189. if (err) {
  190. inode = (err == -ENOENT) ? NULL : ERR_PTR(err);
  191. } else {
  192. inode = vboxsf_new_inode(parent->i_sb);
  193. if (!IS_ERR(inode))
  194. vboxsf_init_inode(sbi, inode, &fsinfo, false);
  195. }
  196. return d_splice_alias(inode, dentry);
  197. }
  198. static int vboxsf_dir_instantiate(struct inode *parent, struct dentry *dentry,
  199. struct shfl_fsobjinfo *info)
  200. {
  201. struct vboxsf_sbi *sbi = VBOXSF_SBI(parent->i_sb);
  202. struct vboxsf_inode *sf_i;
  203. struct inode *inode;
  204. inode = vboxsf_new_inode(parent->i_sb);
  205. if (IS_ERR(inode))
  206. return PTR_ERR(inode);
  207. sf_i = VBOXSF_I(inode);
  208. /* The host may have given us different attr then requested */
  209. sf_i->force_restat = 1;
  210. vboxsf_init_inode(sbi, inode, info, false);
  211. d_instantiate(dentry, inode);
  212. return 0;
  213. }
  214. static int vboxsf_dir_create(struct inode *parent, struct dentry *dentry,
  215. umode_t mode, bool is_dir, bool excl, u64 *handle_ret)
  216. {
  217. struct vboxsf_inode *sf_parent_i = VBOXSF_I(parent);
  218. struct vboxsf_sbi *sbi = VBOXSF_SBI(parent->i_sb);
  219. struct shfl_createparms params = {};
  220. int err;
  221. params.handle = SHFL_HANDLE_NIL;
  222. params.create_flags = SHFL_CF_ACT_CREATE_IF_NEW | SHFL_CF_ACCESS_READWRITE;
  223. if (is_dir)
  224. params.create_flags |= SHFL_CF_DIRECTORY;
  225. if (excl)
  226. params.create_flags |= SHFL_CF_ACT_FAIL_IF_EXISTS;
  227. params.info.attr.mode = (mode & 0777) |
  228. (is_dir ? SHFL_TYPE_DIRECTORY : SHFL_TYPE_FILE);
  229. params.info.attr.additional = SHFLFSOBJATTRADD_NOTHING;
  230. err = vboxsf_create_at_dentry(dentry, &params);
  231. if (err)
  232. return err;
  233. if (params.result != SHFL_FILE_CREATED)
  234. return -EPERM;
  235. err = vboxsf_dir_instantiate(parent, dentry, &params.info);
  236. if (err)
  237. goto out;
  238. /* parent directory access/change time changed */
  239. sf_parent_i->force_restat = 1;
  240. out:
  241. if (err == 0 && handle_ret)
  242. *handle_ret = params.handle;
  243. else
  244. vboxsf_close(sbi->root, params.handle);
  245. return err;
  246. }
  247. static int vboxsf_dir_mkfile(struct mnt_idmap *idmap,
  248. struct inode *parent, struct dentry *dentry,
  249. umode_t mode, bool excl)
  250. {
  251. return vboxsf_dir_create(parent, dentry, mode, false, excl, NULL);
  252. }
  253. static int vboxsf_dir_mkdir(struct mnt_idmap *idmap,
  254. struct inode *parent, struct dentry *dentry,
  255. umode_t mode)
  256. {
  257. return vboxsf_dir_create(parent, dentry, mode, true, true, NULL);
  258. }
  259. static int vboxsf_dir_atomic_open(struct inode *parent, struct dentry *dentry,
  260. struct file *file, unsigned int flags, umode_t mode)
  261. {
  262. struct vboxsf_sbi *sbi = VBOXSF_SBI(parent->i_sb);
  263. struct vboxsf_handle *sf_handle;
  264. struct dentry *res = NULL;
  265. u64 handle;
  266. int err;
  267. if (d_in_lookup(dentry)) {
  268. res = vboxsf_dir_lookup(parent, dentry, 0);
  269. if (IS_ERR(res))
  270. return PTR_ERR(res);
  271. if (res)
  272. dentry = res;
  273. }
  274. /* Only creates */
  275. if (!(flags & O_CREAT) || d_really_is_positive(dentry))
  276. return finish_no_open(file, res);
  277. err = vboxsf_dir_create(parent, dentry, mode, false, flags & O_EXCL, &handle);
  278. if (err)
  279. goto out;
  280. sf_handle = vboxsf_create_sf_handle(d_inode(dentry), handle, SHFL_CF_ACCESS_READWRITE);
  281. if (IS_ERR(sf_handle)) {
  282. vboxsf_close(sbi->root, handle);
  283. err = PTR_ERR(sf_handle);
  284. goto out;
  285. }
  286. err = finish_open(file, dentry, generic_file_open);
  287. if (err) {
  288. /* This also closes the handle passed to vboxsf_create_sf_handle() */
  289. vboxsf_release_sf_handle(d_inode(dentry), sf_handle);
  290. goto out;
  291. }
  292. file->private_data = sf_handle;
  293. file->f_mode |= FMODE_CREATED;
  294. out:
  295. dput(res);
  296. return err;
  297. }
  298. static int vboxsf_dir_unlink(struct inode *parent, struct dentry *dentry)
  299. {
  300. struct vboxsf_sbi *sbi = VBOXSF_SBI(parent->i_sb);
  301. struct vboxsf_inode *sf_parent_i = VBOXSF_I(parent);
  302. struct inode *inode = d_inode(dentry);
  303. struct shfl_string *path;
  304. u32 flags;
  305. int err;
  306. if (S_ISDIR(inode->i_mode))
  307. flags = SHFL_REMOVE_DIR;
  308. else
  309. flags = SHFL_REMOVE_FILE;
  310. if (S_ISLNK(inode->i_mode))
  311. flags |= SHFL_REMOVE_SYMLINK;
  312. path = vboxsf_path_from_dentry(sbi, dentry);
  313. if (IS_ERR(path))
  314. return PTR_ERR(path);
  315. err = vboxsf_remove(sbi->root, path, flags);
  316. __putname(path);
  317. if (err)
  318. return err;
  319. /* parent directory access/change time changed */
  320. sf_parent_i->force_restat = 1;
  321. return 0;
  322. }
  323. static int vboxsf_dir_rename(struct mnt_idmap *idmap,
  324. struct inode *old_parent,
  325. struct dentry *old_dentry,
  326. struct inode *new_parent,
  327. struct dentry *new_dentry,
  328. unsigned int flags)
  329. {
  330. struct vboxsf_sbi *sbi = VBOXSF_SBI(old_parent->i_sb);
  331. struct vboxsf_inode *sf_old_parent_i = VBOXSF_I(old_parent);
  332. struct vboxsf_inode *sf_new_parent_i = VBOXSF_I(new_parent);
  333. u32 shfl_flags = SHFL_RENAME_FILE | SHFL_RENAME_REPLACE_IF_EXISTS;
  334. struct shfl_string *old_path, *new_path;
  335. int err;
  336. if (flags)
  337. return -EINVAL;
  338. old_path = vboxsf_path_from_dentry(sbi, old_dentry);
  339. if (IS_ERR(old_path))
  340. return PTR_ERR(old_path);
  341. new_path = vboxsf_path_from_dentry(sbi, new_dentry);
  342. if (IS_ERR(new_path)) {
  343. err = PTR_ERR(new_path);
  344. goto err_put_old_path;
  345. }
  346. if (d_inode(old_dentry)->i_mode & S_IFDIR)
  347. shfl_flags = 0;
  348. err = vboxsf_rename(sbi->root, old_path, new_path, shfl_flags);
  349. if (err == 0) {
  350. /* parent directories access/change time changed */
  351. sf_new_parent_i->force_restat = 1;
  352. sf_old_parent_i->force_restat = 1;
  353. }
  354. __putname(new_path);
  355. err_put_old_path:
  356. __putname(old_path);
  357. return err;
  358. }
  359. static int vboxsf_dir_symlink(struct mnt_idmap *idmap,
  360. struct inode *parent, struct dentry *dentry,
  361. const char *symname)
  362. {
  363. struct vboxsf_inode *sf_parent_i = VBOXSF_I(parent);
  364. struct vboxsf_sbi *sbi = VBOXSF_SBI(parent->i_sb);
  365. int symname_size = strlen(symname) + 1;
  366. struct shfl_string *path, *ssymname;
  367. struct shfl_fsobjinfo info;
  368. int err;
  369. path = vboxsf_path_from_dentry(sbi, dentry);
  370. if (IS_ERR(path))
  371. return PTR_ERR(path);
  372. ssymname = kmalloc(SHFLSTRING_HEADER_SIZE + symname_size, GFP_KERNEL);
  373. if (!ssymname) {
  374. __putname(path);
  375. return -ENOMEM;
  376. }
  377. ssymname->length = symname_size - 1;
  378. ssymname->size = symname_size;
  379. memcpy(ssymname->string.utf8, symname, symname_size);
  380. err = vboxsf_symlink(sbi->root, path, ssymname, &info);
  381. kfree(ssymname);
  382. __putname(path);
  383. if (err) {
  384. /* -EROFS means symlinks are note support -> -EPERM */
  385. return (err == -EROFS) ? -EPERM : err;
  386. }
  387. err = vboxsf_dir_instantiate(parent, dentry, &info);
  388. if (err)
  389. return err;
  390. /* parent directory access/change time changed */
  391. sf_parent_i->force_restat = 1;
  392. return 0;
  393. }
  394. const struct inode_operations vboxsf_dir_iops = {
  395. .lookup = vboxsf_dir_lookup,
  396. .create = vboxsf_dir_mkfile,
  397. .mkdir = vboxsf_dir_mkdir,
  398. .atomic_open = vboxsf_dir_atomic_open,
  399. .rmdir = vboxsf_dir_unlink,
  400. .unlink = vboxsf_dir_unlink,
  401. .rename = vboxsf_dir_rename,
  402. .symlink = vboxsf_dir_symlink,
  403. .getattr = vboxsf_getattr,
  404. .setattr = vboxsf_setattr,
  405. };