1 /* BGP FlowSpec for packet handling
3 * Copyright (C) 2017 ChinaTelecom SDN Group
4 * Copyright (C) 2018 6WIND
6 * FRRouting is free software; you can redistribute it and/or modify it
7 * under the terms of the GNU General Public License as published by the
8 * Free Software Foundation; either version 2, or (at your option) any
11 * FRRouting is distributed in the hope that it will be useful, but
12 * WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * General Public License for more details.
16 * You should have received a copy of the GNU General Public License along
17 * with this program; see the file COPYING; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
25 #include "lib_errors.h"
27 #include "bgpd/bgpd.h"
28 #include "bgpd/bgp_route.h"
29 #include "bgpd/bgp_flowspec.h"
30 #include "bgpd/bgp_flowspec_util.h"
31 #include "bgpd/bgp_flowspec_private.h"
32 #include "bgpd/bgp_ecommunity.h"
33 #include "bgpd/bgp_debug.h"
34 #include "bgpd/bgp_errors.h"
36 static int bgp_fs_nlri_validate(uint8_t *nlri_content
, uint32_t len
,
41 int ret
= 0, error
= 0;
43 while (offset
< len
-1) {
44 type
= nlri_content
[offset
];
47 case FLOWSPEC_DEST_PREFIX
:
48 case FLOWSPEC_SRC_PREFIX
:
49 ret
= bgp_flowspec_ip_address(
50 BGP_FLOWSPEC_VALIDATE_ONLY
,
51 nlri_content
+ offset
,
52 len
- offset
, NULL
, &error
,
55 case FLOWSPEC_FLOW_LABEL
:
58 ret
= bgp_flowspec_op_decode(BGP_FLOWSPEC_VALIDATE_ONLY
,
59 nlri_content
+ offset
,
60 len
- offset
, NULL
, &error
);
62 case FLOWSPEC_IP_PROTOCOL
:
64 case FLOWSPEC_DEST_PORT
:
65 case FLOWSPEC_SRC_PORT
:
66 case FLOWSPEC_ICMP_TYPE
:
67 case FLOWSPEC_ICMP_CODE
:
68 ret
= bgp_flowspec_op_decode(BGP_FLOWSPEC_VALIDATE_ONLY
,
69 nlri_content
+ offset
,
70 len
- offset
, NULL
, &error
);
72 case FLOWSPEC_TCP_FLAGS
:
73 case FLOWSPEC_FRAGMENT
:
74 ret
= bgp_flowspec_bitmask_decode(
75 BGP_FLOWSPEC_VALIDATE_ONLY
,
76 nlri_content
+ offset
,
77 len
- offset
, NULL
, &error
);
79 case FLOWSPEC_PKT_LEN
:
81 ret
= bgp_flowspec_op_decode(
82 BGP_FLOWSPEC_VALIDATE_ONLY
,
83 nlri_content
+ offset
,
84 len
- offset
, NULL
, &error
);
97 int bgp_nlri_parse_flowspec(struct peer
*peer
, struct attr
*attr
,
98 struct bgp_nlri
*packet
, int withdraw
)
108 /* Start processing the NLRI - there may be multiple in the MP_REACH */
110 lim
= pnt
+ packet
->length
;
114 if (packet
->length
>= FLOWSPEC_NLRI_SIZELIMIT_EXTENDED
) {
115 flog_err(EC_BGP_FLOWSPEC_PACKET
,
116 "BGP flowspec nlri length maximum reached (%u)",
118 return BGP_NLRI_PARSE_ERROR_FLOWSPEC_NLRI_SIZELIMIT
;
121 for (; pnt
< lim
; pnt
+= psize
) {
122 /* Clear prefix structure. */
123 memset(&p
, 0, sizeof(p
));
125 /* All FlowSpec NLRI begin with length. */
127 return BGP_NLRI_PARSE_ERROR_PACKET_OVERFLOW
;
130 if (psize
>= FLOWSPEC_NLRI_SIZELIMIT
) {
135 /* When packet overflow occur return immediately. */
136 if (pnt
+ psize
> lim
) {
138 EC_BGP_FLOWSPEC_PACKET
,
139 "Flowspec NLRI length inconsistent ( size %u seen)",
141 return BGP_NLRI_PARSE_ERROR_PACKET_OVERFLOW
;
143 if (bgp_fs_nlri_validate(pnt
, psize
, afi
) < 0) {
145 EC_BGP_FLOWSPEC_PACKET
,
146 "Bad flowspec format or NLRI options not supported");
147 return BGP_NLRI_PARSE_ERROR_FLOWSPEC_BAD_FORMAT
;
149 p
.family
= AF_FLOWSPEC
;
151 /* Flowspec encoding is in bytes */
152 p
.u
.prefix_flowspec
.prefixlen
= psize
;
153 p
.u
.prefix_flowspec
.family
= afi2family(afi
);
154 temp
= XCALLOC(MTYPE_TMP
, psize
);
155 memcpy(temp
, pnt
, psize
);
156 p
.u
.prefix_flowspec
.ptr
= (uintptr_t) temp
;
158 if (BGP_DEBUG(flowspec
, FLOWSPEC
)) {
159 char return_string
[BGP_FLOWSPEC_NLRI_STRING_MAX
];
160 char local_string
[BGP_FLOWSPEC_NLRI_STRING_MAX
*2+16];
161 char ec_string
[BGP_FLOWSPEC_NLRI_STRING_MAX
];
164 bgp_fs_nlri_get_string((unsigned char *)
165 p
.u
.prefix_flowspec
.ptr
,
166 p
.u
.prefix_flowspec
.prefixlen
,
168 NLRI_STRING_FORMAT_MIN
, NULL
,
170 snprintf(ec_string
, sizeof(ec_string
),
172 if (attr
&& bgp_attr_get_ecommunity(attr
)) {
173 s
= ecommunity_ecom2str(
174 bgp_attr_get_ecommunity(attr
),
175 ECOMMUNITY_FORMAT_ROUTE_MAP
, 0);
176 snprintf(ec_string
, sizeof(ec_string
),
178 s
== NULL
? "none" : s
);
181 ecommunity_strfree(&s
);
183 snprintf(local_string
, sizeof(local_string
),
184 "FS Rx %s %s %s %s", withdraw
?
186 afi2str(afi
), return_string
,
187 attr
!= NULL
? ec_string
: "");
188 zlog_info("%s", local_string
);
190 /* Process the route. */
192 bgp_update(peer
, &p
, 0, attr
, afi
, safi
,
193 ZEBRA_ROUTE_BGP
, BGP_ROUTE_NORMAL
, NULL
,
196 bgp_withdraw(peer
, &p
, 0, attr
, afi
, safi
,
197 ZEBRA_ROUTE_BGP
, BGP_ROUTE_NORMAL
, NULL
,
200 return BGP_NLRI_PARSE_OK
;