66 lines
2.1 KiB
C
66 lines
2.1 KiB
C
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||
|
/*
|
||
|
* Copyright (c) 2022-2024 Oracle. All Rights Reserved.
|
||
|
* Author: Darrick J. Wong <djwong@kernel.org>
|
||
|
*/
|
||
|
#ifndef XFS_HOOKS_H_
|
||
|
#define XFS_HOOKS_H_
|
||
|
|
||
|
#ifdef CONFIG_XFS_LIVE_HOOKS
|
||
|
struct xfs_hooks {
|
||
|
struct blocking_notifier_head head;
|
||
|
};
|
||
|
|
||
|
/*
|
||
|
* If jump labels are enabled in Kconfig, the static key uses nop sleds and
|
||
|
* code patching to eliminate the overhead of taking the rwsem in
|
||
|
* blocking_notifier_call_chain when there are no hooks configured. If not,
|
||
|
* the static key per-call overhead is an atomic read. Most arches that can
|
||
|
* handle XFS also support jump labels.
|
||
|
*
|
||
|
* Note: Patching the kernel code requires taking the cpu hotplug lock. Other
|
||
|
* parts of the kernel allocate memory with that lock held, which means that
|
||
|
* XFS callers cannot hold any locks that might be used by memory reclaim or
|
||
|
* writeback when calling the static_branch_{inc,dec} functions.
|
||
|
*/
|
||
|
# define DEFINE_STATIC_XFS_HOOK_SWITCH(name) \
|
||
|
static DEFINE_STATIC_KEY_FALSE(name)
|
||
|
# define xfs_hooks_switch_on(name) static_branch_inc(name)
|
||
|
# define xfs_hooks_switch_off(name) static_branch_dec(name)
|
||
|
# define xfs_hooks_switched_on(name) static_branch_unlikely(name)
|
||
|
|
||
|
struct xfs_hook {
|
||
|
/* This must come at the start of the structure. */
|
||
|
struct notifier_block nb;
|
||
|
};
|
||
|
|
||
|
typedef int (*xfs_hook_fn_t)(struct xfs_hook *hook, unsigned long action,
|
||
|
void *data);
|
||
|
|
||
|
void xfs_hooks_init(struct xfs_hooks *chain);
|
||
|
int xfs_hooks_add(struct xfs_hooks *chain, struct xfs_hook *hook);
|
||
|
void xfs_hooks_del(struct xfs_hooks *chain, struct xfs_hook *hook);
|
||
|
int xfs_hooks_call(struct xfs_hooks *chain, unsigned long action,
|
||
|
void *priv);
|
||
|
|
||
|
static inline void xfs_hook_setup(struct xfs_hook *hook, notifier_fn_t fn)
|
||
|
{
|
||
|
hook->nb.notifier_call = fn;
|
||
|
hook->nb.priority = 0;
|
||
|
}
|
||
|
|
||
|
#else
|
||
|
|
||
|
struct xfs_hooks { /* empty */ };
|
||
|
|
||
|
# define DEFINE_STATIC_XFS_HOOK_SWITCH(name)
|
||
|
# define xfs_hooks_switch_on(name) ((void)0)
|
||
|
# define xfs_hooks_switch_off(name) ((void)0)
|
||
|
# define xfs_hooks_switched_on(name) (false)
|
||
|
|
||
|
# define xfs_hooks_init(chain) ((void)0)
|
||
|
# define xfs_hooks_call(chain, val, priv) (NOTIFY_DONE)
|
||
|
#endif
|
||
|
|
||
|
#endif /* XFS_HOOKS_H_ */
|