mirror of
https://github.com/torvalds/linux.git
synced 2026-05-28 09:04:39 +02:00
Being a module parameter, it's possible to do:
# modprobe cifs drop_dir_cache=1
Which will lead to a crash, because cifs_tcp_ses_list hasn't been
initialized yet:
[ 168.242624] BUG: kernel NULL pointer dereference, address: 0000000000000010
[ 168.242952] #PF: supervisor read access in kernel mode
[ 168.243175] #PF: error_code(0x0000) - not-present page
[ 168.243394] PGD 0 P4D 0
[ 168.243524] Oops: Oops: 0000 [#1] SMP NOPTI
[ 168.243703] CPU: 2 UID: 0 PID: 1105 Comm: modprobe Not tainted 7.0.0-lku #5 PREEMPT(lazy)
[ 168.244054] Hardware name: QEMU Standard PC (Q35 + ICH9, 2009), BIOS rel-1.17.0-2-g4f253b9b-prebuilt.qemu.org 04/01/2014
[ 168.244557] RIP: 0010:cifs_param_set_drop_dir_cache+0x7c/0x100 [cifs]
...
[ 168.248785] Call Trace:
[ 168.248915] <TASK>
[ 168.249023] parse_args+0x285/0x3a0
[ 168.249204] ? __pfx_unknown_module_param_cb+0x10/0x10
[ 168.249448] load_module+0x192b/0x1bb0
[ 168.249637] ? __pfx_unknown_module_param_cb+0x10/0x10
[ 168.249882] ? kernel_read_file+0x27d/0x2b0
[ 168.250088] init_module_from_file+0xce/0xf0
[ 168.250291] idempotent_init_module+0xfb/0x2f0
[ 168.250496] __x64_sys_finit_module+0x5a/0xa0
[ 168.250694] do_syscall_64+0xe0/0x5a0
[ 168.250863] ? exc_page_fault+0x65/0x160
[ 168.251050] entry_SYSCALL_64_after_hwframe+0x77/0x7f
[ 168.251284] RIP: 0033:0x7fcaa12b774d
Instead of fixing this with some kind of "is module initialized"
approach, this patch instead moves that functionality to procfs,
setting a write op for the existing open_dirs entry, where
writing a 0 to it will drop the cached directory entries.
Also make it available only when CONFIG_CIFS_DEBUG=y.
A small change needed now is to not call flush_delayed_work()
on invalidate_all_cached_dirs() when called from procfs (can't sleep in
that context).
So add a @sync arg to invalidate_all_cached_dirs() to control when to
flush the delayed works.
Fixes: dde6667fa3 ("smb: client: add drop_dir_cache module parameter to invalidate cached dirents")
Signed-off-by: Enzo Matsumiya <ematsumiya@suse.de>
Signed-off-by: Steve French <stfrench@microsoft.com>
97 lines
2.7 KiB
C
97 lines
2.7 KiB
C
/* SPDX-License-Identifier: GPL-2.0 */
|
|
/*
|
|
* Functions to handle the cached directory entries
|
|
*
|
|
* Copyright (c) 2022, Ronnie Sahlberg <lsahlber@redhat.com>
|
|
*/
|
|
|
|
#ifndef _CACHED_DIR_H
|
|
#define _CACHED_DIR_H
|
|
|
|
|
|
struct cached_dirent {
|
|
struct list_head entry;
|
|
char *name;
|
|
int namelen;
|
|
loff_t pos;
|
|
struct cifs_fattr fattr;
|
|
};
|
|
|
|
struct cached_dirents {
|
|
bool is_valid:1;
|
|
bool is_failed:1;
|
|
struct file *file; /*
|
|
* Used to associate the cache with a single
|
|
* open file instance.
|
|
*/
|
|
struct mutex de_mutex;
|
|
loff_t pos; /* Expected ctx->pos */
|
|
struct list_head entries;
|
|
/* accounting for cached entries in this directory */
|
|
unsigned long entries_count;
|
|
unsigned long bytes_used;
|
|
};
|
|
|
|
struct cached_fid {
|
|
struct list_head entry;
|
|
struct cached_fids *cfids;
|
|
const char *path;
|
|
bool has_lease;
|
|
bool is_open;
|
|
bool on_list;
|
|
bool file_all_info_is_valid;
|
|
unsigned long time; /* jiffies of when lease was taken */
|
|
unsigned long last_access_time; /* jiffies of when last accessed */
|
|
struct kref refcount;
|
|
struct cifs_fid fid;
|
|
struct cifs_tcon *tcon;
|
|
struct dentry *dentry;
|
|
struct work_struct put_work;
|
|
struct work_struct close_work;
|
|
struct cached_dirents dirents;
|
|
|
|
/* Must be last as it ends in a flexible-array member. */
|
|
struct smb2_file_all_info file_all_info;
|
|
};
|
|
|
|
/* default MAX_CACHED_FIDS is 16 */
|
|
struct cached_fids {
|
|
/* Must be held when:
|
|
* - accessing the cfids->entries list
|
|
* - accessing the cfids->dying list
|
|
*/
|
|
spinlock_t cfid_list_lock;
|
|
int num_entries;
|
|
struct list_head entries;
|
|
struct list_head dying;
|
|
struct delayed_work laundromat_work;
|
|
/* aggregate accounting for all cached dirents under this tcon */
|
|
atomic_long_t total_dirents_entries;
|
|
atomic64_t total_dirents_bytes;
|
|
};
|
|
|
|
/* Module-wide directory cache accounting (defined in cifsfs.c) */
|
|
extern atomic64_t cifs_dircache_bytes_used; /* bytes across all mounts */
|
|
|
|
static inline bool
|
|
is_valid_cached_dir(struct cached_fid *cfid)
|
|
{
|
|
return cfid->time && cfid->has_lease;
|
|
}
|
|
|
|
struct cached_fids *init_cached_dirs(void);
|
|
void free_cached_dirs(struct cached_fids *cfids);
|
|
int open_cached_dir(unsigned int xid, struct cifs_tcon *tcon, const char *path,
|
|
struct cifs_sb_info *cifs_sb, bool lookup_only,
|
|
struct cached_fid **ret_cfid);
|
|
int open_cached_dir_by_dentry(struct cifs_tcon *tcon, struct dentry *dentry,
|
|
struct cached_fid **ret_cfid);
|
|
void close_cached_dir(struct cached_fid *cfid);
|
|
void drop_cached_dir_by_name(const unsigned int xid, struct cifs_tcon *tcon,
|
|
const char *name, struct cifs_sb_info *cifs_sb);
|
|
void close_all_cached_dirs(struct cifs_sb_info *cifs_sb);
|
|
void invalidate_all_cached_dirs(struct cifs_tcon *tcon, bool sync);
|
|
bool cached_dir_lease_break(struct cifs_tcon *tcon, __u8 lease_key[16]);
|
|
|
|
#endif /* _CACHED_DIR_H */
|