4 * (c) 1996 Hans-Joachim Widmaier - Rewritten
6 * (C) 1993 Ray Burr - Modified for Amiga FFS filesystem.
8 * (C) 1992 Eric Youngdale Modified for ISO 9660 filesystem.
10 * (C) 1991 Linus Torvalds - minix filesystem
12 * affs directory handling functions
18 static int affs_readdir(struct file
*, struct dir_context
*);
20 const struct file_operations affs_dir_operations
= {
21 .read
= generic_read_dir
,
22 .llseek
= generic_file_llseek
,
23 .iterate
= affs_readdir
,
24 .fsync
= affs_file_fsync
,
28 * directories can handle most operations...
30 const struct inode_operations affs_dir_inode_operations
= {
31 .create
= affs_create
,
32 .lookup
= affs_lookup
,
34 .unlink
= affs_unlink
,
35 .symlink
= affs_symlink
,
38 .rename
= affs_rename
,
39 .setattr
= affs_notify_change
,
43 affs_readdir(struct file
*file
, struct dir_context
*ctx
)
45 struct inode
*inode
= file_inode(file
);
46 struct super_block
*sb
= inode
->i_sb
;
47 struct buffer_head
*dir_bh
= NULL
;
48 struct buffer_head
*fh_bh
= NULL
;
56 pr_debug("AFFS: readdir(ino=%lu,f_pos=%lx)\n",inode
->i_ino
,(unsigned long)ctx
->pos
);
59 file
->private_data
= (void *)0;
60 if (!dir_emit_dots(file
, ctx
))
65 chain_pos
= (ctx
->pos
- 2) & 0xffff;
66 hash_pos
= (ctx
->pos
- 2) >> 16;
67 if (chain_pos
== 0xffff) {
68 affs_warning(sb
, "readdir", "More than 65535 entries in chain");
71 ctx
->pos
= ((hash_pos
<< 16) | chain_pos
) + 2;
73 dir_bh
= affs_bread(sb
, inode
->i_ino
);
77 /* If the directory hasn't changed since the last call to readdir(),
78 * we can jump directly to where we left off.
80 ino
= (u32
)(long)file
->private_data
;
81 if (ino
&& file
->f_version
== inode
->i_version
) {
82 pr_debug("AFFS: readdir() left off=%d\n", ino
);
86 ino
= be32_to_cpu(AFFS_HEAD(dir_bh
)->table
[hash_pos
]);
87 for (i
= 0; ino
&& i
< chain_pos
; i
++) {
88 fh_bh
= affs_bread(sb
, ino
);
90 affs_error(sb
, "readdir","Cannot read block %d", i
);
93 ino
= be32_to_cpu(AFFS_TAIL(sb
, fh_bh
)->hash_chain
);
101 for (; hash_pos
< AFFS_SB(sb
)->s_hashsize
; hash_pos
++) {
102 ino
= be32_to_cpu(AFFS_HEAD(dir_bh
)->table
[hash_pos
]);
105 ctx
->pos
= (hash_pos
<< 16) + 2;
108 fh_bh
= affs_bread(sb
, ino
);
110 affs_error(sb
, "readdir","Cannot read block %d", ino
);
114 namelen
= min(AFFS_TAIL(sb
, fh_bh
)->name
[0], (u8
)30);
115 name
= AFFS_TAIL(sb
, fh_bh
)->name
+ 1;
116 pr_debug("AFFS: readdir(): filldir(\"%.*s\", ino=%u), hash=%d, f_pos=%x\n",
117 namelen
, name
, ino
, hash_pos
, (u32
)ctx
->pos
);
118 if (!dir_emit(ctx
, name
, namelen
, ino
, DT_UNKNOWN
))
121 ino
= be32_to_cpu(AFFS_TAIL(sb
, fh_bh
)->hash_chain
);
127 file
->f_version
= inode
->i_version
;
128 file
->private_data
= (void *)(long)ino
;
133 affs_unlock_dir(inode
);