summaryrefslogtreecommitdiff
path: root/fs/xfs/xfs_hooks.h
blob: 7e5ef53f582937c926de408787004aac160431b5 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
// SPDX-License-Identifier: GPL-2.0-or-later
/*
 * Copyright (C) 2022 Oracle.  All Rights Reserved.
 * Author: Darrick J. Wong <djwong@kernel.org>
 */
#ifndef XFS_HOOKS_H_
#define XFS_HOOKS_H_

#if defined(CONFIG_XFS_LIVE_HOOKS_SRCU)
struct xfs_hooks {
	struct srcu_notifier_head	head;
};
#elif defined(CONFIG_XFS_LIVE_HOOKS_BLOCKING)
struct xfs_hooks {
	struct blocking_notifier_head	head;
};
#else
struct xfs_hooks { /* empty */ };
#endif

/*
 * If hooks and jump labels are enabled, we use jump labels (aka patching of
 * the code segment) to avoid the minute overhead of calling an empty notifier
 * chain when we know there are no callers.  If hooks are enabled without jump
 * labels, hardwire the predicate to true because calling an empty srcu
 * notifier chain isn't so expensive.
 */
#if defined(CONFIG_JUMP_LABEL) && defined(CONFIG_XFS_LIVE_HOOKS)
# 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)
#elif defined(CONFIG_XFS_LIVE_HOOKS)
# 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)	(true)
#else
# 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)
#endif /* JUMP_LABEL && XFS_LIVE_HOOKS */

#ifdef CONFIG_XFS_LIVE_HOOKS
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, xfs_hook_fn_t fn)
{
	hook->nb.notifier_call = (notifier_fn_t)fn;
	hook->nb.priority = 0;
}

#else
# define xfs_hooks_init(chain)			((void)0)
# define xfs_hooks_call(chain, val, priv)	(NOTIFY_DONE)
#endif

#endif /* XFS_HOOKS_H_ */