dir_fplus.c 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188
  1. /*
  2. * linux/fs/adfs/dir_fplus.c
  3. *
  4. * Copyright (C) 1997-1999 Russell King
  5. *
  6. * This program is free software; you can redistribute it and/or modify
  7. * it under the terms of the GNU General Public License version 2 as
  8. * published by the Free Software Foundation.
  9. */
  10. #include <linux/buffer_head.h>
  11. #include "adfs.h"
  12. #include "dir_fplus.h"
  13. static int
  14. adfs_fplus_read(struct super_block *sb, unsigned int id, unsigned int sz, struct adfs_dir *dir)
  15. {
  16. struct adfs_bigdirheader *h;
  17. struct adfs_bigdirtail *t;
  18. unsigned long block;
  19. unsigned int blk, size;
  20. int i, ret = -EIO;
  21. dir->nr_buffers = 0;
  22. block = __adfs_block_map(sb, id, 0);
  23. if (!block) {
  24. adfs_error(sb, "dir object %X has a hole at offset 0", id);
  25. goto out;
  26. }
  27. dir->bh[0] = sb_bread(sb, block);
  28. if (!dir->bh[0])
  29. goto out;
  30. dir->nr_buffers += 1;
  31. h = (struct adfs_bigdirheader *)dir->bh[0]->b_data;
  32. size = le32_to_cpu(h->bigdirsize);
  33. if (size != sz) {
  34. printk(KERN_WARNING "adfs: adfs_fplus_read: directory header size\n"
  35. " does not match directory size\n");
  36. }
  37. if (h->bigdirversion[0] != 0 || h->bigdirversion[1] != 0 ||
  38. h->bigdirversion[2] != 0 || size & 2047 ||
  39. h->bigdirstartname != cpu_to_le32(BIGDIRSTARTNAME))
  40. goto out;
  41. size >>= sb->s_blocksize_bits;
  42. for (blk = 1; blk < size; blk++) {
  43. block = __adfs_block_map(sb, id, blk);
  44. if (!block) {
  45. adfs_error(sb, "dir object %X has a hole at offset %d", id, blk);
  46. goto out;
  47. }
  48. dir->bh[blk] = sb_bread(sb, block);
  49. if (!dir->bh[blk])
  50. goto out;
  51. dir->nr_buffers = blk;
  52. }
  53. t = (struct adfs_bigdirtail *)(dir->bh[size - 1]->b_data + (sb->s_blocksize - 8));
  54. if (t->bigdirendname != cpu_to_le32(BIGDIRENDNAME) ||
  55. t->bigdirendmasseq != h->startmasseq ||
  56. t->reserved[0] != 0 || t->reserved[1] != 0)
  57. goto out;
  58. dir->parent_id = le32_to_cpu(h->bigdirparent);
  59. dir->sb = sb;
  60. return 0;
  61. out:
  62. for (i = 0; i < dir->nr_buffers; i++)
  63. brelse(dir->bh[i]);
  64. dir->sb = NULL;
  65. return ret;
  66. }
  67. static int
  68. adfs_fplus_setpos(struct adfs_dir *dir, unsigned int fpos)
  69. {
  70. struct adfs_bigdirheader *h = (struct adfs_bigdirheader *)dir->bh[0]->b_data;
  71. int ret = -ENOENT;
  72. if (fpos <= le32_to_cpu(h->bigdirentries)) {
  73. dir->pos = fpos;
  74. ret = 0;
  75. }
  76. return ret;
  77. }
  78. static void
  79. dir_memcpy(struct adfs_dir *dir, unsigned int offset, void *to, int len)
  80. {
  81. struct super_block *sb = dir->sb;
  82. unsigned int buffer, partial, remainder;
  83. buffer = offset >> sb->s_blocksize_bits;
  84. offset &= sb->s_blocksize - 1;
  85. partial = sb->s_blocksize - offset;
  86. if (partial >= len)
  87. memcpy(to, dir->bh[buffer]->b_data + offset, len);
  88. else {
  89. char *c = (char *)to;
  90. remainder = len - partial;
  91. memcpy(c, dir->bh[buffer]->b_data + offset, partial);
  92. memcpy(c + partial, dir->bh[buffer + 1]->b_data, remainder);
  93. }
  94. }
  95. static int
  96. adfs_fplus_getnext(struct adfs_dir *dir, struct object_info *obj)
  97. {
  98. struct adfs_bigdirheader *h = (struct adfs_bigdirheader *)dir->bh[0]->b_data;
  99. struct adfs_bigdirentry bde;
  100. unsigned int offset;
  101. int i, ret = -ENOENT;
  102. if (dir->pos >= le32_to_cpu(h->bigdirentries))
  103. goto out;
  104. offset = offsetof(struct adfs_bigdirheader, bigdirname);
  105. offset += ((le32_to_cpu(h->bigdirnamelen) + 4) & ~3);
  106. offset += dir->pos * sizeof(struct adfs_bigdirentry);
  107. dir_memcpy(dir, offset, &bde, sizeof(struct adfs_bigdirentry));
  108. obj->loadaddr = le32_to_cpu(bde.bigdirload);
  109. obj->execaddr = le32_to_cpu(bde.bigdirexec);
  110. obj->size = le32_to_cpu(bde.bigdirlen);
  111. obj->file_id = le32_to_cpu(bde.bigdirindaddr);
  112. obj->attr = le32_to_cpu(bde.bigdirattr);
  113. obj->name_len = le32_to_cpu(bde.bigdirobnamelen);
  114. offset = offsetof(struct adfs_bigdirheader, bigdirname);
  115. offset += ((le32_to_cpu(h->bigdirnamelen) + 4) & ~3);
  116. offset += le32_to_cpu(h->bigdirentries) * sizeof(struct adfs_bigdirentry);
  117. offset += le32_to_cpu(bde.bigdirobnameptr);
  118. dir_memcpy(dir, offset, obj->name, obj->name_len);
  119. for (i = 0; i < obj->name_len; i++)
  120. if (obj->name[i] == '/')
  121. obj->name[i] = '.';
  122. dir->pos += 1;
  123. ret = 0;
  124. out:
  125. return ret;
  126. }
  127. static int
  128. adfs_fplus_sync(struct adfs_dir *dir)
  129. {
  130. int err = 0;
  131. int i;
  132. for (i = dir->nr_buffers - 1; i >= 0; i--) {
  133. struct buffer_head *bh = dir->bh[i];
  134. sync_dirty_buffer(bh);
  135. if (buffer_req(bh) && !buffer_uptodate(bh))
  136. err = -EIO;
  137. }
  138. return err;
  139. }
  140. static void
  141. adfs_fplus_free(struct adfs_dir *dir)
  142. {
  143. int i;
  144. for (i = 0; i < dir->nr_buffers; i++)
  145. brelse(dir->bh[i]);
  146. dir->sb = NULL;
  147. }
  148. struct adfs_dir_ops adfs_fplus_dir_ops = {
  149. .read = adfs_fplus_read,
  150. .setpos = adfs_fplus_setpos,
  151. .getnext = adfs_fplus_getnext,
  152. .sync = adfs_fplus_sync,
  153. .free = adfs_fplus_free
  154. };