From patchwork Thu Jan 3 20:33:13 2013 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Weston Andros Adamson X-Patchwork-Id: 1929271 Return-Path: X-Original-To: patchwork-linux-nfs@patchwork.kernel.org Delivered-To: patchwork-process-083081@patchwork1.kernel.org Received: from vger.kernel.org (vger.kernel.org [209.132.180.67]) by patchwork1.kernel.org (Postfix) with ESMTP id A4A103FC33 for ; Thu, 3 Jan 2013 20:33:17 +0000 (UTC) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1753710Ab3ACUdQ (ORCPT ); Thu, 3 Jan 2013 15:33:16 -0500 Received: from mx12.netapp.com ([216.240.18.77]:38712 "EHLO mx12.netapp.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1753624Ab3ACUdQ (ORCPT ); Thu, 3 Jan 2013 15:33:16 -0500 X-IronPort-AV: E=Sophos;i="4.84,404,1355126400"; d="scan'208";a="4531577" Received: from smtp1.corp.netapp.com ([10.57.156.124]) by mx12-out.netapp.com with ESMTP; 03 Jan 2013 12:33:15 -0800 Received: from vpn2ntap-610999.vpn.netapp.com (vpn2ntap-610999.vpn.netapp.com [10.55.64.236]) by smtp1.corp.netapp.com (8.13.1/8.13.1/NTAP-1.6) with ESMTP id r03KXEox020131; Thu, 3 Jan 2013 12:33:14 -0800 (PST) From: Weston Andros Adamson To: Trond.Myklebust@netapp.com Cc: linux-nfs@vger.kernel.org, Weston Andros Adamson Subject: [PATCH] NFS: Fix access to suid/sgid executables Date: Thu, 3 Jan 2013 15:33:13 -0500 Message-Id: <1357245193-10375-1-git-send-email-dros@netapp.com> X-Mailer: git-send-email 1.7.9.6 (Apple Git-31.1) Sender: linux-nfs-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-nfs@vger.kernel.org nfs_open_permission_mask() shouldn't check MAY_READ for suid/sgid files that are opened with __FMODE_EXEC. Also fix NFSv4 access-in-open path in a similar way -- openflags must be used because fmode will not always have FMODE_EXEC set. Signed-off-by: Weston Andros Adamson --- This patch fixes https://bugzilla.kernel.org/show_bug.cgi?id=49101 The fix is not as obvious or clean as I'd like, so here is a little background: Not checking MAY_READ when a S_ISUID or S_ISGID file has MAY_EXEC causes suid/sgid executables to behave the same as on a local FS. The second part is fixing open-in-access for NFSv4 - we want to do the same thing as in nfs_open_permission_mask, but oddly enough fmode doesn't always indicate that the file is being opened for execution. Instead we have to use the openflags for this. Adding some debug prints show that the first exec of a suid/sgid file will open with fmode containing FMODE_EXEC and FMODE_READ, but subsequent execs will open with fmode containing FMODE_READ, but not FMODE_EXEC. openflags always has __FMODE_EXEC set in these examples. The first exec has these values in nfs4_opendata_access(): fmode = 0x21: contains read exec openflags = 0x8020: contains exec The second (and subsequent) execs have these values in nfs4_opendata_access(): fmode = 0x1: contains read openflags = 0x8020: contains exec -dros fs/nfs/dir.c | 12 +++++++++--- fs/nfs/nfs4proc.c | 13 +++++++++++-- 2 files changed, 20 insertions(+), 5 deletions(-) diff --git a/fs/nfs/dir.c b/fs/nfs/dir.c index 32e6c53..5141243 100644 --- a/fs/nfs/dir.c +++ b/fs/nfs/dir.c @@ -2149,7 +2149,7 @@ out: return -EACCES; } -static int nfs_open_permission_mask(int openflags) +static int nfs_open_permission_mask(struct inode *inode, int openflags) { int mask = 0; @@ -2157,14 +2157,20 @@ static int nfs_open_permission_mask(int openflags) mask |= MAY_READ; if ((openflags & O_ACCMODE) != O_RDONLY) mask |= MAY_WRITE; - if (openflags & __FMODE_EXEC) + if (openflags & __FMODE_EXEC) { mask |= MAY_EXEC; + /* don't check MAY_READ for exec of suid/sgid */ + if (inode->i_mode & (S_ISUID | S_ISGID)) + mask &= ~MAY_READ; + } + return mask; } int nfs_may_open(struct inode *inode, struct rpc_cred *cred, int openflags) { - return nfs_do_access(inode, cred, nfs_open_permission_mask(openflags)); + return nfs_do_access(inode, cred, + nfs_open_permission_mask(inode, openflags)); } EXPORT_SYMBOL_GPL(nfs_may_open); diff --git a/fs/nfs/nfs4proc.c b/fs/nfs/nfs4proc.c index 5d864fb..23cf1a8 100644 --- a/fs/nfs/nfs4proc.c +++ b/fs/nfs/nfs4proc.c @@ -1626,7 +1626,8 @@ static int _nfs4_recover_proc_open(struct nfs4_opendata *data) static int nfs4_opendata_access(struct rpc_cred *cred, struct nfs4_opendata *opendata, - struct nfs4_state *state, fmode_t fmode) + struct nfs4_state *state, fmode_t fmode, + int openflags) { struct nfs_access_entry cache; u32 mask; @@ -1644,6 +1645,14 @@ static int nfs4_opendata_access(struct rpc_cred *cred, if (fmode & FMODE_EXEC) mask |= MAY_EXEC; + /* use openflags to check for exec of suid/sgid, because fmode won't + * always have FMODE_EXEC set by VFS. + * Also, don't check MAY_READ on a suid/sgid file being executed */ + if ((openflags & __FMODE_EXEC) && + (state->inode->i_mode & (S_ISUID | S_ISGID))) { + mask = MAY_EXEC; + } + cache.cred = cred; cache.jiffies = jiffies; nfs_access_set_mask(&cache, opendata->o_res.access_result); @@ -1896,7 +1905,7 @@ static int _nfs4_do_open(struct inode *dir, if (server->caps & NFS_CAP_POSIX_LOCK) set_bit(NFS_STATE_POSIX_LOCKS, &state->flags); - status = nfs4_opendata_access(cred, opendata, state, fmode); + status = nfs4_opendata_access(cred, opendata, state, fmode, flags); if (status != 0) goto err_opendata_put;