1 // SPDX-License-Identifier: GPL-2.0
5 * (c) 1996 Hans-Joachim Widmaier - Rewritten
7 * (C) 1993 Ray Burr - Modified for Amiga FFS filesystem.
9 * (C) 1992 Eric Youngdale Modified for ISO 9660 filesystem.
11 * (C) 1991 Linus Torvalds - minix filesystem
13 * affs directory handling functions
17 #include <linux/iversion.h>
20 struct affs_dir_data
{
25 static int affs_readdir(struct file
*, struct dir_context
*);
27 static loff_t
affs_dir_llseek(struct file
*file
, loff_t offset
, int whence
)
29 struct affs_dir_data
*data
= file
->private_data
;
31 return generic_llseek_cookie(file
, offset
, whence
, &data
->cookie
);
34 static int affs_dir_open(struct inode
*inode
, struct file
*file
)
36 struct affs_dir_data
*data
;
38 data
= kzalloc(sizeof(struct affs_dir_data
), GFP_KERNEL
);
41 file
->private_data
= data
;
45 static int affs_dir_release(struct inode
*inode
, struct file
*file
)
47 kfree(file
->private_data
);
51 const struct file_operations affs_dir_operations
= {
52 .open
= affs_dir_open
,
53 .read
= generic_read_dir
,
54 .llseek
= affs_dir_llseek
,
55 .iterate_shared
= affs_readdir
,
56 .fsync
= affs_file_fsync
,
57 .release
= affs_dir_release
,
61 * directories can handle most operations...
63 const struct inode_operations affs_dir_inode_operations
= {
64 .create
= affs_create
,
65 .lookup
= affs_lookup
,
67 .unlink
= affs_unlink
,
68 .symlink
= affs_symlink
,
71 .rename
= affs_rename2
,
72 .setattr
= affs_notify_change
,
76 affs_readdir(struct file
*file
, struct dir_context
*ctx
)
78 struct inode
*inode
= file_inode(file
);
79 struct affs_dir_data
*data
= file
->private_data
;
80 struct super_block
*sb
= inode
->i_sb
;
81 struct buffer_head
*dir_bh
= NULL
;
82 struct buffer_head
*fh_bh
= NULL
;
91 pr_debug("%s(ino=%lu,f_pos=%llx)\n", __func__
, inode
->i_ino
, ctx
->pos
);
95 if (!dir_emit_dots(file
, ctx
))
100 chain_pos
= (ctx
->pos
- 2) & 0xffff;
101 hash_pos
= (ctx
->pos
- 2) >> 16;
102 if (chain_pos
== 0xffff) {
103 affs_warning(sb
, "readdir", "More than 65535 entries in chain");
106 ctx
->pos
= ((hash_pos
<< 16) | chain_pos
) + 2;
108 dir_bh
= affs_bread(sb
, inode
->i_ino
);
112 /* If the directory hasn't changed since the last call to readdir(),
113 * we can jump directly to where we left off.
116 if (ino
&& inode_eq_iversion(inode
, data
->cookie
)) {
117 pr_debug("readdir() left off=%d\n", ino
);
121 ino
= be32_to_cpu(AFFS_HEAD(dir_bh
)->table
[hash_pos
]);
122 for (i
= 0; ino
&& i
< chain_pos
; i
++) {
123 fh_bh
= affs_bread(sb
, ino
);
125 affs_error(sb
, "readdir","Cannot read block %d", i
);
129 ino
= be32_to_cpu(AFFS_TAIL(sb
, fh_bh
)->hash_chain
);
137 for (; hash_pos
< AFFS_SB(sb
)->s_hashsize
; hash_pos
++) {
138 ino
= be32_to_cpu(AFFS_HEAD(dir_bh
)->table
[hash_pos
]);
141 ctx
->pos
= (hash_pos
<< 16) + 2;
144 fh_bh
= affs_bread(sb
, ino
);
146 affs_error(sb
, "readdir",
147 "Cannot read block %d", ino
);
151 namelen
= min(AFFS_TAIL(sb
, fh_bh
)->name
[0],
153 name
= AFFS_TAIL(sb
, fh_bh
)->name
+ 1;
154 pr_debug("readdir(): dir_emit(\"%.*s\", ino=%u), hash=%d, f_pos=%llx\n",
155 namelen
, name
, ino
, hash_pos
, ctx
->pos
);
157 if (!dir_emit(ctx
, name
, namelen
, ino
, DT_UNKNOWN
))
160 ino
= be32_to_cpu(AFFS_TAIL(sb
, fh_bh
)->hash_chain
);
166 data
->cookie
= inode_query_iversion(inode
);
174 affs_unlock_dir(inode
);