From nobody Fri Dec 19 16:23:01 2025 Received: from szxga04-in.huawei.com (szxga04-in.huawei.com [45.249.212.190]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by smtp.subspace.kernel.org (Postfix) with ESMTPS id 69419149DE3; Sat, 24 Aug 2024 09:26:48 +0000 (UTC) Authentication-Results: smtp.subspace.kernel.org; arc=none smtp.client-ip=45.249.212.190 ARC-Seal: i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1724491611; cv=none; b=Lfyys3h0jxZDyrE3ftxsHCJQ5Mj023saGn13AA5S75LIqm80L67gjVSn7Cuj8w52gyLGkkLwvbv2sqUUa7JMn8nv8/M5CrzhzWaKB2AUCj+iZv5XY93sNQvqTgeNcFEf12Ci6bmBFqMp91HSm9Eu8lABDAs3v+U29cVCvmHAqiU= ARC-Message-Signature: i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1724491611; c=relaxed/simple; bh=srNPvtAZ9UW/EwtnZw8yfgSO/mKPmLLLOJzUsP4g2dM=; h=From:To:CC:Subject:Date:Message-ID:In-Reply-To:References: MIME-Version:Content-Type; b=TiTNy2y+c3a3M7uTWPc2/uBDgqnG5iqpFox1QE1HM0KNqIP2UX+I8goh1FVHRdisYwN78ruSSqGy2/W0R8pgewW+/Y8YcEK8XEUy1nLnsh8Es1bBjjpwKOSyXr69RNAlI8ZvH2yXvsj7bueQWMx4Bb5z68Yd5do2kFNSW/+d3BM= ARC-Authentication-Results: i=1; smtp.subspace.kernel.org; dmarc=pass (p=quarantine dis=none) header.from=huawei.com; spf=pass smtp.mailfrom=huawei.com; arc=none smtp.client-ip=45.249.212.190 Authentication-Results: smtp.subspace.kernel.org; dmarc=pass (p=quarantine dis=none) header.from=huawei.com Authentication-Results: smtp.subspace.kernel.org; spf=pass smtp.mailfrom=huawei.com Received: from mail.maildlp.com (unknown [172.19.88.234]) by szxga04-in.huawei.com (SkyGuard) with ESMTP id 4WrWm01jkNz2Cn8n; Sat, 24 Aug 2024 17:26:40 +0800 (CST) Received: from kwepemd100024.china.huawei.com (unknown [7.221.188.41]) by mail.maildlp.com (Postfix) with ESMTPS id 532DD1400FD; Sat, 24 Aug 2024 17:26:46 +0800 (CST) Received: from huawei.com (10.175.124.27) by kwepemd100024.china.huawei.com (7.221.188.41) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384) id 15.2.1544.11; Sat, 24 Aug 2024 17:26:45 +0800 From: yangyun To: Miklos Szeredi CC: , , , Subject: [PATCH v2 1/2] fuse: move fuse_forget_link allocation inside fuse_queue_forget() Date: Sat, 24 Aug 2024 17:25:52 +0800 Message-ID: <20240824092553.730338-2-yangyun50@huawei.com> X-Mailer: git-send-email 2.33.0 In-Reply-To: <20240824092553.730338-1-yangyun50@huawei.com> References: <20240824092553.730338-1-yangyun50@huawei.com> Precedence: bulk X-Mailing-List: linux-kernel@vger.kernel.org List-Id: List-Subscribe: List-Unsubscribe: MIME-Version: 1.0 Content-Transfer-Encoding: quoted-printable X-ClientProxiedBy: dggems706-chm.china.huawei.com (10.3.19.183) To kwepemd100024.china.huawei.com (7.221.188.41) Content-Type: text/plain; charset="utf-8" The `struct fuse_forget_link` is allocated outside `fuse_queue_forget()` before this patch. This requires the allocation in advance. In some cases, this struct is not needed but allocated, which contributes to memory usage and performance degradation. Besides, this messes up the code to some extent. So move the `fuse_forget_link` allocation inside fuse_queue_forget with __GFP_NOFAIL. `fuse_force_forget()` is used by `readdirplus` before this patch for the reason that we do not know how many 'fuse_forget_link' structures will be allocated in advance when error happens. After this patch, this function is not needed any more and can be removed. By this way, all FUSE_FORGET requests are sent by using `fuse_queue_forget()` function as e.g. virtiofs handles them differently from regular requests. Signed-off-by: yangyun --- fs/fuse/dev.c | 10 ++++++-- fs/fuse/dir.c | 59 +++++++++++------------------------------------ fs/fuse/fuse_i.h | 11 +-------- fs/fuse/inode.c | 33 ++++---------------------- fs/fuse/readdir.c | 29 +++++------------------ 5 files changed, 32 insertions(+), 110 deletions(-) diff --git a/fs/fuse/dev.c b/fs/fuse/dev.c index 9eb191b5c4de..4b106db2f97f 100644 --- a/fs/fuse/dev.c +++ b/fs/fuse/dev.c @@ -75,6 +75,12 @@ static void __fuse_put_request(struct fuse_req *req) refcount_dec(&req->count); } =20 +/* Use __GFP_NOFAIL to force the allocation success */ +static struct fuse_forget_link *fuse_alloc_forget(void) +{ + return kzalloc(sizeof(struct fuse_forget_link), GFP_KERNEL_ACCOUNT | __GF= P_NOFAIL); +} + void fuse_set_initialized(struct fuse_conn *fc) { /* Make sure stores before this are seen on another CPU */ @@ -233,10 +239,10 @@ __releases(fiq->lock) fiq->ops->wake_pending_and_unlock(fiq); } =20 -void fuse_queue_forget(struct fuse_conn *fc, struct fuse_forget_link *forg= et, - u64 nodeid, u64 nlookup) +void fuse_queue_forget(struct fuse_conn *fc, u64 nodeid, u64 nlookup) { struct fuse_iqueue *fiq =3D &fc->iq; + struct fuse_forget_link *forget =3D fuse_alloc_forget(); =20 forget->forget_one.nodeid =3D nodeid; forget->forget_one.nlookup =3D nlookup; diff --git a/fs/fuse/dir.c b/fs/fuse/dir.c index 2b0d4781f394..f070e380fd7b 100644 --- a/fs/fuse/dir.c +++ b/fs/fuse/dir.c @@ -207,7 +207,6 @@ static int fuse_dentry_revalidate(struct dentry *entry,= unsigned int flags) (flags & (LOOKUP_EXCL | LOOKUP_REVAL | LOOKUP_RENAME_TARGET))) { struct fuse_entry_out outarg; FUSE_ARGS(args); - struct fuse_forget_link *forget; u64 attr_version; =20 /* For negative dentries, always do a fresh lookup */ @@ -220,11 +219,6 @@ static int fuse_dentry_revalidate(struct dentry *entry= , unsigned int flags) =20 fm =3D get_fuse_mount(inode); =20 - forget =3D fuse_alloc_forget(); - ret =3D -ENOMEM; - if (!forget) - goto out; - attr_version =3D fuse_get_attr_version(fm->fc); =20 parent =3D dget_parent(entry); @@ -239,15 +233,13 @@ static int fuse_dentry_revalidate(struct dentry *entr= y, unsigned int flags) fi =3D get_fuse_inode(inode); if (outarg.nodeid !=3D get_node_id(inode) || (bool) IS_AUTOMOUNT(inode) !=3D (bool) (outarg.attr.flags & FUSE_AT= TR_SUBMOUNT)) { - fuse_queue_forget(fm->fc, forget, - outarg.nodeid, 1); + fuse_queue_forget(fm->fc, outarg.nodeid, 1); goto invalid; } spin_lock(&fi->lock); fi->nlookup++; spin_unlock(&fi->lock); } - kfree(forget); if (ret =3D=3D -ENOMEM || ret =3D=3D -EINTR) goto out; if (ret || fuse_invalid_attr(&outarg.attr) || @@ -365,7 +357,6 @@ int fuse_lookup_name(struct super_block *sb, u64 nodeid= , const struct qstr *name { struct fuse_mount *fm =3D get_fuse_mount_super(sb); FUSE_ARGS(args); - struct fuse_forget_link *forget; u64 attr_version; int err; =20 @@ -374,23 +365,17 @@ int fuse_lookup_name(struct super_block *sb, u64 node= id, const struct qstr *name if (name->len > FUSE_NAME_MAX) goto out; =20 - - forget =3D fuse_alloc_forget(); - err =3D -ENOMEM; - if (!forget) - goto out; - attr_version =3D fuse_get_attr_version(fm->fc); =20 fuse_lookup_init(fm->fc, &args, nodeid, name, outarg); err =3D fuse_simple_request(fm, &args); /* Zero nodeid is same as -ENOENT, but with valid timeout */ if (err || !outarg->nodeid) - goto out_put_forget; + goto out; =20 err =3D -EIO; if (fuse_invalid_attr(&outarg->attr)) - goto out_put_forget; + goto out; if (outarg->nodeid =3D=3D FUSE_ROOT_ID && outarg->generation !=3D 0) { pr_warn_once("root generation should be zero\n"); outarg->generation =3D 0; @@ -401,13 +386,11 @@ int fuse_lookup_name(struct super_block *sb, u64 node= id, const struct qstr *name attr_version); err =3D -ENOMEM; if (!*inode) { - fuse_queue_forget(fm->fc, forget, outarg->nodeid, 1); + fuse_queue_forget(fm->fc, outarg->nodeid, 1); goto out; } err =3D 0; =20 - out_put_forget: - kfree(forget); out: return err; } @@ -617,7 +600,6 @@ static int fuse_create_open(struct inode *dir, struct d= entry *entry, struct inode *inode; struct fuse_mount *fm =3D get_fuse_mount(dir); FUSE_ARGS(args); - struct fuse_forget_link *forget; struct fuse_create_in inarg; struct fuse_open_out *outopenp; struct fuse_entry_out outentry; @@ -628,15 +610,10 @@ static int fuse_create_open(struct inode *dir, struct= dentry *entry, /* Userspace expects S_IFREG in create mode */ BUG_ON((mode & S_IFMT) !=3D S_IFREG); =20 - forget =3D fuse_alloc_forget(); - err =3D -ENOMEM; - if (!forget) - goto out_err; - err =3D -ENOMEM; ff =3D fuse_file_alloc(fm, true); if (!ff) - goto out_put_forget_req; + goto out_err; =20 if (!fm->fc->dont_mask) mode &=3D ~current_umask(); @@ -670,7 +647,7 @@ static int fuse_create_open(struct inode *dir, struct d= entry *entry, =20 err =3D get_create_ext(&args, dir, entry, mode); if (err) - goto out_put_forget_req; + goto out_err; =20 err =3D fuse_simple_request(fm, &args); free_ext_value(&args); @@ -690,11 +667,10 @@ static int fuse_create_open(struct inode *dir, struct= dentry *entry, if (!inode) { flags &=3D ~(O_CREAT | O_EXCL | O_TRUNC); fuse_sync_release(NULL, ff, flags); - fuse_queue_forget(fm->fc, forget, outentry.nodeid, 1); + fuse_queue_forget(fm->fc, outentry.nodeid, 1); err =3D -ENOMEM; goto out_err; } - kfree(forget); d_instantiate(entry, inode); fuse_change_entry_timeout(entry, &outentry); fuse_dir_changed(dir); @@ -716,8 +692,6 @@ static int fuse_create_open(struct inode *dir, struct d= entry *entry, =20 out_free_ff: fuse_file_free(ff); -out_put_forget_req: - kfree(forget); out_err: return err; } @@ -782,15 +756,10 @@ static int create_new_entry(struct fuse_mount *fm, st= ruct fuse_args *args, struct inode *inode; struct dentry *d; int err; - struct fuse_forget_link *forget; =20 if (fuse_is_bad(dir)) return -EIO; =20 - forget =3D fuse_alloc_forget(); - if (!forget) - return -ENOMEM; - memset(&outarg, 0, sizeof(outarg)); args->nodeid =3D get_node_id(dir); args->out_numargs =3D 1; @@ -800,28 +769,27 @@ static int create_new_entry(struct fuse_mount *fm, st= ruct fuse_args *args, if (args->opcode !=3D FUSE_LINK) { err =3D get_create_ext(args, dir, entry, mode); if (err) - goto out_put_forget_req; + goto out_err; } =20 err =3D fuse_simple_request(fm, args); free_ext_value(args); if (err) - goto out_put_forget_req; + goto out_err; =20 err =3D -EIO; if (invalid_nodeid(outarg.nodeid) || fuse_invalid_attr(&outarg.attr)) - goto out_put_forget_req; + goto out_err; =20 if ((outarg.attr.mode ^ mode) & S_IFMT) - goto out_put_forget_req; + goto out_err; =20 inode =3D fuse_iget(dir->i_sb, outarg.nodeid, outarg.generation, &outarg.attr, ATTR_TIMEOUT(&outarg), 0); if (!inode) { - fuse_queue_forget(fm->fc, forget, outarg.nodeid, 1); + fuse_queue_forget(fm->fc, outarg.nodeid, 1); return -ENOMEM; } - kfree(forget); =20 d_drop(entry); d =3D d_splice_alias(inode, entry); @@ -837,10 +805,9 @@ static int create_new_entry(struct fuse_mount *fm, str= uct fuse_args *args, fuse_dir_changed(dir); return 0; =20 - out_put_forget_req: + out_err: if (err =3D=3D -EEXIST) fuse_invalidate_entry(entry); - kfree(forget); return err; } =20 diff --git a/fs/fuse/fuse_i.h b/fs/fuse/fuse_i.h index f23919610313..90176133209d 100644 --- a/fs/fuse/fuse_i.h +++ b/fs/fuse/fuse_i.h @@ -71,9 +71,6 @@ struct fuse_submount_lookup { /** Unique ID, which identifies the inode between userspace * and kernel */ u64 nodeid; - - /** The request used for sending the FORGET message */ - struct fuse_forget_link *forget; }; =20 /** Container for data related to mapping to backing file */ @@ -98,9 +95,6 @@ struct fuse_inode { /** Number of lookups on this inode */ u64 nlookup; =20 - /** The request used for sending the FORGET message */ - struct fuse_forget_link *forget; - /** Time in jiffies until the file attributes are valid */ u64 i_time; =20 @@ -1048,10 +1042,7 @@ int fuse_lookup_name(struct super_block *sb, u64 nod= eid, const struct qstr *name /** * Send FORGET command */ -void fuse_queue_forget(struct fuse_conn *fc, struct fuse_forget_link *forg= et, - u64 nodeid, u64 nlookup); - -struct fuse_forget_link *fuse_alloc_forget(void); +void fuse_queue_forget(struct fuse_conn *fc, u64 nodeid, u64 nlookup); =20 struct fuse_forget_link *fuse_dequeue_forget(struct fuse_iqueue *fiq, unsigned int max, diff --git a/fs/fuse/inode.c b/fs/fuse/inode.c index 99e44ea7d875..da3e5d4c032c 100644 --- a/fs/fuse/inode.c +++ b/fs/fuse/inode.c @@ -63,27 +63,13 @@ MODULE_PARM_DESC(max_user_congthresh, static struct file_system_type fuseblk_fs_type; #endif =20 -struct fuse_forget_link *fuse_alloc_forget(void) -{ - return kzalloc(sizeof(struct fuse_forget_link), GFP_KERNEL_ACCOUNT); -} - static struct fuse_submount_lookup *fuse_alloc_submount_lookup(void) { struct fuse_submount_lookup *sl; =20 sl =3D kzalloc(sizeof(struct fuse_submount_lookup), GFP_KERNEL_ACCOUNT); - if (!sl) - return NULL; - sl->forget =3D fuse_alloc_forget(); - if (!sl->forget) - goto out_free; =20 return sl; - -out_free: - kfree(sl); - return NULL; } =20 static struct inode *fuse_alloc_inode(struct super_block *sb) @@ -104,20 +90,15 @@ static struct inode *fuse_alloc_inode(struct super_blo= ck *sb) fi->submount_lookup =3D NULL; mutex_init(&fi->mutex); spin_lock_init(&fi->lock); - fi->forget =3D fuse_alloc_forget(); - if (!fi->forget) - goto out_free; =20 if (IS_ENABLED(CONFIG_FUSE_DAX) && !fuse_dax_inode_alloc(sb, fi)) - goto out_free_forget; + goto out_free; =20 if (IS_ENABLED(CONFIG_FUSE_PASSTHROUGH)) fuse_inode_backing_set(fi, NULL); =20 return &fi->inode; =20 -out_free_forget: - kfree(fi->forget); out_free: kmem_cache_free(fuse_inode_cachep, fi); return NULL; @@ -128,7 +109,6 @@ static void fuse_free_inode(struct inode *inode) struct fuse_inode *fi =3D get_fuse_inode(inode); =20 mutex_destroy(&fi->mutex); - kfree(fi->forget); #ifdef CONFIG_FUSE_DAX kfree(fi->dax); #endif @@ -144,8 +124,7 @@ static void fuse_cleanup_submount_lookup(struct fuse_co= nn *fc, if (!refcount_dec_and_test(&sl->count)) return; =20 - fuse_queue_forget(fc, sl->forget, sl->nodeid, 1); - sl->forget =3D NULL; + fuse_queue_forget(fc, sl->nodeid, 1); kfree(sl); } =20 @@ -163,12 +142,8 @@ static void fuse_evict_inode(struct inode *inode) =20 if (FUSE_IS_DAX(inode)) fuse_dax_inode_cleanup(inode); - if (fi->nlookup) { - fuse_queue_forget(fc, fi->forget, fi->nodeid, - fi->nlookup); - fi->forget =3D NULL; - } - + if (fi->nlookup) + fuse_queue_forget(fc, fi->nodeid, fi->nlookup); if (fi->submount_lookup) { fuse_cleanup_submount_lookup(fc, fi->submount_lookup); fi->submount_lookup =3D NULL; diff --git a/fs/fuse/readdir.c b/fs/fuse/readdir.c index 0377b6dc24c8..721fae563c84 100644 --- a/fs/fuse/readdir.c +++ b/fs/fuse/readdir.c @@ -262,27 +262,6 @@ static int fuse_direntplus_link(struct file *file, return 0; } =20 -static void fuse_force_forget(struct file *file, u64 nodeid) -{ - struct inode *inode =3D file_inode(file); - struct fuse_mount *fm =3D get_fuse_mount(inode); - struct fuse_forget_in inarg; - FUSE_ARGS(args); - - memset(&inarg, 0, sizeof(inarg)); - inarg.nlookup =3D 1; - args.opcode =3D FUSE_FORGET; - args.nodeid =3D nodeid; - args.in_numargs =3D 1; - args.in_args[0].size =3D sizeof(inarg); - args.in_args[0].value =3D &inarg; - args.force =3D true; - args.noreply =3D true; - - fuse_simple_request(fm, &args); - /* ignore errors */ -} - static int parse_dirplusfile(char *buf, size_t nbytes, struct file *file, struct dir_context *ctx, u64 attr_version) { @@ -320,8 +299,12 @@ static int parse_dirplusfile(char *buf, size_t nbytes,= struct file *file, nbytes -=3D reclen; =20 ret =3D fuse_direntplus_link(file, direntplus, attr_version); - if (ret) - fuse_force_forget(file, direntplus->entry_out.nodeid); + if (ret) { + struct inode *inode =3D file_inode(file); + struct fuse_mount *fm =3D get_fuse_mount(inode); + + fuse_queue_forget(fm->fc, direntplus->entry_out.nodeid, 1); + } } =20 return 0; --=20 2.33.0 From nobody Fri Dec 19 16:23:01 2025 Received: from szxga02-in.huawei.com (szxga02-in.huawei.com [45.249.212.188]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by smtp.subspace.kernel.org (Postfix) with ESMTPS id E3951153BD7; Sat, 24 Aug 2024 09:26:49 +0000 (UTC) Authentication-Results: smtp.subspace.kernel.org; arc=none smtp.client-ip=45.249.212.188 ARC-Seal: i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1724491612; cv=none; b=QVFZ1NQ1/4J92Xf4RkHl5Nm7+gky1Onk2tIVRKR9p99xT7mOI60xr95Ij8eEnPu77oS30OvneSirhiV8Ff8HzSC9OmQFuqM9KFA7Y0znlC/2gfb8Ys2uUc+qozGQ84r4I0DPAVn26pxsjmIEHkhFHI+Mnu5nRJy4TR9h7VLSPcA= ARC-Message-Signature: i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1724491612; c=relaxed/simple; bh=7NGoICPxYTf1zWyf5Qgu/5OeHqKWWvm0Wiv8aeOWaqo=; h=From:To:CC:Subject:Date:Message-ID:In-Reply-To:References: MIME-Version:Content-Type; b=J2ri30TlXOF+bHophZMcIja6NkEOYJVjiiL6Tks8MPA8XqdjAiHaW8AnAr2PuF7SAviINex9++I29tQj8K6hyGsE4igQxiioFF9YEeQPnay8A0pt3fCVOKsqBvCNsUip+QdHM/472RqhlP8C6VWKPIP/yz3akOMwSbacCkm6wQ4= ARC-Authentication-Results: i=1; smtp.subspace.kernel.org; dmarc=pass (p=quarantine dis=none) header.from=huawei.com; spf=pass smtp.mailfrom=huawei.com; arc=none smtp.client-ip=45.249.212.188 Authentication-Results: smtp.subspace.kernel.org; dmarc=pass (p=quarantine dis=none) header.from=huawei.com Authentication-Results: smtp.subspace.kernel.org; spf=pass smtp.mailfrom=huawei.com Received: from mail.maildlp.com (unknown [172.19.88.105]) by szxga02-in.huawei.com (SkyGuard) with ESMTP id 4WrWjn3t14zhXZb; Sat, 24 Aug 2024 17:24:45 +0800 (CST) Received: from kwepemd100024.china.huawei.com (unknown [7.221.188.41]) by mail.maildlp.com (Postfix) with ESMTPS id 7D3AD140137; Sat, 24 Aug 2024 17:26:47 +0800 (CST) Received: from huawei.com (10.175.124.27) by kwepemd100024.china.huawei.com (7.221.188.41) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384) id 15.2.1544.11; Sat, 24 Aug 2024 17:26:46 +0800 From: yangyun To: Miklos Szeredi CC: , , , Subject: [PATCH v2 2/2] fuse: add support for no forget requests Date: Sat, 24 Aug 2024 17:25:53 +0800 Message-ID: <20240824092553.730338-3-yangyun50@huawei.com> X-Mailer: git-send-email 2.33.0 In-Reply-To: <20240824092553.730338-1-yangyun50@huawei.com> References: <20240824092553.730338-1-yangyun50@huawei.com> Precedence: bulk X-Mailing-List: linux-kernel@vger.kernel.org List-Id: List-Subscribe: List-Unsubscribe: MIME-Version: 1.0 Content-Transfer-Encoding: quoted-printable X-ClientProxiedBy: dggems706-chm.china.huawei.com (10.3.19.183) To kwepemd100024.china.huawei.com (7.221.188.41) Content-Type: text/plain; charset="utf-8" FUSE_FORGET requests are not used if the fuse file system does not implement the forget operation in userspace (e.g., fuse file system does not cache any inodes). However, the kernel is invisible to the userspace implementation and always sends FUSE_FORGET requests, which can lead to performance degradation because of useless contex switch and memory copy in some cases (e.g., many inodes are evicted from icache which was described in commit 07e77dca8a1f ("fuse: separate queue for FORGET requests")). Just like 'no_interrupt' in 'struct fuse_conn', we add 'no_forget'. But since FUSE_FORGET request does not have a reply from userspace, we can not use ENOSYS to reflect the 'no_forget' assignment. So add the FUSE_NO_FORGET_SUPPORT init flag. Besides, if no_forget is enabled, 'nlookup' in 'struct fuse_inode' does not used and its value change can be disabled which are protected by spin_lock to reduce lock contention. Signed-off-by: yangyun --- fs/fuse/dev.c | 6 +++++- fs/fuse/dir.c | 4 +--- fs/fuse/fuse_i.h | 23 +++++++++++++++++++++++ fs/fuse/inode.c | 10 +++++----- fs/fuse/readdir.c | 8 ++------ include/uapi/linux/fuse.h | 3 +++ 6 files changed, 39 insertions(+), 15 deletions(-) diff --git a/fs/fuse/dev.c b/fs/fuse/dev.c index 4b106db2f97f..449c29ef1bce 100644 --- a/fs/fuse/dev.c +++ b/fs/fuse/dev.c @@ -242,8 +242,12 @@ __releases(fiq->lock) void fuse_queue_forget(struct fuse_conn *fc, u64 nodeid, u64 nlookup) { struct fuse_iqueue *fiq =3D &fc->iq; - struct fuse_forget_link *forget =3D fuse_alloc_forget(); + struct fuse_forget_link *forget; =20 + if (fc->no_forget) + return; + + forget =3D fuse_alloc_forget(); forget->forget_one.nodeid =3D nodeid; forget->forget_one.nlookup =3D nlookup; =20 diff --git a/fs/fuse/dir.c b/fs/fuse/dir.c index f070e380fd7b..2cbf96bb8022 100644 --- a/fs/fuse/dir.c +++ b/fs/fuse/dir.c @@ -236,9 +236,7 @@ static int fuse_dentry_revalidate(struct dentry *entry,= unsigned int flags) fuse_queue_forget(fm->fc, outarg.nodeid, 1); goto invalid; } - spin_lock(&fi->lock); - fi->nlookup++; - spin_unlock(&fi->lock); + fuse_inc_nlookup(fm->fc, fi); } if (ret =3D=3D -ENOMEM || ret =3D=3D -EINTR) goto out; diff --git a/fs/fuse/fuse_i.h b/fs/fuse/fuse_i.h index 90176133209d..570decd4b8d6 100644 --- a/fs/fuse/fuse_i.h +++ b/fs/fuse/fuse_i.h @@ -854,6 +854,9 @@ struct fuse_conn { /** Passthrough support for read/write IO */ unsigned int passthrough:1; =20 + /** Is forget not implemented by fs? */ + unsigned int no_forget:1; + /** Maximum stack depth for passthrough backing files */ int max_stack_depth; =20 @@ -1023,6 +1026,26 @@ static inline void fuse_sync_bucket_dec(struct fuse_= sync_bucket *bucket) rcu_read_unlock(); } =20 +static inline void fuse_inc_nlookup(struct fuse_conn *fc, struct fuse_inod= e *fi) +{ + if (fc->no_forget) + return; + + spin_lock(&fi->lock); + fi->nlookup++; + spin_lock(&fi->lock); +} + +static inline void fuse_dec_nlookup(struct fuse_conn *fc, struct fuse_inod= e *fi) +{ + if (fc->no_forget) + return; + + spin_lock(&fi->lock); + fi->nlookup--; + spin_lock(&fi->lock); +} + /** Device operations */ extern const struct file_operations fuse_dev_operations; =20 diff --git a/fs/fuse/inode.c b/fs/fuse/inode.c index da3e5d4c032c..57fe1bbc9a83 100644 --- a/fs/fuse/inode.c +++ b/fs/fuse/inode.c @@ -458,9 +458,7 @@ struct inode *fuse_iget(struct super_block *sb, u64 nod= eid, } } fi =3D get_fuse_inode(inode); - spin_lock(&fi->lock); - fi->nlookup++; - spin_unlock(&fi->lock); + fuse_inc_nlookup(fc, fi); done: fuse_change_attributes(inode, attr, NULL, attr_valid, attr_version); =20 @@ -1306,6 +1304,8 @@ static void process_init_reply(struct fuse_mount *fm,= struct fuse_args *args, } if (flags & FUSE_NO_EXPORT_SUPPORT) fm->sb->s_export_op =3D &fuse_export_fid_operations; + if (flags & FUSE_NO_FORGET_SUPPORT) + fc->no_forget =3D 1; } else { ra_pages =3D fc->max_read / PAGE_SIZE; fc->no_lock =3D 1; @@ -1353,7 +1353,7 @@ void fuse_send_init(struct fuse_mount *fm) FUSE_HANDLE_KILLPRIV_V2 | FUSE_SETXATTR_EXT | FUSE_INIT_EXT | FUSE_SECURITY_CTX | FUSE_CREATE_SUPP_GROUP | FUSE_HAS_EXPIRE_ONLY | FUSE_DIRECT_IO_ALLOW_MMAP | - FUSE_NO_EXPORT_SUPPORT | FUSE_HAS_RESEND; + FUSE_NO_EXPORT_SUPPORT | FUSE_HAS_RESEND | FUSE_NO_FORGET_SUPPORT; #ifdef CONFIG_FUSE_DAX if (fm->fc->dax) flags |=3D FUSE_MAP_ALIGNMENT; @@ -1568,7 +1568,7 @@ static int fuse_fill_super_submount(struct super_bloc= k *sb, * that, though, so undo it here. */ fi =3D get_fuse_inode(root); - fi->nlookup--; + fuse_dec_nlookup(fm->fc, fi); =20 sb->s_d_op =3D &fuse_dentry_operations; sb->s_root =3D d_make_root(root); diff --git a/fs/fuse/readdir.c b/fs/fuse/readdir.c index 721fae563c84..6cda9b34a0e1 100644 --- a/fs/fuse/readdir.c +++ b/fs/fuse/readdir.c @@ -218,9 +218,7 @@ static int fuse_direntplus_link(struct file *file, } =20 fi =3D get_fuse_inode(inode); - spin_lock(&fi->lock); - fi->nlookup++; - spin_unlock(&fi->lock); + fuse_inc_nlookup(fc, fi); =20 forget_all_cached_acls(inode); fuse_change_attributes(inode, &o->attr, NULL, @@ -247,9 +245,7 @@ static int fuse_direntplus_link(struct file *file, if (!IS_ERR(inode)) { struct fuse_inode *fi =3D get_fuse_inode(inode); =20 - spin_lock(&fi->lock); - fi->nlookup--; - spin_unlock(&fi->lock); + fuse_dec_nlookup(fc, fi); } return PTR_ERR(dentry); } diff --git a/include/uapi/linux/fuse.h b/include/uapi/linux/fuse.h index d08b99d60f6f..bf660880bc7a 100644 --- a/include/uapi/linux/fuse.h +++ b/include/uapi/linux/fuse.h @@ -217,6 +217,7 @@ * - add backing_id to fuse_open_out, add FOPEN_PASSTHROUGH open flag * - add FUSE_NO_EXPORT_SUPPORT init flag * - add FUSE_NOTIFY_RESEND, add FUSE_HAS_RESEND init flag + * - add FUSE_NO_FORGET_SUPPORT init flag */ =20 #ifndef _LINUX_FUSE_H @@ -421,6 +422,7 @@ struct fuse_file_lock { * FUSE_NO_EXPORT_SUPPORT: explicitly disable export support * FUSE_HAS_RESEND: kernel supports resending pending requests, and the hi= gh bit * of the request ID indicates resend requests + * FUSE_NO_FORGET_SUPPORT: disable forget requests */ #define FUSE_ASYNC_READ (1 << 0) #define FUSE_POSIX_LOCKS (1 << 1) @@ -463,6 +465,7 @@ struct fuse_file_lock { #define FUSE_PASSTHROUGH (1ULL << 37) #define FUSE_NO_EXPORT_SUPPORT (1ULL << 38) #define FUSE_HAS_RESEND (1ULL << 39) +#define FUSE_NO_FORGET_SUPPORT (1ULL << 40) =20 /* Obsolete alias for FUSE_DIRECT_IO_ALLOW_MMAP */ #define FUSE_DIRECT_IO_RELAX FUSE_DIRECT_IO_ALLOW_MMAP --=20 2.33.0