1 /* SPDX-License-Identifier: GPL-2.0 */
2 #ifndef __NET_TC_WRAPPER_H
3 #define __NET_TC_WRAPPER_H
5 #include <net/pkt_cls.h>
7 #if IS_ENABLED(CONFIG_MITIGATION_RETPOLINE)
9 #include <linux/cpufeature.h>
10 #include <linux/static_key.h>
11 #include <linux/indirect_call_wrapper.h>
13 #define TC_INDIRECT_SCOPE
15 extern struct static_key_false tc_skip_wrapper
;
18 #ifdef CONFIG_NET_CLS_ACT
20 #define TC_INDIRECT_ACTION_DECLARE(fname) \
21 INDIRECT_CALLABLE_DECLARE(int fname(struct sk_buff *skb, \
22 const struct tc_action *a, \
23 struct tcf_result *res))
25 TC_INDIRECT_ACTION_DECLARE(tcf_bpf_act
);
26 TC_INDIRECT_ACTION_DECLARE(tcf_connmark_act
);
27 TC_INDIRECT_ACTION_DECLARE(tcf_csum_act
);
28 TC_INDIRECT_ACTION_DECLARE(tcf_ct_act
);
29 TC_INDIRECT_ACTION_DECLARE(tcf_ctinfo_act
);
30 TC_INDIRECT_ACTION_DECLARE(tcf_gact_act
);
31 TC_INDIRECT_ACTION_DECLARE(tcf_gate_act
);
32 TC_INDIRECT_ACTION_DECLARE(tcf_ife_act
);
33 TC_INDIRECT_ACTION_DECLARE(tcf_ipt_act
);
34 TC_INDIRECT_ACTION_DECLARE(tcf_mirred_act
);
35 TC_INDIRECT_ACTION_DECLARE(tcf_mpls_act
);
36 TC_INDIRECT_ACTION_DECLARE(tcf_nat_act
);
37 TC_INDIRECT_ACTION_DECLARE(tcf_pedit_act
);
38 TC_INDIRECT_ACTION_DECLARE(tcf_police_act
);
39 TC_INDIRECT_ACTION_DECLARE(tcf_sample_act
);
40 TC_INDIRECT_ACTION_DECLARE(tcf_simp_act
);
41 TC_INDIRECT_ACTION_DECLARE(tcf_skbedit_act
);
42 TC_INDIRECT_ACTION_DECLARE(tcf_skbmod_act
);
43 TC_INDIRECT_ACTION_DECLARE(tcf_vlan_act
);
44 TC_INDIRECT_ACTION_DECLARE(tunnel_key_act
);
46 static inline int tc_act(struct sk_buff
*skb
, const struct tc_action
*a
,
47 struct tcf_result
*res
)
49 if (static_branch_likely(&tc_skip_wrapper
))
52 #if IS_BUILTIN(CONFIG_NET_ACT_GACT)
53 if (a
->ops
->act
== tcf_gact_act
)
54 return tcf_gact_act(skb
, a
, res
);
56 #if IS_BUILTIN(CONFIG_NET_ACT_MIRRED)
57 if (a
->ops
->act
== tcf_mirred_act
)
58 return tcf_mirred_act(skb
, a
, res
);
60 #if IS_BUILTIN(CONFIG_NET_ACT_PEDIT)
61 if (a
->ops
->act
== tcf_pedit_act
)
62 return tcf_pedit_act(skb
, a
, res
);
64 #if IS_BUILTIN(CONFIG_NET_ACT_SKBEDIT)
65 if (a
->ops
->act
== tcf_skbedit_act
)
66 return tcf_skbedit_act(skb
, a
, res
);
68 #if IS_BUILTIN(CONFIG_NET_ACT_SKBMOD)
69 if (a
->ops
->act
== tcf_skbmod_act
)
70 return tcf_skbmod_act(skb
, a
, res
);
72 #if IS_BUILTIN(CONFIG_NET_ACT_POLICE)
73 if (a
->ops
->act
== tcf_police_act
)
74 return tcf_police_act(skb
, a
, res
);
76 #if IS_BUILTIN(CONFIG_NET_ACT_BPF)
77 if (a
->ops
->act
== tcf_bpf_act
)
78 return tcf_bpf_act(skb
, a
, res
);
80 #if IS_BUILTIN(CONFIG_NET_ACT_CONNMARK)
81 if (a
->ops
->act
== tcf_connmark_act
)
82 return tcf_connmark_act(skb
, a
, res
);
84 #if IS_BUILTIN(CONFIG_NET_ACT_CSUM)
85 if (a
->ops
->act
== tcf_csum_act
)
86 return tcf_csum_act(skb
, a
, res
);
88 #if IS_BUILTIN(CONFIG_NET_ACT_CT)
89 if (a
->ops
->act
== tcf_ct_act
)
90 return tcf_ct_act(skb
, a
, res
);
92 #if IS_BUILTIN(CONFIG_NET_ACT_CTINFO)
93 if (a
->ops
->act
== tcf_ctinfo_act
)
94 return tcf_ctinfo_act(skb
, a
, res
);
96 #if IS_BUILTIN(CONFIG_NET_ACT_GATE)
97 if (a
->ops
->act
== tcf_gate_act
)
98 return tcf_gate_act(skb
, a
, res
);
100 #if IS_BUILTIN(CONFIG_NET_ACT_MPLS)
101 if (a
->ops
->act
== tcf_mpls_act
)
102 return tcf_mpls_act(skb
, a
, res
);
104 #if IS_BUILTIN(CONFIG_NET_ACT_NAT)
105 if (a
->ops
->act
== tcf_nat_act
)
106 return tcf_nat_act(skb
, a
, res
);
108 #if IS_BUILTIN(CONFIG_NET_ACT_TUNNEL_KEY)
109 if (a
->ops
->act
== tunnel_key_act
)
110 return tunnel_key_act(skb
, a
, res
);
112 #if IS_BUILTIN(CONFIG_NET_ACT_VLAN)
113 if (a
->ops
->act
== tcf_vlan_act
)
114 return tcf_vlan_act(skb
, a
, res
);
116 #if IS_BUILTIN(CONFIG_NET_ACT_IFE)
117 if (a
->ops
->act
== tcf_ife_act
)
118 return tcf_ife_act(skb
, a
, res
);
120 #if IS_BUILTIN(CONFIG_NET_ACT_SIMP)
121 if (a
->ops
->act
== tcf_simp_act
)
122 return tcf_simp_act(skb
, a
, res
);
124 #if IS_BUILTIN(CONFIG_NET_ACT_SAMPLE)
125 if (a
->ops
->act
== tcf_sample_act
)
126 return tcf_sample_act(skb
, a
, res
);
130 return a
->ops
->act(skb
, a
, res
);
133 #endif /* CONFIG_NET_CLS_ACT */
136 #ifdef CONFIG_NET_CLS
138 #define TC_INDIRECT_FILTER_DECLARE(fname) \
139 INDIRECT_CALLABLE_DECLARE(int fname(struct sk_buff *skb, \
140 const struct tcf_proto *tp, \
141 struct tcf_result *res))
143 TC_INDIRECT_FILTER_DECLARE(basic_classify
);
144 TC_INDIRECT_FILTER_DECLARE(cls_bpf_classify
);
145 TC_INDIRECT_FILTER_DECLARE(cls_cgroup_classify
);
146 TC_INDIRECT_FILTER_DECLARE(fl_classify
);
147 TC_INDIRECT_FILTER_DECLARE(flow_classify
);
148 TC_INDIRECT_FILTER_DECLARE(fw_classify
);
149 TC_INDIRECT_FILTER_DECLARE(mall_classify
);
150 TC_INDIRECT_FILTER_DECLARE(route4_classify
);
151 TC_INDIRECT_FILTER_DECLARE(u32_classify
);
153 static inline int tc_classify(struct sk_buff
*skb
, const struct tcf_proto
*tp
,
154 struct tcf_result
*res
)
156 if (static_branch_likely(&tc_skip_wrapper
))
159 #if IS_BUILTIN(CONFIG_NET_CLS_BPF)
160 if (tp
->classify
== cls_bpf_classify
)
161 return cls_bpf_classify(skb
, tp
, res
);
163 #if IS_BUILTIN(CONFIG_NET_CLS_U32)
164 if (tp
->classify
== u32_classify
)
165 return u32_classify(skb
, tp
, res
);
167 #if IS_BUILTIN(CONFIG_NET_CLS_FLOWER)
168 if (tp
->classify
== fl_classify
)
169 return fl_classify(skb
, tp
, res
);
171 #if IS_BUILTIN(CONFIG_NET_CLS_FW)
172 if (tp
->classify
== fw_classify
)
173 return fw_classify(skb
, tp
, res
);
175 #if IS_BUILTIN(CONFIG_NET_CLS_MATCHALL)
176 if (tp
->classify
== mall_classify
)
177 return mall_classify(skb
, tp
, res
);
179 #if IS_BUILTIN(CONFIG_NET_CLS_BASIC)
180 if (tp
->classify
== basic_classify
)
181 return basic_classify(skb
, tp
, res
);
183 #if IS_BUILTIN(CONFIG_NET_CLS_CGROUP)
184 if (tp
->classify
== cls_cgroup_classify
)
185 return cls_cgroup_classify(skb
, tp
, res
);
187 #if IS_BUILTIN(CONFIG_NET_CLS_FLOW)
188 if (tp
->classify
== flow_classify
)
189 return flow_classify(skb
, tp
, res
);
191 #if IS_BUILTIN(CONFIG_NET_CLS_ROUTE4)
192 if (tp
->classify
== route4_classify
)
193 return route4_classify(skb
, tp
, res
);
197 return tp
->classify(skb
, tp
, res
);
200 #endif /* CONFIG_NET_CLS */
202 static inline void tc_wrapper_init(void)
205 if (!cpu_feature_enabled(X86_FEATURE_RETPOLINE
))
206 static_branch_enable(&tc_skip_wrapper
);
212 #define TC_INDIRECT_SCOPE static
214 static inline int tc_act(struct sk_buff
*skb
, const struct tc_action
*a
,
215 struct tcf_result
*res
)
217 return a
->ops
->act(skb
, a
, res
);
220 static inline int tc_classify(struct sk_buff
*skb
, const struct tcf_proto
*tp
,
221 struct tcf_result
*res
)
223 return tp
->classify(skb
, tp
, res
);
226 static inline void tc_wrapper_init(void)
232 #endif /* __NET_TC_WRAPPER_H */