dir_edit.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505
  1. /* AFS filesystem directory editing
  2. *
  3. * Copyright (C) 2018 Red Hat, Inc. All Rights Reserved.
  4. * Written by David Howells (dhowells@redhat.com)
  5. *
  6. * This program is free software; you can redistribute it and/or
  7. * modify it under the terms of the GNU General Public Licence
  8. * as published by the Free Software Foundation; either version
  9. * 2 of the Licence, or (at your option) any later version.
  10. */
  11. #include <linux/kernel.h>
  12. #include <linux/fs.h>
  13. #include <linux/namei.h>
  14. #include <linux/pagemap.h>
  15. #include <linux/iversion.h>
  16. #include "internal.h"
  17. #include "xdr_fs.h"
  18. /*
  19. * Find a number of contiguous clear bits in a directory block bitmask.
  20. *
  21. * There are 64 slots, which means we can load the entire bitmap into a
  22. * variable. The first bit doesn't count as it corresponds to the block header
  23. * slot. nr_slots is between 1 and 9.
  24. */
  25. static int afs_find_contig_bits(union afs_xdr_dir_block *block, unsigned int nr_slots)
  26. {
  27. u64 bitmap;
  28. u32 mask;
  29. int bit, n;
  30. bitmap = (u64)block->hdr.bitmap[0] << 0 * 8;
  31. bitmap |= (u64)block->hdr.bitmap[1] << 1 * 8;
  32. bitmap |= (u64)block->hdr.bitmap[2] << 2 * 8;
  33. bitmap |= (u64)block->hdr.bitmap[3] << 3 * 8;
  34. bitmap |= (u64)block->hdr.bitmap[4] << 4 * 8;
  35. bitmap |= (u64)block->hdr.bitmap[5] << 5 * 8;
  36. bitmap |= (u64)block->hdr.bitmap[6] << 6 * 8;
  37. bitmap |= (u64)block->hdr.bitmap[7] << 7 * 8;
  38. bitmap >>= 1; /* The first entry is metadata */
  39. bit = 1;
  40. mask = (1 << nr_slots) - 1;
  41. do {
  42. if (sizeof(unsigned long) == 8)
  43. n = ffz(bitmap);
  44. else
  45. n = ((u32)bitmap) != 0 ?
  46. ffz((u32)bitmap) :
  47. ffz((u32)(bitmap >> 32)) + 32;
  48. bitmap >>= n;
  49. bit += n;
  50. if ((bitmap & mask) == 0) {
  51. if (bit > 64 - nr_slots)
  52. return -1;
  53. return bit;
  54. }
  55. n = __ffs(bitmap);
  56. bitmap >>= n;
  57. bit += n;
  58. } while (bitmap);
  59. return -1;
  60. }
  61. /*
  62. * Set a number of contiguous bits in the directory block bitmap.
  63. */
  64. static void afs_set_contig_bits(union afs_xdr_dir_block *block,
  65. int bit, unsigned int nr_slots)
  66. {
  67. u64 mask, before, after;
  68. mask = (1 << nr_slots) - 1;
  69. mask <<= bit;
  70. before = *(u64 *)block->hdr.bitmap;
  71. block->hdr.bitmap[0] |= (u8)(mask >> 0 * 8);
  72. block->hdr.bitmap[1] |= (u8)(mask >> 1 * 8);
  73. block->hdr.bitmap[2] |= (u8)(mask >> 2 * 8);
  74. block->hdr.bitmap[3] |= (u8)(mask >> 3 * 8);
  75. block->hdr.bitmap[4] |= (u8)(mask >> 4 * 8);
  76. block->hdr.bitmap[5] |= (u8)(mask >> 5 * 8);
  77. block->hdr.bitmap[6] |= (u8)(mask >> 6 * 8);
  78. block->hdr.bitmap[7] |= (u8)(mask >> 7 * 8);
  79. after = *(u64 *)block->hdr.bitmap;
  80. }
  81. /*
  82. * Clear a number of contiguous bits in the directory block bitmap.
  83. */
  84. static void afs_clear_contig_bits(union afs_xdr_dir_block *block,
  85. int bit, unsigned int nr_slots)
  86. {
  87. u64 mask, before, after;
  88. mask = (1 << nr_slots) - 1;
  89. mask <<= bit;
  90. before = *(u64 *)block->hdr.bitmap;
  91. block->hdr.bitmap[0] &= ~(u8)(mask >> 0 * 8);
  92. block->hdr.bitmap[1] &= ~(u8)(mask >> 1 * 8);
  93. block->hdr.bitmap[2] &= ~(u8)(mask >> 2 * 8);
  94. block->hdr.bitmap[3] &= ~(u8)(mask >> 3 * 8);
  95. block->hdr.bitmap[4] &= ~(u8)(mask >> 4 * 8);
  96. block->hdr.bitmap[5] &= ~(u8)(mask >> 5 * 8);
  97. block->hdr.bitmap[6] &= ~(u8)(mask >> 6 * 8);
  98. block->hdr.bitmap[7] &= ~(u8)(mask >> 7 * 8);
  99. after = *(u64 *)block->hdr.bitmap;
  100. }
  101. /*
  102. * Scan a directory block looking for a dirent of the right name.
  103. */
  104. static int afs_dir_scan_block(union afs_xdr_dir_block *block, struct qstr *name,
  105. unsigned int blocknum)
  106. {
  107. union afs_xdr_dirent *de;
  108. u64 bitmap;
  109. int d, len, n;
  110. _enter("");
  111. bitmap = (u64)block->hdr.bitmap[0] << 0 * 8;
  112. bitmap |= (u64)block->hdr.bitmap[1] << 1 * 8;
  113. bitmap |= (u64)block->hdr.bitmap[2] << 2 * 8;
  114. bitmap |= (u64)block->hdr.bitmap[3] << 3 * 8;
  115. bitmap |= (u64)block->hdr.bitmap[4] << 4 * 8;
  116. bitmap |= (u64)block->hdr.bitmap[5] << 5 * 8;
  117. bitmap |= (u64)block->hdr.bitmap[6] << 6 * 8;
  118. bitmap |= (u64)block->hdr.bitmap[7] << 7 * 8;
  119. for (d = (blocknum == 0 ? AFS_DIR_RESV_BLOCKS0 : AFS_DIR_RESV_BLOCKS);
  120. d < AFS_DIR_SLOTS_PER_BLOCK;
  121. d++) {
  122. if (!((bitmap >> d) & 1))
  123. continue;
  124. de = &block->dirents[d];
  125. if (de->u.valid != 1)
  126. continue;
  127. /* The block was NUL-terminated by afs_dir_check_page(). */
  128. len = strlen(de->u.name);
  129. if (len == name->len &&
  130. memcmp(de->u.name, name->name, name->len) == 0)
  131. return d;
  132. n = round_up(12 + len + 1 + 4, AFS_DIR_DIRENT_SIZE);
  133. n /= AFS_DIR_DIRENT_SIZE;
  134. d += n - 1;
  135. }
  136. return -1;
  137. }
  138. /*
  139. * Initialise a new directory block. Note that block 0 is special and contains
  140. * some extra metadata.
  141. */
  142. static void afs_edit_init_block(union afs_xdr_dir_block *meta,
  143. union afs_xdr_dir_block *block, int block_num)
  144. {
  145. memset(block, 0, sizeof(*block));
  146. block->hdr.npages = htons(1);
  147. block->hdr.magic = AFS_DIR_MAGIC;
  148. block->hdr.bitmap[0] = 1;
  149. if (block_num == 0) {
  150. block->hdr.bitmap[0] = 0xff;
  151. block->hdr.bitmap[1] = 0x1f;
  152. memset(block->meta.alloc_ctrs,
  153. AFS_DIR_SLOTS_PER_BLOCK,
  154. sizeof(block->meta.alloc_ctrs));
  155. meta->meta.alloc_ctrs[0] =
  156. AFS_DIR_SLOTS_PER_BLOCK - AFS_DIR_RESV_BLOCKS0;
  157. }
  158. if (block_num < AFS_DIR_BLOCKS_WITH_CTR)
  159. meta->meta.alloc_ctrs[block_num] =
  160. AFS_DIR_SLOTS_PER_BLOCK - AFS_DIR_RESV_BLOCKS;
  161. }
  162. /*
  163. * Edit a directory's file data to add a new directory entry. Doing this after
  164. * create, mkdir, symlink, link or rename if the data version number is
  165. * incremented by exactly one avoids the need to re-download the entire
  166. * directory contents.
  167. *
  168. * The caller must hold the inode locked.
  169. */
  170. void afs_edit_dir_add(struct afs_vnode *vnode,
  171. struct qstr *name, struct afs_fid *new_fid,
  172. enum afs_edit_dir_reason why)
  173. {
  174. union afs_xdr_dir_block *meta, *block;
  175. struct afs_xdr_dir_page *meta_page, *dir_page;
  176. union afs_xdr_dirent *de;
  177. struct page *page0, *page;
  178. unsigned int need_slots, nr_blocks, b;
  179. pgoff_t index;
  180. loff_t i_size;
  181. gfp_t gfp;
  182. int slot;
  183. _enter(",,{%d,%s},", name->len, name->name);
  184. i_size = i_size_read(&vnode->vfs_inode);
  185. if (i_size > AFS_DIR_BLOCK_SIZE * AFS_DIR_MAX_BLOCKS ||
  186. (i_size & (AFS_DIR_BLOCK_SIZE - 1))) {
  187. clear_bit(AFS_VNODE_DIR_VALID, &vnode->flags);
  188. return;
  189. }
  190. gfp = vnode->vfs_inode.i_mapping->gfp_mask;
  191. page0 = find_or_create_page(vnode->vfs_inode.i_mapping, 0, gfp);
  192. if (!page0) {
  193. clear_bit(AFS_VNODE_DIR_VALID, &vnode->flags);
  194. _leave(" [fgp]");
  195. return;
  196. }
  197. /* Work out how many slots we're going to need. */
  198. need_slots = round_up(12 + name->len + 1 + 4, AFS_DIR_DIRENT_SIZE);
  199. need_slots /= AFS_DIR_DIRENT_SIZE;
  200. meta_page = kmap(page0);
  201. meta = &meta_page->blocks[0];
  202. if (i_size == 0)
  203. goto new_directory;
  204. nr_blocks = i_size / AFS_DIR_BLOCK_SIZE;
  205. /* Find a block that has sufficient slots available. Each VM page
  206. * contains two or more directory blocks.
  207. */
  208. for (b = 0; b < nr_blocks + 1; b++) {
  209. /* If the directory extended into a new page, then we need to
  210. * tack a new page on the end.
  211. */
  212. index = b / AFS_DIR_BLOCKS_PER_PAGE;
  213. if (index == 0) {
  214. page = page0;
  215. dir_page = meta_page;
  216. } else {
  217. if (nr_blocks >= AFS_DIR_MAX_BLOCKS)
  218. goto error;
  219. gfp = vnode->vfs_inode.i_mapping->gfp_mask;
  220. page = find_or_create_page(vnode->vfs_inode.i_mapping,
  221. index, gfp);
  222. if (!page)
  223. goto error;
  224. if (!PagePrivate(page)) {
  225. set_page_private(page, 1);
  226. SetPagePrivate(page);
  227. }
  228. dir_page = kmap(page);
  229. }
  230. /* Abandon the edit if we got a callback break. */
  231. if (!test_bit(AFS_VNODE_DIR_VALID, &vnode->flags))
  232. goto invalidated;
  233. block = &dir_page->blocks[b % AFS_DIR_BLOCKS_PER_PAGE];
  234. _debug("block %u: %2u %3u %u",
  235. b,
  236. (b < AFS_DIR_BLOCKS_WITH_CTR) ? meta->meta.alloc_ctrs[b] : 99,
  237. ntohs(block->hdr.npages),
  238. ntohs(block->hdr.magic));
  239. /* Initialise the block if necessary. */
  240. if (b == nr_blocks) {
  241. _debug("init %u", b);
  242. afs_edit_init_block(meta, block, b);
  243. i_size_write(&vnode->vfs_inode, (b + 1) * AFS_DIR_BLOCK_SIZE);
  244. }
  245. /* Only lower dir pages have a counter in the header. */
  246. if (b >= AFS_DIR_BLOCKS_WITH_CTR ||
  247. meta->meta.alloc_ctrs[b] >= need_slots) {
  248. /* We need to try and find one or more consecutive
  249. * slots to hold the entry.
  250. */
  251. slot = afs_find_contig_bits(block, need_slots);
  252. if (slot >= 0) {
  253. _debug("slot %u", slot);
  254. goto found_space;
  255. }
  256. }
  257. if (page != page0) {
  258. unlock_page(page);
  259. kunmap(page);
  260. put_page(page);
  261. }
  262. }
  263. /* There are no spare slots of sufficient size, yet the operation
  264. * succeeded. Download the directory again.
  265. */
  266. trace_afs_edit_dir(vnode, why, afs_edit_dir_create_nospc, 0, 0, 0, 0, name->name);
  267. clear_bit(AFS_VNODE_DIR_VALID, &vnode->flags);
  268. goto out_unmap;
  269. new_directory:
  270. afs_edit_init_block(meta, meta, 0);
  271. i_size = AFS_DIR_BLOCK_SIZE;
  272. i_size_write(&vnode->vfs_inode, i_size);
  273. slot = AFS_DIR_RESV_BLOCKS0;
  274. page = page0;
  275. block = meta;
  276. nr_blocks = 1;
  277. b = 0;
  278. found_space:
  279. /* Set the dirent slot. */
  280. trace_afs_edit_dir(vnode, why, afs_edit_dir_create, b, slot,
  281. new_fid->vnode, new_fid->unique, name->name);
  282. de = &block->dirents[slot];
  283. de->u.valid = 1;
  284. de->u.unused[0] = 0;
  285. de->u.hash_next = 0; // TODO: Really need to maintain this
  286. de->u.vnode = htonl(new_fid->vnode);
  287. de->u.unique = htonl(new_fid->unique);
  288. memcpy(de->u.name, name->name, name->len + 1);
  289. de->u.name[name->len] = 0;
  290. /* Adjust the bitmap. */
  291. afs_set_contig_bits(block, slot, need_slots);
  292. if (page != page0) {
  293. unlock_page(page);
  294. kunmap(page);
  295. put_page(page);
  296. }
  297. /* Adjust the allocation counter. */
  298. if (b < AFS_DIR_BLOCKS_WITH_CTR)
  299. meta->meta.alloc_ctrs[b] -= need_slots;
  300. inode_inc_iversion_raw(&vnode->vfs_inode);
  301. afs_stat_v(vnode, n_dir_cr);
  302. _debug("Insert %s in %u[%u]", name->name, b, slot);
  303. out_unmap:
  304. unlock_page(page0);
  305. kunmap(page0);
  306. put_page(page0);
  307. _leave("");
  308. return;
  309. invalidated:
  310. trace_afs_edit_dir(vnode, why, afs_edit_dir_create_inval, 0, 0, 0, 0, name->name);
  311. clear_bit(AFS_VNODE_DIR_VALID, &vnode->flags);
  312. if (page != page0) {
  313. kunmap(page);
  314. put_page(page);
  315. }
  316. goto out_unmap;
  317. error:
  318. trace_afs_edit_dir(vnode, why, afs_edit_dir_create_error, 0, 0, 0, 0, name->name);
  319. clear_bit(AFS_VNODE_DIR_VALID, &vnode->flags);
  320. goto out_unmap;
  321. }
  322. /*
  323. * Edit a directory's file data to remove a new directory entry. Doing this
  324. * after unlink, rmdir or rename if the data version number is incremented by
  325. * exactly one avoids the need to re-download the entire directory contents.
  326. *
  327. * The caller must hold the inode locked.
  328. */
  329. void afs_edit_dir_remove(struct afs_vnode *vnode,
  330. struct qstr *name, enum afs_edit_dir_reason why)
  331. {
  332. struct afs_xdr_dir_page *meta_page, *dir_page;
  333. union afs_xdr_dir_block *meta, *block;
  334. union afs_xdr_dirent *de;
  335. struct page *page0, *page;
  336. unsigned int need_slots, nr_blocks, b;
  337. pgoff_t index;
  338. loff_t i_size;
  339. int slot;
  340. _enter(",,{%d,%s},", name->len, name->name);
  341. i_size = i_size_read(&vnode->vfs_inode);
  342. if (i_size < AFS_DIR_BLOCK_SIZE ||
  343. i_size > AFS_DIR_BLOCK_SIZE * AFS_DIR_MAX_BLOCKS ||
  344. (i_size & (AFS_DIR_BLOCK_SIZE - 1))) {
  345. clear_bit(AFS_VNODE_DIR_VALID, &vnode->flags);
  346. return;
  347. }
  348. nr_blocks = i_size / AFS_DIR_BLOCK_SIZE;
  349. page0 = find_lock_page(vnode->vfs_inode.i_mapping, 0);
  350. if (!page0) {
  351. clear_bit(AFS_VNODE_DIR_VALID, &vnode->flags);
  352. _leave(" [fgp]");
  353. return;
  354. }
  355. /* Work out how many slots we're going to discard. */
  356. need_slots = round_up(12 + name->len + 1 + 4, AFS_DIR_DIRENT_SIZE);
  357. need_slots /= AFS_DIR_DIRENT_SIZE;
  358. meta_page = kmap(page0);
  359. meta = &meta_page->blocks[0];
  360. /* Find a page that has sufficient slots available. Each VM page
  361. * contains two or more directory blocks.
  362. */
  363. for (b = 0; b < nr_blocks; b++) {
  364. index = b / AFS_DIR_BLOCKS_PER_PAGE;
  365. if (index != 0) {
  366. page = find_lock_page(vnode->vfs_inode.i_mapping, index);
  367. if (!page)
  368. goto error;
  369. dir_page = kmap(page);
  370. } else {
  371. page = page0;
  372. dir_page = meta_page;
  373. }
  374. /* Abandon the edit if we got a callback break. */
  375. if (!test_bit(AFS_VNODE_DIR_VALID, &vnode->flags))
  376. goto invalidated;
  377. block = &dir_page->blocks[b % AFS_DIR_BLOCKS_PER_PAGE];
  378. if (b > AFS_DIR_BLOCKS_WITH_CTR ||
  379. meta->meta.alloc_ctrs[b] <= AFS_DIR_SLOTS_PER_BLOCK - 1 - need_slots) {
  380. slot = afs_dir_scan_block(block, name, b);
  381. if (slot >= 0)
  382. goto found_dirent;
  383. }
  384. if (page != page0) {
  385. unlock_page(page);
  386. kunmap(page);
  387. put_page(page);
  388. }
  389. }
  390. /* Didn't find the dirent to clobber. Download the directory again. */
  391. trace_afs_edit_dir(vnode, why, afs_edit_dir_delete_noent,
  392. 0, 0, 0, 0, name->name);
  393. clear_bit(AFS_VNODE_DIR_VALID, &vnode->flags);
  394. goto out_unmap;
  395. found_dirent:
  396. de = &block->dirents[slot];
  397. trace_afs_edit_dir(vnode, why, afs_edit_dir_delete, b, slot,
  398. ntohl(de->u.vnode), ntohl(de->u.unique),
  399. name->name);
  400. memset(de, 0, sizeof(*de) * need_slots);
  401. /* Adjust the bitmap. */
  402. afs_clear_contig_bits(block, slot, need_slots);
  403. if (page != page0) {
  404. unlock_page(page);
  405. kunmap(page);
  406. put_page(page);
  407. }
  408. /* Adjust the allocation counter. */
  409. if (b < AFS_DIR_BLOCKS_WITH_CTR)
  410. meta->meta.alloc_ctrs[b] += need_slots;
  411. inode_set_iversion_raw(&vnode->vfs_inode, vnode->status.data_version);
  412. afs_stat_v(vnode, n_dir_rm);
  413. _debug("Remove %s from %u[%u]", name->name, b, slot);
  414. out_unmap:
  415. unlock_page(page0);
  416. kunmap(page0);
  417. put_page(page0);
  418. _leave("");
  419. return;
  420. invalidated:
  421. trace_afs_edit_dir(vnode, why, afs_edit_dir_delete_inval,
  422. 0, 0, 0, 0, name->name);
  423. clear_bit(AFS_VNODE_DIR_VALID, &vnode->flags);
  424. if (page != page0) {
  425. unlock_page(page);
  426. kunmap(page);
  427. put_page(page);
  428. }
  429. goto out_unmap;
  430. error:
  431. trace_afs_edit_dir(vnode, why, afs_edit_dir_delete_error,
  432. 0, 0, 0, 0, name->name);
  433. clear_bit(AFS_VNODE_DIR_VALID, &vnode->flags);
  434. goto out_unmap;
  435. }