* [patch 2/2] cifs: dereferencing first then checking
@ 2010-10-27 21:22 Dan Carpenter
2010-10-27 23:51 ` Jeff Layton
0 siblings, 1 reply; 5+ messages in thread
From: Dan Carpenter @ 2010-10-27 21:22 UTC (permalink / raw)
To: Steve French
Cc: linux-cifs-u79uwXL29TY76Z2rM5mHXA,
samba-technical-w/Ol4Ecudpl8XjKLYN78aQ,
kernel-janitors-u79uwXL29TY76Z2rM5mHXA
Smatch complained about a couple checking for NULL after dereferencing
bugs. I'm not super familiar with the code so I did the conservative
thing and move the dereferences after the checks.
The dereferences in cifs_lock() and cifs_fsync() were added in
ba00ba64cf0 "cifs: make various routines use the cifsFileInfo->tcon
pointer". The dereference in find_writable_file() was added in
6508d904e6f "cifs: have find_readable/writable_file filter by fsuid".
The comments there say it's possible to trigger the NULL dereference
under stress.
Signed-off-by: Dan Carpenter <error27@gmail.com>
diff --git a/fs/cifs/file.c b/fs/cifs/file.c
index 06a006b..33a19b4 100644
--- a/fs/cifs/file.c
+++ b/fs/cifs/file.c
@@ -775,13 +775,13 @@ int cifs_lock(struct file *file, int cmd, struct file_lock *pfLock)
cFYI(1, "Unknown type of lock");
cifs_sb = CIFS_SB(file->f_path.dentry->d_sb);
- tcon = tlink_tcon(((struct cifsFileInfo *)file->private_data)->tlink);
if (file->private_data = NULL) {
rc = -EBADF;
FreeXid(xid);
return rc;
}
+ tcon = tlink_tcon(((struct cifsFileInfo *)file->private_data)->tlink);
netfid = ((struct cifsFileInfo *)file->private_data)->netfid;
if ((tcon->ses->capabilities & CAP_UNIX) &&
@@ -1179,7 +1179,7 @@ struct cifsFileInfo *find_writable_file(struct cifsInodeInfo *cifs_inode,
bool fsuid_only)
{
struct cifsFileInfo *open_file;
- struct cifs_sb_info *cifs_sb = CIFS_SB(cifs_inode->vfs_inode.i_sb);
+ struct cifs_sb_info *cifs_sb;
bool any_available = false;
int rc;
@@ -1192,6 +1192,7 @@ struct cifsFileInfo *find_writable_file(struct cifsInodeInfo *cifs_inode,
dump_stack();
return NULL;
}
+ cifs_sb = CIFS_SB(cifs_inode->vfs_inode.i_sb);
/* only filter by fsuid on multiuser mounts */
if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MULTIUSER))
@@ -1628,15 +1629,26 @@ int cifs_fsync(struct file *file, int datasync)
file->f_path.dentry->d_name.name, datasync);
rc = filemap_write_and_wait(inode->i_mapping);
- if (rc = 0) {
- rc = CIFS_I(inode)->write_behind_rc;
- CIFS_I(inode)->write_behind_rc = 0;
- tcon = tlink_tcon(smbfile->tlink);
- if (!rc && tcon && smbfile &&
- !(CIFS_SB(inode->i_sb)->mnt_cifs_flags & CIFS_MOUNT_NOSSYNC))
- rc = CIFSSMBFlush(xid, tcon, smbfile->netfid);
- }
+ if (rc)
+ goto err_out;
+
+ rc = CIFS_I(inode)->write_behind_rc;
+ CIFS_I(inode)->write_behind_rc = 0;
+ if (rc)
+ goto err_out;
+
+ if (CIFS_SB(inode->i_sb)->mnt_cifs_flags & CIFS_MOUNT_NOSSYNC)
+ goto done;
+ if (!smbfile)
+ goto done;
+ tcon = tlink_tcon(smbfile->tlink);
+ if (!tcon)
+ goto done;
+
+ rc = CIFSSMBFlush(xid, tcon, smbfile->netfid);
+err_out:
+done:
FreeXid(xid);
return rc;
}
^ permalink raw reply related [flat|nested] 5+ messages in thread
* Re: [patch 2/2] cifs: dereferencing first then checking
2010-10-27 21:22 [patch 2/2] cifs: dereferencing first then checking Dan Carpenter
@ 2010-10-27 23:51 ` Jeff Layton
[not found] ` <20101027195123.0a712a53-9yPaYZwiELC+kQycOl6kW4xkIHaj4LzF@public.gmane.org>
0 siblings, 1 reply; 5+ messages in thread
From: Jeff Layton @ 2010-10-27 23:51 UTC (permalink / raw)
To: Dan Carpenter
Cc: Steve French, linux-cifs-u79uwXL29TY76Z2rM5mHXA,
samba-technical-w/Ol4Ecudpl8XjKLYN78aQ,
kernel-janitors-u79uwXL29TY76Z2rM5mHXA
On Wed, 27 Oct 2010 23:22:53 +0200
Dan Carpenter <error27@gmail.com> wrote:
> Smatch complained about a couple checking for NULL after dereferencing
> bugs. I'm not super familiar with the code so I did the conservative
> thing and move the dereferences after the checks.
>
> The dereferences in cifs_lock() and cifs_fsync() were added in
> ba00ba64cf0 "cifs: make various routines use the cifsFileInfo->tcon
> pointer". The dereference in find_writable_file() was added in
> 6508d904e6f "cifs: have find_readable/writable_file filter by fsuid".
> The comments there say it's possible to trigger the NULL dereference
> under stress.
>
> Signed-off-by: Dan Carpenter <error27@gmail.com>
>
> diff --git a/fs/cifs/file.c b/fs/cifs/file.c
> index 06a006b..33a19b4 100644
> --- a/fs/cifs/file.c
> +++ b/fs/cifs/file.c
> @@ -775,13 +775,13 @@ int cifs_lock(struct file *file, int cmd, struct file_lock *pfLock)
> cFYI(1, "Unknown type of lock");
>
> cifs_sb = CIFS_SB(file->f_path.dentry->d_sb);
> - tcon = tlink_tcon(((struct cifsFileInfo *)file->private_data)->tlink);
>
> if (file->private_data = NULL) {
> rc = -EBADF;
> FreeXid(xid);
> return rc;
> }
^^^^^^^
I think the above check can just go away now. If this pointer is NULL
then something is badly wrong.
> + tcon = tlink_tcon(((struct cifsFileInfo *)file->private_data)->tlink);
> netfid = ((struct cifsFileInfo *)file->private_data)->netfid;
>
> if ((tcon->ses->capabilities & CAP_UNIX) &&
> @@ -1179,7 +1179,7 @@ struct cifsFileInfo *find_writable_file(struct cifsInodeInfo *cifs_inode,
> bool fsuid_only)
> {
> struct cifsFileInfo *open_file;
> - struct cifs_sb_info *cifs_sb = CIFS_SB(cifs_inode->vfs_inode.i_sb);
> + struct cifs_sb_info *cifs_sb;
> bool any_available = false;
> int rc;
>
> @@ -1192,6 +1192,7 @@ struct cifsFileInfo *find_writable_file(struct cifsInodeInfo *cifs_inode,
> dump_stack();
> return NULL;
> }
^^^^^^
I think we ought to rid ourselves of the above check too. It's
not clear how we'd get a NULL pointer there anyway -- mostly we
get cifsInodeInfo pointers from the CIFS_I(inode) macro which does a
container_of() on the inode. That should never return a NULL
pointer anyway, unless the inode pointer happens to be corrupt
in a very specific way.
> + cifs_sb = CIFS_SB(cifs_inode->vfs_inode.i_sb);
>
> /* only filter by fsuid on multiuser mounts */
> if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MULTIUSER))
> @@ -1628,15 +1629,26 @@ int cifs_fsync(struct file *file, int datasync)
> file->f_path.dentry->d_name.name, datasync);
>
> rc = filemap_write_and_wait(inode->i_mapping);
> - if (rc = 0) {
> - rc = CIFS_I(inode)->write_behind_rc;
> - CIFS_I(inode)->write_behind_rc = 0;
> - tcon = tlink_tcon(smbfile->tlink);
> - if (!rc && tcon && smbfile &&
> - !(CIFS_SB(inode->i_sb)->mnt_cifs_flags & CIFS_MOUNT_NOSSYNC))
> - rc = CIFSSMBFlush(xid, tcon, smbfile->netfid);
> - }
> + if (rc)
> + goto err_out;
> +
> + rc = CIFS_I(inode)->write_behind_rc;
> + CIFS_I(inode)->write_behind_rc = 0;
> + if (rc)
> + goto err_out;
> +
> + if (CIFS_SB(inode->i_sb)->mnt_cifs_flags & CIFS_MOUNT_NOSSYNC)
> + goto done;
> + if (!smbfile)
> + goto done;
> + tcon = tlink_tcon(smbfile->tlink);
> + if (!tcon)
> + goto done;
> +
> + rc = CIFSSMBFlush(xid, tcon, smbfile->netfid);
>
> +err_out:
> +done:
> FreeXid(xid);
> return rc;
> }
The above delta shouldn't be needed once the patches in Steve's tree
are pushed to Linus. I think what we should probably do is wait for
Steve to push to Linus and then do a patch to remove the first two
checks.
Sound like a plan?
--
Jeff Layton <jlayton@redhat.com>
^ permalink raw reply [flat|nested] 5+ messages in thread
* Re: [patch 2/2] cifs: dereferencing first then checking
[not found] ` <20101027195123.0a712a53-9yPaYZwiELC+kQycOl6kW4xkIHaj4LzF@public.gmane.org>
@ 2010-10-28 3:44 ` Dan Carpenter
2010-11-02 20:22 ` Jeff Layton
0 siblings, 1 reply; 5+ messages in thread
From: Dan Carpenter @ 2010-10-28 3:44 UTC (permalink / raw)
To: Jeff Layton
Cc: Steve French, linux-cifs-u79uwXL29TY76Z2rM5mHXA,
samba-technical-w/Ol4Ecudpl8XjKLYN78aQ,
kernel-janitors-u79uwXL29TY76Z2rM5mHXA
On Wed, Oct 27, 2010 at 07:51:23PM -0400, Jeff Layton wrote:
>
> The above delta shouldn't be needed once the patches in Steve's tree
> are pushed to Linus. I think what we should probably do is wait for
> Steve to push to Linus and then do a patch to remove the first two
> checks.
>
> Sound like a plan?
>
Sounds like a plan.
regards,
dan carpenter
> --
> Jeff Layton <jlayton@redhat.com>
^ permalink raw reply [flat|nested] 5+ messages in thread
* Re: [patch 2/2] cifs: dereferencing first then checking
2010-10-28 3:44 ` Dan Carpenter
@ 2010-11-02 20:22 ` Jeff Layton
[not found] ` <20101102162250.3a2aa670-9yPaYZwiELC+kQycOl6kW4xkIHaj4LzF@public.gmane.org>
0 siblings, 1 reply; 5+ messages in thread
From: Jeff Layton @ 2010-11-02 20:22 UTC (permalink / raw)
To: Dan Carpenter
Cc: Steve French, linux-cifs-u79uwXL29TY76Z2rM5mHXA,
samba-technical-w/Ol4Ecudpl8XjKLYN78aQ,
kernel-janitors-u79uwXL29TY76Z2rM5mHXA
On Thu, 28 Oct 2010 05:44:34 +0200
Dan Carpenter <error27@gmail.com> wrote:
> On Wed, Oct 27, 2010 at 07:51:23PM -0400, Jeff Layton wrote:
> >
> > The above delta shouldn't be needed once the patches in Steve's tree
> > are pushed to Linus. I think what we should probably do is wait for
> > Steve to push to Linus and then do a patch to remove the first two
> > checks.
> >
> > Sound like a plan?
> >
>
> Sounds like a plan.
>
> regards,
> dan carpenter
>
Here's a respun patch that should fix these problems. I chickened out
on removing the check in find_writable_file.
It still seems like a bogus cargo-cult sort of thing. It should
probably be a BUG() if you call that function with a NULL pointer, but
I don't feel like tackling that just yet.
Dan, does this look ok to you? If so, I'll resend to Steve as an
"official" patch.
---------------------[snip]-----------------------
cifs: dereferencing first then checking (try #2)
This patch is based on Dan's original patch. His original description is
below:
Smatch complained about a couple checking for NULL after dereferencing
bugs. I'm not super familiar with the code so I did the conservative
thing and move the dereferences after the checks.
The dereferences in cifs_lock() and cifs_fsync() were added in
ba00ba64cf0 "cifs: make various routines use the cifsFileInfo->tcon
pointer". The dereference in find_writable_file() was added in
6508d904e6f "cifs: have find_readable/writable_file filter by fsuid".
The comments there say it's possible to trigger the NULL dereference
under stress.
Signed-off-by: Dan Carpenter <error27@gmail.com>
Signed-off-by: Jeff Layton <jlayton@redhat.com>
---
fs/cifs/file.c | 10 +++-------
1 files changed, 3 insertions(+), 7 deletions(-)
diff --git a/fs/cifs/file.c b/fs/cifs/file.c
index a566f15..a59b53f 100644
--- a/fs/cifs/file.c
+++ b/fs/cifs/file.c
@@ -755,12 +755,6 @@ int cifs_lock(struct file *file, int cmd, struct file_lock *pfLock)
cifs_sb = CIFS_SB(file->f_path.dentry->d_sb);
tcon = tlink_tcon(((struct cifsFileInfo *)file->private_data)->tlink);
-
- if (file->private_data = NULL) {
- rc = -EBADF;
- FreeXid(xid);
- return rc;
- }
netfid = ((struct cifsFileInfo *)file->private_data)->netfid;
if ((tcon->ses->capabilities & CAP_UNIX) &&
@@ -1155,7 +1149,7 @@ struct cifsFileInfo *find_writable_file(struct cifsInodeInfo *cifs_inode,
bool fsuid_only)
{
struct cifsFileInfo *open_file;
- struct cifs_sb_info *cifs_sb = CIFS_SB(cifs_inode->vfs_inode.i_sb);
+ struct cifs_sb_info *cifs_sb;
bool any_available = false;
int rc;
@@ -1169,6 +1163,8 @@ struct cifsFileInfo *find_writable_file(struct cifsInodeInfo *cifs_inode,
return NULL;
}
+ cifs_sb = CIFS_SB(cifs_inode->vfs_inode.i_sb);
+
/* only filter by fsuid on multiuser mounts */
if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MULTIUSER))
fsuid_only = false;
--
1.7.2.3
^ permalink raw reply related [flat|nested] 5+ messages in thread
* Re: [patch 2/2] cifs: dereferencing first then checking
[not found] ` <20101102162250.3a2aa670-9yPaYZwiELC+kQycOl6kW4xkIHaj4LzF@public.gmane.org>
@ 2010-11-03 16:40 ` Dan Carpenter
0 siblings, 0 replies; 5+ messages in thread
From: Dan Carpenter @ 2010-11-03 16:40 UTC (permalink / raw)
To: Jeff Layton
Cc: Steve French, linux-cifs-u79uwXL29TY76Z2rM5mHXA,
samba-technical-w/Ol4Ecudpl8XjKLYN78aQ,
kernel-janitors-u79uwXL29TY76Z2rM5mHXA
On Tue, Nov 02, 2010 at 04:22:50PM -0400, Jeff Layton wrote:
> Here's a respun patch that should fix these problems. I chickened out
> on removing the check in find_writable_file.
>
> It still seems like a bogus cargo-cult sort of thing. It should
> probably be a BUG() if you call that function with a NULL pointer, but
> I don't feel like tackling that just yet.
>
> Dan, does this look ok to you? If so, I'll resend to Steve as an
> "official" patch.
>
Looks fine to me. Obviously, I've already said that I don't know if the
checks are needed or not. I defer to your greater knowledge. ;)
regards,
dan carpenter
^ permalink raw reply [flat|nested] 5+ messages in thread
end of thread, other threads:[~2010-11-03 16:40 UTC | newest]
Thread overview: 5+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2010-10-27 21:22 [patch 2/2] cifs: dereferencing first then checking Dan Carpenter
2010-10-27 23:51 ` Jeff Layton
[not found] ` <20101027195123.0a712a53-9yPaYZwiELC+kQycOl6kW4xkIHaj4LzF@public.gmane.org>
2010-10-28 3:44 ` Dan Carpenter
2010-11-02 20:22 ` Jeff Layton
[not found] ` <20101102162250.3a2aa670-9yPaYZwiELC+kQycOl6kW4xkIHaj4LzF@public.gmane.org>
2010-11-03 16:40 ` Dan Carpenter
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).