blob: a5d5b7ffcf0cf86476155899f2acaccd1f56a01f [file] [log] [blame]
paul718e3742002-12-13 20:15:29 +00001/* Zebra's client library.
2 * Copyright (C) 1999 Kunihiro Ishiguro
ajs634f9ea2005-04-11 15:51:40 +00003 * Copyright (C) 2005 Andrew J. Schorr
paul718e3742002-12-13 20:15:29 +00004 *
5 * This file is part of GNU Zebra.
6 *
7 * GNU Zebra is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published
9 * by the Free Software Foundation; either version 2, or (at your
10 * option) any later version.
11 *
12 * GNU Zebra is distributed in the hope that it will be useful, but
13 * WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with GNU Zebra; see the file COPYING. If not, write to the
19 * Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston,
20 * MA 02111-1307, USA.
21 */
22
23#include <zebra.h>
24
25#include "prefix.h"
26#include "stream.h"
ajs634f9ea2005-04-11 15:51:40 +000027#include "buffer.h"
paul718e3742002-12-13 20:15:29 +000028#include "network.h"
29#include "if.h"
30#include "log.h"
31#include "thread.h"
32#include "zclient.h"
33#include "memory.h"
34#include "table.h"
paul718e3742002-12-13 20:15:29 +000035
36/* Zebra client events. */
37enum event {ZCLIENT_SCHEDULE, ZCLIENT_READ, ZCLIENT_CONNECT};
38
39/* Prototype for event manager. */
40static void zclient_event (enum event, struct zclient *);
41
ajs634f9ea2005-04-11 15:51:40 +000042extern struct thread_master *master;
43
Vyacheslav Trushkin271ee732011-11-25 18:51:48 +040044char *zclient_serv_path = NULL;
45
paul718e3742002-12-13 20:15:29 +000046/* This file local debug flag. */
47int zclient_debug = 0;
48
49/* Allocate zclient structure. */
50struct zclient *
51zclient_new ()
52{
53 struct zclient *zclient;
Stephen Hemminger393deb92008-08-18 14:13:29 -070054 zclient = XCALLOC (MTYPE_ZCLIENT, sizeof (struct zclient));
paul718e3742002-12-13 20:15:29 +000055
56 zclient->ibuf = stream_new (ZEBRA_MAX_PACKET_SIZ);
57 zclient->obuf = stream_new (ZEBRA_MAX_PACKET_SIZ);
ajs634f9ea2005-04-11 15:51:40 +000058 zclient->wb = buffer_new(0);
paul718e3742002-12-13 20:15:29 +000059
60 return zclient;
61}
62
Chris Caputo228da422009-07-18 05:44:03 +000063/* This function is only called when exiting, because
ajs634f9ea2005-04-11 15:51:40 +000064 many parts of the code do not check for I/O errors, so they could
65 reference an invalid pointer if the structure was ever freed.
ajs634f9ea2005-04-11 15:51:40 +000066
Chris Caputo228da422009-07-18 05:44:03 +000067 Free zclient structure. */
paul718e3742002-12-13 20:15:29 +000068void
69zclient_free (struct zclient *zclient)
70{
ajs634f9ea2005-04-11 15:51:40 +000071 if (zclient->ibuf)
72 stream_free(zclient->ibuf);
73 if (zclient->obuf)
74 stream_free(zclient->obuf);
75 if (zclient->wb)
76 buffer_free(zclient->wb);
77
paul718e3742002-12-13 20:15:29 +000078 XFREE (MTYPE_ZCLIENT, zclient);
79}
80
81/* Initialize zebra client. Argument redist_default is unwanted
82 redistribute route type. */
83void
84zclient_init (struct zclient *zclient, int redist_default)
85{
86 int i;
87
88 /* Enable zebra client connection by default. */
89 zclient->enable = 1;
90
91 /* Set -1 to the default socket value. */
92 zclient->sock = -1;
93
94 /* Clear redistribution flags. */
95 for (i = 0; i < ZEBRA_ROUTE_MAX; i++)
96 zclient->redist[i] = 0;
97
98 /* Set unwanted redistribute route. bgpd does not need BGP route
99 redistribution. */
100 zclient->redist_default = redist_default;
101 zclient->redist[redist_default] = 1;
102
103 /* Set default-information redistribute to zero. */
104 zclient->default_information = 0;
105
106 /* Schedule first zclient connection. */
107 if (zclient_debug)
ajs8ddca702004-12-07 18:53:52 +0000108 zlog_debug ("zclient start scheduled");
paul718e3742002-12-13 20:15:29 +0000109
110 zclient_event (ZCLIENT_SCHEDULE, zclient);
111}
112
113/* Stop zebra client services. */
114void
115zclient_stop (struct zclient *zclient)
116{
117 if (zclient_debug)
ajs8ddca702004-12-07 18:53:52 +0000118 zlog_debug ("zclient stopped");
paul718e3742002-12-13 20:15:29 +0000119
120 /* Stop threads. */
ajs634f9ea2005-04-11 15:51:40 +0000121 THREAD_OFF(zclient->t_read);
122 THREAD_OFF(zclient->t_connect);
123 THREAD_OFF(zclient->t_write);
124
125 /* Reset streams. */
126 stream_reset(zclient->ibuf);
127 stream_reset(zclient->obuf);
128
129 /* Empty the write buffer. */
130 buffer_reset(zclient->wb);
paul718e3742002-12-13 20:15:29 +0000131
132 /* Close socket. */
133 if (zclient->sock >= 0)
134 {
135 close (zclient->sock);
136 zclient->sock = -1;
137 }
138 zclient->fail = 0;
139}
140
141void
142zclient_reset (struct zclient *zclient)
143{
144 zclient_stop (zclient);
145 zclient_init (zclient, zclient->redist_default);
146}
147
Vyacheslav Trushkin271ee732011-11-25 18:51:48 +0400148#ifdef HAVE_TCP_ZEBRA
149
paul718e3742002-12-13 20:15:29 +0000150/* Make socket to zebra daemon. Return zebra socket. */
Vyacheslav Trushkin271ee732011-11-25 18:51:48 +0400151static int
ajs634f9ea2005-04-11 15:51:40 +0000152zclient_socket(void)
paul718e3742002-12-13 20:15:29 +0000153{
154 int sock;
155 int ret;
156 struct sockaddr_in serv;
157
158 /* We should think about IPv6 connection. */
159 sock = socket (AF_INET, SOCK_STREAM, 0);
160 if (sock < 0)
161 return -1;
162
163 /* Make server socket. */
164 memset (&serv, 0, sizeof (struct sockaddr_in));
165 serv.sin_family = AF_INET;
166 serv.sin_port = htons (ZEBRA_PORT);
Paul Jakma6f0e3f62007-05-10 02:38:51 +0000167#ifdef HAVE_STRUCT_SOCKADDR_IN_SIN_LEN
paul718e3742002-12-13 20:15:29 +0000168 serv.sin_len = sizeof (struct sockaddr_in);
Paul Jakma6f0e3f62007-05-10 02:38:51 +0000169#endif /* HAVE_STRUCT_SOCKADDR_IN_SIN_LEN */
paul718e3742002-12-13 20:15:29 +0000170 serv.sin_addr.s_addr = htonl (INADDR_LOOPBACK);
171
172 /* Connect to zebra. */
173 ret = connect (sock, (struct sockaddr *) &serv, sizeof (serv));
174 if (ret < 0)
175 {
176 close (sock);
177 return -1;
178 }
179 return sock;
180}
181
Vyacheslav Trushkind5cd9a92011-11-30 21:03:44 +0400182#else
Vyacheslav Trushkin271ee732011-11-25 18:51:48 +0400183
paul718e3742002-12-13 20:15:29 +0000184/* For sockaddr_un. */
185#include <sys/un.h>
186
Vyacheslav Trushkin271ee732011-11-25 18:51:48 +0400187static int
hasso8c328f12004-10-05 21:01:23 +0000188zclient_socket_un (const char *path)
paul718e3742002-12-13 20:15:29 +0000189{
190 int ret;
191 int sock, len;
192 struct sockaddr_un addr;
193
194 sock = socket (AF_UNIX, SOCK_STREAM, 0);
195 if (sock < 0)
196 return -1;
197
198 /* Make server socket. */
199 memset (&addr, 0, sizeof (struct sockaddr_un));
200 addr.sun_family = AF_UNIX;
201 strncpy (addr.sun_path, path, strlen (path));
Paul Jakma6f0e3f62007-05-10 02:38:51 +0000202#ifdef HAVE_STRUCT_SOCKADDR_UN_SUN_LEN
paul718e3742002-12-13 20:15:29 +0000203 len = addr.sun_len = SUN_LEN(&addr);
204#else
205 len = sizeof (addr.sun_family) + strlen (addr.sun_path);
Paul Jakma6f0e3f62007-05-10 02:38:51 +0000206#endif /* HAVE_STRUCT_SOCKADDR_UN_SUN_LEN */
paul718e3742002-12-13 20:15:29 +0000207
208 ret = connect (sock, (struct sockaddr *) &addr, len);
209 if (ret < 0)
210 {
211 close (sock);
212 return -1;
213 }
214 return sock;
215}
216
Vyacheslav Trushkind5cd9a92011-11-30 21:03:44 +0400217#endif /* HAVE_TCP_ZEBRA */
218
Vyacheslav Trushkin271ee732011-11-25 18:51:48 +0400219/**
220 * Connect to zebra daemon.
221 * @param zclient a pointer to zclient structure
222 * @return socket fd just to make sure that connection established
223 * @see zclient_init
224 * @see zclient_new
225 */
226int
227zclient_socket_connect (struct zclient *zclient)
228{
229#ifdef HAVE_TCP_ZEBRA
230 zclient->sock = zclient_socket ();
231#else
232 zclient->sock = zclient_socket_un (zclient_serv_path ? zclient_serv_path : ZEBRA_SERV_PATH);
233#endif
234 return zclient->sock;
235}
236
ajs634f9ea2005-04-11 15:51:40 +0000237static int
238zclient_failed(struct zclient *zclient)
239{
240 zclient->fail++;
241 zclient_stop(zclient);
242 zclient_event(ZCLIENT_CONNECT, zclient);
243 return -1;
244}
245
246static int
247zclient_flush_data(struct thread *thread)
248{
249 struct zclient *zclient = THREAD_ARG(thread);
250
251 zclient->t_write = NULL;
252 if (zclient->sock < 0)
253 return -1;
254 switch (buffer_flush_available(zclient->wb, zclient->sock))
255 {
256 case BUFFER_ERROR:
257 zlog_warn("%s: buffer_flush_available failed on zclient fd %d, closing",
258 __func__, zclient->sock);
259 return zclient_failed(zclient);
260 break;
261 case BUFFER_PENDING:
262 zclient->t_write = thread_add_write(master, zclient_flush_data,
263 zclient, zclient->sock);
264 break;
265 case BUFFER_EMPTY:
266 break;
267 }
268 return 0;
269}
270
paul718e3742002-12-13 20:15:29 +0000271int
ajs634f9ea2005-04-11 15:51:40 +0000272zclient_send_message(struct zclient *zclient)
273{
274 if (zclient->sock < 0)
275 return -1;
276 switch (buffer_write(zclient->wb, zclient->sock, STREAM_DATA(zclient->obuf),
277 stream_get_endp(zclient->obuf)))
278 {
279 case BUFFER_ERROR:
280 zlog_warn("%s: buffer_write failed to zclient fd %d, closing",
281 __func__, zclient->sock);
282 return zclient_failed(zclient);
283 break;
284 case BUFFER_EMPTY:
285 THREAD_OFF(zclient->t_write);
286 break;
287 case BUFFER_PENDING:
288 THREAD_WRITE_ON(master, zclient->t_write,
289 zclient_flush_data, zclient, zclient->sock);
290 break;
291 }
292 return 0;
293}
294
pauld2110862006-01-17 17:43:18 +0000295void
paulc1b98002006-01-16 01:54:02 +0000296zclient_create_header (struct stream *s, uint16_t command)
297{
298 /* length placeholder, caller can update */
299 stream_putw (s, ZEBRA_HEADER_SIZE);
300 stream_putc (s, ZEBRA_HEADER_MARKER);
301 stream_putc (s, ZSERV_VERSION);
302 stream_putw (s, command);
303}
304
ajs634f9ea2005-04-11 15:51:40 +0000305/* Send simple Zebra message. */
306static int
paul718e3742002-12-13 20:15:29 +0000307zebra_message_send (struct zclient *zclient, int command)
308{
309 struct stream *s;
310
311 /* Get zclient output buffer. */
312 s = zclient->obuf;
313 stream_reset (s);
314
315 /* Send very simple command only Zebra message. */
paulc1b98002006-01-16 01:54:02 +0000316 zclient_create_header (s, command);
317
ajs634f9ea2005-04-11 15:51:40 +0000318 return zclient_send_message(zclient);
paul718e3742002-12-13 20:15:29 +0000319}
320
321/* Make connection to zebra daemon. */
322int
323zclient_start (struct zclient *zclient)
324{
325 int i;
326
327 if (zclient_debug)
ajs8ddca702004-12-07 18:53:52 +0000328 zlog_debug ("zclient_start is called");
paul718e3742002-12-13 20:15:29 +0000329
330 /* zclient is disabled. */
331 if (! zclient->enable)
332 return 0;
333
334 /* If already connected to the zebra. */
335 if (zclient->sock >= 0)
336 return 0;
337
338 /* Check connect thread. */
339 if (zclient->t_connect)
340 return 0;
341
Vyacheslav Trushkin271ee732011-11-25 18:51:48 +0400342 if (zclient_socket_connect(zclient) < 0)
paul718e3742002-12-13 20:15:29 +0000343 {
344 if (zclient_debug)
ajs8ddca702004-12-07 18:53:52 +0000345 zlog_debug ("zclient connection fail");
paul718e3742002-12-13 20:15:29 +0000346 zclient->fail++;
347 zclient_event (ZCLIENT_CONNECT, zclient);
348 return -1;
349 }
350
ajs634f9ea2005-04-11 15:51:40 +0000351 if (set_nonblocking(zclient->sock) < 0)
352 zlog_warn("%s: set_nonblocking(%d) failed", __func__, zclient->sock);
353
paul718e3742002-12-13 20:15:29 +0000354 /* Clear fail count. */
355 zclient->fail = 0;
356 if (zclient_debug)
ajs8ddca702004-12-07 18:53:52 +0000357 zlog_debug ("zclient connect success with socket [%d]", zclient->sock);
paul718e3742002-12-13 20:15:29 +0000358
359 /* Create read thread. */
360 zclient_event (ZCLIENT_READ, zclient);
361
hasso18a6dce2004-10-03 18:18:34 +0000362 /* We need router-id information. */
363 zebra_message_send (zclient, ZEBRA_ROUTER_ID_ADD);
364
Dmitry Tejblum08a7a912010-10-18 19:05:39 +0400365 /* We need interface information. */
366 zebra_message_send (zclient, ZEBRA_INTERFACE_ADD);
367
paul718e3742002-12-13 20:15:29 +0000368 /* Flush all redistribute request. */
369 for (i = 0; i < ZEBRA_ROUTE_MAX; i++)
370 if (i != zclient->redist_default && zclient->redist[i])
ajs634f9ea2005-04-11 15:51:40 +0000371 zebra_redistribute_send (ZEBRA_REDISTRIBUTE_ADD, zclient, i);
paul718e3742002-12-13 20:15:29 +0000372
373 /* If default information is needed. */
374 if (zclient->default_information)
375 zebra_message_send (zclient, ZEBRA_REDISTRIBUTE_DEFAULT_ADD);
376
377 return 0;
378}
379
380/* This function is a wrapper function for calling zclient_start from
381 timer or event thread. */
ajs634f9ea2005-04-11 15:51:40 +0000382static int
paul718e3742002-12-13 20:15:29 +0000383zclient_connect (struct thread *t)
384{
385 struct zclient *zclient;
386
387 zclient = THREAD_ARG (t);
388 zclient->t_connect = NULL;
389
390 if (zclient_debug)
ajs8ddca702004-12-07 18:53:52 +0000391 zlog_debug ("zclient_connect is called");
paul718e3742002-12-13 20:15:29 +0000392
393 return zclient_start (zclient);
394}
395
paul0a589352004-05-08 11:48:26 +0000396 /*
397 * "xdr_encode"-like interface that allows daemon (client) to send
398 * a message to zebra server for a route that needs to be
399 * added/deleted to the kernel. Info about the route is specified
400 * by the caller in a struct zapi_ipv4. zapi_ipv4_read() then writes
401 * the info down the zclient socket using the stream_* functions.
402 *
403 * The corresponding read ("xdr_decode") function on the server
404 * side is zread_ipv4_add()/zread_ipv4_delete().
405 *
406 * 0 1 2 3 4 5 6 7 8 9 A B C D E F 0 1 2 3 4 5 6 7 8 9 A B C D E F
407 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
408 * | Length (2) | Command | Route Type |
409 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
410 * | ZEBRA Flags | Message Flags | Prefix length |
411 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
412 * | Destination IPv4 Prefix for route |
413 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
414 * | Nexthop count |
415 * +-+-+-+-+-+-+-+-+
416 *
417 *
418 * A number of IPv4 nexthop(s) or nexthop interface index(es) are then
419 * described, as per the Nexthop count. Each nexthop described as:
420 *
421 * +-+-+-+-+-+-+-+-+
422 * | Nexthop Type | Set to one of ZEBRA_NEXTHOP_*
423 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
424 * | IPv4 Nexthop address or Interface Index number |
425 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
426 *
427 * Alternatively, if the flags field has ZEBRA_FLAG_BLACKHOLE or
428 * ZEBRA_FLAG_REJECT is set then Nexthop count is set to 1, then _no_
429 * nexthop information is provided, and the message describes a prefix
430 * to blackhole or reject route.
431 *
432 * If ZAPI_MESSAGE_DISTANCE is set, the distance value is written as a 1
433 * byte value.
434 *
435 * If ZAPI_MESSAGE_METRIC is set, the metric value is written as an 8
436 * byte value.
437 *
438 * XXX: No attention paid to alignment.
439 */
paul718e3742002-12-13 20:15:29 +0000440int
paul0a589352004-05-08 11:48:26 +0000441zapi_ipv4_route (u_char cmd, struct zclient *zclient, struct prefix_ipv4 *p,
442 struct zapi_ipv4 *api)
paul718e3742002-12-13 20:15:29 +0000443{
444 int i;
445 int psize;
446 struct stream *s;
447
448 /* Reset stream. */
449 s = zclient->obuf;
450 stream_reset (s);
paulc1b98002006-01-16 01:54:02 +0000451
452 zclient_create_header (s, cmd);
453
454 /* Put type and nexthop. */
paul718e3742002-12-13 20:15:29 +0000455 stream_putc (s, api->type);
456 stream_putc (s, api->flags);
457 stream_putc (s, api->message);
paul0a589352004-05-08 11:48:26 +0000458
paul718e3742002-12-13 20:15:29 +0000459 /* Put prefix information. */
460 psize = PSIZE (p->prefixlen);
461 stream_putc (s, p->prefixlen);
paul0a589352004-05-08 11:48:26 +0000462 stream_write (s, (u_char *) & p->prefix, psize);
paul718e3742002-12-13 20:15:29 +0000463
464 /* Nexthop, ifindex, distance and metric information. */
465 if (CHECK_FLAG (api->message, ZAPI_MESSAGE_NEXTHOP))
466 {
paul595db7f2003-05-25 21:35:06 +0000467 if (CHECK_FLAG (api->flags, ZEBRA_FLAG_BLACKHOLE))
468 {
469 stream_putc (s, 1);
470 stream_putc (s, ZEBRA_NEXTHOP_BLACKHOLE);
paul0a589352004-05-08 11:48:26 +0000471 /* XXX assert(api->nexthop_num == 0); */
472 /* XXX assert(api->ifindex_num == 0); */
paul595db7f2003-05-25 21:35:06 +0000473 }
474 else
475 stream_putc (s, api->nexthop_num + api->ifindex_num);
paul718e3742002-12-13 20:15:29 +0000476
477 for (i = 0; i < api->nexthop_num; i++)
paul595db7f2003-05-25 21:35:06 +0000478 {
479 stream_putc (s, ZEBRA_NEXTHOP_IPV4);
480 stream_put_in_addr (s, api->nexthop[i]);
481 }
paul718e3742002-12-13 20:15:29 +0000482 for (i = 0; i < api->ifindex_num; i++)
paul595db7f2003-05-25 21:35:06 +0000483 {
484 stream_putc (s, ZEBRA_NEXTHOP_IFINDEX);
485 stream_putl (s, api->ifindex[i]);
486 }
paul718e3742002-12-13 20:15:29 +0000487 }
488
489 if (CHECK_FLAG (api->message, ZAPI_MESSAGE_DISTANCE))
490 stream_putc (s, api->distance);
491 if (CHECK_FLAG (api->message, ZAPI_MESSAGE_METRIC))
492 stream_putl (s, api->metric);
493
494 /* Put length at the first point of the stream. */
495 stream_putw_at (s, 0, stream_get_endp (s));
496
ajs634f9ea2005-04-11 15:51:40 +0000497 return zclient_send_message(zclient);
paul718e3742002-12-13 20:15:29 +0000498}
499
500#ifdef HAVE_IPV6
501int
paul0a589352004-05-08 11:48:26 +0000502zapi_ipv6_route (u_char cmd, struct zclient *zclient, struct prefix_ipv6 *p,
paul718e3742002-12-13 20:15:29 +0000503 struct zapi_ipv6 *api)
504{
505 int i;
506 int psize;
507 struct stream *s;
508
509 /* Reset stream. */
510 s = zclient->obuf;
511 stream_reset (s);
512
paulc1b98002006-01-16 01:54:02 +0000513 zclient_create_header (s, cmd);
paul718e3742002-12-13 20:15:29 +0000514
paulc1b98002006-01-16 01:54:02 +0000515 /* Put type and nexthop. */
paul718e3742002-12-13 20:15:29 +0000516 stream_putc (s, api->type);
517 stream_putc (s, api->flags);
518 stream_putc (s, api->message);
519
520 /* Put prefix information. */
521 psize = PSIZE (p->prefixlen);
522 stream_putc (s, p->prefixlen);
523 stream_write (s, (u_char *)&p->prefix, psize);
524
525 /* Nexthop, ifindex, distance and metric information. */
526 if (CHECK_FLAG (api->message, ZAPI_MESSAGE_NEXTHOP))
527 {
528 stream_putc (s, api->nexthop_num + api->ifindex_num);
529
530 for (i = 0; i < api->nexthop_num; i++)
531 {
532 stream_putc (s, ZEBRA_NEXTHOP_IPV6);
533 stream_write (s, (u_char *)api->nexthop[i], 16);
534 }
535 for (i = 0; i < api->ifindex_num; i++)
536 {
537 stream_putc (s, ZEBRA_NEXTHOP_IFINDEX);
538 stream_putl (s, api->ifindex[i]);
539 }
540 }
541
542 if (CHECK_FLAG (api->message, ZAPI_MESSAGE_DISTANCE))
543 stream_putc (s, api->distance);
544 if (CHECK_FLAG (api->message, ZAPI_MESSAGE_METRIC))
545 stream_putl (s, api->metric);
546
547 /* Put length at the first point of the stream. */
548 stream_putw_at (s, 0, stream_get_endp (s));
549
ajs634f9ea2005-04-11 15:51:40 +0000550 return zclient_send_message(zclient);
paul718e3742002-12-13 20:15:29 +0000551}
paul718e3742002-12-13 20:15:29 +0000552#endif /* HAVE_IPV6 */
553
paul0a589352004-05-08 11:48:26 +0000554/*
555 * send a ZEBRA_REDISTRIBUTE_ADD or ZEBRA_REDISTRIBUTE_DELETE
556 * for the route type (ZEBRA_ROUTE_KERNEL etc.). The zebra server will
557 * then set/unset redist[type] in the client handle (a struct zserv) for the
558 * sending client
559 */
paul718e3742002-12-13 20:15:29 +0000560int
ajs634f9ea2005-04-11 15:51:40 +0000561zebra_redistribute_send (int command, struct zclient *zclient, int type)
paul718e3742002-12-13 20:15:29 +0000562{
paul718e3742002-12-13 20:15:29 +0000563 struct stream *s;
564
ajs634f9ea2005-04-11 15:51:40 +0000565 s = zclient->obuf;
566 stream_reset(s);
paul718e3742002-12-13 20:15:29 +0000567
paulc1b98002006-01-16 01:54:02 +0000568 zclient_create_header (s, command);
paul718e3742002-12-13 20:15:29 +0000569 stream_putc (s, type);
paulc1b98002006-01-16 01:54:02 +0000570
571 stream_putw_at (s, 0, stream_get_endp (s));
572
ajs634f9ea2005-04-11 15:51:40 +0000573 return zclient_send_message(zclient);
paul718e3742002-12-13 20:15:29 +0000574}
575
hasso18a6dce2004-10-03 18:18:34 +0000576/* Router-id update from zebra daemon. */
577void
578zebra_router_id_update_read (struct stream *s, struct prefix *rid)
579{
580 int plen;
581
582 /* Fetch interface address. */
583 rid->family = stream_getc (s);
584
585 plen = prefix_blen (rid);
586 stream_get (&rid->u.prefix, s, plen);
587 rid->prefixlen = stream_getc (s);
588}
589
paul718e3742002-12-13 20:15:29 +0000590/* Interface addition from zebra daemon. */
paul0a589352004-05-08 11:48:26 +0000591/*
592 * The format of the message sent with type ZEBRA_INTERFACE_ADD or
593 * ZEBRA_INTERFACE_DELETE from zebra to the client is:
594 * 0 1 2 3
595 * 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
596 * +-+-+-+-+-+-+-+-+
597 * | type |
598 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
599 * | ifname |
600 * | |
601 * | |
602 * | |
603 * | |
604 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
605 * | ifindex |
606 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
607 * | if_flags |
paulc77d4542006-01-11 01:59:04 +0000608 * | |
paul0a589352004-05-08 11:48:26 +0000609 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
610 * | metric |
611 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
612 * | ifmtu |
613 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
614 * | ifmtu6 |
615 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
616 * | bandwidth |
617 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
618 * | sockaddr_dl |
619 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
620 */
621
paul718e3742002-12-13 20:15:29 +0000622struct interface *
623zebra_interface_add_read (struct stream *s)
624{
625 struct interface *ifp;
paul02ff83c2004-06-11 11:27:03 +0000626 char ifname_tmp[INTERFACE_NAMSIZ];
paul718e3742002-12-13 20:15:29 +0000627
628 /* Read interface name. */
629 stream_get (ifname_tmp, s, INTERFACE_NAMSIZ);
630
ajsa3491982005-04-02 22:50:38 +0000631 /* Lookup/create interface by name. */
632 ifp = if_get_by_name_len (ifname_tmp, strnlen(ifname_tmp, INTERFACE_NAMSIZ));
paul718e3742002-12-13 20:15:29 +0000633
634 /* Read interface's index. */
635 ifp->ifindex = stream_getl (s);
636
637 /* Read interface's value. */
paul2e3b2e42002-12-13 21:03:13 +0000638 ifp->status = stream_getc (s);
paulc77d4542006-01-11 01:59:04 +0000639 ifp->flags = stream_getq (s);
paul718e3742002-12-13 20:15:29 +0000640 ifp->metric = stream_getl (s);
641 ifp->mtu = stream_getl (s);
paul0a589352004-05-08 11:48:26 +0000642 ifp->mtu6 = stream_getl (s);
paul718e3742002-12-13 20:15:29 +0000643 ifp->bandwidth = stream_getl (s);
Paul Jakma6f0e3f62007-05-10 02:38:51 +0000644#ifdef HAVE_STRUCT_SOCKADDR_DL
paul718e3742002-12-13 20:15:29 +0000645 stream_get (&ifp->sdl, s, sizeof (ifp->sdl));
646#else
647 ifp->hw_addr_len = stream_getl (s);
648 if (ifp->hw_addr_len)
649 stream_get (ifp->hw_addr, s, ifp->hw_addr_len);
Paul Jakma6f0e3f62007-05-10 02:38:51 +0000650#endif /* HAVE_STRUCT_SOCKADDR_DL */
paul718e3742002-12-13 20:15:29 +0000651
652 return ifp;
653}
654
paul0a589352004-05-08 11:48:26 +0000655/*
656 * Read interface up/down msg (ZEBRA_INTERFACE_UP/ZEBRA_INTERFACE_DOWN)
657 * from zebra server. The format of this message is the same as
658 * that sent for ZEBRA_INTERFACE_ADD/ZEBRA_INTERFACE_DELETE (see
659 * comments for zebra_interface_add_read), except that no sockaddr_dl
660 * is sent at the tail of the message.
661 */
paul718e3742002-12-13 20:15:29 +0000662struct interface *
663zebra_interface_state_read (struct stream *s)
664{
665 struct interface *ifp;
paul02ff83c2004-06-11 11:27:03 +0000666 char ifname_tmp[INTERFACE_NAMSIZ];
paul718e3742002-12-13 20:15:29 +0000667
668 /* Read interface name. */
669 stream_get (ifname_tmp, s, INTERFACE_NAMSIZ);
670
671 /* Lookup this by interface index. */
ajsa3491982005-04-02 22:50:38 +0000672 ifp = if_lookup_by_name_len (ifname_tmp,
673 strnlen(ifname_tmp, INTERFACE_NAMSIZ));
paul718e3742002-12-13 20:15:29 +0000674
675 /* If such interface does not exist, indicate an error */
676 if (! ifp)
677 return NULL;
678
679 /* Read interface's index. */
680 ifp->ifindex = stream_getl (s);
681
682 /* Read interface's value. */
paul2e3b2e42002-12-13 21:03:13 +0000683 ifp->status = stream_getc (s);
paulc77d4542006-01-11 01:59:04 +0000684 ifp->flags = stream_getq (s);
paul718e3742002-12-13 20:15:29 +0000685 ifp->metric = stream_getl (s);
686 ifp->mtu = stream_getl (s);
paul0a589352004-05-08 11:48:26 +0000687 ifp->mtu6 = stream_getl (s);
paul718e3742002-12-13 20:15:29 +0000688 ifp->bandwidth = stream_getl (s);
689
690 return ifp;
691}
692
paul0a589352004-05-08 11:48:26 +0000693/*
694 * format of message for address additon is:
695 * 0
696 * 0 1 2 3 4 5 6 7
697 * +-+-+-+-+-+-+-+-+
698 * | type | ZEBRA_INTERFACE_ADDRESS_ADD or
699 * +-+-+-+-+-+-+-+-+ ZEBRA_INTERFACE_ADDRES_DELETE
700 * | |
701 * + +
702 * | ifindex |
703 * + +
704 * | |
705 * + +
706 * | |
707 * +-+-+-+-+-+-+-+-+
708 * | ifc_flags | flags for connected address
709 * +-+-+-+-+-+-+-+-+
710 * | addr_family |
711 * +-+-+-+-+-+-+-+-+
712 * | addr... |
713 * : :
714 * | |
715 * +-+-+-+-+-+-+-+-+
716 * | addr_len | len of addr. E.g., addr_len = 4 for ipv4 addrs.
717 * +-+-+-+-+-+-+-+-+
718 * | daddr.. |
719 * : :
720 * | |
721 * +-+-+-+-+-+-+-+-+
722 *
723 */
724
hasso18a6dce2004-10-03 18:18:34 +0000725void
726zebra_interface_if_set_value (struct stream *s, struct interface *ifp)
727{
728 /* Read interface's index. */
729 ifp->ifindex = stream_getl (s);
hasso508ec912004-10-23 14:26:49 +0000730 ifp->status = stream_getc (s);
hasso18a6dce2004-10-03 18:18:34 +0000731
732 /* Read interface's value. */
paulc77d4542006-01-11 01:59:04 +0000733 ifp->flags = stream_getq (s);
hasso18a6dce2004-10-03 18:18:34 +0000734 ifp->metric = stream_getl (s);
735 ifp->mtu = stream_getl (s);
hasso508ec912004-10-23 14:26:49 +0000736 ifp->mtu6 = stream_getl (s);
hasso18a6dce2004-10-03 18:18:34 +0000737 ifp->bandwidth = stream_getl (s);
738}
739
hasso3fb9cd62004-10-19 19:44:43 +0000740static int
741memconstant(const void *s, int c, size_t n)
742{
743 const u_char *p = s;
744
745 while (n-- > 0)
746 if (*p++ != c)
747 return 0;
748 return 1;
749}
750
paul718e3742002-12-13 20:15:29 +0000751struct connected *
paul0a589352004-05-08 11:48:26 +0000752zebra_interface_address_read (int type, struct stream *s)
paul718e3742002-12-13 20:15:29 +0000753{
754 unsigned int ifindex;
755 struct interface *ifp;
756 struct connected *ifc;
paul0a589352004-05-08 11:48:26 +0000757 struct prefix p, d;
paul718e3742002-12-13 20:15:29 +0000758 int family;
759 int plen;
paul0a589352004-05-08 11:48:26 +0000760 u_char ifc_flags;
761
762 memset (&p, 0, sizeof(p));
763 memset (&d, 0, sizeof(d));
paul718e3742002-12-13 20:15:29 +0000764
765 /* Get interface index. */
766 ifindex = stream_getl (s);
767
768 /* Lookup index. */
769 ifp = if_lookup_by_index (ifindex);
770 if (ifp == NULL)
771 {
paul0a589352004-05-08 11:48:26 +0000772 zlog_warn ("zebra_interface_address_read(%s): "
773 "Can't find interface by ifindex: %d ",
774 (type == ZEBRA_INTERFACE_ADDRESS_ADD? "ADD" : "DELETE"),
775 ifindex);
paul718e3742002-12-13 20:15:29 +0000776 return NULL;
777 }
778
779 /* Fetch flag. */
paul0a589352004-05-08 11:48:26 +0000780 ifc_flags = stream_getc (s);
paul718e3742002-12-13 20:15:29 +0000781
782 /* Fetch interface address. */
783 family = p.family = stream_getc (s);
784
paul0a589352004-05-08 11:48:26 +0000785 plen = prefix_blen (&p);
786 stream_get (&p.u.prefix, s, plen);
paul718e3742002-12-13 20:15:29 +0000787 p.prefixlen = stream_getc (s);
788
789 /* Fetch destination address. */
paul0a589352004-05-08 11:48:26 +0000790 stream_get (&d.u.prefix, s, plen);
paul718e3742002-12-13 20:15:29 +0000791 d.family = family;
792
paul0a589352004-05-08 11:48:26 +0000793 if (type == ZEBRA_INTERFACE_ADDRESS_ADD)
794 {
hasso3fb9cd62004-10-19 19:44:43 +0000795 /* N.B. NULL destination pointers are encoded as all zeroes */
796 ifc = connected_add_by_prefix(ifp, &p,(memconstant(&d.u.prefix,0,plen) ?
797 NULL : &d));
paul0a589352004-05-08 11:48:26 +0000798 if (ifc != NULL)
Andrew J. Schorre4529632006-12-12 19:18:21 +0000799 {
800 ifc->flags = ifc_flags;
801 if (ifc->destination)
802 ifc->destination->prefixlen = ifc->address->prefixlen;
803 }
paul0a589352004-05-08 11:48:26 +0000804 }
805 else
806 {
807 assert (type == ZEBRA_INTERFACE_ADDRESS_DELETE);
808 ifc = connected_delete_by_prefix(ifp, &p);
809 }
paul718e3742002-12-13 20:15:29 +0000810
811 return ifc;
812}
paul0a589352004-05-08 11:48:26 +0000813
paul718e3742002-12-13 20:15:29 +0000814
815/* Zebra client message read function. */
ajs634f9ea2005-04-11 15:51:40 +0000816static int
paul718e3742002-12-13 20:15:29 +0000817zclient_read (struct thread *thread)
818{
819 int ret;
ajs634f9ea2005-04-11 15:51:40 +0000820 size_t already;
paulc1b98002006-01-16 01:54:02 +0000821 uint16_t length, command;
822 uint8_t marker, version;
paul718e3742002-12-13 20:15:29 +0000823 struct zclient *zclient;
824
825 /* Get socket to zebra. */
paul718e3742002-12-13 20:15:29 +0000826 zclient = THREAD_ARG (thread);
827 zclient->t_read = NULL;
828
ajs634f9ea2005-04-11 15:51:40 +0000829 /* Read zebra header (if we don't have it already). */
830 if ((already = stream_get_endp(zclient->ibuf)) < ZEBRA_HEADER_SIZE)
paul718e3742002-12-13 20:15:29 +0000831 {
ajs634f9ea2005-04-11 15:51:40 +0000832 ssize_t nbyte;
833 if (((nbyte = stream_read_try(zclient->ibuf, zclient->sock,
834 ZEBRA_HEADER_SIZE-already)) == 0) ||
835 (nbyte == -1))
836 {
837 if (zclient_debug)
838 zlog_debug ("zclient connection closed socket [%d].", zclient->sock);
839 return zclient_failed(zclient);
840 }
841 if (nbyte != (ssize_t)(ZEBRA_HEADER_SIZE-already))
842 {
843 /* Try again later. */
844 zclient_event (ZCLIENT_READ, zclient);
845 return 0;
846 }
847 already = ZEBRA_HEADER_SIZE;
paul718e3742002-12-13 20:15:29 +0000848 }
849
ajs634f9ea2005-04-11 15:51:40 +0000850 /* Reset to read from the beginning of the incoming packet. */
851 stream_set_getp(zclient->ibuf, 0);
paul718e3742002-12-13 20:15:29 +0000852
paulc1b98002006-01-16 01:54:02 +0000853 /* Fetch header values. */
paul718e3742002-12-13 20:15:29 +0000854 length = stream_getw (zclient->ibuf);
paulc1b98002006-01-16 01:54:02 +0000855 marker = stream_getc (zclient->ibuf);
856 version = stream_getc (zclient->ibuf);
857 command = stream_getw (zclient->ibuf);
858
859 if (marker != ZEBRA_HEADER_MARKER || version != ZSERV_VERSION)
860 {
861 zlog_err("%s: socket %d version mismatch, marker %d, version %d",
862 __func__, zclient->sock, marker, version);
863 return zclient_failed(zclient);
864 }
865
ajs634f9ea2005-04-11 15:51:40 +0000866 if (length < ZEBRA_HEADER_SIZE)
paul718e3742002-12-13 20:15:29 +0000867 {
ajs634f9ea2005-04-11 15:51:40 +0000868 zlog_err("%s: socket %d message length %u is less than %d ",
869 __func__, zclient->sock, length, ZEBRA_HEADER_SIZE);
870 return zclient_failed(zclient);
paul718e3742002-12-13 20:15:29 +0000871 }
ajs634f9ea2005-04-11 15:51:40 +0000872
873 /* Length check. */
874 if (length > STREAM_SIZE(zclient->ibuf))
875 {
876 struct stream *ns;
877 zlog_warn("%s: message size %u exceeds buffer size %lu, expanding...",
878 __func__, length, (u_long)STREAM_SIZE(zclient->ibuf));
879 ns = stream_new(length);
880 stream_copy(ns, zclient->ibuf);
881 stream_free (zclient->ibuf);
882 zclient->ibuf = ns;
883 }
paul718e3742002-12-13 20:15:29 +0000884
885 /* Read rest of zebra packet. */
ajs634f9ea2005-04-11 15:51:40 +0000886 if (already < length)
887 {
888 ssize_t nbyte;
889 if (((nbyte = stream_read_try(zclient->ibuf, zclient->sock,
890 length-already)) == 0) ||
891 (nbyte == -1))
892 {
893 if (zclient_debug)
894 zlog_debug("zclient connection closed socket [%d].", zclient->sock);
895 return zclient_failed(zclient);
896 }
897 if (nbyte != (ssize_t)(length-already))
898 {
899 /* Try again later. */
900 zclient_event (ZCLIENT_READ, zclient);
901 return 0;
902 }
903 }
904
905 length -= ZEBRA_HEADER_SIZE;
paul718e3742002-12-13 20:15:29 +0000906
paul0a589352004-05-08 11:48:26 +0000907 if (zclient_debug)
ajs8ddca702004-12-07 18:53:52 +0000908 zlog_debug("zclient 0x%p command 0x%x \n", zclient, command);
paul0a589352004-05-08 11:48:26 +0000909
paul718e3742002-12-13 20:15:29 +0000910 switch (command)
911 {
hasso18a6dce2004-10-03 18:18:34 +0000912 case ZEBRA_ROUTER_ID_UPDATE:
913 if (zclient->router_id_update)
914 ret = (*zclient->router_id_update) (command, zclient, length);
915 break;
paul718e3742002-12-13 20:15:29 +0000916 case ZEBRA_INTERFACE_ADD:
917 if (zclient->interface_add)
918 ret = (*zclient->interface_add) (command, zclient, length);
919 break;
920 case ZEBRA_INTERFACE_DELETE:
921 if (zclient->interface_delete)
922 ret = (*zclient->interface_delete) (command, zclient, length);
923 break;
924 case ZEBRA_INTERFACE_ADDRESS_ADD:
925 if (zclient->interface_address_add)
926 ret = (*zclient->interface_address_add) (command, zclient, length);
927 break;
928 case ZEBRA_INTERFACE_ADDRESS_DELETE:
929 if (zclient->interface_address_delete)
930 ret = (*zclient->interface_address_delete) (command, zclient, length);
931 break;
932 case ZEBRA_INTERFACE_UP:
933 if (zclient->interface_up)
934 ret = (*zclient->interface_up) (command, zclient, length);
935 break;
936 case ZEBRA_INTERFACE_DOWN:
937 if (zclient->interface_down)
938 ret = (*zclient->interface_down) (command, zclient, length);
939 break;
940 case ZEBRA_IPV4_ROUTE_ADD:
941 if (zclient->ipv4_route_add)
942 ret = (*zclient->ipv4_route_add) (command, zclient, length);
943 break;
944 case ZEBRA_IPV4_ROUTE_DELETE:
945 if (zclient->ipv4_route_delete)
946 ret = (*zclient->ipv4_route_delete) (command, zclient, length);
947 break;
948 case ZEBRA_IPV6_ROUTE_ADD:
949 if (zclient->ipv6_route_add)
950 ret = (*zclient->ipv6_route_add) (command, zclient, length);
951 break;
952 case ZEBRA_IPV6_ROUTE_DELETE:
953 if (zclient->ipv6_route_delete)
954 ret = (*zclient->ipv6_route_delete) (command, zclient, length);
955 break;
956 default:
957 break;
958 }
959
ajs634f9ea2005-04-11 15:51:40 +0000960 if (zclient->sock < 0)
961 /* Connection was closed during packet processing. */
962 return -1;
963
paul718e3742002-12-13 20:15:29 +0000964 /* Register read thread. */
ajs634f9ea2005-04-11 15:51:40 +0000965 stream_reset(zclient->ibuf);
paul718e3742002-12-13 20:15:29 +0000966 zclient_event (ZCLIENT_READ, zclient);
967
968 return 0;
969}
970
971void
paul0a589352004-05-08 11:48:26 +0000972zclient_redistribute (int command, struct zclient *zclient, int type)
paul718e3742002-12-13 20:15:29 +0000973{
paul718e3742002-12-13 20:15:29 +0000974
paul0a589352004-05-08 11:48:26 +0000975 if (command == ZEBRA_REDISTRIBUTE_ADD)
976 {
977 if (zclient->redist[type])
978 return;
979 zclient->redist[type] = 1;
980 }
981 else
982 {
983 if (!zclient->redist[type])
984 return;
985 zclient->redist[type] = 0;
986 }
paul718e3742002-12-13 20:15:29 +0000987
988 if (zclient->sock > 0)
ajs634f9ea2005-04-11 15:51:40 +0000989 zebra_redistribute_send (command, zclient, type);
paul718e3742002-12-13 20:15:29 +0000990}
991
paul0a589352004-05-08 11:48:26 +0000992
paul718e3742002-12-13 20:15:29 +0000993void
paul0a589352004-05-08 11:48:26 +0000994zclient_redistribute_default (int command, struct zclient *zclient)
paul718e3742002-12-13 20:15:29 +0000995{
paul718e3742002-12-13 20:15:29 +0000996
paul0a589352004-05-08 11:48:26 +0000997 if (command == ZEBRA_REDISTRIBUTE_DEFAULT_ADD)
998 {
999 if (zclient->default_information)
1000 return;
1001 zclient->default_information = 1;
1002 }
1003 else
1004 {
1005 if (!zclient->default_information)
1006 return;
1007 zclient->default_information = 0;
1008 }
paul718e3742002-12-13 20:15:29 +00001009
1010 if (zclient->sock > 0)
paul0a589352004-05-08 11:48:26 +00001011 zebra_message_send (zclient, command);
paul718e3742002-12-13 20:15:29 +00001012}
1013
paul718e3742002-12-13 20:15:29 +00001014static void
1015zclient_event (enum event event, struct zclient *zclient)
1016{
1017 switch (event)
1018 {
1019 case ZCLIENT_SCHEDULE:
1020 if (! zclient->t_connect)
1021 zclient->t_connect =
1022 thread_add_event (master, zclient_connect, zclient, 0);
1023 break;
1024 case ZCLIENT_CONNECT:
1025 if (zclient->fail >= 10)
1026 return;
1027 if (zclient_debug)
ajs8ddca702004-12-07 18:53:52 +00001028 zlog_debug ("zclient connect schedule interval is %d",
paul718e3742002-12-13 20:15:29 +00001029 zclient->fail < 3 ? 10 : 60);
1030 if (! zclient->t_connect)
1031 zclient->t_connect =
1032 thread_add_timer (master, zclient_connect, zclient,
1033 zclient->fail < 3 ? 10 : 60);
1034 break;
1035 case ZCLIENT_READ:
1036 zclient->t_read =
1037 thread_add_read (master, zclient_read, zclient, zclient->sock);
1038 break;
1039 }
1040}
Vyacheslav Trushkin271ee732011-11-25 18:51:48 +04001041
1042void
1043zclient_serv_path_set (char *path)
1044{
1045 struct stat sb;
1046
1047 /* reset */
1048 zclient_serv_path = NULL;
1049
1050 /* test if `path' is socket. don't set it otherwise. */
1051 if (stat(path, &sb) == -1)
1052 {
1053 zlog_warn ("%s: zebra socket `%s' does not exist", __func__, path);
1054 return;
1055 }
1056
1057 if ((sb.st_mode & S_IFMT) != S_IFSOCK)
1058 {
1059 zlog_warn ("%s: `%s' is not unix socket, sir", __func__, path);
1060 return;
1061 }
1062
1063 /* it seems that path is unix socket */
1064 zclient_serv_path = path;
1065}
1066