Received: by 2002:a25:4158:0:0:0:0:0 with SMTP id o85csp453292yba; Thu, 9 May 2019 00:09:05 -0700 (PDT) X-Google-Smtp-Source: APXvYqy694gpviDGCSN18Q7QXTerB1EgZ+9LFtZtuBnmuEw0VrkezJRWFRtdE8b6Ncu0K2aQZwYE X-Received: by 2002:aa7:91c8:: with SMTP id z8mr2954584pfa.110.1557385744976; Thu, 09 May 2019 00:09:04 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1557385744; cv=none; d=google.com; s=arc-20160816; b=wjeBkzlwAr3GLcAyinHfk6tKkYsIgkcj8Px8fDdLA6sj55+o45LorAlg0jLg/llmnX a5/3HjzKWbvP8ryhgq8G2jEWcQYD9lROYtUb7RyVPhJyJ+eEtGnTk1GfQCbhvcsntxGk Peq8nr1VSggcIjv5nwnyoM6JD8FsEg5DCf6q2vnvUAlqB0oj930AtHPigqfIi65Zn8AG 6M0EXHCwO+MNhGCjiaXAvZJMwVFsgpUykyWbN4MU7tAFWh1i6gvni1ZceVxxRxL3WBdA VdXxqUoM/fBqShELKIetL9ouJOL4953o6um0d5vmzD02LqQRZjyiqsB8U62BsEajC2pd u6rQ== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding :content-language:in-reply-to:mime-version:user-agent:date :message-id:from:references:cc:to:subject; bh=h6BuLWVmoCBwmD/NSZB15BvR2AOQ099gWrE/2yxpe/U=; b=BLRTjiDNvtDmeizlRDNu0b/pQruCJv7ZMX8jPStVBbxB2hVOngpuEhWZFQt30f0RYn yHLpCKSuCoa6z5JUWlHy/4Ad2Pac/R/xWmgBpfN+mzUCWco3gbYjTODRh0EsTnhdWWRI kfdiCI3tQjj6Heo2OQbCT3uFHsfK8ypOg2U36Byj9qUcgl42LRi0fytsKvvC877mjKm6 bx28Ikeuv72eb5V0jR96EZtpd8cgemSxZQfltNWuO/YCuLQb+2vKOcQJ3GsB3cSdNOMr zOeEJEmyzvMwQgZ2xahO+4x/Ud2/IXgG1pwtz3tgRxgOEWTmElSkcE1VgjZEuGW/WzGA zofA== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id cg1si1818199plb.247.2019.05.09.00.08.48; Thu, 09 May 2019 00:09:04 -0700 (PDT) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1726571AbfEIHH1 (ORCPT + 99 others); Thu, 9 May 2019 03:07:27 -0400 Received: from szxga03-in.huawei.com ([45.249.212.189]:2581 "EHLO huawei.com" rhost-flags-OK-OK-OK-FAIL) by vger.kernel.org with ESMTP id S1726192AbfEIHH1 (ORCPT ); Thu, 9 May 2019 03:07:27 -0400 Received: from DGGEMM401-HUB.china.huawei.com (unknown [172.30.72.54]) by Forcepoint Email with ESMTP id 2F89AF8233877CFF8656; Thu, 9 May 2019 15:07:24 +0800 (CST) Received: from dggeme763-chm.china.huawei.com (10.3.19.109) by DGGEMM401-HUB.china.huawei.com (10.3.20.209) with Microsoft SMTP Server (TLS) id 14.3.408.0; Thu, 9 May 2019 15:07:23 +0800 Received: from [10.134.22.195] (10.134.22.195) by dggeme763-chm.china.huawei.com (10.3.19.109) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256_P256) id 15.1.1591.10; Thu, 9 May 2019 15:07:23 +0800 Subject: Re: [PATCH v5] f2fs: fix to avoid accessing xattr across the boundary To: Jaegeuk Kim , Randall Huang CC: , References: <20190411082646.169977-1-huangrandall@google.com> <20190509041535.GA62877@jaegeuk-macbookpro.roam.corp.google.com> From: Chao Yu Message-ID: Date: Thu, 9 May 2019 15:07:18 +0800 User-Agent: Mozilla/5.0 (Windows NT 6.1; WOW64; rv:52.0) Gecko/20100101 Thunderbird/52.9.1 MIME-Version: 1.0 In-Reply-To: <20190509041535.GA62877@jaegeuk-macbookpro.roam.corp.google.com> Content-Type: text/plain; charset="windows-1252" Content-Language: en-US Content-Transfer-Encoding: 7bit X-Originating-IP: [10.134.22.195] X-ClientProxiedBy: dggeme707-chm.china.huawei.com (10.1.199.103) To dggeme763-chm.china.huawei.com (10.3.19.109) X-CFilter-Loop: Reflected Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On 2019/5/9 12:15, Jaegeuk Kim wrote: > On 04/11, Randall Huang wrote: >> When we traverse xattr entries via __find_xattr(), >> if the raw filesystem content is faked or any hardware failure occurs, >> out-of-bound error can be detected by KASAN. >> Fix the issue by introducing boundary check. >> >> [ 38.402878] c7 1827 BUG: KASAN: slab-out-of-bounds in f2fs_getxattr+0x518/0x68c >> [ 38.402891] c7 1827 Read of size 4 at addr ffffffc0b6fb35dc by task >> [ 38.402935] c7 1827 Call trace: >> [ 38.402952] c7 1827 [] dump_backtrace+0x0/0x6bc >> [ 38.402966] c7 1827 [] show_stack+0x20/0x2c >> [ 38.402981] c7 1827 [] dump_stack+0xfc/0x140 >> [ 38.402995] c7 1827 [] print_address_description+0x80/0x2d8 >> [ 38.403009] c7 1827 [] kasan_report_error+0x198/0x1fc >> [ 38.403022] c7 1827 [] kasan_report_error+0x0/0x1fc >> [ 38.403037] c7 1827 [] __asan_load4+0x1b0/0x1b8 >> [ 38.403051] c7 1827 [] f2fs_getxattr+0x518/0x68c >> [ 38.403066] c7 1827 [] f2fs_xattr_generic_get+0xb0/0xd0 >> [ 38.403080] c7 1827 [] __vfs_getxattr+0x1f4/0x1fc >> [ 38.403096] c7 1827 [] inode_doinit_with_dentry+0x360/0x938 >> [ 38.403109] c7 1827 [] selinux_d_instantiate+0x2c/0x38 >> [ 38.403123] c7 1827 [] security_d_instantiate+0x68/0x98 >> [ 38.403136] c7 1827 [] d_splice_alias+0x58/0x348 >> [ 38.403149] c7 1827 [] f2fs_lookup+0x608/0x774 >> [ 38.403163] c7 1827 [] lookup_slow+0x1e0/0x2cc >> [ 38.403177] c7 1827 [] walk_component+0x160/0x520 >> [ 38.403190] c7 1827 [] path_lookupat+0x110/0x2b4 >> [ 38.403203] c7 1827 [] filename_lookup+0x1d8/0x3a8 >> [ 38.403216] c7 1827 [] user_path_at_empty+0x54/0x68 >> [ 38.403229] c7 1827 [] SyS_getxattr+0xb4/0x18c >> [ 38.403241] c7 1827 [] el0_svc_naked+0x34/0x38 >> >> Bug: 126558260 >> >> Signed-off-by: Randall Huang >> --- >> v2: >> * return EFAULT if OOB error is detected >> >> v3: >> * fix typo in setxattr() >> >> v4: >> * change boundry definition >> >> v5: >> * revise boundry definition >> --- >> fs/f2fs/xattr.c | 32 +++++++++++++++++++++++++++----- >> 1 file changed, 27 insertions(+), 5 deletions(-) >> >> diff --git a/fs/f2fs/xattr.c b/fs/f2fs/xattr.c >> index 848a785abe25..587429e29a69 100644 >> --- a/fs/f2fs/xattr.c >> +++ b/fs/f2fs/xattr.c >> @@ -202,12 +202,18 @@ static inline const struct xattr_handler *f2fs_xattr_handler(int index) >> return handler; >> } >> >> -static struct f2fs_xattr_entry *__find_xattr(void *base_addr, int index, >> - size_t len, const char *name) >> +static struct f2fs_xattr_entry *__find_xattr(void *base_addr, >> + void *last_base_addr, int index, >> + size_t len, const char *name) >> { >> struct f2fs_xattr_entry *entry; >> >> list_for_each_xattr(entry, base_addr) { >> + if ((void *)(entry) + sizeof(__u32) > last_base_addr || >> + (void *)XATTR_NEXT_ENTRY(entry) + >> + sizeof(__u32) > last_base_addr) >> + return NULL; >> + >> if (entry->e_name_index != index) >> continue; >> if (entry->e_name_len != len) >> @@ -298,6 +304,7 @@ static int lookup_all_xattrs(struct inode *inode, struct page *ipage, >> void **base_addr, int *base_size) >> { >> void *cur_addr, *txattr_addr, *last_addr = NULL; >> + void *last_txattr_addr = NULL; >> nid_t xnid = F2FS_I(inode)->i_xattr_nid; >> unsigned int size = xnid ? VALID_XATTR_BLOCK_SIZE : 0; >> unsigned int inline_size = inline_xattr_size(inode); >> @@ -311,6 +318,8 @@ static int lookup_all_xattrs(struct inode *inode, struct page *ipage, >> if (!txattr_addr) >> return -ENOMEM; >> >> + last_txattr_addr = (void *)txattr_addr + inline_size + size; > > I just found this should be + XATTR_PADDING_SIZE. Otherwise, generic/026 fails. > Let me know, if there is any other concern below. We're trying to use [txattr_addr, last_txattr_addr] to indicate valid range of xattr datas, any valid entries across the boundary is not allowed, so I think it's correct to exclude padding space for last_txattr_addr. I added log and tested generic/026, found that: [365804.868431] base:ffff89e391f8e000, last:ffff89e391f8f0b0 [365804.868433] entry:ffff89e391f8e018, next:ffff89e391f8f0b0 The root cause of this issue is caused by below condition: (void *)XATTR_NEXT_ENTRY(entry) + sizeof(__u32) > last_base_addr) It occurs when the end address of last valid entry is located in range (last_base_addr - XATTR_PADDING_SIZE, last_base_addr], then next invalid entry's head will across last_base_addr, result in returning -EFAULT. So anyway, please try below diff instead. ;) diff --git a/fs/f2fs/xattr.c b/fs/f2fs/xattr.c index 019778fb9a0d..217bf7c2bb7a 100644 --- a/fs/f2fs/xattr.c +++ b/fs/f2fs/xattr.c @@ -210,10 +210,8 @@ static struct f2fs_xattr_entry *__find_xattr(void *base_addr, list_for_each_xattr(entry, base_addr) { if ((void *)(entry) + sizeof(__u32) > last_base_addr || - (void *)XATTR_NEXT_ENTRY(entry) + - sizeof(__u32) > last_base_addr) + (void *)XATTR_NEXT_ENTRY(entry) > last_base_addr) return NULL; if (entry->e_name_index != index) Thanks, > > --- > fs/f2fs/xattr.c | 12 ++++-------- > fs/f2fs/xattr.h | 2 ++ > 2 files changed, 6 insertions(+), 8 deletions(-) > > diff --git a/fs/f2fs/xattr.c b/fs/f2fs/xattr.c > index 019778fb9a0d..cd199a09d436 100644 > --- a/fs/f2fs/xattr.c > +++ b/fs/f2fs/xattr.c > @@ -306,19 +306,18 @@ static int lookup_all_xattrs(struct inode *inode, struct page *ipage, > void *cur_addr, *txattr_addr, *last_addr = NULL; > void *last_txattr_addr = NULL; > nid_t xnid = F2FS_I(inode)->i_xattr_nid; > - unsigned int size = xnid ? VALID_XATTR_BLOCK_SIZE : 0; > unsigned int inline_size = inline_xattr_size(inode); > int err = 0; > > - if (!size && !inline_size) > + if (!xnid && !inline_size) > return -ENODATA; > > - *base_size = inline_size + size + XATTR_PADDING_SIZE; > + *base_size = XATTR_SIZE(xnid, inode); > txattr_addr = f2fs_kzalloc(F2FS_I_SB(inode), *base_size, GFP_NOFS); > if (!txattr_addr) > return -ENOMEM; > > - last_txattr_addr = (void *)txattr_addr + inline_size + size; > + last_txattr_addr = (void *)txattr_addr + *base_size; > > /* read from inline xattr */ > if (inline_size) { > @@ -599,9 +598,6 @@ static int __f2fs_setxattr(struct inode *inode, int index, > size_t len; > __u32 new_hsize; > nid_t xnid = F2FS_I(inode)->i_xattr_nid; > - unsigned int xnode_size = xnid ? VALID_XATTR_BLOCK_SIZE : 0; > - unsigned int inline_size = inline_xattr_size(inode); > - > int error = 0; > > if (name == NULL) > @@ -621,7 +617,7 @@ static int __f2fs_setxattr(struct inode *inode, int index, > error = read_all_xattrs(inode, ipage, &base_addr); > if (error) > return error; > - last_base_addr = (void *)base_addr + xnode_size + inline_size; > + last_base_addr = (void *)base_addr + XATTR_SIZE(xnid, inode); > > /* find entry with wanted name. */ > here = __find_xattr(base_addr, last_base_addr, index, len, name); > diff --git a/fs/f2fs/xattr.h b/fs/f2fs/xattr.h > index 9172ee082ca8..1eca1a2d996a 100644 > --- a/fs/f2fs/xattr.h > +++ b/fs/f2fs/xattr.h > @@ -71,6 +71,8 @@ struct f2fs_xattr_entry { > entry = XATTR_NEXT_ENTRY(entry)) > #define VALID_XATTR_BLOCK_SIZE (PAGE_SIZE - sizeof(struct node_footer)) > #define XATTR_PADDING_SIZE (sizeof(__u32)) > +#define XATTR_SIZE(x,i) (((x) ? VALID_XATTR_BLOCK_SIZE : 0) + \ > + (inline_xattr_size(i)) + XATTR_PADDING_SIZE) > #define MIN_OFFSET(i) XATTR_ALIGN(inline_xattr_size(i) + \ > VALID_XATTR_BLOCK_SIZE) > >