2 * SR-IPv6 implementation
5 * David Lebrun <david.lebrun@uclouvain.be>
6 * eBPF support: Mathieu Xhonneux <m.xhonneux@gmail.com>
9 * This program is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU General Public License
11 * as published by the Free Software Foundation; either version
12 * 2 of the License, or (at your option) any later version.
15 #include <linux/types.h>
16 #include <linux/skbuff.h>
17 #include <linux/net.h>
18 #include <linux/module.h>
20 #include <net/lwtunnel.h>
21 #include <net/netevent.h>
22 #include <net/netns/generic.h>
23 #include <net/ip6_fib.h>
24 #include <net/route.h>
26 #include <linux/seg6.h>
27 #include <linux/seg6_local.h>
28 #include <net/addrconf.h>
29 #include <net/ip6_route.h>
30 #include <net/dst_cache.h>
31 #ifdef CONFIG_IPV6_SEG6_HMAC
32 #include <net/seg6_hmac.h>
34 #include <net/seg6_local.h>
35 #include <linux/etherdevice.h>
36 #include <linux/bpf.h>
38 struct seg6_local_lwt
;
40 struct seg6_action_desc
{
43 int (*input
)(struct sk_buff
*skb
, struct seg6_local_lwt
*slwt
);
48 struct bpf_prog
*prog
;
52 struct seg6_local_lwt
{
54 struct ipv6_sr_hdr
*srh
;
60 struct bpf_lwt_prog bpf
;
63 struct seg6_action_desc
*desc
;
66 static struct seg6_local_lwt
*seg6_local_lwtunnel(struct lwtunnel_state
*lwt
)
68 return (struct seg6_local_lwt
*)lwt
->data
;
71 static struct ipv6_sr_hdr
*get_srh(struct sk_buff
*skb
)
73 struct ipv6_sr_hdr
*srh
;
76 if (ipv6_find_hdr(skb
, &srhoff
, IPPROTO_ROUTING
, NULL
, NULL
) < 0)
79 if (!pskb_may_pull(skb
, srhoff
+ sizeof(*srh
)))
82 srh
= (struct ipv6_sr_hdr
*)(skb
->data
+ srhoff
);
84 len
= (srh
->hdrlen
+ 1) << 3;
86 if (!pskb_may_pull(skb
, srhoff
+ len
))
89 if (!seg6_validate_srh(srh
, len
))
95 static struct ipv6_sr_hdr
*get_and_validate_srh(struct sk_buff
*skb
)
97 struct ipv6_sr_hdr
*srh
;
103 if (srh
->segments_left
== 0)
106 #ifdef CONFIG_IPV6_SEG6_HMAC
107 if (!seg6_hmac_validate_skb(skb
))
114 static bool decap_and_validate(struct sk_buff
*skb
, int proto
)
116 struct ipv6_sr_hdr
*srh
;
117 unsigned int off
= 0;
120 if (srh
&& srh
->segments_left
> 0)
123 #ifdef CONFIG_IPV6_SEG6_HMAC
124 if (srh
&& !seg6_hmac_validate_skb(skb
))
128 if (ipv6_find_hdr(skb
, &off
, proto
, NULL
, NULL
) < 0)
131 if (!pskb_pull(skb
, off
))
134 skb_postpull_rcsum(skb
, skb_network_header(skb
), off
);
136 skb_reset_network_header(skb
);
137 skb_reset_transport_header(skb
);
138 skb
->encapsulation
= 0;
143 static void advance_nextseg(struct ipv6_sr_hdr
*srh
, struct in6_addr
*daddr
)
145 struct in6_addr
*addr
;
147 srh
->segments_left
--;
148 addr
= srh
->segments
+ srh
->segments_left
;
152 int seg6_lookup_nexthop(struct sk_buff
*skb
, struct in6_addr
*nhaddr
,
155 struct net
*net
= dev_net(skb
->dev
);
156 struct ipv6hdr
*hdr
= ipv6_hdr(skb
);
157 int flags
= RT6_LOOKUP_F_HAS_SADDR
;
158 struct dst_entry
*dst
= NULL
;
162 fl6
.flowi6_iif
= skb
->dev
->ifindex
;
163 fl6
.daddr
= nhaddr
? *nhaddr
: hdr
->daddr
;
164 fl6
.saddr
= hdr
->saddr
;
165 fl6
.flowlabel
= ip6_flowinfo(hdr
);
166 fl6
.flowi6_mark
= skb
->mark
;
167 fl6
.flowi6_proto
= hdr
->nexthdr
;
170 fl6
.flowi6_flags
= FLOWI_FLAG_KNOWN_NH
;
173 dst
= ip6_route_input_lookup(net
, skb
->dev
, &fl6
, skb
, flags
);
175 struct fib6_table
*table
;
177 table
= fib6_get_table(net
, tbl_id
);
181 rt
= ip6_pol_route(net
, table
, 0, &fl6
, skb
, flags
);
185 if (dst
&& dst
->dev
->flags
& IFF_LOOPBACK
&& !dst
->error
) {
192 rt
= net
->ipv6
.ip6_blk_hole_entry
;
198 skb_dst_set(skb
, dst
);
202 /* regular endpoint function */
203 static int input_action_end(struct sk_buff
*skb
, struct seg6_local_lwt
*slwt
)
205 struct ipv6_sr_hdr
*srh
;
207 srh
= get_and_validate_srh(skb
);
211 advance_nextseg(srh
, &ipv6_hdr(skb
)->daddr
);
213 seg6_lookup_nexthop(skb
, NULL
, 0);
215 return dst_input(skb
);
222 /* regular endpoint, and forward to specified nexthop */
223 static int input_action_end_x(struct sk_buff
*skb
, struct seg6_local_lwt
*slwt
)
225 struct ipv6_sr_hdr
*srh
;
227 srh
= get_and_validate_srh(skb
);
231 advance_nextseg(srh
, &ipv6_hdr(skb
)->daddr
);
233 seg6_lookup_nexthop(skb
, &slwt
->nh6
, 0);
235 return dst_input(skb
);
242 static int input_action_end_t(struct sk_buff
*skb
, struct seg6_local_lwt
*slwt
)
244 struct ipv6_sr_hdr
*srh
;
246 srh
= get_and_validate_srh(skb
);
250 advance_nextseg(srh
, &ipv6_hdr(skb
)->daddr
);
252 seg6_lookup_nexthop(skb
, NULL
, slwt
->table
);
254 return dst_input(skb
);
261 /* decapsulate and forward inner L2 frame on specified interface */
262 static int input_action_end_dx2(struct sk_buff
*skb
,
263 struct seg6_local_lwt
*slwt
)
265 struct net
*net
= dev_net(skb
->dev
);
266 struct net_device
*odev
;
269 if (!decap_and_validate(skb
, NEXTHDR_NONE
))
272 if (!pskb_may_pull(skb
, ETH_HLEN
))
275 skb_reset_mac_header(skb
);
276 eth
= (struct ethhdr
*)skb
->data
;
278 /* To determine the frame's protocol, we assume it is 802.3. This avoids
279 * a call to eth_type_trans(), which is not really relevant for our
282 if (!eth_proto_is_802_3(eth
->h_proto
))
285 odev
= dev_get_by_index_rcu(net
, slwt
->oif
);
289 /* As we accept Ethernet frames, make sure the egress device is of
292 if (odev
->type
!= ARPHRD_ETHER
)
295 if (!(odev
->flags
& IFF_UP
) || !netif_carrier_ok(odev
))
300 if (skb_warn_if_lro(skb
))
303 skb_forward_csum(skb
);
305 if (skb
->len
- ETH_HLEN
> odev
->mtu
)
309 skb
->protocol
= eth
->h_proto
;
311 return dev_queue_xmit(skb
);
318 /* decapsulate and forward to specified nexthop */
319 static int input_action_end_dx6(struct sk_buff
*skb
,
320 struct seg6_local_lwt
*slwt
)
322 struct in6_addr
*nhaddr
= NULL
;
324 /* this function accepts IPv6 encapsulated packets, with either
325 * an SRH with SL=0, or no SRH.
328 if (!decap_and_validate(skb
, IPPROTO_IPV6
))
331 if (!pskb_may_pull(skb
, sizeof(struct ipv6hdr
)))
334 /* The inner packet is not associated to any local interface,
335 * so we do not call netif_rx().
337 * If slwt->nh6 is set to ::, then lookup the nexthop for the
338 * inner packet's DA. Otherwise, use the specified nexthop.
341 if (!ipv6_addr_any(&slwt
->nh6
))
344 seg6_lookup_nexthop(skb
, nhaddr
, 0);
346 return dst_input(skb
);
352 static int input_action_end_dx4(struct sk_buff
*skb
,
353 struct seg6_local_lwt
*slwt
)
359 if (!decap_and_validate(skb
, IPPROTO_IPIP
))
362 if (!pskb_may_pull(skb
, sizeof(struct iphdr
)))
365 skb
->protocol
= htons(ETH_P_IP
);
369 nhaddr
= slwt
->nh4
.s_addr
?: iph
->daddr
;
373 err
= ip_route_input(skb
, nhaddr
, iph
->saddr
, 0, skb
->dev
);
377 return dst_input(skb
);
384 static int input_action_end_dt6(struct sk_buff
*skb
,
385 struct seg6_local_lwt
*slwt
)
387 if (!decap_and_validate(skb
, IPPROTO_IPV6
))
390 if (!pskb_may_pull(skb
, sizeof(struct ipv6hdr
)))
393 seg6_lookup_nexthop(skb
, NULL
, slwt
->table
);
395 return dst_input(skb
);
402 /* push an SRH on top of the current one */
403 static int input_action_end_b6(struct sk_buff
*skb
, struct seg6_local_lwt
*slwt
)
405 struct ipv6_sr_hdr
*srh
;
408 srh
= get_and_validate_srh(skb
);
412 err
= seg6_do_srh_inline(skb
, slwt
->srh
);
416 ipv6_hdr(skb
)->payload_len
= htons(skb
->len
- sizeof(struct ipv6hdr
));
417 skb_set_transport_header(skb
, sizeof(struct ipv6hdr
));
419 seg6_lookup_nexthop(skb
, NULL
, 0);
421 return dst_input(skb
);
428 /* encapsulate within an outer IPv6 header and a specified SRH */
429 static int input_action_end_b6_encap(struct sk_buff
*skb
,
430 struct seg6_local_lwt
*slwt
)
432 struct ipv6_sr_hdr
*srh
;
435 srh
= get_and_validate_srh(skb
);
439 advance_nextseg(srh
, &ipv6_hdr(skb
)->daddr
);
441 skb_reset_inner_headers(skb
);
442 skb
->encapsulation
= 1;
444 err
= seg6_do_srh_encap(skb
, slwt
->srh
, IPPROTO_IPV6
);
448 ipv6_hdr(skb
)->payload_len
= htons(skb
->len
- sizeof(struct ipv6hdr
));
449 skb_set_transport_header(skb
, sizeof(struct ipv6hdr
));
451 seg6_lookup_nexthop(skb
, NULL
, 0);
453 return dst_input(skb
);
460 DEFINE_PER_CPU(struct seg6_bpf_srh_state
, seg6_bpf_srh_states
);
462 bool seg6_bpf_has_valid_srh(struct sk_buff
*skb
)
464 struct seg6_bpf_srh_state
*srh_state
=
465 this_cpu_ptr(&seg6_bpf_srh_states
);
466 struct ipv6_sr_hdr
*srh
= srh_state
->srh
;
468 if (unlikely(srh
== NULL
))
471 if (unlikely(!srh_state
->valid
)) {
472 if ((srh_state
->hdrlen
& 7) != 0)
475 srh
->hdrlen
= (u8
)(srh_state
->hdrlen
>> 3);
476 if (!seg6_validate_srh(srh
, (srh
->hdrlen
+ 1) << 3))
479 srh_state
->valid
= true;
485 static int input_action_end_bpf(struct sk_buff
*skb
,
486 struct seg6_local_lwt
*slwt
)
488 struct seg6_bpf_srh_state
*srh_state
=
489 this_cpu_ptr(&seg6_bpf_srh_states
);
490 struct ipv6_sr_hdr
*srh
;
493 srh
= get_and_validate_srh(skb
);
498 advance_nextseg(srh
, &ipv6_hdr(skb
)->daddr
);
500 /* preempt_disable is needed to protect the per-CPU buffer srh_state,
501 * which is also accessed by the bpf_lwt_seg6_* helpers
504 srh_state
->srh
= srh
;
505 srh_state
->hdrlen
= srh
->hdrlen
<< 3;
506 srh_state
->valid
= true;
509 bpf_compute_data_pointers(skb
);
510 ret
= bpf_prog_run_save_cb(slwt
->bpf
.prog
, skb
);
520 pr_warn_once("bpf-seg6local: Illegal return value %u\n", ret
);
524 if (srh_state
->srh
&& !seg6_bpf_has_valid_srh(skb
))
528 if (ret
!= BPF_REDIRECT
)
529 seg6_lookup_nexthop(skb
, NULL
, 0);
531 return dst_input(skb
);
539 static struct seg6_action_desc seg6_action_table
[] = {
541 .action
= SEG6_LOCAL_ACTION_END
,
543 .input
= input_action_end
,
546 .action
= SEG6_LOCAL_ACTION_END_X
,
547 .attrs
= (1 << SEG6_LOCAL_NH6
),
548 .input
= input_action_end_x
,
551 .action
= SEG6_LOCAL_ACTION_END_T
,
552 .attrs
= (1 << SEG6_LOCAL_TABLE
),
553 .input
= input_action_end_t
,
556 .action
= SEG6_LOCAL_ACTION_END_DX2
,
557 .attrs
= (1 << SEG6_LOCAL_OIF
),
558 .input
= input_action_end_dx2
,
561 .action
= SEG6_LOCAL_ACTION_END_DX6
,
562 .attrs
= (1 << SEG6_LOCAL_NH6
),
563 .input
= input_action_end_dx6
,
566 .action
= SEG6_LOCAL_ACTION_END_DX4
,
567 .attrs
= (1 << SEG6_LOCAL_NH4
),
568 .input
= input_action_end_dx4
,
571 .action
= SEG6_LOCAL_ACTION_END_DT6
,
572 .attrs
= (1 << SEG6_LOCAL_TABLE
),
573 .input
= input_action_end_dt6
,
576 .action
= SEG6_LOCAL_ACTION_END_B6
,
577 .attrs
= (1 << SEG6_LOCAL_SRH
),
578 .input
= input_action_end_b6
,
581 .action
= SEG6_LOCAL_ACTION_END_B6_ENCAP
,
582 .attrs
= (1 << SEG6_LOCAL_SRH
),
583 .input
= input_action_end_b6_encap
,
584 .static_headroom
= sizeof(struct ipv6hdr
),
587 .action
= SEG6_LOCAL_ACTION_END_BPF
,
588 .attrs
= (1 << SEG6_LOCAL_BPF
),
589 .input
= input_action_end_bpf
,
594 static struct seg6_action_desc
*__get_action_desc(int action
)
596 struct seg6_action_desc
*desc
;
599 count
= ARRAY_SIZE(seg6_action_table
);
600 for (i
= 0; i
< count
; i
++) {
601 desc
= &seg6_action_table
[i
];
602 if (desc
->action
== action
)
609 static int seg6_local_input(struct sk_buff
*skb
)
611 struct dst_entry
*orig_dst
= skb_dst(skb
);
612 struct seg6_action_desc
*desc
;
613 struct seg6_local_lwt
*slwt
;
615 if (skb
->protocol
!= htons(ETH_P_IPV6
)) {
620 slwt
= seg6_local_lwtunnel(orig_dst
->lwtstate
);
623 return desc
->input(skb
, slwt
);
626 static const struct nla_policy seg6_local_policy
[SEG6_LOCAL_MAX
+ 1] = {
627 [SEG6_LOCAL_ACTION
] = { .type
= NLA_U32
},
628 [SEG6_LOCAL_SRH
] = { .type
= NLA_BINARY
},
629 [SEG6_LOCAL_TABLE
] = { .type
= NLA_U32
},
630 [SEG6_LOCAL_NH4
] = { .type
= NLA_BINARY
,
631 .len
= sizeof(struct in_addr
) },
632 [SEG6_LOCAL_NH6
] = { .type
= NLA_BINARY
,
633 .len
= sizeof(struct in6_addr
) },
634 [SEG6_LOCAL_IIF
] = { .type
= NLA_U32
},
635 [SEG6_LOCAL_OIF
] = { .type
= NLA_U32
},
636 [SEG6_LOCAL_BPF
] = { .type
= NLA_NESTED
},
639 static int parse_nla_srh(struct nlattr
**attrs
, struct seg6_local_lwt
*slwt
)
641 struct ipv6_sr_hdr
*srh
;
644 srh
= nla_data(attrs
[SEG6_LOCAL_SRH
]);
645 len
= nla_len(attrs
[SEG6_LOCAL_SRH
]);
647 /* SRH must contain at least one segment */
648 if (len
< sizeof(*srh
) + sizeof(struct in6_addr
))
651 if (!seg6_validate_srh(srh
, len
))
654 slwt
->srh
= kmemdup(srh
, len
, GFP_KERNEL
);
658 slwt
->headroom
+= len
;
663 static int put_nla_srh(struct sk_buff
*skb
, struct seg6_local_lwt
*slwt
)
665 struct ipv6_sr_hdr
*srh
;
670 len
= (srh
->hdrlen
+ 1) << 3;
672 nla
= nla_reserve(skb
, SEG6_LOCAL_SRH
, len
);
676 memcpy(nla_data(nla
), srh
, len
);
681 static int cmp_nla_srh(struct seg6_local_lwt
*a
, struct seg6_local_lwt
*b
)
683 int len
= (a
->srh
->hdrlen
+ 1) << 3;
685 if (len
!= ((b
->srh
->hdrlen
+ 1) << 3))
688 return memcmp(a
->srh
, b
->srh
, len
);
691 static int parse_nla_table(struct nlattr
**attrs
, struct seg6_local_lwt
*slwt
)
693 slwt
->table
= nla_get_u32(attrs
[SEG6_LOCAL_TABLE
]);
698 static int put_nla_table(struct sk_buff
*skb
, struct seg6_local_lwt
*slwt
)
700 if (nla_put_u32(skb
, SEG6_LOCAL_TABLE
, slwt
->table
))
706 static int cmp_nla_table(struct seg6_local_lwt
*a
, struct seg6_local_lwt
*b
)
708 if (a
->table
!= b
->table
)
714 static int parse_nla_nh4(struct nlattr
**attrs
, struct seg6_local_lwt
*slwt
)
716 memcpy(&slwt
->nh4
, nla_data(attrs
[SEG6_LOCAL_NH4
]),
717 sizeof(struct in_addr
));
722 static int put_nla_nh4(struct sk_buff
*skb
, struct seg6_local_lwt
*slwt
)
726 nla
= nla_reserve(skb
, SEG6_LOCAL_NH4
, sizeof(struct in_addr
));
730 memcpy(nla_data(nla
), &slwt
->nh4
, sizeof(struct in_addr
));
735 static int cmp_nla_nh4(struct seg6_local_lwt
*a
, struct seg6_local_lwt
*b
)
737 return memcmp(&a
->nh4
, &b
->nh4
, sizeof(struct in_addr
));
740 static int parse_nla_nh6(struct nlattr
**attrs
, struct seg6_local_lwt
*slwt
)
742 memcpy(&slwt
->nh6
, nla_data(attrs
[SEG6_LOCAL_NH6
]),
743 sizeof(struct in6_addr
));
748 static int put_nla_nh6(struct sk_buff
*skb
, struct seg6_local_lwt
*slwt
)
752 nla
= nla_reserve(skb
, SEG6_LOCAL_NH6
, sizeof(struct in6_addr
));
756 memcpy(nla_data(nla
), &slwt
->nh6
, sizeof(struct in6_addr
));
761 static int cmp_nla_nh6(struct seg6_local_lwt
*a
, struct seg6_local_lwt
*b
)
763 return memcmp(&a
->nh6
, &b
->nh6
, sizeof(struct in6_addr
));
766 static int parse_nla_iif(struct nlattr
**attrs
, struct seg6_local_lwt
*slwt
)
768 slwt
->iif
= nla_get_u32(attrs
[SEG6_LOCAL_IIF
]);
773 static int put_nla_iif(struct sk_buff
*skb
, struct seg6_local_lwt
*slwt
)
775 if (nla_put_u32(skb
, SEG6_LOCAL_IIF
, slwt
->iif
))
781 static int cmp_nla_iif(struct seg6_local_lwt
*a
, struct seg6_local_lwt
*b
)
783 if (a
->iif
!= b
->iif
)
789 static int parse_nla_oif(struct nlattr
**attrs
, struct seg6_local_lwt
*slwt
)
791 slwt
->oif
= nla_get_u32(attrs
[SEG6_LOCAL_OIF
]);
796 static int put_nla_oif(struct sk_buff
*skb
, struct seg6_local_lwt
*slwt
)
798 if (nla_put_u32(skb
, SEG6_LOCAL_OIF
, slwt
->oif
))
804 static int cmp_nla_oif(struct seg6_local_lwt
*a
, struct seg6_local_lwt
*b
)
806 if (a
->oif
!= b
->oif
)
812 #define MAX_PROG_NAME 256
813 static const struct nla_policy bpf_prog_policy
[SEG6_LOCAL_BPF_PROG_MAX
+ 1] = {
814 [SEG6_LOCAL_BPF_PROG
] = { .type
= NLA_U32
, },
815 [SEG6_LOCAL_BPF_PROG_NAME
] = { .type
= NLA_NUL_STRING
,
816 .len
= MAX_PROG_NAME
},
819 static int parse_nla_bpf(struct nlattr
**attrs
, struct seg6_local_lwt
*slwt
)
821 struct nlattr
*tb
[SEG6_LOCAL_BPF_PROG_MAX
+ 1];
826 ret
= nla_parse_nested(tb
, SEG6_LOCAL_BPF_PROG_MAX
,
827 attrs
[SEG6_LOCAL_BPF
], bpf_prog_policy
, NULL
);
831 if (!tb
[SEG6_LOCAL_BPF_PROG
] || !tb
[SEG6_LOCAL_BPF_PROG_NAME
])
834 slwt
->bpf
.name
= nla_memdup(tb
[SEG6_LOCAL_BPF_PROG_NAME
], GFP_KERNEL
);
838 fd
= nla_get_u32(tb
[SEG6_LOCAL_BPF_PROG
]);
839 p
= bpf_prog_get_type(fd
, BPF_PROG_TYPE_LWT_SEG6LOCAL
);
841 kfree(slwt
->bpf
.name
);
849 static int put_nla_bpf(struct sk_buff
*skb
, struct seg6_local_lwt
*slwt
)
856 nest
= nla_nest_start(skb
, SEG6_LOCAL_BPF
);
860 if (nla_put_u32(skb
, SEG6_LOCAL_BPF_PROG
, slwt
->bpf
.prog
->aux
->id
))
863 if (slwt
->bpf
.name
&&
864 nla_put_string(skb
, SEG6_LOCAL_BPF_PROG_NAME
, slwt
->bpf
.name
))
867 return nla_nest_end(skb
, nest
);
870 static int cmp_nla_bpf(struct seg6_local_lwt
*a
, struct seg6_local_lwt
*b
)
872 if (!a
->bpf
.name
&& !b
->bpf
.name
)
875 if (!a
->bpf
.name
|| !b
->bpf
.name
)
878 return strcmp(a
->bpf
.name
, b
->bpf
.name
);
881 struct seg6_action_param
{
882 int (*parse
)(struct nlattr
**attrs
, struct seg6_local_lwt
*slwt
);
883 int (*put
)(struct sk_buff
*skb
, struct seg6_local_lwt
*slwt
);
884 int (*cmp
)(struct seg6_local_lwt
*a
, struct seg6_local_lwt
*b
);
887 static struct seg6_action_param seg6_action_params
[SEG6_LOCAL_MAX
+ 1] = {
888 [SEG6_LOCAL_SRH
] = { .parse
= parse_nla_srh
,
890 .cmp
= cmp_nla_srh
},
892 [SEG6_LOCAL_TABLE
] = { .parse
= parse_nla_table
,
893 .put
= put_nla_table
,
894 .cmp
= cmp_nla_table
},
896 [SEG6_LOCAL_NH4
] = { .parse
= parse_nla_nh4
,
898 .cmp
= cmp_nla_nh4
},
900 [SEG6_LOCAL_NH6
] = { .parse
= parse_nla_nh6
,
902 .cmp
= cmp_nla_nh6
},
904 [SEG6_LOCAL_IIF
] = { .parse
= parse_nla_iif
,
906 .cmp
= cmp_nla_iif
},
908 [SEG6_LOCAL_OIF
] = { .parse
= parse_nla_oif
,
910 .cmp
= cmp_nla_oif
},
912 [SEG6_LOCAL_BPF
] = { .parse
= parse_nla_bpf
,
914 .cmp
= cmp_nla_bpf
},
918 static int parse_nla_action(struct nlattr
**attrs
, struct seg6_local_lwt
*slwt
)
920 struct seg6_action_param
*param
;
921 struct seg6_action_desc
*desc
;
924 desc
= __get_action_desc(slwt
->action
);
932 slwt
->headroom
+= desc
->static_headroom
;
934 for (i
= 0; i
< SEG6_LOCAL_MAX
+ 1; i
++) {
935 if (desc
->attrs
& (1 << i
)) {
939 param
= &seg6_action_params
[i
];
941 err
= param
->parse(attrs
, slwt
);
950 static int seg6_local_build_state(struct nlattr
*nla
, unsigned int family
,
951 const void *cfg
, struct lwtunnel_state
**ts
,
952 struct netlink_ext_ack
*extack
)
954 struct nlattr
*tb
[SEG6_LOCAL_MAX
+ 1];
955 struct lwtunnel_state
*newts
;
956 struct seg6_local_lwt
*slwt
;
959 if (family
!= AF_INET6
)
962 err
= nla_parse_nested(tb
, SEG6_LOCAL_MAX
, nla
, seg6_local_policy
,
968 if (!tb
[SEG6_LOCAL_ACTION
])
971 newts
= lwtunnel_state_alloc(sizeof(*slwt
));
975 slwt
= seg6_local_lwtunnel(newts
);
976 slwt
->action
= nla_get_u32(tb
[SEG6_LOCAL_ACTION
]);
978 err
= parse_nla_action(tb
, slwt
);
982 newts
->type
= LWTUNNEL_ENCAP_SEG6_LOCAL
;
983 newts
->flags
= LWTUNNEL_STATE_INPUT_REDIRECT
;
984 newts
->headroom
= slwt
->headroom
;
996 static void seg6_local_destroy_state(struct lwtunnel_state
*lwt
)
998 struct seg6_local_lwt
*slwt
= seg6_local_lwtunnel(lwt
);
1002 if (slwt
->desc
->attrs
& (1 << SEG6_LOCAL_BPF
)) {
1003 kfree(slwt
->bpf
.name
);
1004 bpf_prog_put(slwt
->bpf
.prog
);
1010 static int seg6_local_fill_encap(struct sk_buff
*skb
,
1011 struct lwtunnel_state
*lwt
)
1013 struct seg6_local_lwt
*slwt
= seg6_local_lwtunnel(lwt
);
1014 struct seg6_action_param
*param
;
1017 if (nla_put_u32(skb
, SEG6_LOCAL_ACTION
, slwt
->action
))
1020 for (i
= 0; i
< SEG6_LOCAL_MAX
+ 1; i
++) {
1021 if (slwt
->desc
->attrs
& (1 << i
)) {
1022 param
= &seg6_action_params
[i
];
1023 err
= param
->put(skb
, slwt
);
1032 static int seg6_local_get_encap_size(struct lwtunnel_state
*lwt
)
1034 struct seg6_local_lwt
*slwt
= seg6_local_lwtunnel(lwt
);
1035 unsigned long attrs
;
1038 nlsize
= nla_total_size(4); /* action */
1040 attrs
= slwt
->desc
->attrs
;
1042 if (attrs
& (1 << SEG6_LOCAL_SRH
))
1043 nlsize
+= nla_total_size((slwt
->srh
->hdrlen
+ 1) << 3);
1045 if (attrs
& (1 << SEG6_LOCAL_TABLE
))
1046 nlsize
+= nla_total_size(4);
1048 if (attrs
& (1 << SEG6_LOCAL_NH4
))
1049 nlsize
+= nla_total_size(4);
1051 if (attrs
& (1 << SEG6_LOCAL_NH6
))
1052 nlsize
+= nla_total_size(16);
1054 if (attrs
& (1 << SEG6_LOCAL_IIF
))
1055 nlsize
+= nla_total_size(4);
1057 if (attrs
& (1 << SEG6_LOCAL_OIF
))
1058 nlsize
+= nla_total_size(4);
1060 if (attrs
& (1 << SEG6_LOCAL_BPF
))
1061 nlsize
+= nla_total_size(sizeof(struct nlattr
)) +
1062 nla_total_size(MAX_PROG_NAME
) +
1068 static int seg6_local_cmp_encap(struct lwtunnel_state
*a
,
1069 struct lwtunnel_state
*b
)
1071 struct seg6_local_lwt
*slwt_a
, *slwt_b
;
1072 struct seg6_action_param
*param
;
1075 slwt_a
= seg6_local_lwtunnel(a
);
1076 slwt_b
= seg6_local_lwtunnel(b
);
1078 if (slwt_a
->action
!= slwt_b
->action
)
1081 if (slwt_a
->desc
->attrs
!= slwt_b
->desc
->attrs
)
1084 for (i
= 0; i
< SEG6_LOCAL_MAX
+ 1; i
++) {
1085 if (slwt_a
->desc
->attrs
& (1 << i
)) {
1086 param
= &seg6_action_params
[i
];
1087 if (param
->cmp(slwt_a
, slwt_b
))
1095 static const struct lwtunnel_encap_ops seg6_local_ops
= {
1096 .build_state
= seg6_local_build_state
,
1097 .destroy_state
= seg6_local_destroy_state
,
1098 .input
= seg6_local_input
,
1099 .fill_encap
= seg6_local_fill_encap
,
1100 .get_encap_size
= seg6_local_get_encap_size
,
1101 .cmp_encap
= seg6_local_cmp_encap
,
1102 .owner
= THIS_MODULE
,
1105 int __init
seg6_local_init(void)
1107 return lwtunnel_encap_add_ops(&seg6_local_ops
,
1108 LWTUNNEL_ENCAP_SEG6_LOCAL
);
1111 void seg6_local_exit(void)
1113 lwtunnel_encap_del_ops(&seg6_local_ops
, LWTUNNEL_ENCAP_SEG6_LOCAL
);