xref: /openbmc/linux/include/net/ip_fib.h (revision 4b2b6060)
12874c5fdSThomas Gleixner /* SPDX-License-Identifier: GPL-2.0-or-later */
21da177e4SLinus Torvalds /*
31da177e4SLinus Torvalds  * INET		An implementation of the TCP/IP protocol suite for the LINUX
41da177e4SLinus Torvalds  *		operating system.  INET  is implemented using the  BSD Socket
51da177e4SLinus Torvalds  *		interface as the means of communication with the user level.
61da177e4SLinus Torvalds  *
71da177e4SLinus Torvalds  *		Definitions for the Forwarding Information Base.
81da177e4SLinus Torvalds  *
91da177e4SLinus Torvalds  * Authors:	A.N.Kuznetsov, <kuznet@ms2.inr.ac.ru>
101da177e4SLinus Torvalds  */
111da177e4SLinus Torvalds 
121da177e4SLinus Torvalds #ifndef _NET_IP_FIB_H
131da177e4SLinus Torvalds #define _NET_IP_FIB_H
141da177e4SLinus Torvalds 
151da177e4SLinus Torvalds #include <net/flow.h>
161da177e4SLinus Torvalds #include <linux/seq_file.h>
174895c771SDavid S. Miller #include <linux/rcupdate.h>
1804b1d4e5SIdo Schimmel #include <net/fib_notifier.h>
19e1ef4bf2SThomas Graf #include <net/fib_rules.h>
20f55fbb6aSGuillaume Nault #include <net/inet_dscp.h>
218e773277SDavid S. Miller #include <net/inetpeer.h>
22d26b3a7cSEric Dumazet #include <linux/percpu.h>
23b90eb754SJiri Pirko #include <linux/notifier.h>
240029c0deSReshetova, Elena #include <linux/refcount.h>
251da177e4SLinus Torvalds 
264e902c57SThomas Graf struct fib_config {
274e902c57SThomas Graf 	u8			fc_dst_len;
28f55fbb6aSGuillaume Nault 	dscp_t			fc_dscp;
294e902c57SThomas Graf 	u8			fc_protocol;
304e902c57SThomas Graf 	u8			fc_scope;
314e902c57SThomas Graf 	u8			fc_type;
32f35b794bSDavid Ahern 	u8			fc_gw_family;
33f35b794bSDavid Ahern 	/* 2 bytes unused */
344e902c57SThomas Graf 	u32			fc_table;
356d85c10aSAl Viro 	__be32			fc_dst;
36a4ea5d43SDavid Ahern 	union {
37f35b794bSDavid Ahern 		__be32		fc_gw4;
38a4ea5d43SDavid Ahern 		struct in6_addr	fc_gw6;
39a4ea5d43SDavid Ahern 	};
404e902c57SThomas Graf 	int			fc_oif;
414e902c57SThomas Graf 	u32			fc_flags;
424e902c57SThomas Graf 	u32			fc_priority;
436d85c10aSAl Viro 	__be32			fc_prefsrc;
44493ced1aSDavid Ahern 	u32			fc_nh_id;
454e902c57SThomas Graf 	struct nlattr		*fc_mx;
464e902c57SThomas Graf 	struct rtnexthop	*fc_mp;
474e902c57SThomas Graf 	int			fc_mx_len;
484e902c57SThomas Graf 	int			fc_mp_len;
494e902c57SThomas Graf 	u32			fc_flow;
504e902c57SThomas Graf 	u32			fc_nlflags;
514e902c57SThomas Graf 	struct nl_info		fc_nlinfo;
52571e7226SRoopa Prabhu 	struct nlattr		*fc_encap;
53571e7226SRoopa Prabhu 	u16			fc_encap_type;
541da177e4SLinus Torvalds };
551da177e4SLinus Torvalds 
561da177e4SLinus Torvalds struct fib_info;
57f2bb4bedSDavid S. Miller struct rtable;
581da177e4SLinus Torvalds 
594895c771SDavid S. Miller struct fib_nh_exception {
604895c771SDavid S. Miller 	struct fib_nh_exception __rcu	*fnhe_next;
615aad1de5STimo Teräs 	int				fnhe_genid;
624895c771SDavid S. Miller 	__be32				fnhe_daddr;
634895c771SDavid S. Miller 	u32				fnhe_pmtu;
64d52e5a7eSSabrina Dubroca 	bool				fnhe_mtu_locked;
65aee06da6SJulian Anastasov 	__be32				fnhe_gw;
664895c771SDavid S. Miller 	unsigned long			fnhe_expires;
672ffae99dSTimo Teräs 	struct rtable __rcu		*fnhe_rth_input;
682ffae99dSTimo Teräs 	struct rtable __rcu		*fnhe_rth_output;
694895c771SDavid S. Miller 	unsigned long			fnhe_stamp;
70deed49dfSXin Long 	struct rcu_head			rcu;
714895c771SDavid S. Miller };
724895c771SDavid S. Miller 
734895c771SDavid S. Miller struct fnhe_hash_bucket {
744895c771SDavid S. Miller 	struct fib_nh_exception __rcu	*chain;
754895c771SDavid S. Miller };
764895c771SDavid S. Miller 
77d546c621SEric Dumazet #define FNHE_HASH_SHIFT		11
78d546c621SEric Dumazet #define FNHE_HASH_SIZE		(1 << FNHE_HASH_SHIFT)
794895c771SDavid S. Miller #define FNHE_RECLAIM_DEPTH	5
804895c771SDavid S. Miller 
81f1741730SDavid Ahern struct fib_nh_common {
82f1741730SDavid Ahern 	struct net_device	*nhc_dev;
83e44b14ebSEric Dumazet 	netdevice_tracker	nhc_dev_tracker;
84f1741730SDavid Ahern 	int			nhc_oif;
85f1741730SDavid Ahern 	unsigned char		nhc_scope;
86f1741730SDavid Ahern 	u8			nhc_family;
87bdf00467SDavid Ahern 	u8			nhc_gw_family;
88ecc5663cSDavid Ahern 	unsigned char		nhc_flags;
89ecc5663cSDavid Ahern 	struct lwtunnel_state	*nhc_lwtstate;
90bdf00467SDavid Ahern 
91f1741730SDavid Ahern 	union {
92f1741730SDavid Ahern 		__be32          ipv4;
93f1741730SDavid Ahern 		struct in6_addr ipv6;
94f1741730SDavid Ahern 	} nhc_gw;
95f1741730SDavid Ahern 
96f1741730SDavid Ahern 	int			nhc_weight;
97f1741730SDavid Ahern 	atomic_t		nhc_upper_bound;
980f457a36SDavid Ahern 
990f457a36SDavid Ahern 	/* v4 specific, but allows fib6_nh with v4 routes */
1000f457a36SDavid Ahern 	struct rtable __rcu * __percpu *nhc_pcpu_rth_output;
1010f457a36SDavid Ahern 	struct rtable __rcu     *nhc_rth_input;
102a5995e71SDavid Ahern 	struct fnhe_hash_bucket	__rcu *nhc_exceptions;
103f1741730SDavid Ahern };
104f1741730SDavid Ahern 
1051da177e4SLinus Torvalds struct fib_nh {
106f1741730SDavid Ahern 	struct fib_nh_common	nh_common;
1071da177e4SLinus Torvalds 	struct hlist_node	nh_hash;
1081da177e4SLinus Torvalds 	struct fib_info		*nh_parent;
109c7066f70SPatrick McHardy #ifdef CONFIG_IP_ROUTE_CLASSID
1101da177e4SLinus Torvalds 	__u32			nh_tclassid;
1111da177e4SLinus Torvalds #endif
1121fc050a1SDavid S. Miller 	__be32			nh_saddr;
113436c3b66SDavid S. Miller 	int			nh_saddr_genid;
114f1741730SDavid Ahern #define fib_nh_family		nh_common.nhc_family
115f1741730SDavid Ahern #define fib_nh_dev		nh_common.nhc_dev
116e44b14ebSEric Dumazet #define fib_nh_dev_tracker	nh_common.nhc_dev_tracker
117f1741730SDavid Ahern #define fib_nh_oif		nh_common.nhc_oif
118f1741730SDavid Ahern #define fib_nh_flags		nh_common.nhc_flags
119f1741730SDavid Ahern #define fib_nh_lws		nh_common.nhc_lwtstate
120f1741730SDavid Ahern #define fib_nh_scope		nh_common.nhc_scope
121bdf00467SDavid Ahern #define fib_nh_gw_family	nh_common.nhc_gw_family
122f1741730SDavid Ahern #define fib_nh_gw4		nh_common.nhc_gw.ipv4
123f1741730SDavid Ahern #define fib_nh_gw6		nh_common.nhc_gw.ipv6
124f1741730SDavid Ahern #define fib_nh_weight		nh_common.nhc_weight
125f1741730SDavid Ahern #define fib_nh_upper_bound	nh_common.nhc_upper_bound
1261da177e4SLinus Torvalds };
1271da177e4SLinus Torvalds 
1281da177e4SLinus Torvalds /*
1291da177e4SLinus Torvalds  * This structure contains data shared by many of routes.
1301da177e4SLinus Torvalds  */
1311da177e4SLinus Torvalds 
1324c7e8084SDavid Ahern struct nexthop;
1334c7e8084SDavid Ahern 
1341da177e4SLinus Torvalds struct fib_info {
1351da177e4SLinus Torvalds 	struct hlist_node	fib_hash;
1361da177e4SLinus Torvalds 	struct hlist_node	fib_lhash;
1374c7e8084SDavid Ahern 	struct list_head	nh_list;
1387462bd74SDenis V. Lunev 	struct net		*fib_net;
13979976892SYajun Deng 	refcount_t		fib_treeref;
1400029c0deSReshetova, Elena 	refcount_t		fib_clntref;
14195c96174SEric Dumazet 	unsigned int		fib_flags;
14237e826c5SDavid S. Miller 	unsigned char		fib_dead;
14337e826c5SDavid S. Miller 	unsigned char		fib_protocol;
14437e826c5SDavid S. Miller 	unsigned char		fib_scope;
145f4ef85bbSEric Dumazet 	unsigned char		fib_type;
146b83738aeSAl Viro 	__be32			fib_prefsrc;
1475a56a0b3SMark Tomlinson 	u32			fib_tb_id;
1481da177e4SLinus Torvalds 	u32			fib_priority;
1493fb07dafSEric Dumazet 	struct dst_metrics	*fib_metrics;
1503fb07dafSEric Dumazet #define fib_mtu fib_metrics->metrics[RTAX_MTU-1]
1513fb07dafSEric Dumazet #define fib_window fib_metrics->metrics[RTAX_WINDOW-1]
1523fb07dafSEric Dumazet #define fib_rtt fib_metrics->metrics[RTAX_RTT-1]
1533fb07dafSEric Dumazet #define fib_advmss fib_metrics->metrics[RTAX_ADVMSS-1]
1541da177e4SLinus Torvalds 	int			fib_nhs;
15519a9d136SDavid Ahern 	bool			fib_nh_is_v6;
1561bff1a0cSDavid Ahern 	bool			nh_updated;
157*4b2b6060SHangbin Liu 	bool			pfsrc_removed;
1584c7e8084SDavid Ahern 	struct nexthop		*nh;
159ebc0ffaeSEric Dumazet 	struct rcu_head		rcu;
160a5311060SGustavo A. R. Silva 	struct fib_nh		fib_nh[];
1611da177e4SLinus Torvalds };
1621da177e4SLinus Torvalds 
1631da177e4SLinus Torvalds 
1641da177e4SLinus Torvalds #ifdef CONFIG_IP_MULTIPLE_TABLES
1651da177e4SLinus Torvalds struct fib_rule;
1661da177e4SLinus Torvalds #endif
1671da177e4SLinus Torvalds 
1685b470441SDavid S. Miller struct fib_table;
1691da177e4SLinus Torvalds struct fib_result {
1706ffd9034SDavid Ahern 	__be32			prefix;
1711da177e4SLinus Torvalds 	unsigned char		prefixlen;
1721da177e4SLinus Torvalds 	unsigned char		nh_sel;
1731da177e4SLinus Torvalds 	unsigned char		type;
1741da177e4SLinus Torvalds 	unsigned char		scope;
17585b91b03SDavid S. Miller 	u32			tclassid;
176eba618abSDavid Ahern 	struct fib_nh_common	*nhc;
1771da177e4SLinus Torvalds 	struct fib_info		*fi;
1785b470441SDavid S. Miller 	struct fib_table	*table;
17956315f9eSAlexander Duyck 	struct hlist_head	*fa_head;
1801da177e4SLinus Torvalds };
1811da177e4SLinus Torvalds 
182246955feSRobert Olsson struct fib_result_nl {
18380e856e1SAl Viro 	__be32		fl_addr;   /* To be looked up*/
1845f300893SThomas Graf 	u32		fl_mark;
185246955feSRobert Olsson 	unsigned char	fl_tos;
186246955feSRobert Olsson 	unsigned char   fl_scope;
187246955feSRobert Olsson 	unsigned char   tb_id_in;
188246955feSRobert Olsson 
189246955feSRobert Olsson 	unsigned char   tb_id;      /* Results */
190246955feSRobert Olsson 	unsigned char	prefixlen;
191246955feSRobert Olsson 	unsigned char	nh_sel;
192246955feSRobert Olsson 	unsigned char	type;
193246955feSRobert Olsson 	unsigned char	scope;
194246955feSRobert Olsson 	int             err;
195246955feSRobert Olsson };
1961da177e4SLinus Torvalds 
1975b9e12dbSDenis V. Lunev #ifdef CONFIG_IP_MULTIPLE_TABLES
1985b9e12dbSDenis V. Lunev #define FIB_TABLE_HASHSZ 256
1995b9e12dbSDenis V. Lunev #else
2005b9e12dbSDenis V. Lunev #define FIB_TABLE_HASHSZ 2
2015b9e12dbSDenis V. Lunev #endif
2025b9e12dbSDenis V. Lunev 
203dcb1ecb5SDavid Ahern __be32 fib_info_update_nhc_saddr(struct net *net, struct fib_nh_common *nhc,
204c3669486SDavid Ahern 				 unsigned char scope);
205eba618abSDavid Ahern __be32 fib_result_prefsrc(struct net *net, struct fib_result *res);
206436c3b66SDavid S. Miller 
207eba618abSDavid Ahern #define FIB_RES_NHC(res)		((res).nhc)
208eba618abSDavid Ahern #define FIB_RES_DEV(res)	(FIB_RES_NHC(res)->nhc_dev)
209eba618abSDavid Ahern #define FIB_RES_OIF(res)	(FIB_RES_NHC(res)->nhc_oif)
2101fc050a1SDavid S. Miller 
2111e301fd0SIdo Schimmel struct fib_rt_info {
2121e301fd0SIdo Schimmel 	struct fib_info		*fi;
2131e301fd0SIdo Schimmel 	u32			tb_id;
2141e301fd0SIdo Schimmel 	__be32			dst;
2151e301fd0SIdo Schimmel 	int			dst_len;
216888ade8fSGuillaume Nault 	dscp_t			dscp;
2171e301fd0SIdo Schimmel 	u8			type;
21890b93f1bSIdo Schimmel 	u8			offload:1,
21990b93f1bSIdo Schimmel 				trap:1,
22036c5100eSAmit Cohen 				offload_failed:1,
22136c5100eSAmit Cohen 				unused:5;
2221e301fd0SIdo Schimmel };
2231e301fd0SIdo Schimmel 
224b90eb754SJiri Pirko struct fib_entry_notifier_info {
225b90eb754SJiri Pirko 	struct fib_notifier_info info; /* must be first */
226b90eb754SJiri Pirko 	u32 dst;
227b90eb754SJiri Pirko 	int dst_len;
228b90eb754SJiri Pirko 	struct fib_info *fi;
229568a3f33SGuillaume Nault 	dscp_t dscp;
230b90eb754SJiri Pirko 	u8 type;
231b90eb754SJiri Pirko 	u32 tb_id;
232b90eb754SJiri Pirko };
233b90eb754SJiri Pirko 
234982acb97SIdo Schimmel struct fib_nh_notifier_info {
235982acb97SIdo Schimmel 	struct fib_notifier_info info; /* must be first */
236982acb97SIdo Schimmel 	struct fib_nh *fib_nh;
237982acb97SIdo Schimmel };
238982acb97SIdo Schimmel 
2397c550dafSJiri Pirko int call_fib4_notifier(struct notifier_block *nb,
240c0243892SIdo Schimmel 		       enum fib_event_type event_type,
241c0243892SIdo Schimmel 		       struct fib_notifier_info *info);
24204b1d4e5SIdo Schimmel int call_fib4_notifiers(struct net *net, enum fib_event_type event_type,
243b90eb754SJiri Pirko 			struct fib_notifier_info *info);
244b90eb754SJiri Pirko 
24504b1d4e5SIdo Schimmel int __net_init fib4_notifier_init(struct net *net);
24604b1d4e5SIdo Schimmel void __net_exit fib4_notifier_exit(struct net *net);
24704b1d4e5SIdo Schimmel 
2481bff1a0cSDavid Ahern void fib_info_notify_update(struct net *net, struct nl_info *info);
249b7a59557SJiri Pirko int fib_notify(struct net *net, struct notifier_block *nb,
250b7a59557SJiri Pirko 	       struct netlink_ext_ack *extack);
251c0243892SIdo Schimmel 
2521da177e4SLinus Torvalds struct fib_table {
2531af5a8c4SPatrick McHardy 	struct hlist_node	tb_hlist;
2542dfe55b4SPatrick McHardy 	u32			tb_id;
25521d8c49eSDavid S. Miller 	int			tb_num_default;
256a7e53531SAlexander Duyck 	struct rcu_head		rcu;
2570ddcf43dSAlexander Duyck 	unsigned long 		*tb_data;
258a5311060SGustavo A. R. Silva 	unsigned long		__data[];
2591da177e4SLinus Torvalds };
2601da177e4SLinus Torvalds 
2614724676dSDavid Ahern struct fib_dump_filter {
2624724676dSDavid Ahern 	u32			table_id;
2634724676dSDavid Ahern 	/* filter_set is an optimization that an entry is set */
2644724676dSDavid Ahern 	bool			filter_set;
265564c91f7SStefano Brivio 	bool			dump_routes;
266564c91f7SStefano Brivio 	bool			dump_exceptions;
2674724676dSDavid Ahern 	unsigned char		protocol;
2684724676dSDavid Ahern 	unsigned char		rt_type;
2694724676dSDavid Ahern 	unsigned int		flags;
2704724676dSDavid Ahern 	struct net_device	*dev;
2714724676dSDavid Ahern };
2724724676dSDavid Ahern 
2735c3a0fd7SJoe Perches int fib_table_lookup(struct fib_table *tb, const struct flowi4 *flp,
274ebc0ffaeSEric Dumazet 		     struct fib_result *res, int fib_flags);
2756d8422a1SDavid Ahern int fib_table_insert(struct net *, struct fib_table *, struct fib_config *,
2766d8422a1SDavid Ahern 		     struct netlink_ext_ack *extack);
27778055998SDavid Ahern int fib_table_delete(struct net *, struct fib_table *, struct fib_config *,
27878055998SDavid Ahern 		     struct netlink_ext_ack *extack);
2795c3a0fd7SJoe Perches int fib_table_dump(struct fib_table *table, struct sk_buff *skb,
28018a8021aSDavid Ahern 		   struct netlink_callback *cb, struct fib_dump_filter *filter);
281f97f4dd8SIdo Schimmel int fib_table_flush(struct net *net, struct fib_table *table, bool flush_all);
2820ddcf43dSAlexander Duyck struct fib_table *fib_trie_unmerge(struct fib_table *main_tb);
2833b709334SAlexander Duyck void fib_table_flush_external(struct fib_table *table);
2845c3a0fd7SJoe Perches void fib_free_table(struct fib_table *tb);
2854aa2c466SPavel Emelyanov 
2861da177e4SLinus Torvalds #ifndef CONFIG_IP_MULTIPLE_TABLES
2871da177e4SLinus Torvalds 
288a5a519b2SAlexander Duyck #define TABLE_LOCAL_INDEX	(RT_TABLE_LOCAL & (FIB_TABLE_HASHSZ - 1))
289a5a519b2SAlexander Duyck #define TABLE_MAIN_INDEX	(RT_TABLE_MAIN  & (FIB_TABLE_HASHSZ - 1))
2901da177e4SLinus Torvalds 
fib_get_table(struct net * net,u32 id)2918ad4942cSDenis V. Lunev static inline struct fib_table *fib_get_table(struct net *net, u32 id)
2921da177e4SLinus Torvalds {
293a7e53531SAlexander Duyck 	struct hlist_node *tb_hlist;
29493456b6dSDenis V. Lunev 	struct hlist_head *ptr;
29593456b6dSDenis V. Lunev 
29693456b6dSDenis V. Lunev 	ptr = id == RT_TABLE_LOCAL ?
297e4aef8aeSDenis V. Lunev 		&net->ipv4.fib_table_hash[TABLE_LOCAL_INDEX] :
298e4aef8aeSDenis V. Lunev 		&net->ipv4.fib_table_hash[TABLE_MAIN_INDEX];
299a7e53531SAlexander Duyck 
300a7e53531SAlexander Duyck 	tb_hlist = rcu_dereference_rtnl(hlist_first_rcu(ptr));
301a7e53531SAlexander Duyck 
302a7e53531SAlexander Duyck 	return hlist_entry(tb_hlist, struct fib_table, tb_hlist);
3031da177e4SLinus Torvalds }
3041da177e4SLinus Torvalds 
fib_new_table(struct net * net,u32 id)3058ad4942cSDenis V. Lunev static inline struct fib_table *fib_new_table(struct net *net, u32 id)
3061da177e4SLinus Torvalds {
3078ad4942cSDenis V. Lunev 	return fib_get_table(net, id);
3081da177e4SLinus Torvalds }
3091da177e4SLinus Torvalds 
fib_lookup(struct net * net,const struct flowi4 * flp,struct fib_result * res,unsigned int flags)31022bd5b9bSDavid S. Miller static inline int fib_lookup(struct net *net, const struct flowi4 *flp,
3110eeb075fSAndy Gospodarek 			     struct fib_result *res, unsigned int flags)
3121da177e4SLinus Torvalds {
313a7e53531SAlexander Duyck 	struct fib_table *tb;
3140ddcf43dSAlexander Duyck 	int err = -ENETUNREACH;
31593456b6dSDenis V. Lunev 
316345e9b54SAlexander Duyck 	rcu_read_lock();
31793456b6dSDenis V. Lunev 
318a7e53531SAlexander Duyck 	tb = fib_get_table(net, RT_TABLE_MAIN);
3190315e382SNikola Forró 	if (tb)
3200315e382SNikola Forró 		err = fib_table_lookup(tb, flp, res, flags | FIB_LOOKUP_NOREF);
3210315e382SNikola Forró 
3220315e382SNikola Forró 	if (err == -EAGAIN)
3230315e382SNikola Forró 		err = -ENETUNREACH;
324345e9b54SAlexander Duyck 
325345e9b54SAlexander Duyck 	rcu_read_unlock();
326345e9b54SAlexander Duyck 
327345e9b54SAlexander Duyck 	return err;
3281da177e4SLinus Torvalds }
3291da177e4SLinus Torvalds 
fib4_has_custom_rules(const struct net * net)330c43c3d76SPaolo Abeni static inline bool fib4_has_custom_rules(const struct net *net)
331c43c3d76SPaolo Abeni {
332c43c3d76SPaolo Abeni 	return false;
333c43c3d76SPaolo Abeni }
334c43c3d76SPaolo Abeni 
fib4_rule_default(const struct fib_rule * rule)3353c71006dSIdo Schimmel static inline bool fib4_rule_default(const struct fib_rule *rule)
3363c71006dSIdo Schimmel {
3373c71006dSIdo Schimmel 	return true;
3383c71006dSIdo Schimmel }
3393c71006dSIdo Schimmel 
fib4_rules_dump(struct net * net,struct notifier_block * nb,struct netlink_ext_ack * extack)340b7a59557SJiri Pirko static inline int fib4_rules_dump(struct net *net, struct notifier_block *nb,
341b7a59557SJiri Pirko 				  struct netlink_ext_ack *extack)
3421b2a4440SIdo Schimmel {
3431b2a4440SIdo Schimmel 	return 0;
3441b2a4440SIdo Schimmel }
3451b2a4440SIdo Schimmel 
fib4_rules_seq_read(struct net * net)3461b2a4440SIdo Schimmel static inline unsigned int fib4_rules_seq_read(struct net *net)
3471b2a4440SIdo Schimmel {
3481b2a4440SIdo Schimmel 	return 0;
3491b2a4440SIdo Schimmel }
3501b2a4440SIdo Schimmel 
fib4_rules_early_flow_dissect(struct net * net,struct sk_buff * skb,struct flowi4 * fl4,struct flow_keys * flkeys)351e37b1e97SRoopa Prabhu static inline bool fib4_rules_early_flow_dissect(struct net *net,
352e37b1e97SRoopa Prabhu 						 struct sk_buff *skb,
353e37b1e97SRoopa Prabhu 						 struct flowi4 *fl4,
354e37b1e97SRoopa Prabhu 						 struct flow_keys *flkeys)
355e37b1e97SRoopa Prabhu {
356e37b1e97SRoopa Prabhu 	return false;
357e37b1e97SRoopa Prabhu }
3581da177e4SLinus Torvalds #else /* CONFIG_IP_MULTIPLE_TABLES */
3595c3a0fd7SJoe Perches int __net_init fib4_rules_init(struct net *net);
3605c3a0fd7SJoe Perches void __net_exit fib4_rules_exit(struct net *net);
361c3e9a353SPavel Emelyanov 
3625c3a0fd7SJoe Perches struct fib_table *fib_new_table(struct net *net, u32 id);
3635c3a0fd7SJoe Perches struct fib_table *fib_get_table(struct net *net, u32 id);
3641da177e4SLinus Torvalds 
3650eeb075fSAndy Gospodarek int __fib_lookup(struct net *net, struct flowi4 *flp,
3660eeb075fSAndy Gospodarek 		 struct fib_result *res, unsigned int flags);
367f4530fa5SDavid S. Miller 
fib_lookup(struct net * net,struct flowi4 * flp,struct fib_result * res,unsigned int flags)368f4530fa5SDavid S. Miller static inline int fib_lookup(struct net *net, struct flowi4 *flp,
3690eeb075fSAndy Gospodarek 			     struct fib_result *res, unsigned int flags)
370f4530fa5SDavid S. Miller {
371a7e53531SAlexander Duyck 	struct fib_table *tb;
3720315e382SNikola Forró 	int err = -ENETUNREACH;
373a7e53531SAlexander Duyck 
3740eeb075fSAndy Gospodarek 	flags |= FIB_LOOKUP_NOREF;
375a7e53531SAlexander Duyck 	if (net->ipv4.fib_has_custom_rules)
3760eeb075fSAndy Gospodarek 		return __fib_lookup(net, flp, res, flags);
377345e9b54SAlexander Duyck 
378345e9b54SAlexander Duyck 	rcu_read_lock();
379345e9b54SAlexander Duyck 
38085b91b03SDavid S. Miller 	res->tclassid = 0;
381a7e53531SAlexander Duyck 
382a7e53531SAlexander Duyck 	tb = rcu_dereference_rtnl(net->ipv4.fib_main);
3830315e382SNikola Forró 	if (tb)
3840315e382SNikola Forró 		err = fib_table_lookup(tb, flp, res, flags);
3850315e382SNikola Forró 
3860315e382SNikola Forró 	if (!err)
3870315e382SNikola Forró 		goto out;
388a7e53531SAlexander Duyck 
389a7e53531SAlexander Duyck 	tb = rcu_dereference_rtnl(net->ipv4.fib_default);
3900315e382SNikola Forró 	if (tb)
3910315e382SNikola Forró 		err = fib_table_lookup(tb, flp, res, flags);
3920315e382SNikola Forró 
3930315e382SNikola Forró out:
3940315e382SNikola Forró 	if (err == -EAGAIN)
3950315e382SNikola Forró 		err = -ENETUNREACH;
396345e9b54SAlexander Duyck 
397345e9b54SAlexander Duyck 	rcu_read_unlock();
398345e9b54SAlexander Duyck 
399345e9b54SAlexander Duyck 	return err;
400f4530fa5SDavid S. Miller }
401f4530fa5SDavid S. Miller 
fib4_has_custom_rules(const struct net * net)402c43c3d76SPaolo Abeni static inline bool fib4_has_custom_rules(const struct net *net)
403c43c3d76SPaolo Abeni {
404c43c3d76SPaolo Abeni 	return net->ipv4.fib_has_custom_rules;
405c43c3d76SPaolo Abeni }
406c43c3d76SPaolo Abeni 
4073c71006dSIdo Schimmel bool fib4_rule_default(const struct fib_rule *rule);
408b7a59557SJiri Pirko int fib4_rules_dump(struct net *net, struct notifier_block *nb,
409b7a59557SJiri Pirko 		    struct netlink_ext_ack *extack);
4101b2a4440SIdo Schimmel unsigned int fib4_rules_seq_read(struct net *net);
4113c71006dSIdo Schimmel 
fib4_rules_early_flow_dissect(struct net * net,struct sk_buff * skb,struct flowi4 * fl4,struct flow_keys * flkeys)412e37b1e97SRoopa Prabhu static inline bool fib4_rules_early_flow_dissect(struct net *net,
413e37b1e97SRoopa Prabhu 						 struct sk_buff *skb,
414e37b1e97SRoopa Prabhu 						 struct flowi4 *fl4,
415e37b1e97SRoopa Prabhu 						 struct flow_keys *flkeys)
416e37b1e97SRoopa Prabhu {
417e37b1e97SRoopa Prabhu 	unsigned int flag = FLOW_DISSECTOR_F_STOP_AT_ENCAP;
418e37b1e97SRoopa Prabhu 
419e37b1e97SRoopa Prabhu 	if (!net->ipv4.fib_rules_require_fldissect)
420e37b1e97SRoopa Prabhu 		return false;
421e37b1e97SRoopa Prabhu 
4228aae7625SFlorian Westphal 	memset(flkeys, 0, sizeof(*flkeys));
4238aae7625SFlorian Westphal 	__skb_flow_dissect(net, skb, &flow_keys_dissector,
4248aae7625SFlorian Westphal 			   flkeys, NULL, 0, 0, 0, flag);
4258aae7625SFlorian Westphal 
426e37b1e97SRoopa Prabhu 	fl4->fl4_sport = flkeys->ports.src;
427e37b1e97SRoopa Prabhu 	fl4->fl4_dport = flkeys->ports.dst;
428e37b1e97SRoopa Prabhu 	fl4->flowi4_proto = flkeys->basic.ip_proto;
429e37b1e97SRoopa Prabhu 
430e37b1e97SRoopa Prabhu 	return true;
431e37b1e97SRoopa Prabhu }
432e37b1e97SRoopa Prabhu 
4331da177e4SLinus Torvalds #endif /* CONFIG_IP_MULTIPLE_TABLES */
4341da177e4SLinus Torvalds 
4351da177e4SLinus Torvalds /* Exported by fib_frontend.c */
436ef7c79edSPatrick McHardy extern const struct nla_policy rtm_ipv4_policy[];
4375c3a0fd7SJoe Perches void ip_fib_init(void);
438d1566268SDavid Ahern int fib_gw_from_via(struct fib_config *cfg, struct nlattr *nla,
439d1566268SDavid Ahern 		    struct netlink_ext_ack *extack);
4405c3a0fd7SJoe Perches __be32 fib_compute_spec_dst(struct sk_buff *skb);
44178f2756cSDavid Ahern bool fib_info_nh_uses_dev(struct fib_info *fi, const struct net_device *dev);
4425c3a0fd7SJoe Perches int fib_validate_source(struct sk_buff *skb, __be32 src, __be32 dst,
4435c04c819SMichael Smith 			u8 tos, int oif, struct net_device *dev,
4449e56e380SDavid S. Miller 			struct in_device *idev, u32 *itag);
4457a9bc9b8SDavid S. Miller #ifdef CONFIG_IP_ROUTE_CLASSID
fib_num_tclassid_users(struct net * net)446f4530fa5SDavid S. Miller static inline int fib_num_tclassid_users(struct net *net)
447f4530fa5SDavid S. Miller {
448213f5f8fSEric Dumazet 	return atomic_read(&net->ipv4.fib_num_tclassid_users);
449f4530fa5SDavid S. Miller }
4507a9bc9b8SDavid S. Miller #else
fib_num_tclassid_users(struct net * net)451f4530fa5SDavid S. Miller static inline int fib_num_tclassid_users(struct net *net)
452f4530fa5SDavid S. Miller {
453f4530fa5SDavid S. Miller 	return 0;
454f4530fa5SDavid S. Miller }
4557a9bc9b8SDavid S. Miller #endif
4560ddcf43dSAlexander Duyck int fib_unmerge(struct net *net);
45714c85021SArnaldo Carvalho de Melo 
nhc_l3mdev_matches_dev(const struct fib_nh_common * nhc,const struct net_device * dev)4581fd1c768SDavid Ahern static inline bool nhc_l3mdev_matches_dev(const struct fib_nh_common *nhc,
4591fd1c768SDavid Ahern const struct net_device *dev)
4601fd1c768SDavid Ahern {
4611fd1c768SDavid Ahern 	if (nhc->nhc_dev == dev ||
4621fd1c768SDavid Ahern 	    l3mdev_master_ifindex_rcu(nhc->nhc_dev) == dev->ifindex)
4631fd1c768SDavid Ahern 		return true;
4641fd1c768SDavid Ahern 
4651fd1c768SDavid Ahern 	return false;
4661fd1c768SDavid Ahern }
4671fd1c768SDavid Ahern 
4681da177e4SLinus Torvalds /* Exported by fib_semantics.c */
4695c3a0fd7SJoe Perches int ip_fib_check_default(__be32 gw, struct net_device *dev);
4704f823defSJulian Anastasov int fib_sync_down_dev(struct net_device *dev, unsigned long event, bool force);
4715a56a0b3SMark Tomlinson int fib_sync_down_addr(struct net_device *dev, __be32 local);
472ecc5663cSDavid Ahern int fib_sync_up(struct net_device *dev, unsigned char nh_flags);
473af7d6cceSSabrina Dubroca void fib_sync_mtu(struct net_device *dev, u32 orig_mtu);
47406c77c3eSDavid Ahern void fib_nhc_update_mtu(struct fib_nh_common *nhc, u32 new, u32 orig);
4750e884c78SPeter Nørlund 
476ce5c9c20SIdo Schimmel /* Fields used for sysctl_fib_multipath_hash_fields.
477ce5c9c20SIdo Schimmel  * Common to IPv4 and IPv6.
478ce5c9c20SIdo Schimmel  *
479ce5c9c20SIdo Schimmel  * Add new fields at the end. This is user API.
480ce5c9c20SIdo Schimmel  */
481ce5c9c20SIdo Schimmel #define FIB_MULTIPATH_HASH_FIELD_SRC_IP			BIT(0)
482ce5c9c20SIdo Schimmel #define FIB_MULTIPATH_HASH_FIELD_DST_IP			BIT(1)
483ce5c9c20SIdo Schimmel #define FIB_MULTIPATH_HASH_FIELD_IP_PROTO		BIT(2)
484ce5c9c20SIdo Schimmel #define FIB_MULTIPATH_HASH_FIELD_FLOWLABEL		BIT(3)
485ce5c9c20SIdo Schimmel #define FIB_MULTIPATH_HASH_FIELD_SRC_PORT		BIT(4)
486ce5c9c20SIdo Schimmel #define FIB_MULTIPATH_HASH_FIELD_DST_PORT		BIT(5)
487ce5c9c20SIdo Schimmel #define FIB_MULTIPATH_HASH_FIELD_INNER_SRC_IP		BIT(6)
488ce5c9c20SIdo Schimmel #define FIB_MULTIPATH_HASH_FIELD_INNER_DST_IP		BIT(7)
489ce5c9c20SIdo Schimmel #define FIB_MULTIPATH_HASH_FIELD_INNER_IP_PROTO		BIT(8)
490ce5c9c20SIdo Schimmel #define FIB_MULTIPATH_HASH_FIELD_INNER_FLOWLABEL	BIT(9)
491ce5c9c20SIdo Schimmel #define FIB_MULTIPATH_HASH_FIELD_INNER_SRC_PORT		BIT(10)
492ce5c9c20SIdo Schimmel #define FIB_MULTIPATH_HASH_FIELD_INNER_DST_PORT		BIT(11)
493ce5c9c20SIdo Schimmel 
494ce5c9c20SIdo Schimmel #define FIB_MULTIPATH_HASH_FIELD_OUTER_MASK		\
495ce5c9c20SIdo Schimmel 	(FIB_MULTIPATH_HASH_FIELD_SRC_IP |		\
496ce5c9c20SIdo Schimmel 	 FIB_MULTIPATH_HASH_FIELD_DST_IP |		\
497ce5c9c20SIdo Schimmel 	 FIB_MULTIPATH_HASH_FIELD_IP_PROTO |		\
498ce5c9c20SIdo Schimmel 	 FIB_MULTIPATH_HASH_FIELD_FLOWLABEL |		\
499ce5c9c20SIdo Schimmel 	 FIB_MULTIPATH_HASH_FIELD_SRC_PORT |		\
500ce5c9c20SIdo Schimmel 	 FIB_MULTIPATH_HASH_FIELD_DST_PORT)
501ce5c9c20SIdo Schimmel 
502ce5c9c20SIdo Schimmel #define FIB_MULTIPATH_HASH_FIELD_INNER_MASK		\
503ce5c9c20SIdo Schimmel 	(FIB_MULTIPATH_HASH_FIELD_INNER_SRC_IP |	\
504ce5c9c20SIdo Schimmel 	 FIB_MULTIPATH_HASH_FIELD_INNER_DST_IP |	\
505ce5c9c20SIdo Schimmel 	 FIB_MULTIPATH_HASH_FIELD_INNER_IP_PROTO |	\
506ce5c9c20SIdo Schimmel 	 FIB_MULTIPATH_HASH_FIELD_INNER_FLOWLABEL |	\
507ce5c9c20SIdo Schimmel 	 FIB_MULTIPATH_HASH_FIELD_INNER_SRC_PORT |	\
508ce5c9c20SIdo Schimmel 	 FIB_MULTIPATH_HASH_FIELD_INNER_DST_PORT)
509ce5c9c20SIdo Schimmel 
510ce5c9c20SIdo Schimmel #define FIB_MULTIPATH_HASH_FIELD_ALL_MASK		\
511ce5c9c20SIdo Schimmel 	(FIB_MULTIPATH_HASH_FIELD_OUTER_MASK |		\
512ce5c9c20SIdo Schimmel 	 FIB_MULTIPATH_HASH_FIELD_INNER_MASK)
513ce5c9c20SIdo Schimmel 
514ce5c9c20SIdo Schimmel #define FIB_MULTIPATH_HASH_FIELD_DEFAULT_MASK		\
515ce5c9c20SIdo Schimmel 	(FIB_MULTIPATH_HASH_FIELD_SRC_IP |		\
516ce5c9c20SIdo Schimmel 	 FIB_MULTIPATH_HASH_FIELD_DST_IP |		\
517ce5c9c20SIdo Schimmel 	 FIB_MULTIPATH_HASH_FIELD_IP_PROTO)
518ce5c9c20SIdo Schimmel 
519bf4e0a3dSNikolay Aleksandrov #ifdef CONFIG_IP_ROUTE_MULTIPATH
5207efc0b6bSDavid Ahern int fib_multipath_hash(const struct net *net, const struct flowi4 *fl4,
521e37b1e97SRoopa Prabhu 		       const struct sk_buff *skb, struct flow_keys *flkeys);
522bf4e0a3dSNikolay Aleksandrov #endif
523ac1fab2dSDavid Ahern int fib_check_nh(struct net *net, struct fib_nh *nh, u32 table, u8 scope,
524ac1fab2dSDavid Ahern 		 struct netlink_ext_ack *extack);
5250e884c78SPeter Nørlund void fib_select_multipath(struct fib_result *res, int hash);
5263ce58d84SDavid Ahern void fib_select_path(struct net *net, struct fib_result *res,
527bf4e0a3dSNikolay Aleksandrov 		     struct flowi4 *fl4, const struct sk_buff *skb);
5281da177e4SLinus Torvalds 
529e4516ef6SDavid Ahern int fib_nh_init(struct net *net, struct fib_nh *fib_nh,
530e4516ef6SDavid Ahern 		struct fib_config *cfg, int nh_weight,
531e4516ef6SDavid Ahern 		struct netlink_ext_ack *extack);
532faa041a4SDavid Ahern void fib_nh_release(struct net *net, struct fib_nh *fib_nh);
533faee6769SAlexander Aring int fib_nh_common_init(struct net *net, struct fib_nh_common *nhc,
534faee6769SAlexander Aring 		       struct nlattr *fc_encap, u16 fc_encap_type,
535faee6769SAlexander Aring 		       void *cfg, gfp_t gfp_flags,
536979e276eSDavid Ahern 		       struct netlink_ext_ack *extack);
537979e276eSDavid Ahern void fib_nh_common_release(struct fib_nh_common *nhc);
538e4516ef6SDavid Ahern 
5395348ba85SDavid S. Miller /* Exported by fib_trie.c */
54090b93f1bSIdo Schimmel void fib_alias_hw_flags_set(struct net *net, const struct fib_rt_info *fri);
5415c3a0fd7SJoe Perches void fib_trie_init(void);
5420ddcf43dSAlexander Duyck struct fib_table *fib_trie_table(u32 id, struct fib_table *alias);
543af7888adSDavid Ahern bool fib_lookup_good_nhc(const struct fib_nh_common *nhc, int fib_flags,
544af7888adSDavid Ahern 			 const struct flowi4 *flp);
5451da177e4SLinus Torvalds 
fib_combine_itag(u32 * itag,const struct fib_result * res)546b6bf3ca0SDavid S. Miller static inline void fib_combine_itag(u32 *itag, const struct fib_result *res)
5471da177e4SLinus Torvalds {
548c7066f70SPatrick McHardy #ifdef CONFIG_IP_ROUTE_CLASSID
549eba618abSDavid Ahern 	struct fib_nh_common *nhc = res->nhc;
5501da177e4SLinus Torvalds #ifdef CONFIG_IP_MULTIPLE_TABLES
5511da177e4SLinus Torvalds 	u32 rtag;
5521da177e4SLinus Torvalds #endif
553dcb1ecb5SDavid Ahern 	if (nhc->nhc_family == AF_INET) {
554dcb1ecb5SDavid Ahern 		struct fib_nh *nh;
555dcb1ecb5SDavid Ahern 
556dcb1ecb5SDavid Ahern 		nh = container_of(nhc, struct fib_nh, nh_common);
557eba618abSDavid Ahern 		*itag = nh->nh_tclassid << 16;
558dcb1ecb5SDavid Ahern 	} else {
559dcb1ecb5SDavid Ahern 		*itag = 0;
560dcb1ecb5SDavid Ahern 	}
561dcb1ecb5SDavid Ahern 
5621da177e4SLinus Torvalds #ifdef CONFIG_IP_MULTIPLE_TABLES
56385b91b03SDavid S. Miller 	rtag = res->tclassid;
5641da177e4SLinus Torvalds 	if (*itag == 0)
5651da177e4SLinus Torvalds 		*itag = (rtag<<16);
5661da177e4SLinus Torvalds 	*itag |= (rtag>>16);
5671da177e4SLinus Torvalds #endif
5681da177e4SLinus Torvalds #endif
5691da177e4SLinus Torvalds }
5701da177e4SLinus Torvalds 
5719bd83667SDavid Ahern void fib_flush(struct net *net);
5725c3a0fd7SJoe Perches void free_fib_info(struct fib_info *fi);
5731da177e4SLinus Torvalds 
fib_info_hold(struct fib_info * fi)5741c677b3dSIdo Schimmel static inline void fib_info_hold(struct fib_info *fi)
5751c677b3dSIdo Schimmel {
5760029c0deSReshetova, Elena 	refcount_inc(&fi->fib_clntref);
5771c677b3dSIdo Schimmel }
5781c677b3dSIdo Schimmel 
fib_info_put(struct fib_info * fi)5791da177e4SLinus Torvalds static inline void fib_info_put(struct fib_info *fi)
5801da177e4SLinus Torvalds {
5810029c0deSReshetova, Elena 	if (refcount_dec_and_test(&fi->fib_clntref))
5821da177e4SLinus Torvalds 		free_fib_info(fi);
5831da177e4SLinus Torvalds }
5841da177e4SLinus Torvalds 
58520380731SArnaldo Carvalho de Melo #ifdef CONFIG_PROC_FS
5865c3a0fd7SJoe Perches int __net_init fib_proc_init(struct net *net);
5875c3a0fd7SJoe Perches void __net_exit fib_proc_exit(struct net *net);
588cc8274f5SLi Zefan #else
fib_proc_init(struct net * net)589cc8274f5SLi Zefan static inline int fib_proc_init(struct net *net)
590cc8274f5SLi Zefan {
591cc8274f5SLi Zefan 	return 0;
592cc8274f5SLi Zefan }
fib_proc_exit(struct net * net)593cc8274f5SLi Zefan static inline void fib_proc_exit(struct net *net)
594cc8274f5SLi Zefan {
595cc8274f5SLi Zefan }
59620380731SArnaldo Carvalho de Melo #endif
59720380731SArnaldo Carvalho de Melo 
59850d889b1SDavid Ahern u32 ip_mtu_from_fib_result(struct fib_result *res, __be32 daddr);
59950d889b1SDavid Ahern 
6004724676dSDavid Ahern int ip_valid_fib_dump_req(struct net *net, const struct nlmsghdr *nlh,
6014724676dSDavid Ahern 			  struct fib_dump_filter *filter,
602effe6792SDavid Ahern 			  struct netlink_callback *cb);
603c0a72077SDavid Ahern 
604c0a72077SDavid Ahern int fib_nexthop_info(struct sk_buff *skb, const struct fib_nh_common *nh,
6057bdf4de1SDonald Sharp 		     u8 rt_family, unsigned char *flags, bool skip_oif);
606c0a72077SDavid Ahern int fib_add_nexthop(struct sk_buff *skb, const struct fib_nh_common *nh,
607597aa16cSXiao Liang 		    int nh_weight, u8 rt_family, u32 nh_tclassid);
6081da177e4SLinus Torvalds #endif  /* _NET_FIB_H */
609