blob: 4ead9b0edec7f654be85ab4aa490ceb8150cfa5a [file] [log] [blame]
paul718e3742002-12-13 20:15:29 +00001/* RIPd main routine.
2 * Copyright (C) 1997, 98 Kunihiro Ishiguro <kunihiro@zebra.org>
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
22#include <zebra.h>
23
gdt5e4fa162004-03-16 14:38:36 +000024#include <lib/version.h>
paul718e3742002-12-13 20:15:29 +000025#include "getopt.h"
26#include "thread.h"
27#include "command.h"
28#include "memory.h"
29#include "prefix.h"
30#include "filter.h"
31#include "keychain.h"
32#include "log.h"
pauledd7c242003-06-04 13:59:38 +000033#include "privs.h"
paul2d75d052004-01-19 21:31:15 +000034#include "sigevent.h"
Vyacheslav Trushkinb5114682011-11-25 18:51:48 +040035#include "zclient.h"
Feng Lu126215c2015-05-22 11:39:58 +020036#include "vrf.h"
paul718e3742002-12-13 20:15:29 +000037
38#include "ripd/ripd.h"
39
40/* ripd options. */
41static struct option longopts[] =
42{
43 { "daemon", no_argument, NULL, 'd'},
44 { "config_file", required_argument, NULL, 'f'},
45 { "pid_file", required_argument, NULL, 'i'},
Vyacheslav Trushkinb5114682011-11-25 18:51:48 +040046 { "socket", required_argument, NULL, 'z'},
paul718e3742002-12-13 20:15:29 +000047 { "help", no_argument, NULL, 'h'},
Paul Jakma876b8be2006-10-15 23:35:57 +000048 { "dryrun", no_argument, NULL, 'C'},
paul718e3742002-12-13 20:15:29 +000049 { "vty_addr", required_argument, NULL, 'A'},
50 { "vty_port", required_argument, NULL, 'P'},
51 { "retain", no_argument, NULL, 'r'},
pauledd7c242003-06-04 13:59:38 +000052 { "user", required_argument, NULL, 'u'},
hassoc0652302004-11-25 19:33:48 +000053 { "group", required_argument, NULL, 'g'},
paul718e3742002-12-13 20:15:29 +000054 { "version", no_argument, NULL, 'v'},
55 { 0 }
56};
57
pauledd7c242003-06-04 13:59:38 +000058/* ripd privileges */
59zebra_capabilities_t _caps_p [] =
60{
paulceacedb2005-09-29 14:39:32 +000061 ZCAP_NET_RAW,
pauledd7c242003-06-04 13:59:38 +000062 ZCAP_BIND
63};
64
65struct zebra_privs_t ripd_privs =
66{
pauld81fadf2003-08-14 05:32:12 +000067#if defined(QUAGGA_USER)
68 .user = QUAGGA_USER,
pauledd7c242003-06-04 13:59:38 +000069#endif
pauld81fadf2003-08-14 05:32:12 +000070#if defined QUAGGA_GROUP
71 .group = QUAGGA_GROUP,
72#endif
73#ifdef VTY_GROUP
74 .vty_group = VTY_GROUP,
pauledd7c242003-06-04 13:59:38 +000075#endif
76 .caps_p = _caps_p,
77 .cap_num_p = 2,
78 .cap_num_i = 0
79};
80
paul718e3742002-12-13 20:15:29 +000081/* Configuration file and directory. */
paul718e3742002-12-13 20:15:29 +000082char config_default[] = SYSCONFDIR RIPD_DEFAULT_CONFIG;
83char *config_file = NULL;
84
85/* ripd program name */
86
87/* Route retain mode flag. */
88int retain_mode = 0;
89
90/* RIP VTY bind address. */
91char *vty_addr = NULL;
92
93/* RIP VTY connection port. */
94int vty_port = RIP_VTY_PORT;
95
96/* Master of threads. */
97struct thread_master *master;
98
99/* Process ID saved for use by init system */
hasso8a676be2004-10-08 06:36:38 +0000100const char *pid_file = PATH_RIPD_PID;
paul718e3742002-12-13 20:15:29 +0000101
102/* Help information display. */
103static void
104usage (char *progname, int status)
105{
106 if (status != 0)
107 fprintf (stderr, "Try `%s --help' for more information.\n", progname);
108 else
109 {
110 printf ("Usage : %s [OPTION...]\n\
111Daemon which manages RIP version 1 and 2.\n\n\
112-d, --daemon Runs in daemon mode\n\
113-f, --config_file Set configuration file name\n\
114-i, --pid_file Set process identifier file name\n\
Vyacheslav Trushkinb5114682011-11-25 18:51:48 +0400115-z, --socket Set path of zebra socket\n\
paul718e3742002-12-13 20:15:29 +0000116-A, --vty_addr Set vty's bind address\n\
117-P, --vty_port Set vty's port number\n\
Paul Jakma876b8be2006-10-15 23:35:57 +0000118-C, --dryrun Check configuration for validity and exit\n\
paul718e3742002-12-13 20:15:29 +0000119-r, --retain When program terminates, retain added route by ripd.\n\
hassoc0652302004-11-25 19:33:48 +0000120-u, --user User to run as\n\
121-g, --group Group to run as\n\
paul718e3742002-12-13 20:15:29 +0000122-v, --version Print program version\n\
123-h, --help Display this help and exit\n\
124\n\
125Report bugs to %s\n", progname, ZEBRA_BUG_ADDRESS);
126 }
127
128 exit (status);
129}
David Lamparter6b0655a2014-06-04 06:53:35 +0200130
paul718e3742002-12-13 20:15:29 +0000131/* SIGHUP handler. */
pauldc63bfd2005-10-25 23:31:05 +0000132static void
paul2d75d052004-01-19 21:31:15 +0000133sighup (void)
paul718e3742002-12-13 20:15:29 +0000134{
135 zlog_info ("SIGHUP received");
136 rip_clean ();
137 rip_reset ();
138 zlog_info ("ripd restarting!");
139
140 /* Reload config file. */
hasso320ec102004-06-20 19:54:37 +0000141 vty_read_config (config_file, config_default);
paul718e3742002-12-13 20:15:29 +0000142
143 /* Create VTY's socket */
144 vty_serv_sock (vty_addr, vty_port, RIP_VTYSH_PATH);
145
146 /* Try to return to normal operation. */
147}
148
149/* SIGINT handler. */
pauldc63bfd2005-10-25 23:31:05 +0000150static void
paul2d75d052004-01-19 21:31:15 +0000151sigint (void)
paul718e3742002-12-13 20:15:29 +0000152{
ajs887c44a2004-12-03 16:36:46 +0000153 zlog_notice ("Terminating on signal");
paul718e3742002-12-13 20:15:29 +0000154
155 if (! retain_mode)
156 rip_clean ();
157
158 exit (0);
159}
160
161/* SIGUSR1 handler. */
pauldc63bfd2005-10-25 23:31:05 +0000162static void
paul2d75d052004-01-19 21:31:15 +0000163sigusr1 (void)
paul718e3742002-12-13 20:15:29 +0000164{
165 zlog_rotate (NULL);
166}
167
pauldc63bfd2005-10-25 23:31:05 +0000168static struct quagga_signal_t ripd_signals[] =
paul718e3742002-12-13 20:15:29 +0000169{
paul2d75d052004-01-19 21:31:15 +0000170 {
171 .signal = SIGHUP,
172 .handler = &sighup,
173 },
174 {
175 .signal = SIGUSR1,
176 .handler = &sigusr1,
177 },
178 {
179 .signal = SIGINT,
hasso8c903fb2004-03-17 20:39:18 +0000180 .handler = &sigint,
paul2d75d052004-01-19 21:31:15 +0000181 },
hassof571dab2004-03-22 08:55:25 +0000182 {
183 .signal = SIGTERM,
184 .handler = &sigint,
185 },
paul2d75d052004-01-19 21:31:15 +0000186};
David Lamparter6b0655a2014-06-04 06:53:35 +0200187
paul718e3742002-12-13 20:15:29 +0000188/* Main routine of ripd. */
189int
190main (int argc, char **argv)
191{
192 char *p;
193 int daemon_mode = 0;
Paul Jakma876b8be2006-10-15 23:35:57 +0000194 int dryrun = 0;
paul718e3742002-12-13 20:15:29 +0000195 char *progname;
196 struct thread thread;
197
198 /* Set umask before anything for security */
199 umask (0027);
200
201 /* Get program name. */
202 progname = ((p = strrchr (argv[0], '/')) ? ++p : argv[0]);
203
204 /* First of all we need logging init. */
ajs274a4a42004-12-07 15:39:31 +0000205 zlog_default = openzlog (progname, ZLOG_RIP,
paul718e3742002-12-13 20:15:29 +0000206 LOG_CONS|LOG_NDELAY|LOG_PID, LOG_DAEMON);
207
208 /* Command line option parse. */
209 while (1)
210 {
211 int opt;
212
Vyacheslav Trushkinb5114682011-11-25 18:51:48 +0400213 opt = getopt_long (argc, argv, "df:i:z:hA:P:u:g:rvC", longopts, 0);
paul718e3742002-12-13 20:15:29 +0000214
215 if (opt == EOF)
216 break;
217
218 switch (opt)
219 {
220 case 0:
221 break;
222 case 'd':
223 daemon_mode = 1;
224 break;
225 case 'f':
226 config_file = optarg;
227 break;
228 case 'A':
229 vty_addr = optarg;
230 break;
231 case 'i':
232 pid_file = optarg;
233 break;
Vyacheslav Trushkinb5114682011-11-25 18:51:48 +0400234 case 'z':
235 zclient_serv_path_set (optarg);
236 break;
paul718e3742002-12-13 20:15:29 +0000237 case 'P':
paul4fc4e7a2003-01-22 19:47:09 +0000238 /* Deal with atoi() returning 0 on failure, and ripd not
239 listening on rip port... */
240 if (strcmp(optarg, "0") == 0)
241 {
242 vty_port = 0;
243 break;
244 }
245 vty_port = atoi (optarg);
Paul Jakma0d6b2ee2008-05-29 18:29:16 +0000246 if (vty_port <= 0 || vty_port > 0xffff)
247 vty_port = RIP_VTY_PORT;
paul718e3742002-12-13 20:15:29 +0000248 break;
249 case 'r':
250 retain_mode = 1;
251 break;
Paul Jakma876b8be2006-10-15 23:35:57 +0000252 case 'C':
253 dryrun = 1;
254 break;
hassoc0652302004-11-25 19:33:48 +0000255 case 'u':
256 ripd_privs.user = optarg;
257 break;
258 case 'g':
259 ripd_privs.group = optarg;
260 break;
paul718e3742002-12-13 20:15:29 +0000261 case 'v':
262 print_version (progname);
263 exit (0);
264 break;
265 case 'h':
266 usage (progname, 0);
267 break;
268 default:
269 usage (progname, 1);
270 break;
271 }
272 }
273
274 /* Prepare master thread. */
275 master = thread_master_create ();
276
277 /* Library initialization. */
pauledd7c242003-06-04 13:59:38 +0000278 zprivs_init (&ripd_privs);
Balaji.G837d16c2012-09-26 14:09:10 +0530279 signal_init (master, array_size(ripd_signals), ripd_signals);
paul718e3742002-12-13 20:15:29 +0000280 cmd_init (1);
paulb21b19c2003-06-15 01:28:29 +0000281 vty_init (master);
paul718e3742002-12-13 20:15:29 +0000282 memory_init ();
283 keychain_init ();
Feng Lu126215c2015-05-22 11:39:58 +0200284 vrf_init ();
paul718e3742002-12-13 20:15:29 +0000285
286 /* RIP related initialization. */
287 rip_init ();
288 rip_if_init ();
Donald Sharp71252932015-09-24 09:25:19 -0400289 rip_zclient_init (master);
paul718e3742002-12-13 20:15:29 +0000290 rip_peer_init ();
291
paul718e3742002-12-13 20:15:29 +0000292 /* Get configuration file. */
hasso320ec102004-06-20 19:54:37 +0000293 vty_read_config (config_file, config_default);
paul718e3742002-12-13 20:15:29 +0000294
Paul Jakma876b8be2006-10-15 23:35:57 +0000295 /* Start execution only if not in dry-run mode */
296 if(dryrun)
297 return (0);
298
paul718e3742002-12-13 20:15:29 +0000299 /* Change to the daemon program. */
Stephen Hemminger065de902009-08-07 11:13:49 -0700300 if (daemon_mode && daemon (0, 0) < 0)
301 {
302 zlog_err("RIPd daemon failed: %s", strerror(errno));
303 exit (1);
304 }
paul718e3742002-12-13 20:15:29 +0000305
306 /* Pid file create. */
307 pid_output (pid_file);
308
309 /* Create VTY's socket */
310 vty_serv_sock (vty_addr, vty_port, RIP_VTYSH_PATH);
311
ajs887c44a2004-12-03 16:36:46 +0000312 /* Print banner. */
313 zlog_notice ("RIPd %s starting: vty@%d", QUAGGA_VERSION, vty_port);
314
paul718e3742002-12-13 20:15:29 +0000315 /* Execute each thread. */
316 while (thread_fetch (master, &thread))
317 thread_call (&thread);
318
319 /* Not reached. */
paule8e19462006-01-19 20:16:55 +0000320 return (0);
paul718e3742002-12-13 20:15:29 +0000321}