Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
linux
Commits
f0dd2136
Commit
f0dd2136
authored
Jun 22, 2005
by
Trond Myklebust
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
[PATCH] NFS: Clean up readdir changes.
Signed-off-by:
Trond Myklebust
<
Trond.Myklebust@netapp.com
>
parent
00a92642
Changes
3
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
48 additions
and
39 deletions
+48
-39
fs/nfs/dir.c
fs/nfs/dir.c
+48
-37
fs/nfs/inode.c
fs/nfs/inode.c
+0
-1
include/linux/nfs_fs.h
include/linux/nfs_fs.h
+0
-1
No files found.
fs/nfs/dir.c
View file @
f0dd2136
...
@@ -51,8 +51,10 @@ static int nfs_mknod(struct inode *, struct dentry *, int, dev_t);
...
@@ -51,8 +51,10 @@ static int nfs_mknod(struct inode *, struct dentry *, int, dev_t);
static
int
nfs_rename
(
struct
inode
*
,
struct
dentry
*
,
static
int
nfs_rename
(
struct
inode
*
,
struct
dentry
*
,
struct
inode
*
,
struct
dentry
*
);
struct
inode
*
,
struct
dentry
*
);
static
int
nfs_fsync_dir
(
struct
file
*
,
struct
dentry
*
,
int
);
static
int
nfs_fsync_dir
(
struct
file
*
,
struct
dentry
*
,
int
);
static
loff_t
nfs_llseek_dir
(
struct
file
*
,
loff_t
,
int
);
struct
file_operations
nfs_dir_operations
=
{
struct
file_operations
nfs_dir_operations
=
{
.
llseek
=
nfs_llseek_dir
,
.
read
=
generic_read_dir
,
.
read
=
generic_read_dir
,
.
readdir
=
nfs_readdir
,
.
readdir
=
nfs_readdir
,
.
open
=
nfs_opendir
,
.
open
=
nfs_opendir
,
...
@@ -141,9 +143,8 @@ typedef struct {
...
@@ -141,9 +143,8 @@ typedef struct {
struct
page
*
page
;
struct
page
*
page
;
unsigned
long
page_index
;
unsigned
long
page_index
;
u32
*
ptr
;
u32
*
ptr
;
u64
target_cookie
;
u64
*
dir_cookie
;
int
target_index
;
loff_t
current_index
;
int
current_index
;
struct
nfs_entry
*
entry
;
struct
nfs_entry
*
entry
;
decode_dirent_t
decode
;
decode_dirent_t
decode
;
int
plus
;
int
plus
;
...
@@ -227,7 +228,7 @@ void dir_page_release(nfs_readdir_descriptor_t *desc)
...
@@ -227,7 +228,7 @@ void dir_page_release(nfs_readdir_descriptor_t *desc)
/*
/*
* Given a pointer to a buffer that has already been filled by a call
* Given a pointer to a buffer that has already been filled by a call
* to readdir, find the next entry with cookie '
desc->target
_cookie'.
* to readdir, find the next entry with cookie '
*desc->dir
_cookie'.
*
*
* If the end of the buffer has been reached, return -EAGAIN, if not,
* If the end of the buffer has been reached, return -EAGAIN, if not,
* return the offset within the buffer of the next entry to be
* return the offset within the buffer of the next entry to be
...
@@ -241,8 +242,8 @@ int find_dirent(nfs_readdir_descriptor_t *desc)
...
@@ -241,8 +242,8 @@ int find_dirent(nfs_readdir_descriptor_t *desc)
status
;
status
;
while
((
status
=
dir_decode
(
desc
))
==
0
)
{
while
((
status
=
dir_decode
(
desc
))
==
0
)
{
dfprintk
(
VFS
,
"NFS: found cookie %Lu
\n
"
,
(
long
long
)
entry
->
cookie
);
dfprintk
(
VFS
,
"NFS: found cookie %Lu
\n
"
,
(
unsigned
long
long
)
entry
->
cookie
);
if
(
entry
->
prev_cookie
==
desc
->
target
_cookie
)
if
(
entry
->
prev_cookie
==
*
desc
->
dir
_cookie
)
break
;
break
;
if
(
loop_count
++
>
200
)
{
if
(
loop_count
++
>
200
)
{
loop_count
=
0
;
loop_count
=
0
;
...
@@ -255,7 +256,7 @@ int find_dirent(nfs_readdir_descriptor_t *desc)
...
@@ -255,7 +256,7 @@ int find_dirent(nfs_readdir_descriptor_t *desc)
/*
/*
* Given a pointer to a buffer that has already been filled by a call
* Given a pointer to a buffer that has already been filled by a call
* to readdir, find the entry at offset 'desc->
target_index
'.
* to readdir, find the entry at offset 'desc->
file->f_pos
'.
*
*
* If the end of the buffer has been reached, return -EAGAIN, if not,
* If the end of the buffer has been reached, return -EAGAIN, if not,
* return the offset within the buffer of the next entry to be
* return the offset within the buffer of the next entry to be
...
@@ -273,10 +274,10 @@ int find_dirent_index(nfs_readdir_descriptor_t *desc)
...
@@ -273,10 +274,10 @@ int find_dirent_index(nfs_readdir_descriptor_t *desc)
if
(
status
)
if
(
status
)
break
;
break
;
dfprintk
(
VFS
,
"NFS: found cookie %Lu at index %
d
\n
"
,
(
long
long
)
entry
->
cookie
,
desc
->
current_index
);
dfprintk
(
VFS
,
"NFS: found cookie %Lu at index %
Ld
\n
"
,
(
unsigned
long
long
)
entry
->
cookie
,
desc
->
current_index
);
if
(
desc
->
target_index
==
desc
->
current_index
)
{
if
(
desc
->
file
->
f_pos
==
desc
->
current_index
)
{
desc
->
target
_cookie
=
entry
->
cookie
;
*
desc
->
dir
_cookie
=
entry
->
cookie
;
break
;
break
;
}
}
desc
->
current_index
++
;
desc
->
current_index
++
;
...
@@ -314,7 +315,7 @@ int find_dirent_page(nfs_readdir_descriptor_t *desc)
...
@@ -314,7 +315,7 @@ int find_dirent_page(nfs_readdir_descriptor_t *desc)
/* NOTE: Someone else may have changed the READDIRPLUS flag */
/* NOTE: Someone else may have changed the READDIRPLUS flag */
desc
->
page
=
page
;
desc
->
page
=
page
;
desc
->
ptr
=
kmap
(
page
);
/* matching kunmap in nfs_do_filldir */
desc
->
ptr
=
kmap
(
page
);
/* matching kunmap in nfs_do_filldir */
if
(
desc
->
target_cookie
)
if
(
*
desc
->
dir_cookie
!=
0
)
status
=
find_dirent
(
desc
);
status
=
find_dirent
(
desc
);
else
else
status
=
find_dirent_index
(
desc
);
status
=
find_dirent_index
(
desc
);
...
@@ -332,8 +333,8 @@ int find_dirent_page(nfs_readdir_descriptor_t *desc)
...
@@ -332,8 +333,8 @@ int find_dirent_page(nfs_readdir_descriptor_t *desc)
* Recurse through the page cache pages, and return a
* Recurse through the page cache pages, and return a
* filled nfs_entry structure of the next directory entry if possible.
* filled nfs_entry structure of the next directory entry if possible.
*
*
* The target for the search is '
desc->target
_cookie' if non-0,
* The target for the search is '
*desc->dir
_cookie' if non-0,
* 'desc->
target_index
' otherwise
* 'desc->
file->f_pos
' otherwise
*/
*/
static
inline
static
inline
int
readdir_search_pagecache
(
nfs_readdir_descriptor_t
*
desc
)
int
readdir_search_pagecache
(
nfs_readdir_descriptor_t
*
desc
)
...
@@ -341,18 +342,15 @@ int readdir_search_pagecache(nfs_readdir_descriptor_t *desc)
...
@@ -341,18 +342,15 @@ int readdir_search_pagecache(nfs_readdir_descriptor_t *desc)
int
loop_count
=
0
;
int
loop_count
=
0
;
int
res
;
int
res
;
if
(
desc
->
target_cookie
)
dfprintk
(
VFS
,
"NFS: readdir_search_pagecache() searching for cookie %Lu
\n
"
,
(
long
long
)
desc
->
target_cookie
);
else
dfprintk
(
VFS
,
"NFS: readdir_search_pagecache() searching for cookie number %d
\n
"
,
desc
->
target_index
);
/* Always search-by-index from the beginning of the cache */
/* Always search-by-index from the beginning of the cache */
if
(
!
(
desc
->
target_cookie
))
{
if
(
*
desc
->
dir_cookie
==
0
)
{
dfprintk
(
VFS
,
"NFS: readdir_search_pagecache() searching for offset %Ld
\n
"
,
(
long
long
)
desc
->
file
->
f_pos
);
desc
->
page_index
=
0
;
desc
->
page_index
=
0
;
desc
->
entry
->
cookie
=
desc
->
entry
->
prev_cookie
=
0
;
desc
->
entry
->
cookie
=
desc
->
entry
->
prev_cookie
=
0
;
desc
->
entry
->
eof
=
0
;
desc
->
entry
->
eof
=
0
;
desc
->
current_index
=
0
;
desc
->
current_index
=
0
;
}
}
else
dfprintk
(
VFS
,
"NFS: readdir_search_pagecache() searching for cookie %Lu
\n
"
,
(
unsigned
long
long
)
*
desc
->
dir_cookie
);
for
(;;)
{
for
(;;)
{
res
=
find_dirent_page
(
desc
);
res
=
find_dirent_page
(
desc
);
...
@@ -386,7 +384,6 @@ int nfs_do_filldir(nfs_readdir_descriptor_t *desc, void *dirent,
...
@@ -386,7 +384,6 @@ int nfs_do_filldir(nfs_readdir_descriptor_t *desc, void *dirent,
struct
file
*
file
=
desc
->
file
;
struct
file
*
file
=
desc
->
file
;
struct
nfs_entry
*
entry
=
desc
->
entry
;
struct
nfs_entry
*
entry
=
desc
->
entry
;
struct
dentry
*
dentry
=
NULL
;
struct
dentry
*
dentry
=
NULL
;
struct
nfs_open_context
*
ctx
=
file
->
private_data
;
unsigned
long
fileid
;
unsigned
long
fileid
;
int
loop_count
=
0
,
int
loop_count
=
0
,
res
;
res
;
...
@@ -415,7 +412,7 @@ int nfs_do_filldir(nfs_readdir_descriptor_t *desc, void *dirent,
...
@@ -415,7 +412,7 @@ int nfs_do_filldir(nfs_readdir_descriptor_t *desc, void *dirent,
if
(
res
<
0
)
if
(
res
<
0
)
break
;
break
;
file
->
f_pos
++
;
file
->
f_pos
++
;
desc
->
target
_cookie
=
entry
->
cookie
;
*
desc
->
dir
_cookie
=
entry
->
cookie
;
if
(
dir_decode
(
desc
)
!=
0
)
{
if
(
dir_decode
(
desc
)
!=
0
)
{
desc
->
page_index
++
;
desc
->
page_index
++
;
break
;
break
;
...
@@ -425,12 +422,10 @@ int nfs_do_filldir(nfs_readdir_descriptor_t *desc, void *dirent,
...
@@ -425,12 +422,10 @@ int nfs_do_filldir(nfs_readdir_descriptor_t *desc, void *dirent,
schedule
();
schedule
();
}
}
}
}
ctx
->
dir_pos
=
file
->
f_pos
;
ctx
->
dir_cookie
=
desc
->
target_cookie
;
dir_page_release
(
desc
);
dir_page_release
(
desc
);
if
(
dentry
!=
NULL
)
if
(
dentry
!=
NULL
)
dput
(
dentry
);
dput
(
dentry
);
dfprintk
(
VFS
,
"NFS: nfs_do_filldir() filling ended @ cookie %Lu; returning = %d
\n
"
,
(
long
long
)
desc
->
target
_cookie
,
res
);
dfprintk
(
VFS
,
"NFS: nfs_do_filldir() filling ended @ cookie %Lu; returning = %d
\n
"
,
(
unsigned
long
long
)
*
desc
->
dir
_cookie
,
res
);
return
res
;
return
res
;
}
}
...
@@ -456,14 +451,14 @@ int uncached_readdir(nfs_readdir_descriptor_t *desc, void *dirent,
...
@@ -456,14 +451,14 @@ int uncached_readdir(nfs_readdir_descriptor_t *desc, void *dirent,
struct
page
*
page
=
NULL
;
struct
page
*
page
=
NULL
;
int
status
;
int
status
;
dfprintk
(
VFS
,
"NFS: uncached_readdir() searching for cookie %Lu
\n
"
,
(
long
long
)
desc
->
target
_cookie
);
dfprintk
(
VFS
,
"NFS: uncached_readdir() searching for cookie %Lu
\n
"
,
(
unsigned
long
long
)
*
desc
->
dir
_cookie
);
page
=
alloc_page
(
GFP_HIGHUSER
);
page
=
alloc_page
(
GFP_HIGHUSER
);
if
(
!
page
)
{
if
(
!
page
)
{
status
=
-
ENOMEM
;
status
=
-
ENOMEM
;
goto
out
;
goto
out
;
}
}
desc
->
error
=
NFS_PROTO
(
inode
)
->
readdir
(
file
->
f_dentry
,
cred
,
desc
->
target
_cookie
,
desc
->
error
=
NFS_PROTO
(
inode
)
->
readdir
(
file
->
f_dentry
,
cred
,
*
desc
->
dir
_cookie
,
page
,
page
,
NFS_SERVER
(
inode
)
->
dtsize
,
NFS_SERVER
(
inode
)
->
dtsize
,
desc
->
plus
);
desc
->
plus
);
...
@@ -472,7 +467,7 @@ int uncached_readdir(nfs_readdir_descriptor_t *desc, void *dirent,
...
@@ -472,7 +467,7 @@ int uncached_readdir(nfs_readdir_descriptor_t *desc, void *dirent,
desc
->
ptr
=
kmap
(
page
);
/* matching kunmap in nfs_do_filldir */
desc
->
ptr
=
kmap
(
page
);
/* matching kunmap in nfs_do_filldir */
if
(
desc
->
error
>=
0
)
{
if
(
desc
->
error
>=
0
)
{
if
((
status
=
dir_decode
(
desc
))
==
0
)
if
((
status
=
dir_decode
(
desc
))
==
0
)
desc
->
entry
->
prev_cookie
=
desc
->
target
_cookie
;
desc
->
entry
->
prev_cookie
=
*
desc
->
dir
_cookie
;
}
else
}
else
status
=
-
EIO
;
status
=
-
EIO
;
if
(
status
<
0
)
if
(
status
<
0
)
...
@@ -501,7 +496,6 @@ static int nfs_readdir(struct file *filp, void *dirent, filldir_t filldir)
...
@@ -501,7 +496,6 @@ static int nfs_readdir(struct file *filp, void *dirent, filldir_t filldir)
{
{
struct
dentry
*
dentry
=
filp
->
f_dentry
;
struct
dentry
*
dentry
=
filp
->
f_dentry
;
struct
inode
*
inode
=
dentry
->
d_inode
;
struct
inode
*
inode
=
dentry
->
d_inode
;
struct
nfs_open_context
*
ctx
=
filp
->
private_data
;
nfs_readdir_descriptor_t
my_desc
,
nfs_readdir_descriptor_t
my_desc
,
*
desc
=
&
my_desc
;
*
desc
=
&
my_desc
;
struct
nfs_entry
my_entry
;
struct
nfs_entry
my_entry
;
...
@@ -519,21 +513,16 @@ static int nfs_readdir(struct file *filp, void *dirent, filldir_t filldir)
...
@@ -519,21 +513,16 @@ static int nfs_readdir(struct file *filp, void *dirent, filldir_t filldir)
/*
/*
* filp->f_pos points to the dirent entry number.
* filp->f_pos points to the dirent entry number.
*
ctx->dir_pos has the number of the cached cookie.
We have
*
*desc->dir_cookie has the cookie for the next entry.
We have
* to either find the entry with the appropriate number or
* to either find the entry with the appropriate number or
* revalidate the cookie.
* revalidate the cookie.
*/
*/
memset
(
desc
,
0
,
sizeof
(
*
desc
));
memset
(
desc
,
0
,
sizeof
(
*
desc
));
desc
->
file
=
filp
;
desc
->
file
=
filp
;
desc
->
dir_cookie
=
&
((
struct
nfs_open_context
*
)
filp
->
private_data
)
->
dir_cookie
;
desc
->
decode
=
NFS_PROTO
(
inode
)
->
decode_dirent
;
desc
->
decode
=
NFS_PROTO
(
inode
)
->
decode_dirent
;
desc
->
plus
=
NFS_USE_READDIRPLUS
(
inode
);
desc
->
plus
=
NFS_USE_READDIRPLUS
(
inode
);
desc
->
target_index
=
filp
->
f_pos
;
if
(
filp
->
f_pos
==
ctx
->
dir_pos
)
desc
->
target_cookie
=
ctx
->
dir_cookie
;
else
desc
->
target_cookie
=
0
;
my_entry
.
cookie
=
my_entry
.
prev_cookie
=
0
;
my_entry
.
cookie
=
my_entry
.
prev_cookie
=
0
;
my_entry
.
eof
=
0
;
my_entry
.
eof
=
0
;
...
@@ -546,7 +535,7 @@ static int nfs_readdir(struct file *filp, void *dirent, filldir_t filldir)
...
@@ -546,7 +535,7 @@ static int nfs_readdir(struct file *filp, void *dirent, filldir_t filldir)
if
(
res
==
-
EBADCOOKIE
)
{
if
(
res
==
-
EBADCOOKIE
)
{
/* This means either end of directory */
/* This means either end of directory */
if
(
desc
->
target_cookie
&&
desc
->
entry
->
cookie
!=
desc
->
target
_cookie
)
{
if
(
*
desc
->
dir_cookie
&&
desc
->
entry
->
cookie
!=
*
desc
->
dir
_cookie
)
{
/* Or that the server has 'lost' a cookie */
/* Or that the server has 'lost' a cookie */
res
=
uncached_readdir
(
desc
,
dirent
,
filldir
);
res
=
uncached_readdir
(
desc
,
dirent
,
filldir
);
if
(
res
>=
0
)
if
(
res
>=
0
)
...
@@ -579,6 +568,28 @@ static int nfs_readdir(struct file *filp, void *dirent, filldir_t filldir)
...
@@ -579,6 +568,28 @@ static int nfs_readdir(struct file *filp, void *dirent, filldir_t filldir)
return
0
;
return
0
;
}
}
loff_t
nfs_llseek_dir
(
struct
file
*
filp
,
loff_t
offset
,
int
origin
)
{
down
(
&
filp
->
f_dentry
->
d_inode
->
i_sem
);
switch
(
origin
)
{
case
1
:
offset
+=
filp
->
f_pos
;
case
0
:
if
(
offset
>=
0
)
break
;
default:
offset
=
-
EINVAL
;
goto
out
;
}
if
(
offset
!=
filp
->
f_pos
)
{
filp
->
f_pos
=
offset
;
((
struct
nfs_open_context
*
)
filp
->
private_data
)
->
dir_cookie
=
0
;
}
out:
up
(
&
filp
->
f_dentry
->
d_inode
->
i_sem
);
return
offset
;
}
/*
/*
* All directory operations under NFS are synchronous, so fsync()
* All directory operations under NFS are synchronous, so fsync()
* is a dummy operation.
* is a dummy operation.
...
...
fs/nfs/inode.c
View file @
f0dd2136
...
@@ -891,7 +891,6 @@ struct nfs_open_context *alloc_nfs_open_context(struct dentry *dentry, struct rp
...
@@ -891,7 +891,6 @@ struct nfs_open_context *alloc_nfs_open_context(struct dentry *dentry, struct rp
ctx
->
state
=
NULL
;
ctx
->
state
=
NULL
;
ctx
->
lockowner
=
current
->
files
;
ctx
->
lockowner
=
current
->
files
;
ctx
->
error
=
0
;
ctx
->
error
=
0
;
ctx
->
dir_pos
=
0
;
ctx
->
dir_cookie
=
0
;
ctx
->
dir_cookie
=
0
;
}
}
return
ctx
;
return
ctx
;
...
...
include/linux/nfs_fs.h
View file @
f0dd2136
...
@@ -85,7 +85,6 @@ struct nfs_open_context {
...
@@ -85,7 +85,6 @@ struct nfs_open_context {
struct
list_head
list
;
struct
list_head
list
;
int
dir_pos
;
/* Directory cookie cache */
__u64
dir_cookie
;
__u64
dir_cookie
;
};
};
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment