* [PATCH] dcache: error out if the name buffer is too short
@ 2014-01-24 14:04 Denys Vlasenko
2014-01-24 16:19 ` Oleg Nesterov
0 siblings, 1 reply; 7+ messages in thread
From: Denys Vlasenko @ 2014-01-24 14:04 UTC (permalink / raw)
To: Al Viro; +Cc: Denys Vlasenko, Jan Kratochvil, Oleg Nesterov, linux-kernel
This change makes __dentry_path() and d_path()
immediately return ENAMETOOLONG if buflen < 2.
Cc: Jan Kratochvil <jan.kratochvil@redhat.com>
Cc: Oleg Nesterov <oleg@redhat.com>
Cc: linux-kernel@vger.kernel.org
Signed-off-by: Denys Vlasenko <dvlasenk@redhat.com>
---
fs/dcache.c | 8 ++++++--
1 file changed, 6 insertions(+), 2 deletions(-)
diff --git a/fs/dcache.c b/fs/dcache.c
index cb4a106..2fba276 100644
--- a/fs/dcache.c
+++ b/fs/dcache.c
@@ -3055,6 +3055,9 @@ char *d_path(const struct path *path, char *buf, int buflen)
struct path root;
int error;
+ if (buflen < 2)
+ return ERR_PTR(-ENAMETOOLONG);
+
/*
* We have various synthetic filesystems that never get mounted. On
* these filesystems dentries are never used for lookup purposes, and
@@ -3122,13 +3125,14 @@ static char *__dentry_path(struct dentry *dentry, char *buf, int buflen)
int len, seq = 0;
int error = 0;
+ if (buflen < 2)
+ goto Elong;
+
rcu_read_lock();
restart:
end = buf + buflen;
len = buflen;
prepend(&end, &len, "\0", 1);
- if (buflen < 1)
- goto Elong;
/* Get '/' right */
retval = end-1;
*retval = '/';
--
1.8.1.4
^ permalink raw reply related [flat|nested] 7+ messages in thread
* Re: [PATCH] dcache: error out if the name buffer is too short
2014-01-24 14:04 [PATCH] dcache: error out if the name buffer is too short Denys Vlasenko
@ 2014-01-24 16:19 ` Oleg Nesterov
2014-01-26 15:37 ` Oleg Nesterov
0 siblings, 1 reply; 7+ messages in thread
From: Oleg Nesterov @ 2014-01-24 16:19 UTC (permalink / raw)
To: Denys Vlasenko; +Cc: Al Viro, Jan Kratochvil, linux-kernel
On 01/24, Denys Vlasenko wrote:
>
> This change makes __dentry_path() and d_path()
> immediately return ENAMETOOLONG if buflen < 2.
I am not sure about d_path, but as for __dentry_path:
> @@ -3122,13 +3125,14 @@ static char *__dentry_path(struct dentry *dentry, char *buf, int buflen)
> int len, seq = 0;
> int error = 0;
>
> + if (buflen < 2)
> + goto Elong;
> +
> rcu_read_lock();
> restart:
> end = buf + buflen;
> len = buflen;
> prepend(&end, &len, "\0", 1);
> - if (buflen < 1)
> - goto Elong;
you forgot to mention that this change fixes a bug, this "goto Elong"
leaks rcu_read_lock().
And probably you are right, the fix should be as simple as possible.
But can't we also simplify __dentry_path? Unless I missed something
we can move prepend() up, before rcu_read_lock(), "move Get '/' right"
into that prepend, and even kill retval... OK, most probably I missed
something, but at first glance we can do something like
static char *__dentry_path(struct dentry *dentry, char *buf, int buflen)
{
int len, seq = 0;
int error = 0;
char *end;
buf += buflen;
/* Get '/' right, write "/\0" at the end */
if (prepend(&buf, &buflen, "/", 2))
goto Elong;
rcu_read_lock();
restart:
end = buf;
len = buflen;
read_seqbegin_or_lock(&rename_lock, &seq);
while (!IS_ROOT(dentry)) {
struct dentry *parent = dentry->d_parent;
int error;
prefetch(parent);
error = prepend_name(&end, &len, &dentry->d_name);
if (error)
break;
dentry = parent;
}
if (!(seq & 1))
rcu_read_unlock();
if (need_seqretry(&rename_lock, seq)) {
seq = 1;
goto restart;
}
done_seqretry(&rename_lock, seq);
if (!error)
return end;
Elong:
return ERR_PTR(-ENAMETOOLONG);
}
Oleg.
^ permalink raw reply [flat|nested] 7+ messages in thread
* Re: [PATCH] dcache: error out if the name buffer is too short
2014-01-24 16:19 ` Oleg Nesterov
@ 2014-01-26 15:37 ` Oleg Nesterov
2014-01-26 15:51 ` Oleg Nesterov
0 siblings, 1 reply; 7+ messages in thread
From: Oleg Nesterov @ 2014-01-26 15:37 UTC (permalink / raw)
To: Denys Vlasenko; +Cc: Al Viro, Jan Kratochvil, linux-kernel
On 01/24, Oleg Nesterov wrote:
>
> And probably you are right, the fix should be as simple as possible.
> But can't we also simplify __dentry_path? Unless I missed something
> we can move prepend() up, before rcu_read_lock(), "move Get '/' right"
> into that prepend, and even kill retval... OK, most probably I missed
> something,
Of course I missed something ;)
> but at first glance we can do something like
>
> static char *__dentry_path(struct dentry *dentry, char *buf, int buflen)
> {
> int len, seq = 0;
> int error = 0;
> char *end;
>
> buf += buflen;
> /* Get '/' right, write "/\0" at the end */
> if (prepend(&buf, &buflen, "/", 2))
> goto Elong;
Heh. Not sure what I was thinking about, but this looks obviously wrong
when I re-read my email. This will add the extra "/" at the end, unless
IS_ROOT().
Sorry for noise.
Oleg.
^ permalink raw reply [flat|nested] 7+ messages in thread
* Re: [PATCH] dcache: error out if the name buffer is too short
2014-01-26 15:37 ` Oleg Nesterov
@ 2014-01-26 15:51 ` Oleg Nesterov
2014-01-26 16:35 ` Oleg Nesterov
0 siblings, 1 reply; 7+ messages in thread
From: Oleg Nesterov @ 2014-01-26 15:51 UTC (permalink / raw)
To: Denys Vlasenko; +Cc: Al Viro, Jan Kratochvil, linux-kernel
On 01/26, Oleg Nesterov wrote:
>
> Heh. Not sure what I was thinking about, but this looks obviously wrong
> when I re-read my email. This will add the extra "/" at the end, unless
> IS_ROOT().
And this motivated me to try to actually read __dentry_path().
Al, Denys, unless I am totally confused the "restart" logic is very broken.
We can't simply restart because the main loop changes dentry?
Oleg.
^ permalink raw reply [flat|nested] 7+ messages in thread
* Re: [PATCH] dcache: error out if the name buffer is too short
2014-01-26 15:51 ` Oleg Nesterov
@ 2014-01-26 16:35 ` Oleg Nesterov
2014-01-26 17:41 ` Al Viro
0 siblings, 1 reply; 7+ messages in thread
From: Oleg Nesterov @ 2014-01-26 16:35 UTC (permalink / raw)
To: Denys Vlasenko; +Cc: Al Viro, Jan Kratochvil, linux-kernel
On 01/26, Oleg Nesterov wrote:
>
> Al, Denys, unless I am totally confused the "restart" logic is very broken.
> We can't simply restart because the main loop changes dentry?
Plus prepend_name() can't actually return -ENAMETOOLONG,
"int error" inside the loop is wrong.
I believe the minimal fix is something like below. I'll try to test
this change (currently I can't) later and resend. And I still think
it can be cleanuped a bit, but this is minor.
Oleg.
--- a/fs/dcache.c
+++ b/fs/dcache.c
@@ -3103,7 +3103,7 @@ char *simple_dname(struct dentry *dentry, char *buffer, int buflen)
/* these dentries are never renamed, so d_lock is not needed */
if (prepend(&end, &buflen, " (deleted)", 11) ||
prepend(&end, &buflen, dentry->d_name.name, dentry->d_name.len) ||
- prepend(&end, &buflen, "/", 1))
+ prepend(&end, &buflen, "/", 1))
end = ERR_PTR(-ENAMETOOLONG);
return end;
}
@@ -3113,32 +3113,34 @@ char *simple_dname(struct dentry *dentry, char *buffer, int buflen)
*/
static char *__dentry_path(struct dentry *dentry, char *buf, int buflen)
{
+ struct dentry *de;
char *end, *retval;
int len, seq = 0;
int error = 0;
+ if (buflen < 2)
+ goto Elong;
+
rcu_read_lock();
restart:
end = buf + buflen;
len = buflen;
prepend(&end, &len, "\0", 1);
- if (buflen < 1)
- goto Elong;
/* Get '/' right */
retval = end-1;
*retval = '/';
+ de = dentry;
read_seqbegin_or_lock(&rename_lock, &seq);
- while (!IS_ROOT(dentry)) {
- struct dentry *parent = dentry->d_parent;
- int error;
+ while (!IS_ROOT(de)) {
+ struct dentry *parent = de->d_parent;
prefetch(parent);
- error = prepend_name(&end, &len, &dentry->d_name);
+ error = prepend_name(&end, &len, &de->d_name);
if (error)
break;
retval = end;
- dentry = parent;
+ de = parent;
}
if (!(seq & 1))
rcu_read_unlock();
^ permalink raw reply [flat|nested] 7+ messages in thread
* Re: [PATCH] dcache: error out if the name buffer is too short
2014-01-26 16:35 ` Oleg Nesterov
@ 2014-01-26 17:41 ` Al Viro
2014-01-27 17:40 ` Oleg Nesterov
0 siblings, 1 reply; 7+ messages in thread
From: Al Viro @ 2014-01-26 17:41 UTC (permalink / raw)
To: Oleg Nesterov; +Cc: Denys Vlasenko, Jan Kratochvil, linux-kernel
On Sun, Jan 26, 2014 at 05:35:36PM +0100, Oleg Nesterov wrote:
> On 01/26, Oleg Nesterov wrote:
> >
> > Al, Denys, unless I am totally confused the "restart" logic is very broken.
> > We can't simply restart because the main loop changes dentry?
>
> Plus prepend_name() can't actually return -ENAMETOOLONG,
> "int error" inside the loop is wrong.
... and already fixed. Said that, the point about restarts is definitely
true. See vfs.git#for-linus - it should propagate in a few
^ permalink raw reply [flat|nested] 7+ messages in thread
* Re: [PATCH] dcache: error out if the name buffer is too short
2014-01-26 17:41 ` Al Viro
@ 2014-01-27 17:40 ` Oleg Nesterov
0 siblings, 0 replies; 7+ messages in thread
From: Oleg Nesterov @ 2014-01-27 17:40 UTC (permalink / raw)
To: Al Viro; +Cc: Denys Vlasenko, Jan Kratochvil, linux-kernel
On 01/26, Al Viro wrote:
>
> ... and already fixed. Said that, the point about restarts is definitely
> true. See vfs.git#for-linus - it should propagate in a few
Yes, http://git.kernel.org/cgit/linux/kernel/git/viro/vfs.git/commit/?h=for-linus&id=f6500801522c61782d4990fa1ad96154cb397cd4
should obviously fix both problems.
Thanks,
Oleg.
^ permalink raw reply [flat|nested] 7+ messages in thread
end of thread, other threads:[~2014-01-27 17:40 UTC | newest]
Thread overview: 7+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2014-01-24 14:04 [PATCH] dcache: error out if the name buffer is too short Denys Vlasenko
2014-01-24 16:19 ` Oleg Nesterov
2014-01-26 15:37 ` Oleg Nesterov
2014-01-26 15:51 ` Oleg Nesterov
2014-01-26 16:35 ` Oleg Nesterov
2014-01-26 17:41 ` Al Viro
2014-01-27 17:40 ` Oleg Nesterov
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox