Annotation of embedaddon/quagga/bgpd/bgp_main.c, revision 1.1.1.1
1.1 misho 1: /* Main routine of bgpd.
2: Copyright (C) 1996, 97, 98, 1999 Kunihiro Ishiguro
3:
4: This file is part of GNU Zebra.
5:
6: GNU Zebra is free software; you can redistribute it and/or modify it
7: under the terms of the GNU General Public License as published by the
8: Free Software Foundation; either version 2, or (at your option) any
9: later version.
10:
11: GNU Zebra is distributed in the hope that it will be useful, but
12: WITHOUT ANY WARRANTY; without even the implied warranty of
13: MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14: General Public License for more details.
15:
16: You should have received a copy of the GNU General Public License
17: along with GNU Zebra; see the file COPYING. If not, write to the Free
18: Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
19: 02111-1307, USA. */
20:
21: #include <zebra.h>
22:
23: #include "vector.h"
24: #include "vty.h"
25: #include "command.h"
26: #include "getopt.h"
27: #include "thread.h"
28: #include <lib/version.h>
29: #include "memory.h"
30: #include "prefix.h"
31: #include "log.h"
32: #include "privs.h"
33: #include "sigevent.h"
34: #include "zclient.h"
35: #include "routemap.h"
36: #include "filter.h"
37: #include "plist.h"
38:
39: #include "bgpd/bgpd.h"
40: #include "bgpd/bgp_attr.h"
41: #include "bgpd/bgp_mplsvpn.h"
42: #include "bgpd/bgp_aspath.h"
43: #include "bgpd/bgp_dump.h"
44: #include "bgpd/bgp_route.h"
45: #include "bgpd/bgp_nexthop.h"
46: #include "bgpd/bgp_regex.h"
47: #include "bgpd/bgp_clist.h"
48: #include "bgpd/bgp_debug.h"
49: #include "bgpd/bgp_filter.h"
50:
51: /* bgpd options, we use GNU getopt library. */
52: static const struct option longopts[] =
53: {
54: { "daemon", no_argument, NULL, 'd'},
55: { "config_file", required_argument, NULL, 'f'},
56: { "pid_file", required_argument, NULL, 'i'},
57: { "bgp_port", required_argument, NULL, 'p'},
58: { "listenon", required_argument, NULL, 'l'},
59: { "vty_addr", required_argument, NULL, 'A'},
60: { "vty_port", required_argument, NULL, 'P'},
61: { "retain", no_argument, NULL, 'r'},
62: { "no_kernel", no_argument, NULL, 'n'},
63: { "user", required_argument, NULL, 'u'},
64: { "group", required_argument, NULL, 'g'},
65: { "version", no_argument, NULL, 'v'},
66: { "dryrun", no_argument, NULL, 'C'},
67: { "help", no_argument, NULL, 'h'},
68: { 0 }
69: };
70:
71: /* signal definitions */
72: void sighup (void);
73: void sigint (void);
74: void sigusr1 (void);
75:
76: static void bgp_exit (int);
77:
78: static struct quagga_signal_t bgp_signals[] =
79: {
80: {
81: .signal = SIGHUP,
82: .handler = &sighup,
83: },
84: {
85: .signal = SIGUSR1,
86: .handler = &sigusr1,
87: },
88: {
89: .signal = SIGINT,
90: .handler = &sigint,
91: },
92: {
93: .signal = SIGTERM,
94: .handler = &sigint,
95: },
96: };
97:
98: /* Configuration file and directory. */
99: char config_default[] = SYSCONFDIR BGP_DEFAULT_CONFIG;
100:
101: /* Route retain mode flag. */
102: static int retain_mode = 0;
103:
104: /* Master of threads. */
105: struct thread_master *master;
106:
107: /* Manually specified configuration file name. */
108: char *config_file = NULL;
109:
110: /* Process ID saved for use by init system */
111: static const char *pid_file = PATH_BGPD_PID;
112:
113: /* VTY port number and address. */
114: int vty_port = BGP_VTY_PORT;
115: char *vty_addr = NULL;
116:
117: /* privileges */
118: static zebra_capabilities_t _caps_p [] =
119: {
120: ZCAP_BIND,
121: ZCAP_NET_RAW,
122: };
123:
124: struct zebra_privs_t bgpd_privs =
125: {
126: #if defined(QUAGGA_USER) && defined(QUAGGA_GROUP)
127: .user = QUAGGA_USER,
128: .group = QUAGGA_GROUP,
129: #endif
130: #ifdef VTY_GROUP
131: .vty_group = VTY_GROUP,
132: #endif
133: .caps_p = _caps_p,
134: .cap_num_p = sizeof(_caps_p)/sizeof(_caps_p[0]),
135: .cap_num_i = 0,
136: };
137:
138: /* Help information display. */
139: static void
140: usage (char *progname, int status)
141: {
142: if (status != 0)
143: fprintf (stderr, "Try `%s --help' for more information.\n", progname);
144: else
145: {
146: printf ("Usage : %s [OPTION...]\n\n\
147: Daemon which manages kernel routing table management and \
148: redistribution between different routing protocols.\n\n\
149: -d, --daemon Runs in daemon mode\n\
150: -f, --config_file Set configuration file name\n\
151: -i, --pid_file Set process identifier file name\n\
152: -p, --bgp_port Set bgp protocol's port number\n\
153: -l, --listenon Listen on specified address (implies -n)\n\
154: -A, --vty_addr Set vty's bind address\n\
155: -P, --vty_port Set vty's port number\n\
156: -r, --retain When program terminates, retain added route by bgpd.\n\
157: -n, --no_kernel Do not install route to kernel.\n\
158: -u, --user User to run as\n\
159: -g, --group Group to run as\n\
160: -v, --version Print program version\n\
161: -C, --dryrun Check configuration for validity and exit\n\
162: -h, --help Display this help and exit\n\
163: \n\
164: Report bugs to %s\n", progname, ZEBRA_BUG_ADDRESS);
165: }
166:
167: exit (status);
168: }
169:
170: /* SIGHUP handler. */
171: void
172: sighup (void)
173: {
174: zlog (NULL, LOG_INFO, "SIGHUP received");
175:
176: /* Terminate all thread. */
177: bgp_terminate ();
178: bgp_reset ();
179: zlog_info ("bgpd restarting!");
180:
181: /* Reload config file. */
182: vty_read_config (config_file, config_default);
183:
184: /* Create VTY's socket */
185: vty_serv_sock (vty_addr, vty_port, BGP_VTYSH_PATH);
186:
187: /* Try to return to normal operation. */
188: }
189:
190: /* SIGINT handler. */
191: void
192: sigint (void)
193: {
194: zlog_notice ("Terminating on signal");
195:
196: if (! retain_mode)
197: bgp_terminate ();
198:
199: bgp_exit (0);
200: }
201:
202: /* SIGUSR1 handler. */
203: void
204: sigusr1 (void)
205: {
206: zlog_rotate (NULL);
207: }
208:
209: /*
210: Try to free up allocations we know about so that diagnostic tools such as
211: valgrind are able to better illuminate leaks.
212:
213: Zebra route removal and protocol teardown are not meant to be done here.
214: For example, "retain_mode" may be set.
215: */
216: static void
217: bgp_exit (int status)
218: {
219: struct bgp *bgp;
220: struct listnode *node, *nnode;
221: int *socket;
222: struct interface *ifp;
223: extern struct zclient *zclient;
224: extern struct zclient *zlookup;
225:
226: /* it only makes sense for this to be called on a clean exit */
227: assert (status == 0);
228:
229: /* reverse bgp_master_init */
230: for (ALL_LIST_ELEMENTS (bm->bgp, node, nnode, bgp))
231: bgp_delete (bgp);
232: list_free (bm->bgp);
233:
234: /* reverse bgp_master_init */
235: for (ALL_LIST_ELEMENTS_RO(bm->listen_sockets, node, socket))
236: {
237: if (close ((int)(long)socket) == -1)
238: zlog_err ("close (%d): %s", (int)(long)socket, safe_strerror (errno));
239: }
240: list_delete (bm->listen_sockets);
241:
242: /* reverse bgp_zebra_init/if_init */
243: if (retain_mode)
244: if_add_hook (IF_DELETE_HOOK, NULL);
245: for (ALL_LIST_ELEMENTS (iflist, node, nnode, ifp))
246: {
247: struct listnode *c_node, *c_nnode;
248: struct connected *c;
249:
250: for (ALL_LIST_ELEMENTS (ifp->connected, c_node, c_nnode, c))
251: bgp_connected_delete (c);
252:
253: if_delete (ifp);
254: }
255: list_free (iflist);
256:
257: /* reverse bgp_attr_init */
258: bgp_attr_finish ();
259:
260: /* reverse bgp_dump_init */
261: bgp_dump_finish ();
262:
263: /* reverse bgp_route_init */
264: bgp_route_finish ();
265:
266: /* reverse bgp_route_map_init/route_map_init */
267: route_map_finish ();
268:
269: /* reverse bgp_scan_init */
270: bgp_scan_finish ();
271:
272: /* reverse access_list_init */
273: access_list_add_hook (NULL);
274: access_list_delete_hook (NULL);
275: access_list_reset ();
276:
277: /* reverse bgp_filter_init */
278: as_list_add_hook (NULL);
279: as_list_delete_hook (NULL);
280: bgp_filter_reset ();
281:
282: /* reverse prefix_list_init */
283: prefix_list_add_hook (NULL);
284: prefix_list_delete_hook (NULL);
285: prefix_list_reset ();
286:
287: /* reverse community_list_init */
288: community_list_terminate (bgp_clist);
289:
290: cmd_terminate ();
291: vty_terminate ();
292: if (zclient)
293: zclient_free (zclient);
294: if (zlookup)
295: zclient_free (zlookup);
296:
297: /* reverse bgp_master_init */
298: if (master)
299: thread_master_free (master);
300:
301: if (zlog_default)
302: closezlog (zlog_default);
303:
304: if (CONF_BGP_DEBUG (normal, NORMAL))
305: log_memstats_stderr ("bgpd");
306:
307: exit (status);
308: }
309:
310: /* Main routine of bgpd. Treatment of argument and start bgp finite
311: state machine is handled at here. */
312: int
313: main (int argc, char **argv)
314: {
315: char *p;
316: int opt;
317: int daemon_mode = 0;
318: int dryrun = 0;
319: char *progname;
320: struct thread thread;
321: int tmp_port;
322:
323: /* Set umask before anything for security */
324: umask (0027);
325:
326: /* Preserve name of myself. */
327: progname = ((p = strrchr (argv[0], '/')) ? ++p : argv[0]);
328:
329: zlog_default = openzlog (progname, ZLOG_BGP,
330: LOG_CONS|LOG_NDELAY|LOG_PID, LOG_DAEMON);
331:
332: /* BGP master init. */
333: bgp_master_init ();
334:
335: /* Command line argument treatment. */
336: while (1)
337: {
338: opt = getopt_long (argc, argv, "df:i:hp:l:A:P:rnu:g:vC", longopts, 0);
339:
340: if (opt == EOF)
341: break;
342:
343: switch (opt)
344: {
345: case 0:
346: break;
347: case 'd':
348: daemon_mode = 1;
349: break;
350: case 'f':
351: config_file = optarg;
352: break;
353: case 'i':
354: pid_file = optarg;
355: break;
356: case 'p':
357: tmp_port = atoi (optarg);
358: if (tmp_port <= 0 || tmp_port > 0xffff)
359: bm->port = BGP_PORT_DEFAULT;
360: else
361: bm->port = tmp_port;
362: break;
363: case 'A':
364: vty_addr = optarg;
365: break;
366: case 'P':
367: /* Deal with atoi() returning 0 on failure, and bgpd not
368: listening on bgp port... */
369: if (strcmp(optarg, "0") == 0)
370: {
371: vty_port = 0;
372: break;
373: }
374: vty_port = atoi (optarg);
375: if (vty_port <= 0 || vty_port > 0xffff)
376: vty_port = BGP_VTY_PORT;
377: break;
378: case 'r':
379: retain_mode = 1;
380: break;
381: case 'l':
382: bm->address = optarg;
383: /* listenon implies -n */
384: case 'n':
385: bgp_option_set (BGP_OPT_NO_FIB);
386: break;
387: case 'u':
388: bgpd_privs.user = optarg;
389: break;
390: case 'g':
391: bgpd_privs.group = optarg;
392: break;
393: case 'v':
394: print_version (progname);
395: exit (0);
396: break;
397: case 'C':
398: dryrun = 1;
399: break;
400: case 'h':
401: usage (progname, 0);
402: break;
403: default:
404: usage (progname, 1);
405: break;
406: }
407: }
408:
409: /* Make thread master. */
410: master = bm->master;
411:
412: /* Initializations. */
413: srand (time (NULL));
414: signal_init (master, Q_SIGC(bgp_signals), bgp_signals);
415: zprivs_init (&bgpd_privs);
416: cmd_init (1);
417: vty_init (master);
418: memory_init ();
419:
420: /* BGP related initialization. */
421: bgp_init ();
422:
423: /* Sort CLI commands. */
424: sort_node ();
425:
426: /* Parse config file. */
427: vty_read_config (config_file, config_default);
428:
429: /* Start execution only if not in dry-run mode */
430: if(dryrun)
431: return(0);
432:
433: /* Turn into daemon if daemon_mode is set. */
434: if (daemon_mode && daemon (0, 0) < 0)
435: {
436: zlog_err("BGPd daemon failed: %s", strerror(errno));
437: return (1);
438: }
439:
440:
441: /* Process ID file creation. */
442: pid_output (pid_file);
443:
444: /* Make bgp vty socket. */
445: vty_serv_sock (vty_addr, vty_port, BGP_VTYSH_PATH);
446:
447: /* Print banner. */
448: zlog_notice ("BGPd %s starting: vty@%d, bgp@%s:%d", QUAGGA_VERSION,
449: vty_port,
450: (bm->address ? bm->address : "<all>"),
451: bm->port);
452:
453: /* Start finite state machine, here we go! */
454: while (thread_fetch (master, &thread))
455: thread_call (&thread);
456:
457: /* Not reached. */
458: return (0);
459: }
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>