1 /* SPDX-License-Identifier: GPL-2.0 */
2 #ifndef _NF_CONNTRACK_TIMEOUT_H
3 #define _NF_CONNTRACK_TIMEOUT_H
4
5 #include <net/net_namespace.h>
6 #include <linux/netfilter/nf_conntrack_common.h>
7 #include <linux/netfilter/nf_conntrack_tuple_common.h>
8 #include <linux/refcount.h>
9 #include <net/netfilter/nf_conntrack.h>
10 #include <net/netfilter/nf_conntrack_extend.h>
11
12 #define CTNL_TIMEOUT_NAME_MAX 32
13
14 struct nf_ct_timeout {
15 __u16 l3num;
16 const struct nf_conntrack_l4proto *l4proto;
17 char data[0];
18 };
19
20 struct ctnl_timeout {
21 struct list_head head;
22 struct rcu_head rcu_head;
23 refcount_t refcnt;
24 char name[CTNL_TIMEOUT_NAME_MAX];
25 struct nf_ct_timeout timeout;
26 };
27
28 struct nf_conn_timeout {
29 struct nf_ct_timeout __rcu *timeout;
30 };
31
32 static inline unsigned int *
nf_ct_timeout_data(const struct nf_conn_timeout * t)33 nf_ct_timeout_data(const struct nf_conn_timeout *t)
34 {
35 #ifdef CONFIG_NF_CONNTRACK_TIMEOUT
36 struct nf_ct_timeout *timeout;
37
38 timeout = rcu_dereference(t->timeout);
39 if (timeout == NULL)
40 return NULL;
41
42 return (unsigned int *)timeout->data;
43 #else
44 return NULL;
45 #endif
46 }
47
48 static inline
nf_ct_timeout_find(const struct nf_conn * ct)49 struct nf_conn_timeout *nf_ct_timeout_find(const struct nf_conn *ct)
50 {
51 #ifdef CONFIG_NF_CONNTRACK_TIMEOUT
52 return nf_ct_ext_find(ct, NF_CT_EXT_TIMEOUT);
53 #else
54 return NULL;
55 #endif
56 }
57
58 static inline
nf_ct_timeout_ext_add(struct nf_conn * ct,struct nf_ct_timeout * timeout,gfp_t gfp)59 struct nf_conn_timeout *nf_ct_timeout_ext_add(struct nf_conn *ct,
60 struct nf_ct_timeout *timeout,
61 gfp_t gfp)
62 {
63 #ifdef CONFIG_NF_CONNTRACK_TIMEOUT
64 struct nf_conn_timeout *timeout_ext;
65
66 timeout_ext = nf_ct_ext_add(ct, NF_CT_EXT_TIMEOUT, gfp);
67 if (timeout_ext == NULL)
68 return NULL;
69
70 rcu_assign_pointer(timeout_ext->timeout, timeout);
71
72 return timeout_ext;
73 #else
74 return NULL;
75 #endif
76 };
77
nf_ct_timeout_lookup(const struct nf_conn * ct)78 static inline unsigned int *nf_ct_timeout_lookup(const struct nf_conn *ct)
79 {
80 unsigned int *timeouts = NULL;
81 #ifdef CONFIG_NF_CONNTRACK_TIMEOUT
82 struct nf_conn_timeout *timeout_ext;
83
84 timeout_ext = nf_ct_timeout_find(ct);
85 if (timeout_ext)
86 timeouts = nf_ct_timeout_data(timeout_ext);
87 #endif
88 return timeouts;
89 }
90
91 #ifdef CONFIG_NF_CONNTRACK_TIMEOUT
92 int nf_conntrack_timeout_init(void);
93 void nf_conntrack_timeout_fini(void);
94 void nf_ct_untimeout(struct net *net, struct nf_ct_timeout *timeout);
95 int nf_ct_set_timeout(struct net *net, struct nf_conn *ct, u8 l3num, u8 l4num,
96 const char *timeout_name);
97 void nf_ct_destroy_timeout(struct nf_conn *ct);
98 #else
nf_conntrack_timeout_init(void)99 static inline int nf_conntrack_timeout_init(void)
100 {
101 return 0;
102 }
103
nf_conntrack_timeout_fini(void)104 static inline void nf_conntrack_timeout_fini(void)
105 {
106 return;
107 }
108
nf_ct_set_timeout(struct net * net,struct nf_conn * ct,u8 l3num,u8 l4num,const char * timeout_name)109 static inline int nf_ct_set_timeout(struct net *net, struct nf_conn *ct,
110 u8 l3num, u8 l4num,
111 const char *timeout_name)
112 {
113 return -EOPNOTSUPP;
114 }
115
nf_ct_destroy_timeout(struct nf_conn * ct)116 static inline void nf_ct_destroy_timeout(struct nf_conn *ct)
117 {
118 return;
119 }
120 #endif /* CONFIG_NF_CONNTRACK_TIMEOUT */
121
122 #ifdef CONFIG_NF_CONNTRACK_TIMEOUT
123 extern struct nf_ct_timeout *(*nf_ct_timeout_find_get_hook)(struct net *net, const char *name);
124 extern void (*nf_ct_timeout_put_hook)(struct nf_ct_timeout *timeout);
125 #endif
126
127 #endif /* _NF_CONNTRACK_TIMEOUT_H */
128