blob: 271a21a0f48bf369acbbd7b7b0a8a37dd9bee031 [file] [log] [blame]
paul718e3742002-12-13 20:15:29 +00001/* BGP packet management routine.
2 Copyright (C) 1999 Kunihiro Ishiguro
3
4This file is part of GNU Zebra.
5
6GNU Zebra is free software; you can redistribute it and/or modify it
7under the terms of the GNU General Public License as published by the
8Free Software Foundation; either version 2, or (at your option) any
9later version.
10
11GNU Zebra is distributed in the hope that it will be useful, but
12WITHOUT ANY WARRANTY; without even the implied warranty of
13MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14General Public License for more details.
15
16You should have received a copy of the GNU General Public License
17along with GNU Zebra; see the file COPYING. If not, write to the Free
18Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
1902111-1307, USA. */
20
21#include <zebra.h>
22
23#include "thread.h"
24#include "stream.h"
25#include "network.h"
26#include "prefix.h"
27#include "command.h"
28#include "log.h"
29#include "memory.h"
30#include "sockunion.h" /* for inet_ntop () */
31#include "linklist.h"
32#include "plist.h"
33
34#include "bgpd/bgpd.h"
35#include "bgpd/bgp_table.h"
36#include "bgpd/bgp_dump.h"
37#include "bgpd/bgp_attr.h"
38#include "bgpd/bgp_debug.h"
39#include "bgpd/bgp_fsm.h"
40#include "bgpd/bgp_route.h"
41#include "bgpd/bgp_packet.h"
42#include "bgpd/bgp_open.h"
43#include "bgpd/bgp_aspath.h"
44#include "bgpd/bgp_community.h"
45#include "bgpd/bgp_ecommunity.h"
46#include "bgpd/bgp_network.h"
47#include "bgpd/bgp_mplsvpn.h"
48#include "bgpd/bgp_advertise.h"
hasso93406d82005-02-02 14:40:33 +000049#include "bgpd/bgp_vty.h"
paul718e3742002-12-13 20:15:29 +000050
51int stream_put_prefix (struct stream *, struct prefix *);
52
53/* Set up BGP packet marker and packet type. */
54static int
55bgp_packet_set_marker (struct stream *s, u_char type)
56{
57 int i;
58
59 /* Fill in marker. */
60 for (i = 0; i < BGP_MARKER_SIZE; i++)
61 stream_putc (s, 0xff);
62
63 /* Dummy total length. This field is should be filled in later on. */
64 stream_putw (s, 0);
65
66 /* BGP packet type. */
67 stream_putc (s, type);
68
69 /* Return current stream size. */
paul9985f832005-02-09 15:51:56 +000070 return stream_get_endp (s);
paul718e3742002-12-13 20:15:29 +000071}
72
73/* Set BGP packet header size entry. If size is zero then use current
74 stream size. */
75static int
76bgp_packet_set_size (struct stream *s)
77{
78 int cp;
79
80 /* Preserve current pointer. */
paul9985f832005-02-09 15:51:56 +000081 cp = stream_get_endp (s);
82 stream_putw_at (s, BGP_MARKER_SIZE, cp);
paul718e3742002-12-13 20:15:29 +000083
84 return cp;
85}
86
87/* Add new packet to the peer. */
paul94f2b392005-06-28 12:44:16 +000088static void
paul718e3742002-12-13 20:15:29 +000089bgp_packet_add (struct peer *peer, struct stream *s)
90{
91 /* Add packet to the end of list. */
92 stream_fifo_push (peer->obuf, s);
93}
94
95/* Free first packet. */
paul94f2b392005-06-28 12:44:16 +000096static void
paul718e3742002-12-13 20:15:29 +000097bgp_packet_delete (struct peer *peer)
98{
99 stream_free (stream_fifo_pop (peer->obuf));
100}
101
paul718e3742002-12-13 20:15:29 +0000102/* Check file descriptor whether connect is established. */
103static void
104bgp_connect_check (struct peer *peer)
105{
106 int status;
paul5228ad22004-06-04 17:58:18 +0000107 socklen_t slen;
paul718e3742002-12-13 20:15:29 +0000108 int ret;
109
110 /* Anyway I have to reset read and write thread. */
111 BGP_READ_OFF (peer->t_read);
112 BGP_WRITE_OFF (peer->t_write);
113
114 /* Check file descriptor. */
115 slen = sizeof (status);
pauleb821182004-05-01 08:44:08 +0000116 ret = getsockopt(peer->fd, SOL_SOCKET, SO_ERROR, (void *) &status, &slen);
paul718e3742002-12-13 20:15:29 +0000117
118 /* If getsockopt is fail, this is fatal error. */
119 if (ret < 0)
120 {
121 zlog (peer->log, LOG_INFO, "can't get sockopt for nonblocking connect");
122 BGP_EVENT_ADD (peer, TCP_fatal_error);
123 return;
124 }
125
126 /* When status is 0 then TCP connection is established. */
127 if (status == 0)
128 {
129 BGP_EVENT_ADD (peer, TCP_connection_open);
130 }
131 else
132 {
133 if (BGP_DEBUG (events, EVENTS))
ajs6b514742004-12-08 21:03:23 +0000134 plog_debug (peer->log, "%s [Event] Connect failed (%s)",
ajs6099b3b2004-11-20 02:06:59 +0000135 peer->host, safe_strerror (errno));
paul718e3742002-12-13 20:15:29 +0000136 BGP_EVENT_ADD (peer, TCP_connection_open_failed);
137 }
138}
139
140/* Make BGP update packet. */
paul94f2b392005-06-28 12:44:16 +0000141static struct stream *
paul718e3742002-12-13 20:15:29 +0000142bgp_update_packet (struct peer *peer, afi_t afi, safi_t safi)
143{
144 struct stream *s;
145 struct bgp_adj_out *adj;
146 struct bgp_advertise *adv;
147 struct stream *packet;
148 struct bgp_node *rn = NULL;
149 struct bgp_info *binfo = NULL;
150 bgp_size_t total_attr_len = 0;
151 unsigned long pos;
152 char buf[BUFSIZ];
paul718e3742002-12-13 20:15:29 +0000153
154 s = peer->work;
155 stream_reset (s);
156
157 adv = FIFO_HEAD (&peer->sync[afi][safi]->update);
158
159 while (adv)
160 {
Paul Jakmaed3ebfa2006-10-15 23:50:16 +0000161 assert (adv->rn);
162 rn = adv->rn;
paul718e3742002-12-13 20:15:29 +0000163 adj = adv->adj;
164 if (adv->binfo)
165 binfo = adv->binfo;
paul718e3742002-12-13 20:15:29 +0000166
167 /* When remaining space can't include NLRI and it's length. */
Paul Jakmaed3ebfa2006-10-15 23:50:16 +0000168 if (STREAM_REMAIN (s) <= BGP_NLRI_LENGTH + PSIZE (rn->p.prefixlen))
paul718e3742002-12-13 20:15:29 +0000169 break;
170
171 /* If packet is empty, set attribute. */
172 if (stream_empty (s))
173 {
Paul Jakmaa3b6ea52006-05-04 07:52:12 +0000174 struct prefix_rd *prd = NULL;
175 u_char *tag = NULL;
Paul Jakmaed3ebfa2006-10-15 23:50:16 +0000176 struct peer *from = NULL;
Paul Jakmaa3b6ea52006-05-04 07:52:12 +0000177
178 if (rn->prn)
179 prd = (struct prefix_rd *) &rn->prn->p;
Paul Jakmafb982c22007-05-04 20:15:47 +0000180 if (binfo && binfo->extra)
Paul Jakmaed3ebfa2006-10-15 23:50:16 +0000181 {
Paul Jakmafb982c22007-05-04 20:15:47 +0000182 tag = binfo->extra->tag;
Paul Jakmaed3ebfa2006-10-15 23:50:16 +0000183 from = binfo->peer;
184 }
Paul Jakmaa3b6ea52006-05-04 07:52:12 +0000185
paul718e3742002-12-13 20:15:29 +0000186 bgp_packet_set_marker (s, BGP_MSG_UPDATE);
187 stream_putw (s, 0);
paul9985f832005-02-09 15:51:56 +0000188 pos = stream_get_endp (s);
paul718e3742002-12-13 20:15:29 +0000189 stream_putw (s, 0);
paul5228ad22004-06-04 17:58:18 +0000190 total_attr_len = bgp_packet_attribute (NULL, peer, s,
191 adv->baa->attr,
192 &rn->p, afi, safi,
Paul Jakmaed3ebfa2006-10-15 23:50:16 +0000193 from, prd, tag);
paul718e3742002-12-13 20:15:29 +0000194 stream_putw_at (s, pos, total_attr_len);
195 }
196
197 if (afi == AFI_IP && safi == SAFI_UNICAST)
198 stream_put_prefix (s, &rn->p);
199
200 if (BGP_DEBUG (update, UPDATE_OUT))
ajs6b514742004-12-08 21:03:23 +0000201 zlog (peer->log, LOG_DEBUG, "%s send UPDATE %s/%d",
paul718e3742002-12-13 20:15:29 +0000202 peer->host,
203 inet_ntop (rn->p.family, &(rn->p.u.prefix), buf, BUFSIZ),
204 rn->p.prefixlen);
205
206 /* Synchnorize attribute. */
207 if (adj->attr)
208 bgp_attr_unintern (adj->attr);
209 else
210 peer->scount[afi][safi]++;
211
212 adj->attr = bgp_attr_intern (adv->baa->attr);
213
214 adv = bgp_advertise_clean (peer, adj, afi, safi);
215
216 if (! (afi == AFI_IP && safi == SAFI_UNICAST))
217 break;
218 }
219
220 if (! stream_empty (s))
221 {
222 bgp_packet_set_size (s);
paule83e2082005-05-19 02:12:25 +0000223 packet = stream_dup (s);
paul718e3742002-12-13 20:15:29 +0000224 bgp_packet_add (peer, packet);
pauleb821182004-05-01 08:44:08 +0000225 BGP_WRITE_ON (peer->t_write, bgp_write, peer->fd);
paul718e3742002-12-13 20:15:29 +0000226 stream_reset (s);
227 return packet;
228 }
229 return NULL;
hasso93406d82005-02-02 14:40:33 +0000230}
paul718e3742002-12-13 20:15:29 +0000231
paul94f2b392005-06-28 12:44:16 +0000232static struct stream *
hasso93406d82005-02-02 14:40:33 +0000233bgp_update_packet_eor (struct peer *peer, afi_t afi, safi_t safi)
234{
235 struct stream *s;
236 struct stream *packet;
237
Paul Jakma750e8142008-07-22 21:11:48 +0000238 if (DISABLE_BGP_ANNOUNCE)
239 return NULL;
hasso93406d82005-02-02 14:40:33 +0000240
241 if (BGP_DEBUG (normal, NORMAL))
242 zlog_debug ("send End-of-RIB for %s to %s", afi_safi_print (afi, safi), peer->host);
243
244 s = stream_new (BGP_MAX_PACKET_SIZE);
245
246 /* Make BGP update packet. */
247 bgp_packet_set_marker (s, BGP_MSG_UPDATE);
248
249 /* Unfeasible Routes Length */
250 stream_putw (s, 0);
251
252 if (afi == AFI_IP && safi == SAFI_UNICAST)
253 {
254 /* Total Path Attribute Length */
255 stream_putw (s, 0);
256 }
257 else
258 {
259 /* Total Path Attribute Length */
260 stream_putw (s, 6);
261 stream_putc (s, BGP_ATTR_FLAG_OPTIONAL);
262 stream_putc (s, BGP_ATTR_MP_UNREACH_NLRI);
263 stream_putc (s, 3);
264 stream_putw (s, afi);
265 stream_putc (s, safi);
266 }
267
268 bgp_packet_set_size (s);
paule83e2082005-05-19 02:12:25 +0000269 packet = stream_dup (s);
hasso93406d82005-02-02 14:40:33 +0000270 bgp_packet_add (peer, packet);
271 stream_free (s);
272 return packet;
paul718e3742002-12-13 20:15:29 +0000273}
274
275/* Make BGP withdraw packet. */
paul94f2b392005-06-28 12:44:16 +0000276static struct stream *
paul718e3742002-12-13 20:15:29 +0000277bgp_withdraw_packet (struct peer *peer, afi_t afi, safi_t safi)
278{
279 struct stream *s;
280 struct stream *packet;
281 struct bgp_adj_out *adj;
282 struct bgp_advertise *adv;
283 struct bgp_node *rn;
284 unsigned long pos;
285 bgp_size_t unfeasible_len;
286 bgp_size_t total_attr_len;
287 char buf[BUFSIZ];
paul718e3742002-12-13 20:15:29 +0000288
289 s = peer->work;
290 stream_reset (s);
291
292 while ((adv = FIFO_HEAD (&peer->sync[afi][safi]->withdraw)) != NULL)
293 {
Paul Jakmaed3ebfa2006-10-15 23:50:16 +0000294 assert (adv->rn);
paul718e3742002-12-13 20:15:29 +0000295 adj = adv->adj;
296 rn = adv->rn;
paul718e3742002-12-13 20:15:29 +0000297
298 if (STREAM_REMAIN (s)
hasso4372df72004-05-20 10:20:02 +0000299 < (BGP_NLRI_LENGTH + BGP_TOTAL_ATTR_LEN + PSIZE (rn->p.prefixlen)))
paul718e3742002-12-13 20:15:29 +0000300 break;
301
302 if (stream_empty (s))
303 {
304 bgp_packet_set_marker (s, BGP_MSG_UPDATE);
305 stream_putw (s, 0);
306 }
307
308 if (afi == AFI_IP && safi == SAFI_UNICAST)
309 stream_put_prefix (s, &rn->p);
310 else
311 {
Paul Jakmaa3b6ea52006-05-04 07:52:12 +0000312 struct prefix_rd *prd = NULL;
313
314 if (rn->prn)
315 prd = (struct prefix_rd *) &rn->prn->p;
paul9985f832005-02-09 15:51:56 +0000316 pos = stream_get_endp (s);
paul718e3742002-12-13 20:15:29 +0000317 stream_putw (s, 0);
318 total_attr_len
319 = bgp_packet_withdraw (peer, s, &rn->p, afi, safi, prd, NULL);
320
321 /* Set total path attribute length. */
322 stream_putw_at (s, pos, total_attr_len);
323 }
324
325 if (BGP_DEBUG (update, UPDATE_OUT))
ajs6b514742004-12-08 21:03:23 +0000326 zlog (peer->log, LOG_DEBUG, "%s send UPDATE %s/%d -- unreachable",
paul718e3742002-12-13 20:15:29 +0000327 peer->host,
328 inet_ntop (rn->p.family, &(rn->p.u.prefix), buf, BUFSIZ),
329 rn->p.prefixlen);
330
331 peer->scount[afi][safi]--;
332
333 bgp_adj_out_remove (rn, adj, peer, afi, safi);
334 bgp_unlock_node (rn);
335
336 if (! (afi == AFI_IP && safi == SAFI_UNICAST))
337 break;
338 }
339
340 if (! stream_empty (s))
341 {
342 if (afi == AFI_IP && safi == SAFI_UNICAST)
343 {
344 unfeasible_len
paul9985f832005-02-09 15:51:56 +0000345 = stream_get_endp (s) - BGP_HEADER_SIZE - BGP_UNFEASIBLE_LEN;
paul718e3742002-12-13 20:15:29 +0000346 stream_putw_at (s, BGP_HEADER_SIZE, unfeasible_len);
347 stream_putw (s, 0);
348 }
349 bgp_packet_set_size (s);
paule83e2082005-05-19 02:12:25 +0000350 packet = stream_dup (s);
paul718e3742002-12-13 20:15:29 +0000351 bgp_packet_add (peer, packet);
352 stream_reset (s);
353 return packet;
354 }
355
356 return NULL;
357}
358
359void
360bgp_default_update_send (struct peer *peer, struct attr *attr,
361 afi_t afi, safi_t safi, struct peer *from)
362{
363 struct stream *s;
364 struct stream *packet;
365 struct prefix p;
366 unsigned long pos;
367 bgp_size_t total_attr_len;
368 char attrstr[BUFSIZ];
369 char buf[BUFSIZ];
370
Paul Jakma750e8142008-07-22 21:11:48 +0000371 if (DISABLE_BGP_ANNOUNCE)
372 return;
paul718e3742002-12-13 20:15:29 +0000373
374 if (afi == AFI_IP)
375 str2prefix ("0.0.0.0/0", &p);
376#ifdef HAVE_IPV6
377 else
378 str2prefix ("::/0", &p);
379#endif /* HAVE_IPV6 */
380
381 /* Logging the attribute. */
382 if (BGP_DEBUG (update, UPDATE_OUT))
383 {
384 bgp_dump_attr (peer, attr, attrstr, BUFSIZ);
ajs6b514742004-12-08 21:03:23 +0000385 zlog (peer->log, LOG_DEBUG, "%s send UPDATE %s/%d %s",
paul718e3742002-12-13 20:15:29 +0000386 peer->host, inet_ntop(p.family, &(p.u.prefix), buf, BUFSIZ),
387 p.prefixlen, attrstr);
388 }
389
390 s = stream_new (BGP_MAX_PACKET_SIZE);
391
392 /* Make BGP update packet. */
393 bgp_packet_set_marker (s, BGP_MSG_UPDATE);
394
395 /* Unfeasible Routes Length. */
396 stream_putw (s, 0);
397
398 /* Make place for total attribute length. */
paul9985f832005-02-09 15:51:56 +0000399 pos = stream_get_endp (s);
paul718e3742002-12-13 20:15:29 +0000400 stream_putw (s, 0);
401 total_attr_len = bgp_packet_attribute (NULL, peer, s, attr, &p, afi, safi, from, NULL, NULL);
402
403 /* Set Total Path Attribute Length. */
404 stream_putw_at (s, pos, total_attr_len);
405
406 /* NLRI set. */
407 if (p.family == AF_INET && safi == SAFI_UNICAST)
408 stream_put_prefix (s, &p);
409
410 /* Set size. */
411 bgp_packet_set_size (s);
412
paule83e2082005-05-19 02:12:25 +0000413 packet = stream_dup (s);
paul718e3742002-12-13 20:15:29 +0000414 stream_free (s);
415
416 /* Dump packet if debug option is set. */
417#ifdef DEBUG
jardin2d74db52005-10-01 00:07:50 +0000418 /* bgp_packet_dump (packet); */
paul718e3742002-12-13 20:15:29 +0000419#endif /* DEBUG */
420
421 /* Add packet to the peer. */
422 bgp_packet_add (peer, packet);
423
pauleb821182004-05-01 08:44:08 +0000424 BGP_WRITE_ON (peer->t_write, bgp_write, peer->fd);
paul718e3742002-12-13 20:15:29 +0000425}
426
427void
428bgp_default_withdraw_send (struct peer *peer, afi_t afi, safi_t safi)
429{
430 struct stream *s;
431 struct stream *packet;
432 struct prefix p;
433 unsigned long pos;
434 unsigned long cp;
435 bgp_size_t unfeasible_len;
436 bgp_size_t total_attr_len;
437 char buf[BUFSIZ];
438
Paul Jakma750e8142008-07-22 21:11:48 +0000439 if (DISABLE_BGP_ANNOUNCE)
440 return;
paul718e3742002-12-13 20:15:29 +0000441
442 if (afi == AFI_IP)
443 str2prefix ("0.0.0.0/0", &p);
444#ifdef HAVE_IPV6
445 else
446 str2prefix ("::/0", &p);
447#endif /* HAVE_IPV6 */
448
449 total_attr_len = 0;
450 pos = 0;
451
452 if (BGP_DEBUG (update, UPDATE_OUT))
ajs6b514742004-12-08 21:03:23 +0000453 zlog (peer->log, LOG_DEBUG, "%s send UPDATE %s/%d -- unreachable",
paul718e3742002-12-13 20:15:29 +0000454 peer->host, inet_ntop(p.family, &(p.u.prefix), buf, BUFSIZ),
455 p.prefixlen);
456
457 s = stream_new (BGP_MAX_PACKET_SIZE);
458
459 /* Make BGP update packet. */
460 bgp_packet_set_marker (s, BGP_MSG_UPDATE);
461
462 /* Unfeasible Routes Length. */;
paul9985f832005-02-09 15:51:56 +0000463 cp = stream_get_endp (s);
paul718e3742002-12-13 20:15:29 +0000464 stream_putw (s, 0);
465
466 /* Withdrawn Routes. */
467 if (p.family == AF_INET && safi == SAFI_UNICAST)
468 {
469 stream_put_prefix (s, &p);
470
paul9985f832005-02-09 15:51:56 +0000471 unfeasible_len = stream_get_endp (s) - cp - 2;
paul718e3742002-12-13 20:15:29 +0000472
473 /* Set unfeasible len. */
474 stream_putw_at (s, cp, unfeasible_len);
475
476 /* Set total path attribute length. */
477 stream_putw (s, 0);
478 }
479 else
480 {
paul9985f832005-02-09 15:51:56 +0000481 pos = stream_get_endp (s);
paul718e3742002-12-13 20:15:29 +0000482 stream_putw (s, 0);
483 total_attr_len = bgp_packet_withdraw (peer, s, &p, afi, safi, NULL, NULL);
484
485 /* Set total path attribute length. */
486 stream_putw_at (s, pos, total_attr_len);
487 }
488
489 bgp_packet_set_size (s);
490
paule83e2082005-05-19 02:12:25 +0000491 packet = stream_dup (s);
paul718e3742002-12-13 20:15:29 +0000492 stream_free (s);
493
494 /* Add packet to the peer. */
495 bgp_packet_add (peer, packet);
496
pauleb821182004-05-01 08:44:08 +0000497 BGP_WRITE_ON (peer->t_write, bgp_write, peer->fd);
paul718e3742002-12-13 20:15:29 +0000498}
499
500/* Get next packet to be written. */
paul94f2b392005-06-28 12:44:16 +0000501static struct stream *
paul718e3742002-12-13 20:15:29 +0000502bgp_write_packet (struct peer *peer)
503{
504 afi_t afi;
505 safi_t safi;
506 struct stream *s = NULL;
507 struct bgp_advertise *adv;
508
509 s = stream_fifo_head (peer->obuf);
510 if (s)
511 return s;
512
513 for (afi = AFI_IP; afi < AFI_MAX; afi++)
514 for (safi = SAFI_UNICAST; safi < SAFI_MAX; safi++)
515 {
516 adv = FIFO_HEAD (&peer->sync[afi][safi]->withdraw);
517 if (adv)
518 {
519 s = bgp_withdraw_packet (peer, afi, safi);
520 if (s)
521 return s;
522 }
523 }
524
525 for (afi = AFI_IP; afi < AFI_MAX; afi++)
526 for (safi = SAFI_UNICAST; safi < SAFI_MAX; safi++)
527 {
528 adv = FIFO_HEAD (&peer->sync[afi][safi]->update);
529 if (adv)
530 {
531 if (adv->binfo && adv->binfo->uptime < peer->synctime)
hasso93406d82005-02-02 14:40:33 +0000532 {
533 if (CHECK_FLAG (adv->binfo->peer->cap, PEER_CAP_RESTART_RCV)
534 && CHECK_FLAG (adv->binfo->peer->cap, PEER_CAP_RESTART_ADV)
535 && ! CHECK_FLAG (adv->binfo->flags, BGP_INFO_STALE)
536 && safi != SAFI_MPLS_VPN)
537 {
538 if (CHECK_FLAG (adv->binfo->peer->af_sflags[afi][safi],
539 PEER_STATUS_EOR_RECEIVED))
540 s = bgp_update_packet (peer, afi, safi);
541 }
542 else
543 s = bgp_update_packet (peer, afi, safi);
544 }
paul718e3742002-12-13 20:15:29 +0000545
546 if (s)
547 return s;
548 }
hasso93406d82005-02-02 14:40:33 +0000549
550 if (CHECK_FLAG (peer->cap, PEER_CAP_RESTART_RCV))
551 {
552 if (peer->afc_nego[afi][safi] && peer->synctime
553 && ! CHECK_FLAG (peer->af_sflags[afi][safi], PEER_STATUS_EOR_SEND)
554 && safi != SAFI_MPLS_VPN)
555 {
556 SET_FLAG (peer->af_sflags[afi][safi], PEER_STATUS_EOR_SEND);
557 return bgp_update_packet_eor (peer, afi, safi);
558 }
559 }
paul718e3742002-12-13 20:15:29 +0000560 }
561
562 return NULL;
563}
564
565/* Is there partially written packet or updates we can send right
566 now. */
paul94f2b392005-06-28 12:44:16 +0000567static int
paul718e3742002-12-13 20:15:29 +0000568bgp_write_proceed (struct peer *peer)
569{
570 afi_t afi;
571 safi_t safi;
572 struct bgp_advertise *adv;
573
574 if (stream_fifo_head (peer->obuf))
575 return 1;
576
577 for (afi = AFI_IP; afi < AFI_MAX; afi++)
578 for (safi = SAFI_UNICAST; safi < SAFI_MAX; safi++)
579 if (FIFO_HEAD (&peer->sync[afi][safi]->withdraw))
580 return 1;
581
582 for (afi = AFI_IP; afi < AFI_MAX; afi++)
583 for (safi = SAFI_UNICAST; safi < SAFI_MAX; safi++)
584 if ((adv = FIFO_HEAD (&peer->sync[afi][safi]->update)) != NULL)
585 if (adv->binfo->uptime < peer->synctime)
586 return 1;
587
588 return 0;
589}
590
591/* Write packet to the peer. */
592int
593bgp_write (struct thread *thread)
594{
595 struct peer *peer;
596 u_char type;
597 struct stream *s;
598 int num;
paulfd79ac92004-10-13 05:06:08 +0000599 unsigned int count = 0;
paul718e3742002-12-13 20:15:29 +0000600 int write_errno;
601
602 /* Yes first of all get peer pointer. */
603 peer = THREAD_ARG (thread);
604 peer->t_write = NULL;
605
606 /* For non-blocking IO check. */
607 if (peer->status == Connect)
608 {
609 bgp_connect_check (peer);
610 return 0;
611 }
612
613 /* Nonblocking write until TCP output buffer is full. */
614 while (1)
615 {
616 int writenum;
paula24a7e12005-01-05 08:14:13 +0000617 int val;
paul718e3742002-12-13 20:15:29 +0000618
619 s = bgp_write_packet (peer);
620 if (! s)
621 return 0;
paula24a7e12005-01-05 08:14:13 +0000622
623 /* XXX: FIXME, the socket should be NONBLOCK from the start
624 * status shouldnt need to be toggled on each write
625 */
626 val = fcntl (peer->fd, F_GETFL, 0);
627 fcntl (peer->fd, F_SETFL, val|O_NONBLOCK);
paul718e3742002-12-13 20:15:29 +0000628
629 /* Number of bytes to be sent. */
630 writenum = stream_get_endp (s) - stream_get_getp (s);
631
632 /* Call write() system call. */
pauleb821182004-05-01 08:44:08 +0000633 num = write (peer->fd, STREAM_PNT (s), writenum);
paul718e3742002-12-13 20:15:29 +0000634 write_errno = errno;
paula24a7e12005-01-05 08:14:13 +0000635 fcntl (peer->fd, F_SETFL, val);
paul718e3742002-12-13 20:15:29 +0000636 if (num <= 0)
637 {
638 /* Partial write. */
639 if (write_errno == EWOULDBLOCK || write_errno == EAGAIN)
640 break;
641
Paul Jakmadcdf3992006-10-15 23:39:59 +0000642 BGP_EVENT_ADD (peer, TCP_fatal_error);
paul718e3742002-12-13 20:15:29 +0000643 return 0;
644 }
645 if (num != writenum)
646 {
paul9985f832005-02-09 15:51:56 +0000647 stream_forward_getp (s, num);
paul718e3742002-12-13 20:15:29 +0000648
649 if (write_errno == EAGAIN)
650 break;
651
652 continue;
653 }
654
655 /* Retrieve BGP packet type. */
656 stream_set_getp (s, BGP_MARKER_SIZE + 2);
657 type = stream_getc (s);
658
659 switch (type)
660 {
661 case BGP_MSG_OPEN:
662 peer->open_out++;
663 break;
664 case BGP_MSG_UPDATE:
665 peer->update_out++;
666 break;
667 case BGP_MSG_NOTIFY:
668 peer->notify_out++;
669 /* Double start timer. */
670 peer->v_start *= 2;
671
672 /* Overflow check. */
673 if (peer->v_start >= (60 * 2))
674 peer->v_start = (60 * 2);
675
Paul Jakmaca058a32006-09-14 02:58:49 +0000676 /* Flush any existing events */
Paul Jakmadcdf3992006-10-15 23:39:59 +0000677 BGP_EVENT_ADD (peer, BGP_Stop);
paul718e3742002-12-13 20:15:29 +0000678 return 0;
paul718e3742002-12-13 20:15:29 +0000679 case BGP_MSG_KEEPALIVE:
680 peer->keepalive_out++;
681 break;
682 case BGP_MSG_ROUTE_REFRESH_NEW:
683 case BGP_MSG_ROUTE_REFRESH_OLD:
684 peer->refresh_out++;
685 break;
686 case BGP_MSG_CAPABILITY:
687 peer->dynamic_cap_out++;
688 break;
689 }
690
691 /* OK we send packet so delete it. */
692 bgp_packet_delete (peer);
693
694 if (++count >= BGP_WRITE_PACKET_MAX)
695 break;
696 }
697
698 if (bgp_write_proceed (peer))
pauleb821182004-05-01 08:44:08 +0000699 BGP_WRITE_ON (peer->t_write, bgp_write, peer->fd);
paul718e3742002-12-13 20:15:29 +0000700
701 return 0;
702}
703
704/* This is only for sending NOTIFICATION message to neighbor. */
paul94f2b392005-06-28 12:44:16 +0000705static int
paul718e3742002-12-13 20:15:29 +0000706bgp_write_notify (struct peer *peer)
707{
708 int ret;
709 u_char type;
710 struct stream *s;
711
712 /* There should be at least one packet. */
713 s = stream_fifo_head (peer->obuf);
714 if (!s)
715 return 0;
716 assert (stream_get_endp (s) >= BGP_HEADER_SIZE);
717
718 /* I'm not sure fd is writable. */
pauleb821182004-05-01 08:44:08 +0000719 ret = writen (peer->fd, STREAM_DATA (s), stream_get_endp (s));
paul718e3742002-12-13 20:15:29 +0000720 if (ret <= 0)
721 {
Paul Jakmadcdf3992006-10-15 23:39:59 +0000722 BGP_EVENT_ADD (peer, TCP_fatal_error);
paul718e3742002-12-13 20:15:29 +0000723 return 0;
724 }
725
726 /* Retrieve BGP packet type. */
727 stream_set_getp (s, BGP_MARKER_SIZE + 2);
728 type = stream_getc (s);
729
730 assert (type == BGP_MSG_NOTIFY);
731
732 /* Type should be notify. */
733 peer->notify_out++;
734
735 /* Double start timer. */
736 peer->v_start *= 2;
737
738 /* Overflow check. */
739 if (peer->v_start >= (60 * 2))
740 peer->v_start = (60 * 2);
741
Paul Jakmadcdf3992006-10-15 23:39:59 +0000742 BGP_EVENT_ADD (peer, BGP_Stop);
paul718e3742002-12-13 20:15:29 +0000743
744 return 0;
745}
746
747/* Make keepalive packet and send it to the peer. */
748void
749bgp_keepalive_send (struct peer *peer)
750{
751 struct stream *s;
752 int length;
753
754 s = stream_new (BGP_MAX_PACKET_SIZE);
755
756 /* Make keepalive packet. */
757 bgp_packet_set_marker (s, BGP_MSG_KEEPALIVE);
758
759 /* Set packet size. */
760 length = bgp_packet_set_size (s);
761
762 /* Dump packet if debug option is set. */
763 /* bgp_packet_dump (s); */
764
765 if (BGP_DEBUG (keepalive, KEEPALIVE))
ajs6b514742004-12-08 21:03:23 +0000766 zlog_debug ("%s sending KEEPALIVE", peer->host);
paul718e3742002-12-13 20:15:29 +0000767 if (BGP_DEBUG (normal, NORMAL))
ajs6b514742004-12-08 21:03:23 +0000768 zlog_debug ("%s send message type %d, length (incl. header) %d",
paul718e3742002-12-13 20:15:29 +0000769 peer->host, BGP_MSG_KEEPALIVE, length);
770
771 /* Add packet to the peer. */
772 bgp_packet_add (peer, s);
773
pauleb821182004-05-01 08:44:08 +0000774 BGP_WRITE_ON (peer->t_write, bgp_write, peer->fd);
paul718e3742002-12-13 20:15:29 +0000775}
776
777/* Make open packet and send it to the peer. */
778void
779bgp_open_send (struct peer *peer)
780{
781 struct stream *s;
782 int length;
783 u_int16_t send_holdtime;
784 as_t local_as;
785
786 if (CHECK_FLAG (peer->config, PEER_CONFIG_TIMER))
787 send_holdtime = peer->holdtime;
788 else
789 send_holdtime = peer->bgp->default_holdtime;
790
791 /* local-as Change */
792 if (peer->change_local_as)
793 local_as = peer->change_local_as;
794 else
795 local_as = peer->local_as;
796
797 s = stream_new (BGP_MAX_PACKET_SIZE);
798
799 /* Make open packet. */
800 bgp_packet_set_marker (s, BGP_MSG_OPEN);
801
802 /* Set open packet values. */
803 stream_putc (s, BGP_VERSION_4); /* BGP version */
Paul Jakma0b2aa3a2007-10-14 22:32:21 +0000804 stream_putw (s, (local_as <= BGP_AS_MAX) ? (u_int16_t) local_as
805 : BGP_AS_TRANS);
paul718e3742002-12-13 20:15:29 +0000806 stream_putw (s, send_holdtime); /* Hold Time */
807 stream_put_in_addr (s, &peer->local_id); /* BGP Identifier */
808
809 /* Set capability code. */
810 bgp_open_capability (s, peer);
811
812 /* Set BGP packet length. */
813 length = bgp_packet_set_size (s);
814
815 if (BGP_DEBUG (normal, NORMAL))
ajs6b514742004-12-08 21:03:23 +0000816 zlog_debug ("%s sending OPEN, version %d, my as %d, holdtime %d, id %s",
paul718e3742002-12-13 20:15:29 +0000817 peer->host, BGP_VERSION_4, local_as,
818 send_holdtime, inet_ntoa (peer->local_id));
819
820 if (BGP_DEBUG (normal, NORMAL))
ajs6b514742004-12-08 21:03:23 +0000821 zlog_debug ("%s send message type %d, length (incl. header) %d",
paul718e3742002-12-13 20:15:29 +0000822 peer->host, BGP_MSG_OPEN, length);
823
824 /* Dump packet if debug option is set. */
825 /* bgp_packet_dump (s); */
826
827 /* Add packet to the peer. */
828 bgp_packet_add (peer, s);
829
pauleb821182004-05-01 08:44:08 +0000830 BGP_WRITE_ON (peer->t_write, bgp_write, peer->fd);
paul718e3742002-12-13 20:15:29 +0000831}
832
833/* Send BGP notify packet with data potion. */
834void
835bgp_notify_send_with_data (struct peer *peer, u_char code, u_char sub_code,
836 u_char *data, size_t datalen)
837{
838 struct stream *s;
839 int length;
840
841 /* Allocate new stream. */
842 s = stream_new (BGP_MAX_PACKET_SIZE);
843
844 /* Make nitify packet. */
845 bgp_packet_set_marker (s, BGP_MSG_NOTIFY);
846
847 /* Set notify packet values. */
848 stream_putc (s, code); /* BGP notify code */
849 stream_putc (s, sub_code); /* BGP notify sub_code */
850
851 /* If notify data is present. */
852 if (data)
853 stream_write (s, data, datalen);
854
855 /* Set BGP packet length. */
856 length = bgp_packet_set_size (s);
857
858 /* Add packet to the peer. */
859 stream_fifo_clean (peer->obuf);
860 bgp_packet_add (peer, s);
861
862 /* For debug */
863 {
864 struct bgp_notify bgp_notify;
865 int first = 0;
866 int i;
867 char c[4];
868
869 bgp_notify.code = code;
870 bgp_notify.subcode = sub_code;
871 bgp_notify.data = NULL;
872 bgp_notify.length = length - BGP_MSG_NOTIFY_MIN_SIZE;
873
874 if (bgp_notify.length)
875 {
876 bgp_notify.data = XMALLOC (MTYPE_TMP, bgp_notify.length * 3);
877 for (i = 0; i < bgp_notify.length; i++)
878 if (first)
879 {
880 sprintf (c, " %02x", data[i]);
881 strcat (bgp_notify.data, c);
882 }
883 else
884 {
885 first = 1;
886 sprintf (c, "%02x", data[i]);
887 strcpy (bgp_notify.data, c);
888 }
889 }
890 bgp_notify_print (peer, &bgp_notify, "sending");
891 if (bgp_notify.data)
892 XFREE (MTYPE_TMP, bgp_notify.data);
893 }
894
895 if (BGP_DEBUG (normal, NORMAL))
ajs6b514742004-12-08 21:03:23 +0000896 zlog_debug ("%s send message type %d, length (incl. header) %d",
paul718e3742002-12-13 20:15:29 +0000897 peer->host, BGP_MSG_NOTIFY, length);
898
hassoe0701b72004-05-20 09:19:34 +0000899 /* peer reset cause */
900 if (sub_code != BGP_NOTIFY_CEASE_CONFIG_CHANGE)
901 {
902 if (sub_code == BGP_NOTIFY_CEASE_ADMIN_RESET)
903 peer->last_reset = PEER_DOWN_USER_RESET;
904 else if (sub_code == BGP_NOTIFY_CEASE_ADMIN_SHUTDOWN)
905 peer->last_reset = PEER_DOWN_USER_SHUTDOWN;
906 else
907 peer->last_reset = PEER_DOWN_NOTIFY_SEND;
908 }
909
paul718e3742002-12-13 20:15:29 +0000910 /* Call imidiately. */
911 BGP_WRITE_OFF (peer->t_write);
912
913 bgp_write_notify (peer);
914}
915
916/* Send BGP notify packet. */
917void
918bgp_notify_send (struct peer *peer, u_char code, u_char sub_code)
919{
920 bgp_notify_send_with_data (peer, code, sub_code, NULL, 0);
921}
922
paul94f2b392005-06-28 12:44:16 +0000923static const char *
paul718e3742002-12-13 20:15:29 +0000924afi2str (afi_t afi)
925{
926 if (afi == AFI_IP)
927 return "AFI_IP";
928 else if (afi == AFI_IP6)
929 return "AFI_IP6";
930 else
931 return "Unknown AFI";
932}
933
paul94f2b392005-06-28 12:44:16 +0000934static const char *
paul718e3742002-12-13 20:15:29 +0000935safi2str (safi_t safi)
936{
937 if (safi == SAFI_UNICAST)
938 return "SAFI_UNICAST";
939 else if (safi == SAFI_MULTICAST)
940 return "SAFI_MULTICAST";
941 else if (safi == SAFI_MPLS_VPN || safi == BGP_SAFI_VPNV4)
942 return "SAFI_MPLS_VPN";
943 else
944 return "Unknown SAFI";
945}
946
947/* Send route refresh message to the peer. */
948void
949bgp_route_refresh_send (struct peer *peer, afi_t afi, safi_t safi,
950 u_char orf_type, u_char when_to_refresh, int remove)
951{
952 struct stream *s;
953 struct stream *packet;
954 int length;
955 struct bgp_filter *filter;
956 int orf_refresh = 0;
957
Paul Jakma750e8142008-07-22 21:11:48 +0000958 if (DISABLE_BGP_ANNOUNCE)
959 return;
paul718e3742002-12-13 20:15:29 +0000960
961 filter = &peer->filter[afi][safi];
962
963 /* Adjust safi code. */
964 if (safi == SAFI_MPLS_VPN)
965 safi = BGP_SAFI_VPNV4;
966
967 s = stream_new (BGP_MAX_PACKET_SIZE);
968
969 /* Make BGP update packet. */
970 if (CHECK_FLAG (peer->cap, PEER_CAP_REFRESH_NEW_RCV))
971 bgp_packet_set_marker (s, BGP_MSG_ROUTE_REFRESH_NEW);
972 else
973 bgp_packet_set_marker (s, BGP_MSG_ROUTE_REFRESH_OLD);
974
975 /* Encode Route Refresh message. */
976 stream_putw (s, afi);
977 stream_putc (s, 0);
978 stream_putc (s, safi);
979
980 if (orf_type == ORF_TYPE_PREFIX
981 || orf_type == ORF_TYPE_PREFIX_OLD)
982 if (remove || filter->plist[FILTER_IN].plist)
983 {
984 u_int16_t orf_len;
985 unsigned long orfp;
986
987 orf_refresh = 1;
988 stream_putc (s, when_to_refresh);
989 stream_putc (s, orf_type);
paul9985f832005-02-09 15:51:56 +0000990 orfp = stream_get_endp (s);
paul718e3742002-12-13 20:15:29 +0000991 stream_putw (s, 0);
992
993 if (remove)
994 {
995 UNSET_FLAG (peer->af_sflags[afi][safi], PEER_STATUS_ORF_PREFIX_SEND);
996 stream_putc (s, ORF_COMMON_PART_REMOVE_ALL);
997 if (BGP_DEBUG (normal, NORMAL))
ajs6b514742004-12-08 21:03:23 +0000998 zlog_debug ("%s sending REFRESH_REQ to remove ORF(%d) (%s) for afi/safi: %d/%d",
paul718e3742002-12-13 20:15:29 +0000999 peer->host, orf_type,
1000 (when_to_refresh == REFRESH_DEFER ? "defer" : "immediate"),
1001 afi, safi);
1002 }
1003 else
1004 {
1005 SET_FLAG (peer->af_sflags[afi][safi], PEER_STATUS_ORF_PREFIX_SEND);
1006 prefix_bgp_orf_entry (s, filter->plist[FILTER_IN].plist,
1007 ORF_COMMON_PART_ADD, ORF_COMMON_PART_PERMIT,
1008 ORF_COMMON_PART_DENY);
1009 if (BGP_DEBUG (normal, NORMAL))
ajs6b514742004-12-08 21:03:23 +00001010 zlog_debug ("%s sending REFRESH_REQ with pfxlist ORF(%d) (%s) for afi/safi: %d/%d",
paul718e3742002-12-13 20:15:29 +00001011 peer->host, orf_type,
1012 (when_to_refresh == REFRESH_DEFER ? "defer" : "immediate"),
1013 afi, safi);
1014 }
1015
1016 /* Total ORF Entry Len. */
paul9985f832005-02-09 15:51:56 +00001017 orf_len = stream_get_endp (s) - orfp - 2;
paul718e3742002-12-13 20:15:29 +00001018 stream_putw_at (s, orfp, orf_len);
1019 }
1020
1021 /* Set packet size. */
1022 length = bgp_packet_set_size (s);
1023
1024 if (BGP_DEBUG (normal, NORMAL))
1025 {
1026 if (! orf_refresh)
ajs6b514742004-12-08 21:03:23 +00001027 zlog_debug ("%s sending REFRESH_REQ for afi/safi: %d/%d",
paul718e3742002-12-13 20:15:29 +00001028 peer->host, afi, safi);
ajs6b514742004-12-08 21:03:23 +00001029 zlog_debug ("%s send message type %d, length (incl. header) %d",
paul718e3742002-12-13 20:15:29 +00001030 peer->host, CHECK_FLAG (peer->cap, PEER_CAP_REFRESH_NEW_RCV) ?
1031 BGP_MSG_ROUTE_REFRESH_NEW : BGP_MSG_ROUTE_REFRESH_OLD, length);
1032 }
1033
1034 /* Make real packet. */
paule83e2082005-05-19 02:12:25 +00001035 packet = stream_dup (s);
paul718e3742002-12-13 20:15:29 +00001036 stream_free (s);
1037
1038 /* Add packet to the peer. */
1039 bgp_packet_add (peer, packet);
1040
pauleb821182004-05-01 08:44:08 +00001041 BGP_WRITE_ON (peer->t_write, bgp_write, peer->fd);
paul718e3742002-12-13 20:15:29 +00001042}
1043
1044/* Send capability message to the peer. */
1045void
1046bgp_capability_send (struct peer *peer, afi_t afi, safi_t safi,
1047 int capability_code, int action)
1048{
1049 struct stream *s;
1050 struct stream *packet;
1051 int length;
1052
1053 /* Adjust safi code. */
1054 if (safi == SAFI_MPLS_VPN)
1055 safi = BGP_SAFI_VPNV4;
1056
1057 s = stream_new (BGP_MAX_PACKET_SIZE);
1058
1059 /* Make BGP update packet. */
1060 bgp_packet_set_marker (s, BGP_MSG_CAPABILITY);
1061
1062 /* Encode MP_EXT capability. */
1063 if (capability_code == CAPABILITY_CODE_MP)
1064 {
1065 stream_putc (s, action);
1066 stream_putc (s, CAPABILITY_CODE_MP);
1067 stream_putc (s, CAPABILITY_CODE_MP_LEN);
1068 stream_putw (s, afi);
1069 stream_putc (s, 0);
1070 stream_putc (s, safi);
1071
1072 if (BGP_DEBUG (normal, NORMAL))
ajs6b514742004-12-08 21:03:23 +00001073 zlog_debug ("%s sending CAPABILITY has %s MP_EXT CAP for afi/safi: %d/%d",
paul718e3742002-12-13 20:15:29 +00001074 peer->host, action == CAPABILITY_ACTION_SET ?
1075 "Advertising" : "Removing", afi, safi);
1076 }
1077
paul718e3742002-12-13 20:15:29 +00001078 /* Set packet size. */
1079 length = bgp_packet_set_size (s);
1080
1081 /* Make real packet. */
paule83e2082005-05-19 02:12:25 +00001082 packet = stream_dup (s);
paul718e3742002-12-13 20:15:29 +00001083 stream_free (s);
1084
1085 /* Add packet to the peer. */
1086 bgp_packet_add (peer, packet);
1087
1088 if (BGP_DEBUG (normal, NORMAL))
ajs6b514742004-12-08 21:03:23 +00001089 zlog_debug ("%s send message type %d, length (incl. header) %d",
paul718e3742002-12-13 20:15:29 +00001090 peer->host, BGP_MSG_CAPABILITY, length);
1091
pauleb821182004-05-01 08:44:08 +00001092 BGP_WRITE_ON (peer->t_write, bgp_write, peer->fd);
paul718e3742002-12-13 20:15:29 +00001093}
1094
1095/* RFC1771 6.8 Connection collision detection. */
paul94f2b392005-06-28 12:44:16 +00001096static int
pauleb821182004-05-01 08:44:08 +00001097bgp_collision_detect (struct peer *new, struct in_addr remote_id)
paul718e3742002-12-13 20:15:29 +00001098{
pauleb821182004-05-01 08:44:08 +00001099 struct peer *peer;
paul1eb8ef22005-04-07 07:30:20 +00001100 struct listnode *node, *nnode;
paul718e3742002-12-13 20:15:29 +00001101 struct bgp *bgp;
1102
1103 bgp = bgp_get_default ();
1104 if (! bgp)
1105 return 0;
1106
1107 /* Upon receipt of an OPEN message, the local system must examine
1108 all of its connections that are in the OpenConfirm state. A BGP
1109 speaker may also examine connections in an OpenSent state if it
1110 knows the BGP Identifier of the peer by means outside of the
1111 protocol. If among these connections there is a connection to a
1112 remote BGP speaker whose BGP Identifier equals the one in the
1113 OPEN message, then the local system performs the following
1114 collision resolution procedure: */
1115
paul1eb8ef22005-04-07 07:30:20 +00001116 for (ALL_LIST_ELEMENTS (bgp->peer, node, nnode, peer))
paul718e3742002-12-13 20:15:29 +00001117 {
1118 /* Under OpenConfirm status, local peer structure already hold
1119 remote router ID. */
pauleb821182004-05-01 08:44:08 +00001120
1121 if (peer != new
1122 && (peer->status == OpenConfirm || peer->status == OpenSent)
1123 && sockunion_same (&peer->su, &new->su))
1124 {
paul718e3742002-12-13 20:15:29 +00001125 /* 1. The BGP Identifier of the local system is compared to
1126 the BGP Identifier of the remote system (as specified in
1127 the OPEN message). */
1128
1129 if (ntohl (peer->local_id.s_addr) < ntohl (remote_id.s_addr))
1130 {
1131 /* 2. If the value of the local BGP Identifier is less
1132 than the remote one, the local system closes BGP
1133 connection that already exists (the one that is
1134 already in the OpenConfirm state), and accepts BGP
1135 connection initiated by the remote system. */
1136
pauleb821182004-05-01 08:44:08 +00001137 if (peer->fd >= 0)
hassoe0701b72004-05-20 09:19:34 +00001138 bgp_notify_send (peer, BGP_NOTIFY_CEASE, BGP_NOTIFY_CEASE_COLLISION_RESOLUTION);
paul718e3742002-12-13 20:15:29 +00001139 return 1;
1140 }
1141 else
1142 {
1143 /* 3. Otherwise, the local system closes newly created
1144 BGP connection (the one associated with the newly
1145 received OPEN message), and continues to use the
1146 existing one (the one that is already in the
1147 OpenConfirm state). */
1148
pauleb821182004-05-01 08:44:08 +00001149 if (new->fd >= 0)
paulf5ba3872004-07-09 12:11:31 +00001150 bgp_notify_send (new, BGP_NOTIFY_CEASE,
1151 BGP_NOTIFY_CEASE_COLLISION_RESOLUTION);
paul718e3742002-12-13 20:15:29 +00001152 return -1;
1153 }
pauleb821182004-05-01 08:44:08 +00001154 }
1155 }
paul718e3742002-12-13 20:15:29 +00001156 return 0;
1157}
1158
paul94f2b392005-06-28 12:44:16 +00001159static int
paul718e3742002-12-13 20:15:29 +00001160bgp_open_receive (struct peer *peer, bgp_size_t size)
1161{
1162 int ret;
1163 u_char version;
1164 u_char optlen;
1165 u_int16_t holdtime;
1166 u_int16_t send_holdtime;
1167 as_t remote_as;
Paul Jakma0b2aa3a2007-10-14 22:32:21 +00001168 as_t as4 = 0;
paul718e3742002-12-13 20:15:29 +00001169 struct peer *realpeer;
1170 struct in_addr remote_id;
1171 int capability;
paul5228ad22004-06-04 17:58:18 +00001172 u_int8_t notify_data_remote_as[2];
1173 u_int8_t notify_data_remote_id[4];
paul718e3742002-12-13 20:15:29 +00001174
1175 realpeer = NULL;
1176
1177 /* Parse open packet. */
1178 version = stream_getc (peer->ibuf);
1179 memcpy (notify_data_remote_as, stream_pnt (peer->ibuf), 2);
1180 remote_as = stream_getw (peer->ibuf);
1181 holdtime = stream_getw (peer->ibuf);
1182 memcpy (notify_data_remote_id, stream_pnt (peer->ibuf), 4);
1183 remote_id.s_addr = stream_get_ipv4 (peer->ibuf);
1184
1185 /* Receive OPEN message log */
1186 if (BGP_DEBUG (normal, NORMAL))
Paul Jakma0b2aa3a2007-10-14 22:32:21 +00001187 zlog_debug ("%s rcv OPEN, version %d, remote-as (in open) %d,"
1188 " holdtime %d, id %s",
1189 peer->host, version, remote_as, holdtime,
1190 inet_ntoa (remote_id));
1191
1192 /* BEGIN to read the capability here, but dont do it yet */
1193 capability = 0;
1194 optlen = stream_getc (peer->ibuf);
1195
1196 if (optlen != 0)
1197 {
1198 /* We need the as4 capability value *right now* because
1199 * if it is there, we have not got the remote_as yet, and without
1200 * that we do not know which peer is connecting to us now.
1201 */
1202 as4 = peek_for_as4_capability (peer, optlen);
1203 }
1204
1205 /* Just in case we have a silly peer who sends AS4 capability set to 0 */
1206 if (CHECK_FLAG (peer->cap, PEER_CAP_AS4_RCV) && !as4)
1207 {
1208 zlog_err ("%s bad OPEN, got AS4 capability, but AS4 set to 0",
1209 peer->host);
1210 bgp_notify_send (peer, BGP_NOTIFY_OPEN_ERR,
1211 BGP_NOTIFY_OPEN_BAD_PEER_AS);
1212 return -1;
1213 }
1214
1215 if (remote_as == BGP_AS_TRANS)
1216 {
1217 /* Take the AS4 from the capability. We must have received the
1218 * capability now! Otherwise we have a asn16 peer who uses
1219 * BGP_AS_TRANS, for some unknown reason.
1220 */
1221 if (as4 == BGP_AS_TRANS)
1222 {
1223 zlog_err ("%s [AS4] NEW speaker using AS_TRANS for AS4, not allowed",
1224 peer->host);
1225 bgp_notify_send (peer, BGP_NOTIFY_OPEN_ERR,
1226 BGP_NOTIFY_OPEN_BAD_PEER_AS);
1227 return -1;
1228 }
1229
1230 if (!as4 && BGP_DEBUG (as4, AS4))
1231 zlog_debug ("%s [AS4] OPEN remote_as is AS_TRANS, but no AS4."
1232 " Odd, but proceeding.", peer->host);
1233 else if (as4 < BGP_AS_MAX && BGP_DEBUG (as4, AS4))
Paul Jakma0df7c912008-07-21 21:02:49 +00001234 zlog_debug ("%s [AS4] OPEN remote_as is AS_TRANS, but AS4 (%u) fits "
Paul Jakma0b2aa3a2007-10-14 22:32:21 +00001235 "in 2-bytes, very odd peer.", peer->host, as4);
1236 if (as4)
1237 remote_as = as4;
1238 }
1239 else
1240 {
1241 /* We may have a partner with AS4 who has an asno < BGP_AS_MAX */
1242 /* If we have got the capability, peer->as4cap must match remote_as */
1243 if (CHECK_FLAG (peer->cap, PEER_CAP_AS4_RCV)
1244 && as4 != remote_as)
1245 {
1246 /* raise error, log this, close session */
1247 zlog_err ("%s bad OPEN, got AS4 capability, but remote_as %u"
1248 " mismatch with 16bit 'myasn' %u in open",
1249 peer->host, as4, remote_as);
1250 bgp_notify_send (peer, BGP_NOTIFY_OPEN_ERR,
1251 BGP_NOTIFY_OPEN_BAD_PEER_AS);
1252 return -1;
1253 }
1254 }
1255
paul718e3742002-12-13 20:15:29 +00001256 /* Lookup peer from Open packet. */
1257 if (CHECK_FLAG (peer->sflags, PEER_STATUS_ACCEPT_PEER))
1258 {
1259 int as = 0;
1260
1261 realpeer = peer_lookup_with_open (&peer->su, remote_as, &remote_id, &as);
1262
1263 if (! realpeer)
1264 {
1265 /* Peer's source IP address is check in bgp_accept(), so this
1266 must be AS number mismatch or remote-id configuration
1267 mismatch. */
1268 if (as)
1269 {
1270 if (BGP_DEBUG (normal, NORMAL))
ajs6b514742004-12-08 21:03:23 +00001271 zlog_debug ("%s bad OPEN, wrong router identifier %s",
1272 peer->host, inet_ntoa (remote_id));
1273 bgp_notify_send_with_data (peer, BGP_NOTIFY_OPEN_ERR,
1274 BGP_NOTIFY_OPEN_BAD_BGP_IDENT,
1275 notify_data_remote_id, 4);
paul718e3742002-12-13 20:15:29 +00001276 }
1277 else
1278 {
1279 if (BGP_DEBUG (normal, NORMAL))
ajs6b514742004-12-08 21:03:23 +00001280 zlog_debug ("%s bad OPEN, remote AS is %d, expected %d",
1281 peer->host, remote_as, peer->as);
1282 bgp_notify_send_with_data (peer, BGP_NOTIFY_OPEN_ERR,
1283 BGP_NOTIFY_OPEN_BAD_PEER_AS,
1284 notify_data_remote_as, 2);
paul718e3742002-12-13 20:15:29 +00001285 }
1286 return -1;
1287 }
1288 }
1289
1290 /* When collision is detected and this peer is closed. Retrun
1291 immidiately. */
1292 ret = bgp_collision_detect (peer, remote_id);
1293 if (ret < 0)
1294 return ret;
1295
pauleb821182004-05-01 08:44:08 +00001296 /* Hack part. */
1297 if (CHECK_FLAG (peer->sflags, PEER_STATUS_ACCEPT_PEER))
1298 {
hasso93406d82005-02-02 14:40:33 +00001299 if (realpeer->status == Established
1300 && CHECK_FLAG (realpeer->sflags, PEER_STATUS_NSF_MODE))
1301 {
1302 realpeer->last_reset = PEER_DOWN_NSF_CLOSE_SESSION;
1303 SET_FLAG (realpeer->sflags, PEER_STATUS_NSF_WAIT);
1304 }
1305 else if (ret == 0 && realpeer->status != Active
1306 && realpeer->status != OpenSent
1307 && realpeer->status != OpenConfirm)
1308
pauleb821182004-05-01 08:44:08 +00001309 {
Paul Jakma2b2fc562008-09-06 13:09:35 +01001310 /* XXX: This is an awful problem..
1311 *
1312 * According to the RFC we should just let this connection (of the
1313 * accepted 'peer') continue on to Established if the other
1314 * connection (the 'realpeer' one) is in state Connect, and deal
1315 * with the more larval FSM as/when it gets far enough to receive
1316 * an Open. We don't do that though, we instead close the (more
1317 * developed) accepted connection.
1318 *
1319 * This means there's a race, which if hit, can loop:
1320 *
1321 * FSM for A FSM for B
1322 * realpeer accept-peer realpeer accept-peer
1323 *
1324 * Connect Connect
1325 * Active
1326 * OpenSent OpenSent
1327 * <arrive here,
1328 * Notify, delete>
1329 * Idle Active
1330 * OpenSent OpenSent
1331 * <arrive here,
1332 * Notify, delete>
1333 * Idle
1334 * <wait> <wait>
1335 * Connect Connect
1336 *
1337 *
1338 * If both sides are Quagga, they're almost certain to wait for
1339 * the same amount of time of course (which doesn't preclude other
1340 * implementations also waiting for same time). The race is
1341 * exacerbated by high-latency (in bgpd and/or the network).
1342 *
1343 * The reason we do this is because our FSM is tied to our peer
1344 * structure, which carries our configuration information, etc.
1345 * I.e. we can't let the accepted-peer FSM continue on as it is,
1346 * cause it's not associated with any actual peer configuration -
1347 * it's just a dummy.
1348 *
1349 * It's possible we could hack-fix this by just bgp_stop'ing the
1350 * realpeer and continueing on with the 'transfer FSM' below.
1351 * Ideally, we need to seperate FSMs from struct peer.
1352 *
1353 * Setting one side to passive avoids the race, as a workaround.
1354 */
pauleb821182004-05-01 08:44:08 +00001355 if (BGP_DEBUG (events, EVENTS))
hasso93406d82005-02-02 14:40:33 +00001356 zlog_debug ("%s peer status is %s close connection",
1357 realpeer->host, LOOKUP (bgp_status_msg,
1358 realpeer->status));
1359 bgp_notify_send (peer, BGP_NOTIFY_CEASE,
1360 BGP_NOTIFY_CEASE_CONNECT_REJECT);
1361
pauleb821182004-05-01 08:44:08 +00001362 return -1;
1363 }
1364
1365 if (BGP_DEBUG (events, EVENTS))
ajs6b514742004-12-08 21:03:23 +00001366 zlog_debug ("%s [Event] Transfer temporary BGP peer to existing one",
pauleb821182004-05-01 08:44:08 +00001367 peer->host);
1368
1369 bgp_stop (realpeer);
1370
1371 /* Transfer file descriptor. */
1372 realpeer->fd = peer->fd;
1373 peer->fd = -1;
1374
1375 /* Transfer input buffer. */
1376 stream_free (realpeer->ibuf);
1377 realpeer->ibuf = peer->ibuf;
1378 realpeer->packet_size = peer->packet_size;
1379 peer->ibuf = NULL;
1380
1381 /* Transfer status. */
1382 realpeer->status = peer->status;
1383 bgp_stop (peer);
paul200df112005-06-01 11:17:05 +00001384
pauleb821182004-05-01 08:44:08 +00001385 /* peer pointer change. Open packet send to neighbor. */
1386 peer = realpeer;
1387 bgp_open_send (peer);
1388 if (peer->fd < 0)
1389 {
1390 zlog_err ("bgp_open_receive peer's fd is negative value %d",
1391 peer->fd);
1392 return -1;
1393 }
1394 BGP_READ_ON (peer->t_read, bgp_read, peer->fd);
1395 }
1396
paul718e3742002-12-13 20:15:29 +00001397 /* remote router-id check. */
1398 if (remote_id.s_addr == 0
1399 || ntohl (remote_id.s_addr) >= 0xe0000000
1400 || ntohl (peer->local_id.s_addr) == ntohl (remote_id.s_addr))
1401 {
1402 if (BGP_DEBUG (normal, NORMAL))
ajs6b514742004-12-08 21:03:23 +00001403 zlog_debug ("%s bad OPEN, wrong router identifier %s",
paul718e3742002-12-13 20:15:29 +00001404 peer->host, inet_ntoa (remote_id));
1405 bgp_notify_send_with_data (peer,
1406 BGP_NOTIFY_OPEN_ERR,
1407 BGP_NOTIFY_OPEN_BAD_BGP_IDENT,
1408 notify_data_remote_id, 4);
1409 return -1;
1410 }
1411
1412 /* Set remote router-id */
1413 peer->remote_id = remote_id;
1414
1415 /* Peer BGP version check. */
1416 if (version != BGP_VERSION_4)
1417 {
paul5228ad22004-06-04 17:58:18 +00001418 u_int8_t maxver = BGP_VERSION_4;
paul718e3742002-12-13 20:15:29 +00001419 if (BGP_DEBUG (normal, NORMAL))
ajs6b514742004-12-08 21:03:23 +00001420 zlog_debug ("%s bad protocol version, remote requested %d, local request %d",
paul718e3742002-12-13 20:15:29 +00001421 peer->host, version, BGP_VERSION_4);
1422 bgp_notify_send_with_data (peer,
1423 BGP_NOTIFY_OPEN_ERR,
1424 BGP_NOTIFY_OPEN_UNSUP_VERSION,
paul5228ad22004-06-04 17:58:18 +00001425 &maxver, 1);
paul718e3742002-12-13 20:15:29 +00001426 return -1;
1427 }
1428
1429 /* Check neighbor as number. */
1430 if (remote_as != peer->as)
1431 {
1432 if (BGP_DEBUG (normal, NORMAL))
ajs6b514742004-12-08 21:03:23 +00001433 zlog_debug ("%s bad OPEN, remote AS is %d, expected %d",
paul718e3742002-12-13 20:15:29 +00001434 peer->host, remote_as, peer->as);
1435 bgp_notify_send_with_data (peer,
1436 BGP_NOTIFY_OPEN_ERR,
1437 BGP_NOTIFY_OPEN_BAD_PEER_AS,
1438 notify_data_remote_as, 2);
1439 return -1;
1440 }
1441
1442 /* From the rfc: Upon receipt of an OPEN message, a BGP speaker MUST
1443 calculate the value of the Hold Timer by using the smaller of its
1444 configured Hold Time and the Hold Time received in the OPEN message.
1445 The Hold Time MUST be either zero or at least three seconds. An
1446 implementation may reject connections on the basis of the Hold Time. */
1447
1448 if (holdtime < 3 && holdtime != 0)
1449 {
1450 bgp_notify_send (peer,
1451 BGP_NOTIFY_OPEN_ERR,
1452 BGP_NOTIFY_OPEN_UNACEP_HOLDTIME);
1453 return -1;
1454 }
1455
1456 /* From the rfc: A reasonable maximum time between KEEPALIVE messages
1457 would be one third of the Hold Time interval. KEEPALIVE messages
1458 MUST NOT be sent more frequently than one per second. An
1459 implementation MAY adjust the rate at which it sends KEEPALIVE
1460 messages as a function of the Hold Time interval. */
1461
1462 if (CHECK_FLAG (peer->config, PEER_CONFIG_TIMER))
1463 send_holdtime = peer->holdtime;
1464 else
1465 send_holdtime = peer->bgp->default_holdtime;
1466
1467 if (holdtime < send_holdtime)
1468 peer->v_holdtime = holdtime;
1469 else
1470 peer->v_holdtime = send_holdtime;
1471
1472 peer->v_keepalive = peer->v_holdtime / 3;
1473
1474 /* Open option part parse. */
paul718e3742002-12-13 20:15:29 +00001475 if (optlen != 0)
1476 {
1477 ret = bgp_open_option_parse (peer, optlen, &capability);
1478 if (ret < 0)
1479 return ret;
paul718e3742002-12-13 20:15:29 +00001480 }
1481 else
1482 {
1483 if (BGP_DEBUG (normal, NORMAL))
ajs6b514742004-12-08 21:03:23 +00001484 zlog_debug ("%s rcvd OPEN w/ OPTION parameter len: 0",
paul718e3742002-12-13 20:15:29 +00001485 peer->host);
1486 }
1487
1488 /* Override capability. */
1489 if (! capability || CHECK_FLAG (peer->flags, PEER_FLAG_OVERRIDE_CAPABILITY))
1490 {
1491 peer->afc_nego[AFI_IP][SAFI_UNICAST] = peer->afc[AFI_IP][SAFI_UNICAST];
1492 peer->afc_nego[AFI_IP][SAFI_MULTICAST] = peer->afc[AFI_IP][SAFI_MULTICAST];
1493 peer->afc_nego[AFI_IP6][SAFI_UNICAST] = peer->afc[AFI_IP6][SAFI_UNICAST];
1494 peer->afc_nego[AFI_IP6][SAFI_MULTICAST] = peer->afc[AFI_IP6][SAFI_MULTICAST];
1495 }
1496
1497 /* Get sockname. */
1498 bgp_getsockname (peer);
1499
1500 BGP_EVENT_ADD (peer, Receive_OPEN_message);
1501
1502 peer->packet_size = 0;
1503 if (peer->ibuf)
1504 stream_reset (peer->ibuf);
1505
1506 return 0;
1507}
1508
1509/* Parse BGP Update packet and make attribute object. */
paul94f2b392005-06-28 12:44:16 +00001510static int
paul718e3742002-12-13 20:15:29 +00001511bgp_update_receive (struct peer *peer, bgp_size_t size)
1512{
1513 int ret;
1514 u_char *end;
1515 struct stream *s;
1516 struct attr attr;
1517 bgp_size_t attribute_len;
1518 bgp_size_t update_len;
1519 bgp_size_t withdraw_len;
1520 struct bgp_nlri update;
1521 struct bgp_nlri withdraw;
1522 struct bgp_nlri mp_update;
1523 struct bgp_nlri mp_withdraw;
paule01f9cb2004-07-09 17:48:53 +00001524 char attrstr[BUFSIZ] = "";
paul718e3742002-12-13 20:15:29 +00001525
1526 /* Status must be Established. */
1527 if (peer->status != Established)
1528 {
1529 zlog_err ("%s [FSM] Update packet received under status %s",
1530 peer->host, LOOKUP (bgp_status_msg, peer->status));
1531 bgp_notify_send (peer, BGP_NOTIFY_FSM_ERR, 0);
1532 return -1;
1533 }
1534
1535 /* Set initial values. */
1536 memset (&attr, 0, sizeof (struct attr));
1537 memset (&update, 0, sizeof (struct bgp_nlri));
1538 memset (&withdraw, 0, sizeof (struct bgp_nlri));
1539 memset (&mp_update, 0, sizeof (struct bgp_nlri));
1540 memset (&mp_withdraw, 0, sizeof (struct bgp_nlri));
1541
1542 s = peer->ibuf;
1543 end = stream_pnt (s) + size;
1544
1545 /* RFC1771 6.3 If the Unfeasible Routes Length or Total Attribute
1546 Length is too large (i.e., if Unfeasible Routes Length + Total
1547 Attribute Length + 23 exceeds the message Length), then the Error
1548 Subcode is set to Malformed Attribute List. */
1549 if (stream_pnt (s) + 2 > end)
1550 {
1551 zlog_err ("%s [Error] Update packet error"
1552 " (packet length is short for unfeasible length)",
1553 peer->host);
1554 bgp_notify_send (peer, BGP_NOTIFY_UPDATE_ERR,
1555 BGP_NOTIFY_UPDATE_MAL_ATTR);
1556 return -1;
1557 }
1558
1559 /* Unfeasible Route Length. */
1560 withdraw_len = stream_getw (s);
1561
1562 /* Unfeasible Route Length check. */
1563 if (stream_pnt (s) + withdraw_len > end)
1564 {
1565 zlog_err ("%s [Error] Update packet error"
1566 " (packet unfeasible length overflow %d)",
1567 peer->host, withdraw_len);
1568 bgp_notify_send (peer, BGP_NOTIFY_UPDATE_ERR,
1569 BGP_NOTIFY_UPDATE_MAL_ATTR);
1570 return -1;
1571 }
1572
1573 /* Unfeasible Route packet format check. */
1574 if (withdraw_len > 0)
1575 {
1576 ret = bgp_nlri_sanity_check (peer, AFI_IP, stream_pnt (s), withdraw_len);
1577 if (ret < 0)
1578 return -1;
1579
1580 if (BGP_DEBUG (packet, PACKET_RECV))
ajs6b514742004-12-08 21:03:23 +00001581 zlog_debug ("%s [Update:RECV] Unfeasible NLRI received", peer->host);
paul718e3742002-12-13 20:15:29 +00001582
1583 withdraw.afi = AFI_IP;
1584 withdraw.safi = SAFI_UNICAST;
1585 withdraw.nlri = stream_pnt (s);
1586 withdraw.length = withdraw_len;
paul9985f832005-02-09 15:51:56 +00001587 stream_forward_getp (s, withdraw_len);
paul718e3742002-12-13 20:15:29 +00001588 }
1589
1590 /* Attribute total length check. */
1591 if (stream_pnt (s) + 2 > end)
1592 {
1593 zlog_warn ("%s [Error] Packet Error"
1594 " (update packet is short for attribute length)",
1595 peer->host);
1596 bgp_notify_send (peer, BGP_NOTIFY_UPDATE_ERR,
1597 BGP_NOTIFY_UPDATE_MAL_ATTR);
1598 return -1;
1599 }
1600
1601 /* Fetch attribute total length. */
1602 attribute_len = stream_getw (s);
1603
1604 /* Attribute length check. */
1605 if (stream_pnt (s) + attribute_len > end)
1606 {
1607 zlog_warn ("%s [Error] Packet Error"
1608 " (update packet attribute length overflow %d)",
1609 peer->host, attribute_len);
1610 bgp_notify_send (peer, BGP_NOTIFY_UPDATE_ERR,
1611 BGP_NOTIFY_UPDATE_MAL_ATTR);
1612 return -1;
1613 }
1614
1615 /* Parse attribute when it exists. */
1616 if (attribute_len)
1617 {
1618 ret = bgp_attr_parse (peer, &attr, attribute_len,
1619 &mp_update, &mp_withdraw);
1620 if (ret < 0)
1621 return -1;
1622 }
1623
1624 /* Logging the attribute. */
1625 if (BGP_DEBUG (update, UPDATE_IN))
1626 {
paule01f9cb2004-07-09 17:48:53 +00001627 ret= bgp_dump_attr (peer, &attr, attrstr, BUFSIZ);
1628
1629 if (ret)
ajs6b514742004-12-08 21:03:23 +00001630 zlog (peer->log, LOG_DEBUG, "%s rcvd UPDATE w/ attr: %s",
paule01f9cb2004-07-09 17:48:53 +00001631 peer->host, attrstr);
paul718e3742002-12-13 20:15:29 +00001632 }
1633
1634 /* Network Layer Reachability Information. */
1635 update_len = end - stream_pnt (s);
1636
1637 if (update_len)
1638 {
1639 /* Check NLRI packet format and prefix length. */
1640 ret = bgp_nlri_sanity_check (peer, AFI_IP, stream_pnt (s), update_len);
1641 if (ret < 0)
1642 return -1;
1643
1644 /* Set NLRI portion to structure. */
1645 update.afi = AFI_IP;
1646 update.safi = SAFI_UNICAST;
1647 update.nlri = stream_pnt (s);
1648 update.length = update_len;
paul9985f832005-02-09 15:51:56 +00001649 stream_forward_getp (s, update_len);
paul718e3742002-12-13 20:15:29 +00001650 }
1651
1652 /* NLRI is processed only when the peer is configured specific
1653 Address Family and Subsequent Address Family. */
1654 if (peer->afc[AFI_IP][SAFI_UNICAST])
1655 {
1656 if (withdraw.length)
1657 bgp_nlri_parse (peer, NULL, &withdraw);
1658
1659 if (update.length)
1660 {
1661 /* We check well-known attribute only for IPv4 unicast
1662 update. */
1663 ret = bgp_attr_check (peer, &attr);
1664 if (ret < 0)
1665 return -1;
1666
1667 bgp_nlri_parse (peer, &attr, &update);
1668 }
paule01f9cb2004-07-09 17:48:53 +00001669
hassof4184462005-02-01 20:13:16 +00001670 if (mp_update.length
1671 && mp_update.afi == AFI_IP
1672 && mp_update.safi == SAFI_UNICAST)
1673 bgp_nlri_parse (peer, &attr, &mp_update);
1674
1675 if (mp_withdraw.length
1676 && mp_withdraw.afi == AFI_IP
1677 && mp_withdraw.safi == SAFI_UNICAST)
1678 bgp_nlri_parse (peer, NULL, &mp_withdraw);
1679
paule01f9cb2004-07-09 17:48:53 +00001680 if (! attribute_len && ! withdraw_len)
1681 {
1682 /* End-of-RIB received */
hasso93406d82005-02-02 14:40:33 +00001683 SET_FLAG (peer->af_sflags[AFI_IP][SAFI_UNICAST],
1684 PEER_STATUS_EOR_RECEIVED);
paule01f9cb2004-07-09 17:48:53 +00001685
hasso93406d82005-02-02 14:40:33 +00001686 /* NSF delete stale route */
1687 if (peer->nsf[AFI_IP][SAFI_UNICAST])
1688 bgp_clear_stale_route (peer, AFI_IP, SAFI_UNICAST);
1689
1690 if (BGP_DEBUG (normal, NORMAL))
ajs6b514742004-12-08 21:03:23 +00001691 zlog (peer->log, LOG_DEBUG, "rcvd End-of-RIB for IPv4 Unicast from %s",
paule01f9cb2004-07-09 17:48:53 +00001692 peer->host);
1693 }
paul718e3742002-12-13 20:15:29 +00001694 }
1695 if (peer->afc[AFI_IP][SAFI_MULTICAST])
1696 {
1697 if (mp_update.length
1698 && mp_update.afi == AFI_IP
1699 && mp_update.safi == SAFI_MULTICAST)
1700 bgp_nlri_parse (peer, &attr, &mp_update);
1701
1702 if (mp_withdraw.length
1703 && mp_withdraw.afi == AFI_IP
1704 && mp_withdraw.safi == SAFI_MULTICAST)
1705 bgp_nlri_parse (peer, NULL, &mp_withdraw);
paule01f9cb2004-07-09 17:48:53 +00001706
hasso93406d82005-02-02 14:40:33 +00001707 if (! withdraw_len
paule01f9cb2004-07-09 17:48:53 +00001708 && mp_withdraw.afi == AFI_IP
1709 && mp_withdraw.safi == SAFI_MULTICAST
1710 && mp_withdraw.length == 0)
1711 {
1712 /* End-of-RIB received */
hasso93406d82005-02-02 14:40:33 +00001713 SET_FLAG (peer->af_sflags[AFI_IP][SAFI_MULTICAST],
1714 PEER_STATUS_EOR_RECEIVED);
paule01f9cb2004-07-09 17:48:53 +00001715
hasso93406d82005-02-02 14:40:33 +00001716 /* NSF delete stale route */
1717 if (peer->nsf[AFI_IP][SAFI_MULTICAST])
1718 bgp_clear_stale_route (peer, AFI_IP, SAFI_MULTICAST);
1719
1720 if (BGP_DEBUG (normal, NORMAL))
ajs6b514742004-12-08 21:03:23 +00001721 zlog (peer->log, LOG_DEBUG, "rcvd End-of-RIB for IPv4 Multicast from %s",
paule01f9cb2004-07-09 17:48:53 +00001722 peer->host);
1723 }
paul718e3742002-12-13 20:15:29 +00001724 }
1725 if (peer->afc[AFI_IP6][SAFI_UNICAST])
1726 {
1727 if (mp_update.length
1728 && mp_update.afi == AFI_IP6
1729 && mp_update.safi == SAFI_UNICAST)
1730 bgp_nlri_parse (peer, &attr, &mp_update);
1731
1732 if (mp_withdraw.length
1733 && mp_withdraw.afi == AFI_IP6
1734 && mp_withdraw.safi == SAFI_UNICAST)
1735 bgp_nlri_parse (peer, NULL, &mp_withdraw);
paule01f9cb2004-07-09 17:48:53 +00001736
hasso93406d82005-02-02 14:40:33 +00001737 if (! withdraw_len
paule01f9cb2004-07-09 17:48:53 +00001738 && mp_withdraw.afi == AFI_IP6
1739 && mp_withdraw.safi == SAFI_UNICAST
1740 && mp_withdraw.length == 0)
1741 {
1742 /* End-of-RIB received */
hasso93406d82005-02-02 14:40:33 +00001743 SET_FLAG (peer->af_sflags[AFI_IP6][SAFI_UNICAST], PEER_STATUS_EOR_RECEIVED);
paule01f9cb2004-07-09 17:48:53 +00001744
hasso93406d82005-02-02 14:40:33 +00001745 /* NSF delete stale route */
1746 if (peer->nsf[AFI_IP6][SAFI_UNICAST])
1747 bgp_clear_stale_route (peer, AFI_IP6, SAFI_UNICAST);
1748
1749 if (BGP_DEBUG (normal, NORMAL))
ajs6b514742004-12-08 21:03:23 +00001750 zlog (peer->log, LOG_DEBUG, "rcvd End-of-RIB for IPv6 Unicast from %s",
paule01f9cb2004-07-09 17:48:53 +00001751 peer->host);
1752 }
paul718e3742002-12-13 20:15:29 +00001753 }
1754 if (peer->afc[AFI_IP6][SAFI_MULTICAST])
1755 {
1756 if (mp_update.length
1757 && mp_update.afi == AFI_IP6
1758 && mp_update.safi == SAFI_MULTICAST)
1759 bgp_nlri_parse (peer, &attr, &mp_update);
1760
1761 if (mp_withdraw.length
1762 && mp_withdraw.afi == AFI_IP6
1763 && mp_withdraw.safi == SAFI_MULTICAST)
1764 bgp_nlri_parse (peer, NULL, &mp_withdraw);
paule01f9cb2004-07-09 17:48:53 +00001765
hasso93406d82005-02-02 14:40:33 +00001766 if (! withdraw_len
paule01f9cb2004-07-09 17:48:53 +00001767 && mp_withdraw.afi == AFI_IP6
1768 && mp_withdraw.safi == SAFI_MULTICAST
1769 && mp_withdraw.length == 0)
1770 {
1771 /* End-of-RIB received */
1772
hasso93406d82005-02-02 14:40:33 +00001773 /* NSF delete stale route */
1774 if (peer->nsf[AFI_IP6][SAFI_MULTICAST])
1775 bgp_clear_stale_route (peer, AFI_IP6, SAFI_MULTICAST);
1776
paule01f9cb2004-07-09 17:48:53 +00001777 if (BGP_DEBUG (update, UPDATE_IN))
ajs6b514742004-12-08 21:03:23 +00001778 zlog (peer->log, LOG_DEBUG, "rcvd End-of-RIB for IPv6 Multicast from %s",
paule01f9cb2004-07-09 17:48:53 +00001779 peer->host);
1780 }
paul718e3742002-12-13 20:15:29 +00001781 }
1782 if (peer->afc[AFI_IP][SAFI_MPLS_VPN])
1783 {
1784 if (mp_update.length
1785 && mp_update.afi == AFI_IP
1786 && mp_update.safi == BGP_SAFI_VPNV4)
1787 bgp_nlri_parse_vpnv4 (peer, &attr, &mp_update);
1788
1789 if (mp_withdraw.length
1790 && mp_withdraw.afi == AFI_IP
1791 && mp_withdraw.safi == BGP_SAFI_VPNV4)
1792 bgp_nlri_parse_vpnv4 (peer, NULL, &mp_withdraw);
paule01f9cb2004-07-09 17:48:53 +00001793
hasso93406d82005-02-02 14:40:33 +00001794 if (! withdraw_len
paule01f9cb2004-07-09 17:48:53 +00001795 && mp_withdraw.afi == AFI_IP
1796 && mp_withdraw.safi == BGP_SAFI_VPNV4
1797 && mp_withdraw.length == 0)
1798 {
1799 /* End-of-RIB received */
1800
1801 if (BGP_DEBUG (update, UPDATE_IN))
ajs6b514742004-12-08 21:03:23 +00001802 zlog (peer->log, LOG_DEBUG, "rcvd End-of-RIB for VPNv4 Unicast from %s",
paule01f9cb2004-07-09 17:48:53 +00001803 peer->host);
1804 }
paul718e3742002-12-13 20:15:29 +00001805 }
1806
1807 /* Everything is done. We unintern temporary structures which
1808 interned in bgp_attr_parse(). */
1809 if (attr.aspath)
1810 aspath_unintern (attr.aspath);
1811 if (attr.community)
1812 community_unintern (attr.community);
Paul Jakmafb982c22007-05-04 20:15:47 +00001813 if (attr.extra)
1814 {
1815 if (attr.extra->ecommunity)
1816 ecommunity_unintern (attr.extra->ecommunity);
1817 if (attr.extra->cluster)
1818 cluster_unintern (attr.extra->cluster);
1819 if (attr.extra->transit)
1820 transit_unintern (attr.extra->transit);
1821 bgp_attr_extra_free (&attr);
1822 }
paul718e3742002-12-13 20:15:29 +00001823
1824 /* If peering is stopped due to some reason, do not generate BGP
1825 event. */
1826 if (peer->status != Established)
1827 return 0;
1828
1829 /* Increment packet counter. */
1830 peer->update_in++;
1831 peer->update_time = time (NULL);
1832
1833 /* Generate BGP event. */
1834 BGP_EVENT_ADD (peer, Receive_UPDATE_message);
1835
1836 return 0;
1837}
1838
1839/* Notify message treatment function. */
paul94f2b392005-06-28 12:44:16 +00001840static void
paul718e3742002-12-13 20:15:29 +00001841bgp_notify_receive (struct peer *peer, bgp_size_t size)
1842{
1843 struct bgp_notify bgp_notify;
1844
1845 if (peer->notify.data)
1846 {
1847 XFREE (MTYPE_TMP, peer->notify.data);
1848 peer->notify.data = NULL;
1849 peer->notify.length = 0;
1850 }
1851
1852 bgp_notify.code = stream_getc (peer->ibuf);
1853 bgp_notify.subcode = stream_getc (peer->ibuf);
1854 bgp_notify.length = size - 2;
1855 bgp_notify.data = NULL;
1856
1857 /* Preserv notify code and sub code. */
1858 peer->notify.code = bgp_notify.code;
1859 peer->notify.subcode = bgp_notify.subcode;
1860 /* For further diagnostic record returned Data. */
1861 if (bgp_notify.length)
1862 {
1863 peer->notify.length = size - 2;
1864 peer->notify.data = XMALLOC (MTYPE_TMP, size - 2);
1865 memcpy (peer->notify.data, stream_pnt (peer->ibuf), size - 2);
1866 }
1867
1868 /* For debug */
1869 {
1870 int i;
1871 int first = 0;
1872 char c[4];
1873
1874 if (bgp_notify.length)
1875 {
1876 bgp_notify.data = XMALLOC (MTYPE_TMP, bgp_notify.length * 3);
1877 for (i = 0; i < bgp_notify.length; i++)
1878 if (first)
1879 {
1880 sprintf (c, " %02x", stream_getc (peer->ibuf));
1881 strcat (bgp_notify.data, c);
1882 }
1883 else
1884 {
1885 first = 1;
1886 sprintf (c, "%02x", stream_getc (peer->ibuf));
1887 strcpy (bgp_notify.data, c);
1888 }
1889 }
1890
1891 bgp_notify_print(peer, &bgp_notify, "received");
1892 if (bgp_notify.data)
1893 XFREE (MTYPE_TMP, bgp_notify.data);
1894 }
1895
1896 /* peer count update */
1897 peer->notify_in++;
1898
hassoe0701b72004-05-20 09:19:34 +00001899 if (peer->status == Established)
1900 peer->last_reset = PEER_DOWN_NOTIFY_RECEIVED;
1901
paul718e3742002-12-13 20:15:29 +00001902 /* We have to check for Notify with Unsupported Optional Parameter.
1903 in that case we fallback to open without the capability option.
1904 But this done in bgp_stop. We just mark it here to avoid changing
1905 the fsm tables. */
1906 if (bgp_notify.code == BGP_NOTIFY_OPEN_ERR &&
1907 bgp_notify.subcode == BGP_NOTIFY_OPEN_UNSUP_PARAM )
1908 UNSET_FLAG (peer->sflags, PEER_STATUS_CAPABILITY_OPEN);
1909
1910 /* Also apply to Unsupported Capability until remote router support
1911 capability. */
1912 if (bgp_notify.code == BGP_NOTIFY_OPEN_ERR &&
1913 bgp_notify.subcode == BGP_NOTIFY_OPEN_UNSUP_CAPBL)
1914 UNSET_FLAG (peer->sflags, PEER_STATUS_CAPABILITY_OPEN);
1915
1916 BGP_EVENT_ADD (peer, Receive_NOTIFICATION_message);
1917}
1918
1919/* Keepalive treatment function -- get keepalive send keepalive */
paul94f2b392005-06-28 12:44:16 +00001920static void
paul718e3742002-12-13 20:15:29 +00001921bgp_keepalive_receive (struct peer *peer, bgp_size_t size)
1922{
1923 if (BGP_DEBUG (keepalive, KEEPALIVE))
ajs6b514742004-12-08 21:03:23 +00001924 zlog_debug ("%s KEEPALIVE rcvd", peer->host);
paul718e3742002-12-13 20:15:29 +00001925
1926 BGP_EVENT_ADD (peer, Receive_KEEPALIVE_message);
1927}
1928
1929/* Route refresh message is received. */
paul94f2b392005-06-28 12:44:16 +00001930static void
paul718e3742002-12-13 20:15:29 +00001931bgp_route_refresh_receive (struct peer *peer, bgp_size_t size)
1932{
1933 afi_t afi;
1934 safi_t safi;
1935 u_char reserved;
1936 struct stream *s;
1937
1938 /* If peer does not have the capability, send notification. */
1939 if (! CHECK_FLAG (peer->cap, PEER_CAP_REFRESH_ADV))
1940 {
1941 plog_err (peer->log, "%s [Error] BGP route refresh is not enabled",
1942 peer->host);
1943 bgp_notify_send (peer,
1944 BGP_NOTIFY_HEADER_ERR,
1945 BGP_NOTIFY_HEADER_BAD_MESTYPE);
1946 return;
1947 }
1948
1949 /* Status must be Established. */
1950 if (peer->status != Established)
1951 {
1952 plog_err (peer->log,
1953 "%s [Error] Route refresh packet received under status %s",
1954 peer->host, LOOKUP (bgp_status_msg, peer->status));
1955 bgp_notify_send (peer, BGP_NOTIFY_FSM_ERR, 0);
1956 return;
1957 }
1958
1959 s = peer->ibuf;
1960
1961 /* Parse packet. */
1962 afi = stream_getw (s);
1963 reserved = stream_getc (s);
1964 safi = stream_getc (s);
1965
1966 if (BGP_DEBUG (normal, NORMAL))
ajs6b514742004-12-08 21:03:23 +00001967 zlog_debug ("%s rcvd REFRESH_REQ for afi/safi: %d/%d",
paul718e3742002-12-13 20:15:29 +00001968 peer->host, afi, safi);
1969
1970 /* Check AFI and SAFI. */
1971 if ((afi != AFI_IP && afi != AFI_IP6)
1972 || (safi != SAFI_UNICAST && safi != SAFI_MULTICAST
1973 && safi != BGP_SAFI_VPNV4))
1974 {
1975 if (BGP_DEBUG (normal, NORMAL))
1976 {
ajs6b514742004-12-08 21:03:23 +00001977 zlog_debug ("%s REFRESH_REQ for unrecognized afi/safi: %d/%d - ignored",
paul718e3742002-12-13 20:15:29 +00001978 peer->host, afi, safi);
1979 }
1980 return;
1981 }
1982
1983 /* Adjust safi code. */
1984 if (safi == BGP_SAFI_VPNV4)
1985 safi = SAFI_MPLS_VPN;
1986
1987 if (size != BGP_MSG_ROUTE_REFRESH_MIN_SIZE - BGP_HEADER_SIZE)
1988 {
1989 u_char *end;
1990 u_char when_to_refresh;
1991 u_char orf_type;
1992 u_int16_t orf_len;
1993
1994 if (size - (BGP_MSG_ROUTE_REFRESH_MIN_SIZE - BGP_HEADER_SIZE) < 5)
1995 {
1996 zlog_info ("%s ORF route refresh length error", peer->host);
1997 bgp_notify_send (peer, BGP_NOTIFY_CEASE, 0);
1998 return;
1999 }
2000
2001 when_to_refresh = stream_getc (s);
2002 end = stream_pnt (s) + (size - 5);
2003
Paul Jakma370b64a2007-12-22 16:49:52 +00002004 while ((stream_pnt (s) + 2) < end)
paul718e3742002-12-13 20:15:29 +00002005 {
2006 orf_type = stream_getc (s);
2007 orf_len = stream_getw (s);
Paul Jakma370b64a2007-12-22 16:49:52 +00002008
2009 /* orf_len in bounds? */
2010 if ((stream_pnt (s) + orf_len) > end)
2011 break; /* XXX: Notify instead?? */
paul718e3742002-12-13 20:15:29 +00002012 if (orf_type == ORF_TYPE_PREFIX
2013 || orf_type == ORF_TYPE_PREFIX_OLD)
2014 {
2015 u_char *p_pnt = stream_pnt (s);
2016 u_char *p_end = stream_pnt (s) + orf_len;
2017 struct orf_prefix orfp;
2018 u_char common = 0;
2019 u_int32_t seq;
2020 int psize;
2021 char name[BUFSIZ];
2022 char buf[BUFSIZ];
2023 int ret;
2024
2025 if (BGP_DEBUG (normal, NORMAL))
2026 {
ajs6b514742004-12-08 21:03:23 +00002027 zlog_debug ("%s rcvd Prefixlist ORF(%d) length %d",
paul718e3742002-12-13 20:15:29 +00002028 peer->host, orf_type, orf_len);
2029 }
2030
Paul Jakma370b64a2007-12-22 16:49:52 +00002031 /* we're going to read at least 1 byte of common ORF header,
2032 * and 7 bytes of ORF Address-filter entry from the stream
2033 */
2034 if (orf_len < 7)
2035 break;
2036
paul718e3742002-12-13 20:15:29 +00002037 /* ORF prefix-list name */
2038 sprintf (name, "%s.%d.%d", peer->host, afi, safi);
2039
2040 while (p_pnt < p_end)
2041 {
2042 memset (&orfp, 0, sizeof (struct orf_prefix));
2043 common = *p_pnt++;
2044 if (common & ORF_COMMON_PART_REMOVE_ALL)
2045 {
2046 if (BGP_DEBUG (normal, NORMAL))
ajs6b514742004-12-08 21:03:23 +00002047 zlog_debug ("%s rcvd Remove-All pfxlist ORF request", peer->host);
paul718e3742002-12-13 20:15:29 +00002048 prefix_bgp_orf_remove_all (name);
2049 break;
2050 }
2051 memcpy (&seq, p_pnt, sizeof (u_int32_t));
2052 p_pnt += sizeof (u_int32_t);
2053 orfp.seq = ntohl (seq);
2054 orfp.ge = *p_pnt++;
2055 orfp.le = *p_pnt++;
2056 orfp.p.prefixlen = *p_pnt++;
2057 orfp.p.family = afi2family (afi);
2058 psize = PSIZE (orfp.p.prefixlen);
2059 memcpy (&orfp.p.u.prefix, p_pnt, psize);
2060 p_pnt += psize;
2061
2062 if (BGP_DEBUG (normal, NORMAL))
ajs6b514742004-12-08 21:03:23 +00002063 zlog_debug ("%s rcvd %s %s seq %u %s/%d ge %d le %d",
paul718e3742002-12-13 20:15:29 +00002064 peer->host,
2065 (common & ORF_COMMON_PART_REMOVE ? "Remove" : "Add"),
2066 (common & ORF_COMMON_PART_DENY ? "deny" : "permit"),
2067 orfp.seq,
2068 inet_ntop (orfp.p.family, &orfp.p.u.prefix, buf, BUFSIZ),
2069 orfp.p.prefixlen, orfp.ge, orfp.le);
2070
2071 ret = prefix_bgp_orf_set (name, afi, &orfp,
2072 (common & ORF_COMMON_PART_DENY ? 0 : 1 ),
2073 (common & ORF_COMMON_PART_REMOVE ? 0 : 1));
2074
2075 if (ret != CMD_SUCCESS)
2076 {
2077 if (BGP_DEBUG (normal, NORMAL))
ajs6b514742004-12-08 21:03:23 +00002078 zlog_debug ("%s Received misformatted prefixlist ORF. Remove All pfxlist", peer->host);
paul718e3742002-12-13 20:15:29 +00002079 prefix_bgp_orf_remove_all (name);
2080 break;
2081 }
2082 }
2083 peer->orf_plist[afi][safi] =
2084 prefix_list_lookup (AFI_ORF_PREFIX, name);
2085 }
paul9985f832005-02-09 15:51:56 +00002086 stream_forward_getp (s, orf_len);
paul718e3742002-12-13 20:15:29 +00002087 }
2088 if (BGP_DEBUG (normal, NORMAL))
ajs6b514742004-12-08 21:03:23 +00002089 zlog_debug ("%s rcvd Refresh %s ORF request", peer->host,
paul718e3742002-12-13 20:15:29 +00002090 when_to_refresh == REFRESH_DEFER ? "Defer" : "Immediate");
2091 if (when_to_refresh == REFRESH_DEFER)
2092 return;
2093 }
2094
2095 /* First update is deferred until ORF or ROUTE-REFRESH is received */
2096 if (CHECK_FLAG (peer->af_sflags[afi][safi], PEER_STATUS_ORF_WAIT_REFRESH))
2097 UNSET_FLAG (peer->af_sflags[afi][safi], PEER_STATUS_ORF_WAIT_REFRESH);
2098
2099 /* Perform route refreshment to the peer */
2100 bgp_announce_route (peer, afi, safi);
2101}
2102
paul94f2b392005-06-28 12:44:16 +00002103static int
paul718e3742002-12-13 20:15:29 +00002104bgp_capability_msg_parse (struct peer *peer, u_char *pnt, bgp_size_t length)
2105{
2106 u_char *end;
Paul Jakma6d582722007-08-06 15:21:45 +00002107 struct capability_mp_data mpc;
2108 struct capability_header *hdr;
paul718e3742002-12-13 20:15:29 +00002109 u_char action;
2110 struct bgp *bgp;
2111 afi_t afi;
2112 safi_t safi;
2113
2114 bgp = peer->bgp;
2115 end = pnt + length;
2116
2117 while (pnt < end)
Paul Jakma6d582722007-08-06 15:21:45 +00002118 {
paul718e3742002-12-13 20:15:29 +00002119 /* We need at least action, capability code and capability length. */
2120 if (pnt + 3 > end)
2121 {
2122 zlog_info ("%s Capability length error", peer->host);
2123 bgp_notify_send (peer, BGP_NOTIFY_CEASE, 0);
2124 return -1;
2125 }
paul718e3742002-12-13 20:15:29 +00002126 action = *pnt;
Paul Jakma6d582722007-08-06 15:21:45 +00002127 hdr = (struct capability_header *)(pnt + 1);
2128
paul718e3742002-12-13 20:15:29 +00002129 /* Action value check. */
2130 if (action != CAPABILITY_ACTION_SET
2131 && action != CAPABILITY_ACTION_UNSET)
2132 {
2133 zlog_info ("%s Capability Action Value error %d",
2134 peer->host, action);
2135 bgp_notify_send (peer, BGP_NOTIFY_CEASE, 0);
2136 return -1;
2137 }
2138
2139 if (BGP_DEBUG (normal, NORMAL))
ajs6b514742004-12-08 21:03:23 +00002140 zlog_debug ("%s CAPABILITY has action: %d, code: %u, length %u",
Paul Jakma6d582722007-08-06 15:21:45 +00002141 peer->host, action, hdr->code, hdr->length);
paul718e3742002-12-13 20:15:29 +00002142
2143 /* Capability length check. */
Paul Jakma6d582722007-08-06 15:21:45 +00002144 if ((pnt + hdr->length + 3) > end)
paul718e3742002-12-13 20:15:29 +00002145 {
2146 zlog_info ("%s Capability length error", peer->host);
2147 bgp_notify_send (peer, BGP_NOTIFY_CEASE, 0);
2148 return -1;
2149 }
2150
Paul Jakma6d582722007-08-06 15:21:45 +00002151 /* Fetch structure to the byte stream. */
2152 memcpy (&mpc, pnt + 3, sizeof (struct capability_mp_data));
2153
paul718e3742002-12-13 20:15:29 +00002154 /* We know MP Capability Code. */
Paul Jakma6d582722007-08-06 15:21:45 +00002155 if (hdr->code == CAPABILITY_CODE_MP)
paul718e3742002-12-13 20:15:29 +00002156 {
Paul Jakma6d582722007-08-06 15:21:45 +00002157 afi = ntohs (mpc.afi);
2158 safi = mpc.safi;
paul718e3742002-12-13 20:15:29 +00002159
2160 /* Ignore capability when override-capability is set. */
2161 if (CHECK_FLAG (peer->flags, PEER_FLAG_OVERRIDE_CAPABILITY))
2162 continue;
Paul Jakma6d582722007-08-06 15:21:45 +00002163
2164 if (!bgp_afi_safi_valid_indices (afi, &safi))
2165 {
2166 if (BGP_DEBUG (normal, NORMAL))
Paul Jakma0b2aa3a2007-10-14 22:32:21 +00002167 zlog_debug ("%s Dynamic Capability MP_EXT afi/safi invalid "
2168 "(%u/%u)", peer->host, afi, safi);
Paul Jakma6d582722007-08-06 15:21:45 +00002169 continue;
2170 }
2171
paul718e3742002-12-13 20:15:29 +00002172 /* Address family check. */
Paul Jakma6d582722007-08-06 15:21:45 +00002173 if (BGP_DEBUG (normal, NORMAL))
2174 zlog_debug ("%s CAPABILITY has %s MP_EXT CAP for afi/safi: %u/%u",
2175 peer->host,
2176 action == CAPABILITY_ACTION_SET
2177 ? "Advertising" : "Removing",
2178 ntohs(mpc.afi) , mpc.safi);
2179
2180 if (action == CAPABILITY_ACTION_SET)
2181 {
2182 peer->afc_recv[afi][safi] = 1;
2183 if (peer->afc[afi][safi])
2184 {
2185 peer->afc_nego[afi][safi] = 1;
2186 bgp_announce_route (peer, afi, safi);
2187 }
2188 }
2189 else
2190 {
2191 peer->afc_recv[afi][safi] = 0;
2192 peer->afc_nego[afi][safi] = 0;
paul718e3742002-12-13 20:15:29 +00002193
Paul Jakma6d582722007-08-06 15:21:45 +00002194 if (peer_active_nego (peer))
2195 bgp_clear_route (peer, afi, safi);
2196 else
2197 BGP_EVENT_ADD (peer, BGP_Stop);
2198 }
paul718e3742002-12-13 20:15:29 +00002199 }
paul718e3742002-12-13 20:15:29 +00002200 else
2201 {
2202 zlog_warn ("%s unrecognized capability code: %d - ignored",
Paul Jakma6d582722007-08-06 15:21:45 +00002203 peer->host, hdr->code);
paul718e3742002-12-13 20:15:29 +00002204 }
Paul Jakma6d582722007-08-06 15:21:45 +00002205 pnt += hdr->length + 3;
paul718e3742002-12-13 20:15:29 +00002206 }
2207 return 0;
2208}
2209
2210/* Dynamic Capability is received. */
Paul Jakma6d582722007-08-06 15:21:45 +00002211int
paul718e3742002-12-13 20:15:29 +00002212bgp_capability_receive (struct peer *peer, bgp_size_t size)
2213{
2214 u_char *pnt;
paul718e3742002-12-13 20:15:29 +00002215
2216 /* Fetch pointer. */
2217 pnt = stream_pnt (peer->ibuf);
2218
2219 if (BGP_DEBUG (normal, NORMAL))
ajs6b514742004-12-08 21:03:23 +00002220 zlog_debug ("%s rcv CAPABILITY", peer->host);
paul718e3742002-12-13 20:15:29 +00002221
2222 /* If peer does not have the capability, send notification. */
2223 if (! CHECK_FLAG (peer->cap, PEER_CAP_DYNAMIC_ADV))
2224 {
2225 plog_err (peer->log, "%s [Error] BGP dynamic capability is not enabled",
2226 peer->host);
2227 bgp_notify_send (peer,
2228 BGP_NOTIFY_HEADER_ERR,
2229 BGP_NOTIFY_HEADER_BAD_MESTYPE);
Paul Jakma0b2aa3a2007-10-14 22:32:21 +00002230 return -1;
paul718e3742002-12-13 20:15:29 +00002231 }
2232
2233 /* Status must be Established. */
2234 if (peer->status != Established)
2235 {
2236 plog_err (peer->log,
2237 "%s [Error] Dynamic capability packet received under status %s", peer->host, LOOKUP (bgp_status_msg, peer->status));
2238 bgp_notify_send (peer, BGP_NOTIFY_FSM_ERR, 0);
Paul Jakma0b2aa3a2007-10-14 22:32:21 +00002239 return -1;
paul718e3742002-12-13 20:15:29 +00002240 }
2241
2242 /* Parse packet. */
Paul Jakma6d582722007-08-06 15:21:45 +00002243 return bgp_capability_msg_parse (peer, pnt, size);
paul718e3742002-12-13 20:15:29 +00002244}
2245
2246/* BGP read utility function. */
paul94f2b392005-06-28 12:44:16 +00002247static int
paul718e3742002-12-13 20:15:29 +00002248bgp_read_packet (struct peer *peer)
2249{
2250 int nbytes;
2251 int readsize;
2252
paul9985f832005-02-09 15:51:56 +00002253 readsize = peer->packet_size - stream_get_endp (peer->ibuf);
paul718e3742002-12-13 20:15:29 +00002254
2255 /* If size is zero then return. */
2256 if (! readsize)
2257 return 0;
2258
2259 /* Read packet from fd. */
pauleb821182004-05-01 08:44:08 +00002260 nbytes = stream_read_unblock (peer->ibuf, peer->fd, readsize);
paul718e3742002-12-13 20:15:29 +00002261
2262 /* If read byte is smaller than zero then error occured. */
2263 if (nbytes < 0)
2264 {
2265 if (errno == EAGAIN)
2266 return -1;
2267
2268 plog_err (peer->log, "%s [Error] bgp_read_packet error: %s",
ajs6099b3b2004-11-20 02:06:59 +00002269 peer->host, safe_strerror (errno));
hasso93406d82005-02-02 14:40:33 +00002270
2271 if (peer->status == Established)
2272 {
2273 if (CHECK_FLAG (peer->sflags, PEER_STATUS_NSF_MODE))
2274 {
2275 peer->last_reset = PEER_DOWN_NSF_CLOSE_SESSION;
2276 SET_FLAG (peer->sflags, PEER_STATUS_NSF_WAIT);
2277 }
2278 else
2279 peer->last_reset = PEER_DOWN_CLOSE_SESSION;
2280 }
2281
paul718e3742002-12-13 20:15:29 +00002282 BGP_EVENT_ADD (peer, TCP_fatal_error);
2283 return -1;
2284 }
2285
2286 /* When read byte is zero : clear bgp peer and return */
2287 if (nbytes == 0)
2288 {
2289 if (BGP_DEBUG (events, EVENTS))
ajs6b514742004-12-08 21:03:23 +00002290 plog_debug (peer->log, "%s [Event] BGP connection closed fd %d",
pauleb821182004-05-01 08:44:08 +00002291 peer->host, peer->fd);
hassoe0701b72004-05-20 09:19:34 +00002292
2293 if (peer->status == Established)
hasso93406d82005-02-02 14:40:33 +00002294 {
2295 if (CHECK_FLAG (peer->sflags, PEER_STATUS_NSF_MODE))
2296 {
2297 peer->last_reset = PEER_DOWN_NSF_CLOSE_SESSION;
2298 SET_FLAG (peer->sflags, PEER_STATUS_NSF_WAIT);
2299 }
2300 else
2301 peer->last_reset = PEER_DOWN_CLOSE_SESSION;
2302 }
hassoe0701b72004-05-20 09:19:34 +00002303
paul718e3742002-12-13 20:15:29 +00002304 BGP_EVENT_ADD (peer, TCP_connection_closed);
2305 return -1;
2306 }
2307
2308 /* We read partial packet. */
paul9985f832005-02-09 15:51:56 +00002309 if (stream_get_endp (peer->ibuf) != peer->packet_size)
paul718e3742002-12-13 20:15:29 +00002310 return -1;
2311
2312 return 0;
2313}
2314
2315/* Marker check. */
paul94f2b392005-06-28 12:44:16 +00002316static int
paul718e3742002-12-13 20:15:29 +00002317bgp_marker_all_one (struct stream *s, int length)
2318{
2319 int i;
2320
2321 for (i = 0; i < length; i++)
2322 if (s->data[i] != 0xff)
2323 return 0;
2324
2325 return 1;
2326}
2327
2328/* Starting point of packet process function. */
2329int
2330bgp_read (struct thread *thread)
2331{
2332 int ret;
2333 u_char type = 0;
2334 struct peer *peer;
2335 bgp_size_t size;
2336 char notify_data_length[2];
2337
2338 /* Yes first of all get peer pointer. */
2339 peer = THREAD_ARG (thread);
2340 peer->t_read = NULL;
2341
2342 /* For non-blocking IO check. */
2343 if (peer->status == Connect)
2344 {
2345 bgp_connect_check (peer);
2346 goto done;
2347 }
2348 else
2349 {
pauleb821182004-05-01 08:44:08 +00002350 if (peer->fd < 0)
paul718e3742002-12-13 20:15:29 +00002351 {
pauleb821182004-05-01 08:44:08 +00002352 zlog_err ("bgp_read peer's fd is negative value %d", peer->fd);
paul718e3742002-12-13 20:15:29 +00002353 return -1;
2354 }
pauleb821182004-05-01 08:44:08 +00002355 BGP_READ_ON (peer->t_read, bgp_read, peer->fd);
paul718e3742002-12-13 20:15:29 +00002356 }
2357
2358 /* Read packet header to determine type of the packet */
2359 if (peer->packet_size == 0)
2360 peer->packet_size = BGP_HEADER_SIZE;
2361
paul9985f832005-02-09 15:51:56 +00002362 if (stream_get_endp (peer->ibuf) < BGP_HEADER_SIZE)
paul718e3742002-12-13 20:15:29 +00002363 {
2364 ret = bgp_read_packet (peer);
2365
2366 /* Header read error or partial read packet. */
2367 if (ret < 0)
2368 goto done;
2369
2370 /* Get size and type. */
paul9985f832005-02-09 15:51:56 +00002371 stream_forward_getp (peer->ibuf, BGP_MARKER_SIZE);
paul718e3742002-12-13 20:15:29 +00002372 memcpy (notify_data_length, stream_pnt (peer->ibuf), 2);
2373 size = stream_getw (peer->ibuf);
2374 type = stream_getc (peer->ibuf);
2375
2376 if (BGP_DEBUG (normal, NORMAL) && type != 2 && type != 0)
ajs6b514742004-12-08 21:03:23 +00002377 zlog_debug ("%s rcv message type %d, length (excl. header) %d",
paul718e3742002-12-13 20:15:29 +00002378 peer->host, type, size - BGP_HEADER_SIZE);
2379
2380 /* Marker check */
paulf5ba3872004-07-09 12:11:31 +00002381 if (((type == BGP_MSG_OPEN) || (type == BGP_MSG_KEEPALIVE))
paul718e3742002-12-13 20:15:29 +00002382 && ! bgp_marker_all_one (peer->ibuf, BGP_MARKER_SIZE))
2383 {
2384 bgp_notify_send (peer,
2385 BGP_NOTIFY_HEADER_ERR,
2386 BGP_NOTIFY_HEADER_NOT_SYNC);
2387 goto done;
2388 }
2389
2390 /* BGP type check. */
2391 if (type != BGP_MSG_OPEN && type != BGP_MSG_UPDATE
2392 && type != BGP_MSG_NOTIFY && type != BGP_MSG_KEEPALIVE
2393 && type != BGP_MSG_ROUTE_REFRESH_NEW
2394 && type != BGP_MSG_ROUTE_REFRESH_OLD
2395 && type != BGP_MSG_CAPABILITY)
2396 {
2397 if (BGP_DEBUG (normal, NORMAL))
ajs6b514742004-12-08 21:03:23 +00002398 plog_debug (peer->log,
paul718e3742002-12-13 20:15:29 +00002399 "%s unknown message type 0x%02x",
2400 peer->host, type);
2401 bgp_notify_send_with_data (peer,
2402 BGP_NOTIFY_HEADER_ERR,
2403 BGP_NOTIFY_HEADER_BAD_MESTYPE,
2404 &type, 1);
2405 goto done;
2406 }
2407 /* Mimimum packet length check. */
2408 if ((size < BGP_HEADER_SIZE)
2409 || (size > BGP_MAX_PACKET_SIZE)
2410 || (type == BGP_MSG_OPEN && size < BGP_MSG_OPEN_MIN_SIZE)
2411 || (type == BGP_MSG_UPDATE && size < BGP_MSG_UPDATE_MIN_SIZE)
2412 || (type == BGP_MSG_NOTIFY && size < BGP_MSG_NOTIFY_MIN_SIZE)
2413 || (type == BGP_MSG_KEEPALIVE && size != BGP_MSG_KEEPALIVE_MIN_SIZE)
2414 || (type == BGP_MSG_ROUTE_REFRESH_NEW && size < BGP_MSG_ROUTE_REFRESH_MIN_SIZE)
2415 || (type == BGP_MSG_ROUTE_REFRESH_OLD && size < BGP_MSG_ROUTE_REFRESH_MIN_SIZE)
2416 || (type == BGP_MSG_CAPABILITY && size < BGP_MSG_CAPABILITY_MIN_SIZE))
2417 {
2418 if (BGP_DEBUG (normal, NORMAL))
ajs6b514742004-12-08 21:03:23 +00002419 plog_debug (peer->log,
paul718e3742002-12-13 20:15:29 +00002420 "%s bad message length - %d for %s",
2421 peer->host, size,
2422 type == 128 ? "ROUTE-REFRESH" :
2423 bgp_type_str[(int) type]);
2424 bgp_notify_send_with_data (peer,
2425 BGP_NOTIFY_HEADER_ERR,
2426 BGP_NOTIFY_HEADER_BAD_MESLEN,
hassoc9e52be2004-09-26 16:09:34 +00002427 (u_char *) notify_data_length, 2);
paul718e3742002-12-13 20:15:29 +00002428 goto done;
2429 }
2430
2431 /* Adjust size to message length. */
2432 peer->packet_size = size;
2433 }
2434
2435 ret = bgp_read_packet (peer);
2436 if (ret < 0)
2437 goto done;
2438
2439 /* Get size and type again. */
2440 size = stream_getw_from (peer->ibuf, BGP_MARKER_SIZE);
2441 type = stream_getc_from (peer->ibuf, BGP_MARKER_SIZE + 2);
2442
2443 /* BGP packet dump function. */
2444 bgp_dump_packet (peer, type, peer->ibuf);
2445
2446 size = (peer->packet_size - BGP_HEADER_SIZE);
2447
2448 /* Read rest of the packet and call each sort of packet routine */
2449 switch (type)
2450 {
2451 case BGP_MSG_OPEN:
2452 peer->open_in++;
paulf5ba3872004-07-09 12:11:31 +00002453 bgp_open_receive (peer, size); /* XXX return value ignored! */
paul718e3742002-12-13 20:15:29 +00002454 break;
2455 case BGP_MSG_UPDATE:
2456 peer->readtime = time(NULL); /* Last read timer reset */
2457 bgp_update_receive (peer, size);
2458 break;
2459 case BGP_MSG_NOTIFY:
2460 bgp_notify_receive (peer, size);
2461 break;
2462 case BGP_MSG_KEEPALIVE:
2463 peer->readtime = time(NULL); /* Last read timer reset */
2464 bgp_keepalive_receive (peer, size);
2465 break;
2466 case BGP_MSG_ROUTE_REFRESH_NEW:
2467 case BGP_MSG_ROUTE_REFRESH_OLD:
2468 peer->refresh_in++;
2469 bgp_route_refresh_receive (peer, size);
2470 break;
2471 case BGP_MSG_CAPABILITY:
2472 peer->dynamic_cap_in++;
2473 bgp_capability_receive (peer, size);
2474 break;
2475 }
2476
2477 /* Clear input buffer. */
2478 peer->packet_size = 0;
2479 if (peer->ibuf)
2480 stream_reset (peer->ibuf);
2481
2482 done:
2483 if (CHECK_FLAG (peer->sflags, PEER_STATUS_ACCEPT_PEER))
2484 {
2485 if (BGP_DEBUG (events, EVENTS))
ajs6b514742004-12-08 21:03:23 +00002486 zlog_debug ("%s [Event] Accepting BGP peer delete", peer->host);
paul718e3742002-12-13 20:15:29 +00002487 peer_delete (peer);
paul718e3742002-12-13 20:15:29 +00002488 }
2489 return 0;
2490}