5 * Directory related functions
8 * This file is distributed under the terms of the GNU General Public
9 * License (GPL). Copies of the GPL can be obtained from:
10 * ftp://prep.ai.mit.edu/pub/gnu/GPL
11 * Each contributing author retains all rights to their own work.
18 #include <linux/string.h>
19 #include <linux/bio.h>
20 #include <linux/crc-itu-t.h>
21 #include <linux/iversion.h>
23 static int udf_verify_fi(struct udf_fileident_iter *iter)
27 if (iter->fi.descTag.tagIdent != cpu_to_le16(TAG_IDENT_FID)) {
28 udf_err(iter->dir->i_sb,
29 "directory (ino %lu) has entry at pos %llu with incorrect tag %x\n",
30 iter->dir->i_ino, (unsigned long long)iter->pos,
31 le16_to_cpu(iter->fi.descTag.tagIdent));
34 len = udf_dir_entry_len(&iter->fi);
35 if (le16_to_cpu(iter->fi.lengthOfImpUse) & 3) {
36 udf_err(iter->dir->i_sb,
37 "directory (ino %lu) has entry at pos %llu with unaligned lenght of impUse field\n",
38 iter->dir->i_ino, (unsigned long long)iter->pos);
42 * This is in fact allowed by the spec due to long impUse field but
43 * we don't support it. If there is real media with this large impUse
44 * field, support can be added.
46 if (len > 1 << iter->dir->i_blkbits) {
47 udf_err(iter->dir->i_sb,
48 "directory (ino %lu) has too big (%u) entry at pos %llu\n",
49 iter->dir->i_ino, len, (unsigned long long)iter->pos);
52 if (iter->pos + len > iter->dir->i_size) {
53 udf_err(iter->dir->i_sb,
54 "directory (ino %lu) has entry past directory size at pos %llu\n",
55 iter->dir->i_ino, (unsigned long long)iter->pos);
58 if (udf_dir_entry_len(&iter->fi) !=
59 sizeof(struct tag) + le16_to_cpu(iter->fi.descTag.descCRCLength)) {
60 udf_err(iter->dir->i_sb,
61 "directory (ino %lu) has entry where CRC length (%u) does not match entry length (%u)\n",
63 (unsigned)le16_to_cpu(iter->fi.descTag.descCRCLength),
64 (unsigned)(udf_dir_entry_len(&iter->fi) -
71 static int udf_copy_fi(struct udf_fileident_iter *iter)
73 struct udf_inode_info *iinfo = UDF_I(iter->dir);
74 int blksize = 1 << iter->dir->i_blkbits;
75 int err, off, len, nameoff;
77 /* Skip copying when we are at EOF */
78 if (iter->pos >= iter->dir->i_size) {
82 if (iter->dir->i_size < iter->pos + sizeof(struct fileIdentDesc)) {
83 udf_err(iter->dir->i_sb,
84 "directory (ino %lu) has entry straddling EOF\n",
88 if (iinfo->i_alloc_type == ICBTAG_FLAG_AD_IN_ICB) {
89 memcpy(&iter->fi, iinfo->i_data + iinfo->i_lenEAttr + iter->pos,
90 sizeof(struct fileIdentDesc));
91 err = udf_verify_fi(iter);
94 iter->name = iinfo->i_data + iinfo->i_lenEAttr + iter->pos +
95 sizeof(struct fileIdentDesc) +
96 le16_to_cpu(iter->fi.lengthOfImpUse);
100 off = iter->pos & (blksize - 1);
101 len = min_t(int, sizeof(struct fileIdentDesc), blksize - off);
102 memcpy(&iter->fi, iter->bh[0]->b_data + off, len);
103 if (len < sizeof(struct fileIdentDesc))
104 memcpy((char *)(&iter->fi) + len, iter->bh[1]->b_data,
105 sizeof(struct fileIdentDesc) - len);
106 err = udf_verify_fi(iter);
110 /* Handle directory entry name */
111 nameoff = off + sizeof(struct fileIdentDesc) +
112 le16_to_cpu(iter->fi.lengthOfImpUse);
113 if (off + udf_dir_entry_len(&iter->fi) <= blksize) {
114 iter->name = iter->bh[0]->b_data + nameoff;
115 } else if (nameoff >= blksize) {
116 iter->name = iter->bh[1]->b_data + (nameoff - blksize);
118 iter->name = iter->namebuf;
119 len = blksize - nameoff;
120 memcpy(iter->name, iter->bh[0]->b_data + nameoff, len);
121 memcpy(iter->name + len, iter->bh[1]->b_data,
122 iter->fi.lengthFileIdent - len);
127 /* Readahead 8k once we are at 8k boundary */
128 static void udf_readahead_dir(struct udf_fileident_iter *iter)
130 unsigned int ralen = 16 >> (iter->dir->i_blkbits - 9);
131 struct buffer_head *tmp, *bha[16];
135 if (iter->loffset & (ralen - 1))
138 if (iter->loffset + ralen > (iter->elen >> iter->dir->i_blkbits))
139 ralen = (iter->elen >> iter->dir->i_blkbits) - iter->loffset;
141 for (i = 0; i < ralen; i++) {
142 blk = udf_get_lb_pblock(iter->dir->i_sb, &iter->eloc,
144 tmp = udf_tgetblk(iter->dir->i_sb, blk);
145 if (tmp && !buffer_uptodate(tmp) && !buffer_locked(tmp))
151 bh_readahead_batch(num, bha, REQ_RAHEAD);
152 for (i = 0; i < num; i++)
157 static struct buffer_head *udf_fiiter_bread_blk(struct udf_fileident_iter *iter)
161 udf_readahead_dir(iter);
162 blk = udf_get_lb_pblock(iter->dir->i_sb, &iter->eloc, iter->loffset);
163 return udf_tread(iter->dir->i_sb, blk);
167 * Updates loffset to point to next directory block; eloc, elen & epos are
168 * updated if we need to traverse to the next extent as well.
170 static int udf_fiiter_advance_blk(struct udf_fileident_iter *iter)
173 if (iter->loffset < iter->elen >> iter->dir->i_blkbits)
177 if (udf_next_aext(iter->dir, &iter->epos, &iter->eloc, &iter->elen, 1)
178 != (EXT_RECORDED_ALLOCATED >> 30)) {
179 if (iter->pos == iter->dir->i_size) {
183 udf_err(iter->dir->i_sb,
184 "extent after position %llu not allocated in directory (ino %lu)\n",
185 (unsigned long long)iter->pos, iter->dir->i_ino);
186 return -EFSCORRUPTED;
191 static int udf_fiiter_load_bhs(struct udf_fileident_iter *iter)
193 int blksize = 1 << iter->dir->i_blkbits;
194 int off = iter->pos & (blksize - 1);
196 struct fileIdentDesc *fi;
198 /* Is there any further extent we can map from? */
199 if (!iter->bh[0] && iter->elen) {
200 iter->bh[0] = udf_fiiter_bread_blk(iter);
205 if (!buffer_uptodate(iter->bh[0])) {
210 /* There's no next block so we are done */
211 if (iter->pos >= iter->dir->i_size)
213 /* Need to fetch next block as well? */
214 if (off + sizeof(struct fileIdentDesc) > blksize)
216 fi = (struct fileIdentDesc *)(iter->bh[0]->b_data + off);
217 /* Need to fetch next block to get name? */
218 if (off + udf_dir_entry_len(fi) > blksize) {
220 udf_fiiter_advance_blk(iter);
221 iter->bh[1] = udf_fiiter_bread_blk(iter);
226 if (!buffer_uptodate(iter->bh[1])) {
235 iter->bh[0] = iter->bh[1] = NULL;
239 int udf_fiiter_init(struct udf_fileident_iter *iter, struct inode *dir,
242 struct udf_inode_info *iinfo = UDF_I(dir);
246 iter->bh[0] = iter->bh[1] = NULL;
249 iter->epos.bh = NULL;
251 iter->namebuf = kmalloc(UDF_NAME_LEN_CS0, GFP_KERNEL);
255 if (iinfo->i_alloc_type == ICBTAG_FLAG_AD_IN_ICB) {
256 err = udf_copy_fi(iter);
260 if (inode_bmap(dir, iter->pos >> dir->i_blkbits, &iter->epos,
261 &iter->eloc, &iter->elen, &iter->loffset) !=
262 (EXT_RECORDED_ALLOCATED >> 30)) {
263 if (pos == dir->i_size)
266 "position %llu not allocated in directory (ino %lu)\n",
267 (unsigned long long)pos, dir->i_ino);
271 err = udf_fiiter_load_bhs(iter);
274 err = udf_copy_fi(iter);
277 udf_fiiter_release(iter);
281 int udf_fiiter_advance(struct udf_fileident_iter *iter)
283 unsigned int oldoff, len;
284 int blksize = 1 << iter->dir->i_blkbits;
287 oldoff = iter->pos & (blksize - 1);
288 len = udf_dir_entry_len(&iter->fi);
290 if (UDF_I(iter->dir)->i_alloc_type != ICBTAG_FLAG_AD_IN_ICB) {
291 if (oldoff + len >= blksize) {
294 /* Next block already loaded? */
296 iter->bh[0] = iter->bh[1];
299 udf_fiiter_advance_blk(iter);
302 err = udf_fiiter_load_bhs(iter);
306 return udf_copy_fi(iter);
309 void udf_fiiter_release(struct udf_fileident_iter *iter)
314 iter->bh[0] = iter->bh[1] = NULL;
315 kfree(iter->namebuf);
316 iter->namebuf = NULL;
319 static void udf_copy_to_bufs(void *buf1, int len1, void *buf2, int len2,
320 int off, void *src, int len)
327 copy = min(off + len, len1) - off;
328 memcpy(buf1 + off, src, copy);
334 if (WARN_ON_ONCE(off + len > len2 || !buf2))
336 memcpy(buf2 + off, src, len);
340 static uint16_t udf_crc_fi_bufs(void *buf1, int len1, void *buf2, int len2,
349 copy = min(off + len, len1) - off;
350 crc = crc_itu_t(crc, buf1 + off, copy);
355 if (WARN_ON_ONCE(off + len > len2 || !buf2))
357 crc = crc_itu_t(crc, buf2 + off, len);
362 static void udf_copy_fi_to_bufs(char *buf1, int len1, char *buf2, int len2,
363 int off, struct fileIdentDesc *fi,
364 uint8_t *impuse, uint8_t *name)
368 int crcoff = off + sizeof(struct tag);
369 unsigned int crclen = udf_dir_entry_len(fi) - sizeof(struct tag);
371 udf_copy_to_bufs(buf1, len1, buf2, len2, off, fi,
372 sizeof(struct fileIdentDesc));
373 off += sizeof(struct fileIdentDesc);
375 udf_copy_to_bufs(buf1, len1, buf2, len2, off, impuse,
376 le16_to_cpu(fi->lengthOfImpUse));
377 off += le16_to_cpu(fi->lengthOfImpUse);
379 udf_copy_to_bufs(buf1, len1, buf2, len2, off, name,
380 fi->lengthFileIdent);
382 crc = udf_crc_fi_bufs(buf1, len1, buf2, len2, crcoff, crclen);
383 fi->descTag.descCRC = cpu_to_le16(crc);
384 fi->descTag.descCRCLength = cpu_to_le16(crclen);
385 fi->descTag.tagChecksum = udf_tag_checksum(&fi->descTag);
387 udf_copy_to_bufs(buf1, len1, buf2, len2, fioff, fi, sizeof(struct tag));
390 void udf_fiiter_write_fi(struct udf_fileident_iter *iter, uint8_t *impuse)
392 struct udf_inode_info *iinfo = UDF_I(iter->dir);
393 void *buf1, *buf2 = NULL;
394 int len1, len2 = 0, off;
395 int blksize = 1 << iter->dir->i_blkbits;
397 off = iter->pos & (blksize - 1);
398 if (iinfo->i_alloc_type == ICBTAG_FLAG_AD_IN_ICB) {
399 buf1 = iinfo->i_data + iinfo->i_lenEAttr;
400 len1 = iter->dir->i_size;
402 buf1 = iter->bh[0]->b_data;
405 buf2 = iter->bh[1]->b_data;
410 udf_copy_fi_to_bufs(buf1, len1, buf2, len2, off, &iter->fi, impuse,
411 iter->name == iter->namebuf ? iter->name : NULL);
413 if (iinfo->i_alloc_type == ICBTAG_FLAG_AD_IN_ICB) {
414 mark_inode_dirty(iter->dir);
416 mark_buffer_dirty_inode(iter->bh[0], iter->dir);
418 mark_buffer_dirty_inode(iter->bh[1], iter->dir);
420 inode_inc_iversion(iter->dir);
423 void udf_fiiter_update_elen(struct udf_fileident_iter *iter, uint32_t new_elen)
425 struct udf_inode_info *iinfo = UDF_I(iter->dir);
426 int diff = new_elen - iter->elen;
428 /* Skip update when we already went past the last extent */
431 iter->elen = new_elen;
432 if (iinfo->i_alloc_type == ICBTAG_FLAG_AD_SHORT)
433 iter->epos.offset -= sizeof(struct short_ad);
434 else if (iinfo->i_alloc_type == ICBTAG_FLAG_AD_LONG)
435 iter->epos.offset -= sizeof(struct long_ad);
436 udf_write_aext(iter->dir, &iter->epos, &iter->eloc, iter->elen, 1);
437 iinfo->i_lenExtents += diff;
438 mark_inode_dirty(iter->dir);
441 /* Append new block to directory. @iter is expected to point at EOF */
442 int udf_fiiter_append_blk(struct udf_fileident_iter *iter)
444 struct udf_inode_info *iinfo = UDF_I(iter->dir);
445 int blksize = 1 << iter->dir->i_blkbits;
446 struct buffer_head *bh;
448 uint32_t old_elen = iter->elen;
451 if (WARN_ON_ONCE(iinfo->i_alloc_type == ICBTAG_FLAG_AD_IN_ICB))
454 /* Round up last extent in the file */
455 udf_fiiter_update_elen(iter, ALIGN(iter->elen, blksize));
457 /* Allocate new block and refresh mapping information */
458 block = iinfo->i_lenExtents >> iter->dir->i_blkbits;
459 bh = udf_bread(iter->dir, block, 1, &err);
461 udf_fiiter_update_elen(iter, old_elen);
464 if (inode_bmap(iter->dir, block, &iter->epos, &iter->eloc, &iter->elen,
465 &iter->loffset) != (EXT_RECORDED_ALLOCATED >> 30)) {
466 udf_err(iter->dir->i_sb,
467 "block %llu not allocated in directory (ino %lu)\n",
468 (unsigned long long)block, iter->dir->i_ino);
469 return -EFSCORRUPTED;
471 if (!(iter->pos & (blksize - 1))) {
480 struct short_ad *udf_get_fileshortad(uint8_t *ptr, int maxoffset, uint32_t *offset,
485 if ((!ptr) || (!offset)) {
486 pr_err("%s: invalidparms\n", __func__);
490 if ((*offset + sizeof(struct short_ad)) > maxoffset)
493 sa = (struct short_ad *)ptr;
494 if (sa->extLength == 0)
499 *offset += sizeof(struct short_ad);
503 struct long_ad *udf_get_filelongad(uint8_t *ptr, int maxoffset, uint32_t *offset, int inc)
507 if ((!ptr) || (!offset)) {
508 pr_err("%s: invalidparms\n", __func__);
512 if ((*offset + sizeof(struct long_ad)) > maxoffset)
515 la = (struct long_ad *)ptr;
516 if (la->extLength == 0)
521 *offset += sizeof(struct long_ad);