Received: by 2002:a05:6358:11c7:b0:104:8066:f915 with SMTP id i7csp5249190rwl; Tue, 11 Apr 2023 02:40:08 -0700 (PDT) X-Google-Smtp-Source: AKy350ZGl8XYiV0P2+16XFcCjJ2JoVkyAwc25GI/4D5HAGqLtM9tNGToKRjW9EOui8i1EF2IWKbX X-Received: by 2002:a17:906:1107:b0:94a:7450:9e2d with SMTP id h7-20020a170906110700b0094a74509e2dmr6627508eja.19.1681206008457; Tue, 11 Apr 2023 02:40:08 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1681206008; cv=none; d=google.com; s=arc-20160816; b=EB8m8b0EAdLHd6X9WSJRZbbSqGKitXWJggtOsrAuuaq8ltsSh5tsw5g7sZ/tmSn3LG sDonRrEu1R1weJyr1XNlIlgVnfVEcm+Pt88swOQJbqHyEgXRPzB+D9e42gOj1A0McY4c R2Nmh42Net+Pg7Nah9ZK090ty9HA8g54b6sQK2xtOU2BZ0y62S0QqeNmNeYwzbix22a9 Ieil6RKJzq3zOWKFxMcfi4JdRPeNHkNkEaYwDyx7RgVa+A8K2ySqmTFJNUz+vlk3exKd 1pSwlSrujc01Ko7dHR5RXZEZ+fLmya3nIX8Zw9FkRt+zsmGWQ1BMMk+XaCRw8c+lO/TC S0eg== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:content-transfer-encoding:mime-version :references:in-reply-to:message-id:date:subject:cc:to:from; bh=N2JJtWPro5h1NBh2e/1YYXYkmCdCaIu5pFcI93t3sE4=; b=dh5TjLGpWBfhU/XbCUa9P7MaJk4Fv4Zyn7Vlw9zbc/EUrAZ5MIdJA0qld6hC8wuOrE zLXudqSapJNFSvL6b53nhsZtDOW6CrosGSABGMsvqmfbLbj3tJQH4ASJGnPj7toa3UCy KrBoooDHYI2vLAocIiYoTkZygi96KuvgRJyG/weMvte8OwxNrmbUCGJm9UgI3G7E+MVq T//HM1Mu+DADtM3VXSG9iglgw2mDlceYlbTt20wY8cnx9Hner48iK+MffJ7+AGSjHeTs mYaaacza/bBEm4OOYp4ya/1Bzp3ZgjBF+0i635RdJdLNj+6W6PNBtujNyo+/avJUnvUA uS3g== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=alibaba.com Return-Path: Received: from out1.vger.email (out1.vger.email. [2620:137:e000::1:20]) by mx.google.com with ESMTP id cs21-20020a170906dc9500b0094a659fbf83si6695776ejc.791.2023.04.11.02.39.43; Tue, 11 Apr 2023 02:40:08 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) client-ip=2620:137:e000::1:20; Authentication-Results: mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=alibaba.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S230064AbjDKJjM (ORCPT + 99 others); Tue, 11 Apr 2023 05:39:12 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:35986 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S229953AbjDKJiw (ORCPT ); Tue, 11 Apr 2023 05:38:52 -0400 Received: from out199-14.us.a.mail.aliyun.com (out199-14.us.a.mail.aliyun.com [47.90.199.14]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id AC23026AB for ; Tue, 11 Apr 2023 02:38:04 -0700 (PDT) X-Alimail-AntiSpam: AC=PASS;BC=-1|-1;BR=01201311R181e4;CH=green;DM=||false|;DS=||;FP=0|-1|-1|-1|0|-1|-1|-1;HT=ay29a033018046051;MF=jefflexu@linux.alibaba.com;NM=1;PH=DS;RN=5;SR=0;TI=SMTPD_---0Vfrhsq-_1681205737; Received: from localhost(mailfrom:jefflexu@linux.alibaba.com fp:SMTPD_---0Vfrhsq-_1681205737) by smtp.aliyun-inc.com; Tue, 11 Apr 2023 17:35:38 +0800 From: Jingbo Xu To: xiang@kernel.org, chao@kernel.org, huyue2@coolpad.com, linux-erofs@lists.ozlabs.org Cc: linux-kernel@vger.kernel.org Subject: [PATCH v3 6/7] erofs: handle long xattr name prefixes properly Date: Tue, 11 Apr 2023 17:35:37 +0800 Message-Id: <20230411093537.127286-1-jefflexu@linux.alibaba.com> X-Mailer: git-send-email 2.19.1.6.gb485710b In-Reply-To: <20230410063918.47215-1-jefflexu@linux.alibaba.com> References: <20230410063918.47215-1-jefflexu@linux.alibaba.com> MIME-Version: 1.0 Content-Transfer-Encoding: 8bit X-Spam-Status: No, score=-8.0 required=5.0 tests=ENV_AND_HDR_SPF_MATCH, SPF_HELO_NONE,SPF_PASS,UNPARSEABLE_RELAY,USER_IN_DEF_SPF_WL autolearn=unavailable autolearn_force=no version=3.4.6 X-Spam-Checker-Version: SpamAssassin 3.4.6 (2021-04-09) on lindbergh.monkeyblade.net Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Make .{list,get}xattr routines adapted to long xattr name prefixes. When the bit 7 of erofs_xattr_entry.e_name_index is set, it indicates that it refers to a long xattr name prefix. Signed-off-by: Jingbo Xu --- v3: introduce infix_len to struct getxattr_iter, and refactor the implementation of xattr_entrymatch(), erofs_xattr_long_entrymatch(), and xattr_namematch() accordingly. The erofs_xattr_long_entrymatch() of v2 version will advance it->name.name pointer by pf->infix_len prematurely, as the following xattr_namematch() may fail (-ENOATTR) since mismatching. And then it->name.name will be compared with the next xattr entry, while it->name.name has been mistakenly modified in the previous round. This will cause -ENOATTR error on the existing xattr. --- fs/erofs/xattr.c | 68 +++++++++++++++++++++++++++++++++++++++--------- 1 file changed, 56 insertions(+), 12 deletions(-) diff --git a/fs/erofs/xattr.c b/fs/erofs/xattr.c index 684571e83a2c..a04724c816e5 100644 --- a/fs/erofs/xattr.c +++ b/fs/erofs/xattr.c @@ -297,17 +297,45 @@ struct getxattr_iter { struct xattr_iter it; char *buffer; - int buffer_size, index; + int buffer_size, index, infix_len; struct qstr name; }; +static int erofs_xattr_long_entrymatch(struct getxattr_iter *it, + struct erofs_xattr_entry *entry) +{ + struct erofs_sb_info *sbi = EROFS_SB(it->it.sb); + struct erofs_xattr_prefix_item *pf = sbi->xattr_prefixes + + (entry->e_name_index & EROFS_XATTR_LONG_PREFIX_MASK); + + if (pf >= sbi->xattr_prefixes + sbi->xattr_prefix_count) + return -ENOATTR; + + if (it->index != pf->prefix->base_index || + it->name.len != entry->e_name_len + pf->infix_len) + return -ENOATTR; + + if (memcmp(it->name.name, pf->prefix->infix, pf->infix_len)) + return -ENOATTR; + + it->infix_len = pf->infix_len; + return 0; +} + static int xattr_entrymatch(struct xattr_iter *_it, struct erofs_xattr_entry *entry) { struct getxattr_iter *it = container_of(_it, struct getxattr_iter, it); - return (it->index != entry->e_name_index || - it->name.len != entry->e_name_len) ? -ENOATTR : 0; + /* should also match the infix for long name prefixes */ + if (entry->e_name_index & EROFS_XATTR_LONG_PREFIX) + return erofs_xattr_long_entrymatch(it, entry); + + if (it->index != entry->e_name_index || + it->name.len != entry->e_name_len) + return -ENOATTR; + it->infix_len = 0; + return 0; } static int xattr_namematch(struct xattr_iter *_it, @@ -315,7 +343,9 @@ static int xattr_namematch(struct xattr_iter *_it, { struct getxattr_iter *it = container_of(_it, struct getxattr_iter, it); - return memcmp(buf, it->name.name + processed, len) ? -ENOATTR : 0; + if (memcmp(buf, it->name.name + it->infix_len + processed, len)) + return -ENOATTR; + return 0; } static int xattr_checkbuffer(struct xattr_iter *_it, @@ -487,12 +517,24 @@ static int xattr_entrylist(struct xattr_iter *_it, { struct listxattr_iter *it = container_of(_it, struct listxattr_iter, it); - unsigned int prefix_len; - const char *prefix; - - const struct xattr_handler *h = - erofs_xattr_handler(entry->e_name_index); + unsigned int base_index = entry->e_name_index; + unsigned int prefix_len, infix_len = 0; + const char *prefix, *infix = NULL; + const struct xattr_handler *h; + + if (entry->e_name_index & EROFS_XATTR_LONG_PREFIX) { + struct erofs_sb_info *sbi = EROFS_SB(_it->sb); + struct erofs_xattr_prefix_item *pf = sbi->xattr_prefixes + + (entry->e_name_index & EROFS_XATTR_LONG_PREFIX_MASK); + + if (pf >= sbi->xattr_prefixes + sbi->xattr_prefix_count) + return 1; + infix = pf->prefix->infix; + infix_len = pf->infix_len; + base_index = pf->prefix->base_index; + } + h = erofs_xattr_handler(base_index); if (!h || (h->list && !h->list(it->dentry))) return 1; @@ -500,16 +542,18 @@ static int xattr_entrylist(struct xattr_iter *_it, prefix_len = strlen(prefix); if (!it->buffer) { - it->buffer_ofs += prefix_len + entry->e_name_len + 1; + it->buffer_ofs += prefix_len + infix_len + + entry->e_name_len + 1; return 1; } - if (it->buffer_ofs + prefix_len + if (it->buffer_ofs + prefix_len + infix_len + + entry->e_name_len + 1 > it->buffer_size) return -ERANGE; memcpy(it->buffer + it->buffer_ofs, prefix, prefix_len); - it->buffer_ofs += prefix_len; + memcpy(it->buffer + it->buffer_ofs + prefix_len, infix, infix_len); + it->buffer_ofs += prefix_len + infix_len; return 0; } -- 2.19.1.6.gb485710b