blob: 2567941a2f32ff4896380296463462e43c5d760f [file] [log] [blame]
David Lebrun1ababeb2016-11-08 14:57:39 +01001/*
2 * SR-IPv6 implementation
3 *
4 * Author:
5 * David Lebrun <david.lebrun@uclouvain.be>
6 *
7 *
8 * This program is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU General Public License
10 * as published by the Free Software Foundation; either version
11 * 2 of the License, or (at your option) any later version.
12 */
13
14#ifndef _NET_SEG6_H
15#define _NET_SEG6_H
16
David Lebrun915d7e52016-11-08 14:57:40 +010017#include <linux/net.h>
18#include <linux/ipv6.h>
David Lebrun6c8702c2016-11-08 14:57:41 +010019#include <net/lwtunnel.h>
20#include <linux/seg6.h>
NeilBrown0eb71a92018-06-18 12:52:50 +100021#include <linux/rhashtable-types.h>
David Lebrun915d7e52016-11-08 14:57:40 +010022
David Lebrun1ababeb2016-11-08 14:57:39 +010023static inline void update_csum_diff4(struct sk_buff *skb, __be32 from,
24 __be32 to)
25{
26 __be32 diff[] = { ~from, to };
27
28 skb->csum = ~csum_partial((char *)diff, sizeof(diff), ~skb->csum);
29}
30
31static inline void update_csum_diff16(struct sk_buff *skb, __be32 *from,
32 __be32 *to)
33{
34 __be32 diff[] = {
35 ~from[0], ~from[1], ~from[2], ~from[3],
36 to[0], to[1], to[2], to[3],
37 };
38
39 skb->csum = ~csum_partial((char *)diff, sizeof(diff), ~skb->csum);
40}
41
David Lebrun915d7e52016-11-08 14:57:40 +010042struct seg6_pernet_data {
43 struct mutex lock;
44 struct in6_addr __rcu *tun_src;
David Lebrunbf355b82016-11-08 14:57:42 +010045#ifdef CONFIG_IPV6_SEG6_HMAC
46 struct rhashtable hmac_infos;
47#endif
David Lebrun915d7e52016-11-08 14:57:40 +010048};
49
50static inline struct seg6_pernet_data *seg6_pernet(struct net *net)
51{
Mathieu Xhonneux63526e12018-05-20 14:58:12 +010052#if IS_ENABLED(CONFIG_IPV6)
David Lebrun915d7e52016-11-08 14:57:40 +010053 return net->ipv6.seg6_data;
Mathieu Xhonneux63526e12018-05-20 14:58:12 +010054#else
55 return NULL;
56#endif
David Lebrun915d7e52016-11-08 14:57:40 +010057}
58
59extern int seg6_init(void);
60extern void seg6_exit(void);
David Lebrun6c8702c2016-11-08 14:57:41 +010061extern int seg6_iptunnel_init(void);
62extern void seg6_iptunnel_exit(void);
David Lebrund1df6fd2017-08-05 12:38:26 +020063extern int seg6_local_init(void);
64extern void seg6_local_exit(void);
David Lebrun6c8702c2016-11-08 14:57:41 +010065
66extern bool seg6_validate_srh(struct ipv6_sr_hdr *srh, int len);
David Lebrun32d99d02017-08-25 09:56:44 +020067extern int seg6_do_srh_encap(struct sk_buff *skb, struct ipv6_sr_hdr *osrh,
68 int proto);
David Lebrunb04c80d2017-08-05 12:38:25 +020069extern int seg6_do_srh_inline(struct sk_buff *skb, struct ipv6_sr_hdr *osrh);
Mathieu Xhonneux1c1e7612018-05-20 14:58:13 +010070extern int seg6_lookup_nexthop(struct sk_buff *skb, struct in6_addr *nhaddr,
71 u32 tbl_id);
David Lebrun1ababeb2016-11-08 14:57:39 +010072#endif