From mboxrd@z Thu Jan 1 00:00:00 1970 From: "Darrick J. Wong" Subject: [PATCH v2] fuse: Fix IOC_[GS]ET{FLAGS, VERSION} argument size brokenness. Date: Fri, 20 Dec 2013 15:35:34 -0800 Message-ID: <20131220233534.GC10192@birch.djwong.org> References: <20131219232739.GA10192@birch.djwong.org> Mime-Version: 1.0 Content-Type: text/plain; charset="us-ascii" Content-Transfer-Encoding: 7bit Cc: linux-fsdevel , fuse-devel-5NWGOfrQmneRv+LV9MX5uipxlwaOVQ5f@public.gmane.org, linux-kernel , Richard Hansen To: Miklos Szeredi , Alexander Viro Return-path: Content-Disposition: inline In-Reply-To: <20131219232739.GA10192-PTl6brltDGh4DFYR7WNSRA@public.gmane.org> List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , Errors-To: fuse-devel-bounces-5NWGOfrQmneRv+LV9MX5uipxlwaOVQ5f@public.gmane.org List-Id: linux-fsdevel.vger.kernel.org The IOC_[GS]ETFLAGS and IOC_[GS]ETVERSION ioctls, despite being defined to take a "long" parameter, actually take "int" parameters. FUSE unfortunately assumed that the ioctl definitions never lie, and transfers a long's worth of data in and out of userspace, which causes stack smashing in chattr, and other bugs elsewhere. So, special-case this in FUSE so that we don't crash userland. v2: Do the same for the IOC_[GS]ETVERSION ioctls, as Richard Hansen points out. Signed-off-by: Darrick J. Wong --- fs/fuse/file.c | 16 ++++++++++++++++ 1 file changed, 16 insertions(+) diff --git a/fs/fuse/file.c b/fs/fuse/file.c index 7e70506..f8766ab 100644 --- a/fs/fuse/file.c +++ b/fs/fuse/file.c @@ -2385,6 +2385,22 @@ long fuse_do_ioctl(struct file *file, unsigned int cmd, unsigned long arg, iov->iov_base = (void __user *)arg; iov->iov_len = _IOC_SIZE(cmd); + /* + * The IOC_[GS]ETFLAGS and IOC_[GS]ETVERSION ioctls take int + * parameters even though the ioctl definition specifies long. + * Userland has been expecting int for ages (and chattr + * segfaults on FUSE filesystems), so special case that here. + * The IOC32 variants were declared with int, so they don't + * need this correction. + */ + switch (cmd) { + case FS_IOC_GETFLAGS: + case FS_IOC_SETFLAGS: + case FS_IOC_GETVERSION: + case FS_IOC_SETVERSION: + iov->iov_len = sizeof(int); + } + if (_IOC_DIR(cmd) & _IOC_WRITE) { in_iov = iov; in_iovs = 1; ------------------------------------------------------------------------------ Rapidly troubleshoot problems before they affect your business. Most IT organizations don't have a clear picture of how application performance affects their revenue. With AppDynamics, you get 100% visibility into your Java,.NET, & PHP application. Start your 15-day FREE TRIAL of AppDynamics Pro! http://pubads.g.doubleclick.net/gampad/clk?id=84349831&iu=/4140/ostg.clktrk