Added sanity check function in config parser
[olsrd.git] / src / cfgparser / olsrd_conf.c
1 /*
2  * OLSR ad-hoc routing table management protocol config parser
3  * Copyright (C) 2004 Andreas T√łnnesen (andreto@olsr.org)
4  *
5  * This file is part of the olsr.org OLSR daemon.
6  *
7  * olsr.org is free software; you can redistribute it and/or modify
8  * it under the terms of the GNU General Public License as published by
9  * the Free Software Foundation; either version 2 of the License, or
10  * (at your option) any later version.
11  *
12  * olsr.org is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15  * GNU General Public License for more details.
16  *
17  * You should have received a copy of the GNU General Public License
18  * along with olsr.org; if not, write to the Free Software
19  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
20  * 
21  * 
22  * $Id: olsrd_conf.c,v 1.21 2004/11/20 21:42:35 kattemat Exp $
23  *
24  */
25
26
27 #include <stdio.h>
28 #include <string.h>
29 #include <errno.h>
30 #include <stdlib.h>
31 #include <sys/types.h>
32 #include <sys/socket.h>
33 #include <netinet/in.h>
34 #include <arpa/inet.h>
35
36 #include "olsrd_conf.h"
37
38
39 extern FILE *yyin;
40 extern int yyparse(void);
41
42
43 #ifdef MAKELIB
44
45 /* Build as DLL */
46
47 void __attribute__ ((constructor)) 
48 my_init(void);
49
50 void __attribute__ ((destructor)) 
51 my_fini(void);
52
53
54 /**
55  *Constructor
56  */
57 void
58 my_init()
59 {
60   /* Print plugin info to stdout */
61   printf("olsrd config file parser %s loaded\n", PARSER_VERSION);
62
63   return;
64 }
65
66 /**
67  *Destructor
68  */
69 void
70 my_fini()
71 {
72   printf("See you around!\n");
73   return;
74 }
75
76 #else
77
78 #ifdef MAKEBIN
79
80 /* Build as standalone binary */
81 int 
82 main(int argc, char *argv[])
83 {
84   struct olsrd_config *cnf;
85
86   if(argc == 1)
87     {
88       fprintf(stderr, "Usage: olsrd_cfgparser [filename] -print\n\n");
89       exit(EXIT_FAILURE);
90     }
91
92   if((cnf = olsrd_parse_cnf(argv[1])) != NULL)
93     {
94       if((argc > 2) && (!strcmp(argv[2], "-print")))
95         {
96           olsrd_print_cnf(cnf);  
97           olsrd_write_cnf(cnf, "./out.conf");
98         }
99       else
100         printf("Use -print to view parsed values\n");
101       printf("Configfile parsed OK\n");
102     }
103   else
104     {
105       printf("Failed parsing \"%s\"\n", argv[1]);
106     }
107
108   return 0;
109 }
110
111 #else
112
113 /* Build as part of olsrd */
114
115
116 #endif
117
118 #endif
119
120 struct olsrd_config *
121 olsrd_parse_cnf(char *filename)
122 {
123   struct olsr_if *in;
124
125   cnf = malloc(sizeof(struct olsrd_config));
126   if (cnf == NULL)
127     {
128       fprintf(stderr, "Out of memory %s\n", __func__);
129       return NULL;
130   }
131
132   set_default_cnf(cnf);
133
134   printf("Parsing file: \"%s\"\n", filename);
135
136   yyin = fopen(filename, "r");
137   
138   if (yyin == NULL)
139     {
140       fprintf(stderr, "Cannot open configuration file '%s': %s.\n",
141               filename, strerror(errno));
142       free(cnf);
143       return NULL;
144   }
145
146   current_line = 1;
147
148   if (yyparse() != 0)
149     {
150       fclose(yyin);
151       olsrd_free_cnf(cnf);
152       exit(0);
153     }
154   
155   fclose(yyin);
156
157
158   in = cnf->interfaces;
159
160   while(in)
161     {
162       /* set various stuff */
163       in->index = cnf->ifcnt++;
164       in->configured = OLSR_FALSE;
165       in->interf = NULL;
166       in = in->next;
167     }
168
169
170   return cnf;
171 }
172
173
174 int
175 olsrd_sanity_check_cnf(struct olsrd_config *cnf)
176 {
177   struct olsr_if           *in = cnf->interfaces;
178   struct if_config_options *io;
179
180   /* Debug level */
181   if(cnf->debug_level < MIN_DEBUGLVL ||
182      cnf->debug_level > MAX_DEBUGLVL)
183     {
184       fprintf(stderr, "Debuglevel %d is not allowed\n", cnf->debug_level);
185       return -1;
186     }
187
188   /* TOS */
189   if(//cnf->tos < MIN_TOS ||
190      cnf->tos > MAX_TOS)
191     {
192       fprintf(stderr, "TOS %d is not allowed\n", cnf->tos);
193       return -1;
194     }
195
196   if(cnf->willingness_auto == OLSR_FALSE &&
197      (cnf->willingness < MIN_WILLINGNESS ||
198       cnf->willingness > MAX_WILLINGNESS))
199     {
200       fprintf(stderr, "willingness %d is not allowed\n", cnf->willingness);
201       return -1;
202     }
203
204   /* Hysteresis */
205
206   if(cnf->hysteresis_param.scaling < MIN_HYST_PARAM ||
207      cnf->hysteresis_param.scaling > MAX_HYST_PARAM)
208     {
209       fprintf(stderr, "Hyst scaling %0.2f is not allowed\n", cnf->hysteresis_param.scaling);
210       return -1;
211     }
212
213   if(cnf->hysteresis_param.thr_high <= cnf->hysteresis_param.thr_low)
214     {
215       fprintf(stderr, "Hyst upper(%0.2f) thr must be bigger than lower(%0.2f) threshold!\n", cnf->hysteresis_param.thr_high, cnf->hysteresis_param.thr_low);
216       return -1;
217     }
218
219   if(cnf->hysteresis_param.thr_high < MIN_HYST_PARAM ||
220      cnf->hysteresis_param.thr_high > MAX_HYST_PARAM)
221     {
222       fprintf(stderr, "Hyst upper thr %0.2f is not allowed\n", cnf->hysteresis_param.thr_high);
223       return -1;
224     }
225
226   if(cnf->hysteresis_param.thr_low < MIN_HYST_PARAM ||
227      cnf->hysteresis_param.thr_low > MAX_HYST_PARAM)
228     {
229       fprintf(stderr, "Hyst lower thr %0.2f is not allowed\n", cnf->hysteresis_param.thr_low);
230       return -1;
231     }
232
233
234   /* Pollrate */
235
236   if(cnf->pollrate < MIN_POLLRATE ||
237      cnf->pollrate > MAX_POLLRATE)
238     {
239       fprintf(stderr, "Pollrate %0.2f is not allowed\n", cnf->pollrate);
240       return -1;
241     }
242
243   /* TC redundancy */
244
245   if(//cnf->tc_redundancy < MIN_TC_REDUNDANCY ||
246      cnf->tc_redundancy > MAX_TC_REDUNDANCY)
247     {
248       fprintf(stderr, "TC redundancy %d is not allowed\n", cnf->tc_redundancy);
249       return -1;
250     }
251
252   /* MPR coverage */
253   if(cnf->mpr_coverage < MIN_MPR_COVERAGE ||
254      cnf->mpr_coverage > MAX_MPR_COVERAGE)
255     {
256       fprintf(stderr, "MPR coverage %d is not allowed\n", cnf->mpr_coverage);
257       return -1;
258     }
259
260   /* Interfaces */
261   while(in)
262     {
263       io = in->cnf;
264
265       if(in->name == NULL || !strlen(in->name))
266         {
267           fprintf(stderr, "Interface has no name!\n");
268           return -1;
269         }
270
271       if(io == NULL)
272         {
273           fprintf(stderr, "Interface %s has no configuration!\n", in->name);
274           return -1;
275         }
276
277       /* HELLO interval */
278       if(io->hello_params.emission_interval < cnf->pollrate ||
279          io->hello_params.emission_interval > io->hello_params.validity_time)
280         {
281           fprintf(stderr, "Bad HELLO parameters! (em: %0.2f, vt: %0.2f)\n", io->hello_params.emission_interval, io->hello_params.validity_time);
282           return -1;
283         }
284
285       /* TC interval */
286       if(io->tc_params.emission_interval < cnf->pollrate ||
287          io->tc_params.emission_interval > io->tc_params.validity_time)
288         {
289           fprintf(stderr, "Bad TC parameters! (em: %0.2f, vt: %0.2f)\n", io->tc_params.emission_interval, io->tc_params.validity_time);
290           return -1;
291         }
292
293       /* MID interval */
294       if(io->mid_params.emission_interval < cnf->pollrate ||
295          io->mid_params.emission_interval > io->mid_params.validity_time)
296         {
297           fprintf(stderr, "Bad MID parameters! (em: %0.2f, vt: %0.2f)\n", io->mid_params.emission_interval, io->mid_params.validity_time);
298           return -1;
299         }
300
301       /* HNA interval */
302       if(io->hna_params.emission_interval < cnf->pollrate ||
303          io->hna_params.emission_interval > io->hna_params.validity_time)
304         {
305           fprintf(stderr, "Bad HNA parameters! (em: %0.2f, vt: %0.2f)\n", io->hna_params.emission_interval, io->hna_params.validity_time);
306           return -1;
307         }
308
309       in = in->next;
310     }
311
312   return 0;
313 }
314
315
316 void
317 olsrd_free_cnf(struct olsrd_config *cnf)
318 {
319   struct hna4_entry        *h4d, *h4 = cnf->hna4_entries;
320   struct hna6_entry        *h6d, *h6 = cnf->hna6_entries;
321   struct olsr_if           *ind, *in = cnf->interfaces;
322   struct plugin_entry      *ped, *pe = cnf->plugins;
323   
324   while(h4)
325     {
326       h4d = h4;
327       h4 = h4->next;
328       free(h4d);
329     }
330
331   while(h6)
332     {
333       h6d = h6;
334       h6 = h6->next;
335       free(h6d);
336     }
337
338   while(in)
339     {
340       free(in->cnf);
341       ind = in;
342       in = in->next;
343       free(ind->name);
344       free(ind->config);
345       free(ind);
346     }
347
348   while(pe)
349     {
350       ped = pe;
351       pe = pe->next;
352       free(ped->name);
353       free(ped);
354     }
355
356   return;
357 }
358
359
360
361 struct olsrd_config *
362 olsrd_get_default_cnf()
363 {
364   cnf = malloc(sizeof(struct olsrd_config));
365   if (cnf == NULL)
366     {
367       fprintf(stderr, "Out of memory %s\n", __func__);
368       return NULL;
369   }
370
371   set_default_cnf(cnf);
372
373   return cnf;
374 }
375
376
377
378
379 void
380 set_default_cnf(struct olsrd_config *cnf)
381 {
382     memset(cnf, 0, sizeof(struct olsrd_config));
383     
384     cnf->debug_level = DEF_DEBUGLVL;
385     cnf->ip_version  = AF_INET;
386     cnf->allow_no_interfaces = DEF_ALLOW_NO_INTS;
387     cnf->tos = DEF_TOS;
388     cnf->willingness_auto = DEF_WILL_AUTO;
389     cnf->ipc_connections = DEF_IPC_CONNECTIONS;
390     cnf->open_ipc = cnf->ipc_connections ? OLSR_TRUE : OLSR_FALSE;
391
392     cnf->use_hysteresis = DEF_USE_HYST;
393     cnf->hysteresis_param.scaling = HYST_SCALING;
394     cnf->hysteresis_param.thr_high = HYST_THRESHOLD_HIGH;
395     cnf->hysteresis_param.thr_low = HYST_THRESHOLD_LOW;
396
397     cnf->pollrate = DEF_POLLRATE;
398
399     cnf->tc_redundancy = TC_REDUNDANCY;
400     cnf->mpr_coverage = MPR_COVERAGE;
401     cnf->lq_level = DEF_LQ_LEVEL;
402     cnf->lq_wsize = DEF_LQ_WSIZE;
403     cnf->clear_screen = DEF_CLEAR_SCREEN;
404 }
405
406
407
408
409 struct if_config_options *
410 get_default_if_config()
411 {
412   struct if_config_options *io = malloc(sizeof(struct if_config_options));
413   struct in6_addr in6;
414  
415   memset(io, 0, sizeof(struct if_config_options));
416
417   io->ipv6_addrtype = 1; /* XXX - FixMe */
418
419   if(inet_pton(AF_INET6, OLSR_IPV6_MCAST_SITE_LOCAL, &in6) < 0)
420     {
421       fprintf(stderr, "Failed converting IP address %s\n", OLSR_IPV6_MCAST_SITE_LOCAL);
422       exit(EXIT_FAILURE);
423     }
424   memcpy(&io->ipv6_multi_site.v6, &in6, sizeof(struct in6_addr));
425
426   if(inet_pton(AF_INET6, OLSR_IPV6_MCAST_GLOBAL, &in6) < 0)
427     {
428       fprintf(stderr, "Failed converting IP address %s\n", OLSR_IPV6_MCAST_GLOBAL);
429       exit(EXIT_FAILURE);
430     }
431   memcpy(&io->ipv6_multi_glbl.v6, &in6, sizeof(struct in6_addr));
432
433
434   io->hello_params.emission_interval = HELLO_INTERVAL;
435   io->hello_params.validity_time = NEIGHB_HOLD_TIME;
436   io->tc_params.emission_interval = TC_INTERVAL;
437   io->tc_params.validity_time = TOP_HOLD_TIME;
438   io->mid_params.emission_interval = MID_INTERVAL;
439   io->mid_params.validity_time = MID_HOLD_TIME;
440   io->hna_params.emission_interval = HNA_INTERVAL;
441   io->hna_params.validity_time = HNA_HOLD_TIME;
442
443   return io;
444
445 }
446
447
448
449
450
451 int
452 olsrd_write_cnf(struct olsrd_config *cnf, char *fname)
453 {
454   struct hna4_entry        *h4 = cnf->hna4_entries;
455   struct hna6_entry        *h6 = cnf->hna6_entries;
456   struct olsr_if           *in = cnf->interfaces;
457   struct plugin_entry      *pe = cnf->plugins;
458   struct plugin_param      *pp;
459   struct ipc_host          *ih = cnf->ipc_hosts;
460   struct ipc_net           *ie = cnf->ipc_nets;
461
462   char ipv6_buf[100];             /* buffer for IPv6 inet_htop */
463   struct in_addr in4;
464
465   FILE *fd;
466
467   fd = fopen(fname, "w");
468
469   if(fd == NULL)
470     {
471       fprintf(stderr, "Could not open file %s for writing\n%s\n", fname, strerror(errno));
472       return -1;
473     }
474
475   printf("Writing config to file \"%s\".... ", fname);
476
477   fprintf(fd, "#\n# Configuration file for olsr.org olsrd\n# automatically generated by olsrd-cnf %s\n#\n\n\n", PARSER_VERSION);
478
479   /* Debug level */
480   fprintf(fd, "# Debug level(0-9)\n# If set to 0 the daemon runs in the background\n\nDebugLevel\t%d\n\n", cnf->debug_level);
481
482   /* IP version */
483   if(cnf->ip_version == AF_INET6)
484     fprintf(fd, "# IP version to use (4 or 6)\n\nIpVersion\t6\n\n");
485   else
486     fprintf(fd, "# IP version to use (4 or 6)\n\nIpVersion\t4\n\n");
487
488
489   /* HNA IPv4 */
490   fprintf(fd, "# HNA IPv4 routes\n# syntax: netaddr netmask\n# Example Internet gateway:\n# 0.0.0.0 0.0.0.0\n\nHna4\n{\n");
491   while(h4)
492     {
493       in4.s_addr = h4->net.v4;
494       fprintf(fd, "    %s ", inet_ntoa(in4));
495       in4.s_addr = h4->netmask.v4;
496       fprintf(fd, "%s\n", inet_ntoa(in4));
497       h4 = h4->next;
498     }
499   fprintf(fd, "}\n\n");
500
501
502   /* HNA IPv6 */
503   fprintf(fd, "# HNA IPv6 routes\n# syntax: netaddr prefix\n# Example Internet gateway:\nHna6\n{\n");
504   while(h6)
505     {
506       fprintf(fd, "    %s/%d\n", (char *)inet_ntop(AF_INET6, &h6->net.v6, ipv6_buf, sizeof(ipv6_buf)), h6->prefix_len);
507       h6 = h6->next;
508     }
509
510   fprintf(fd, "}\n\n");
511
512   /* No interfaces */
513   fprintf(fd, "# Should olsrd keep on running even if there are\n# no interfaces available? This is a good idea\n# for a PCMCIA/USB hotswap environment.\n# \"yes\" OR \"no\"\n\nAllowNoInt\t");
514   if(cnf->allow_no_interfaces)
515     fprintf(fd, "yes\n\n");
516   else
517     fprintf(fd, "no\n\n");
518
519   /* TOS */
520   fprintf(fd, "# TOS(type of service) value for\n# the IP header of control traffic.\n# default is 16\n\n");
521   fprintf(fd, "TosValue\t%d\n\n", cnf->tos);
522
523   /* Willingness */
524   fprintf(fd, "# The fixed willingness to use(0-7)\n# If not set willingness will be calculated\n# dynammically based on battery/power status\n\n");
525   if(cnf->willingness_auto)
526     fprintf(fd, "#Willingness\t4\n\n");
527   else
528     fprintf(fd, "Willingness%d\n\n", cnf->willingness);
529
530   /* IPC */
531   fprintf(fd, "# Allow processes like the GUI front-end\n# to connect to the daemon.\n\n");
532   fprintf(fd, "IpcConnect\n{\n");
533   fprintf(fd, "   MaxConnections  %d\n\n", cnf->ipc_connections);
534
535   while(ih)
536     {
537       in4.s_addr = ih->host.v4;
538       fprintf(fd, "   Host          %s\n", inet_ntoa(in4));
539       ih = ih->next;
540     }
541   fprintf(fd, "\n");
542   while(ie)
543     {
544       in4.s_addr = ie->net.v4;
545       fprintf(fd, "   Net           %s ", inet_ntoa(in4));
546       in4.s_addr = ie->mask.v4;
547       fprintf(fd, "%s\n", inet_ntoa(in4));
548       ie = ie->next;
549     }
550
551   fprintf(fd, "}\n\n");
552
553
554
555   /* Hysteresis */
556   fprintf(fd, "# Wether to use hysteresis or not\n# Hysteresis adds more robustness to the\n# link sensing but delays neighbor registration.\n# Used by default. 'yes' or 'no'\n\n");
557
558   if(cnf->use_hysteresis)
559     {
560       fprintf(fd, "UseHysteresis\tyes\n\n");
561       fprintf(fd, "# Hysteresis parameters\n# Do not alter these unless you know \n# what you are doing!\n# Set to auto by default. Allowed\n# values are floating point values\n# in the interval 0,1\n# THR_LOW must always be lower than\n# THR_HIGH!!\n\n");
562       fprintf(fd, "HystScaling\t%0.2f\n", cnf->hysteresis_param.scaling);
563       fprintf(fd, "HystThrHigh\t%0.2f\n", cnf->hysteresis_param.thr_high);
564       fprintf(fd, "HystThrLow\t%0.2f\n", cnf->hysteresis_param.thr_low);
565     }
566   else
567     fprintf(fd, "UseHysteresis\tno\n\n");
568
569   fprintf(fd, "\n\n");
570
571   /* Pollrate */
572   fprintf(fd, "# Polling rate in seconds(float).\n# Auto uses default value 0.05 sec\n\n");
573   fprintf(fd, "Pollrate\t%0.2f\n", cnf->pollrate);
574
575   /* TC redundancy */
576   fprintf(fd, "# TC redundancy\n# Specifies how much neighbor info should\n# be sent in TC messages\n# Possible values are:\n# 0 - only send MPR selectors\n# 1 - send MPR selectors and MPRs\n# 2 - send all neighbors\n#\n# defaults to 0\n\n");
577   fprintf(fd, "TcRedundancy\t%d\n\n", cnf->tc_redundancy);
578
579   /* MPR coverage */
580   fprintf(fd, "# MPR coverage\n# Specifies how many MPRs a node should\n# try select to reach every 2 hop neighbor\n# Can be set to any integer >0\n# defaults to 1\n\n");
581
582   fprintf(fd, "MprCoverage\t%d\n\n", cnf->mpr_coverage);
583
584   fprintf(fd, "# Link quality level\n# 0 = do not use link quality\n# 1 = use link quality for MPR selection\n# 2 = use link quality for MPR selection and routing\n\n");
585   fprintf(fd, "LinkQualityLevel\t%d\n\n", cnf->lq_level);
586
587   fprintf(fd, "# Link quality window size\n\n");
588   fprintf(fd, "LinkQualityWinSize\t%d\n\n", cnf->lq_wsize);
589
590   fprintf(fd, "# Clear screen when printing debug output?\n\n");
591   fprintf(fd, "ClearScreen\t%s\n\n", cnf->clear_screen ? "yes" : "no");
592
593   /* Plugins */
594   fprintf(fd, "# Olsrd plugins to load\n# This must be the absolute path to the file\n# or the loader will use the following scheme:\n# - Try the paths in the LD_LIBRARY_PATH \n#   environment variable.\n# - The list of libraries cached in /etc/ld.so.cache\n# - /lib, followed by /usr/lib\n\n");
595   if(pe)
596     {
597       while(pe)
598         {
599           fprintf(fd, "LoadPlugin \"%s\"\n{\n", pe->name);
600           pp = pe->params;
601           while(pp)
602             {
603               fprintf(fd, "    PlParam \"%s\" \"%s\"\n", pp->key, pp->value);
604               pp = pp->next;
605             }
606           fprintf(fd, "}\n");
607           pe = pe->next;
608         }
609     }
610   fprintf(fd, "\n");
611
612   
613   
614
615   /* Interfaces */
616   fprintf(fd, "# Interfaces\n\n");
617   /* Interfaces */
618   if(in)
619     {
620       while(in)
621         {
622           fprintf(fd, "Interface \"%s\"\n{\n", in->name);
623           fprintf(fd, "\n");
624       
625           fprintf(fd, "    # IPv4 broadcast address to use. The\n    # one usefull example would be 255.255.255.255\n    # If not defined the broadcastaddress\n    # every card is configured with is used\n\n");
626
627           if(in->cnf->ipv4_broadcast.v4)
628             {
629               in4.s_addr = in->cnf->ipv4_broadcast.v4;
630               fprintf(fd, "    Ip4Broadcast\t %s\n\n", inet_ntoa(in4));
631             }
632           else
633             {
634               fprintf(fd, "    #Ip4Broadcast\t255.255.255.255\n\n");
635             }
636           
637           
638           fprintf(fd, "    # IPv6 address scope to use.\n    # Must be 'site-local' or 'global'\n\n");
639           if(in->cnf->ipv6_addrtype)
640             fprintf(fd, "    Ip6AddrType \tsite-local\n\n");
641           else
642             fprintf(fd, "    Ip6AddrType \tglobal\n\n");
643           
644           fprintf(fd, "    # IPv6 multicast address to use when\n    # using site-local addresses.\n    # If not defined, ff05::15 is used\n");
645           fprintf(fd, "    Ip6MulticastSite\t%s\n\n", (char *)inet_ntop(AF_INET6, &in->cnf->ipv6_multi_site.v6, ipv6_buf, sizeof(ipv6_buf)));
646           fprintf(fd, "    # IPv6 multicast address to use when\n    # using global addresses\n    # If not defined, ff0e::1 is used\n");
647           fprintf(fd, "    Ip6MulticastGlobal\t%s\n\n", (char *)inet_ntop(AF_INET6, &in->cnf->ipv6_multi_glbl.v6, ipv6_buf, sizeof(ipv6_buf)));
648           
649           
650           
651           fprintf(fd, "    # Emission and validity intervals.\n    # If not defined, RFC proposed values will\n    # in most cases be used.\n\n");
652           
653           
654           if(in->cnf->hello_params.emission_interval != HELLO_INTERVAL)
655             fprintf(fd, "    HelloInterval\t%0.2f\n", in->cnf->hello_params.emission_interval);
656           else
657             fprintf(fd, "    #HelloInterval\t%0.2f\n", in->cnf->hello_params.emission_interval);
658           if(in->cnf->hello_params.validity_time != NEIGHB_HOLD_TIME)
659             fprintf(fd, "    HelloValidityTime\t%0.2f\n", in->cnf->hello_params.validity_time);
660           else
661             fprintf(fd, "    #HelloValidityTime\t%0.2f\n", in->cnf->hello_params.validity_time);
662           if(in->cnf->tc_params.emission_interval != TC_INTERVAL)
663             fprintf(fd, "    TcInterval\t\t%0.2f\n", in->cnf->tc_params.emission_interval);
664           else
665             fprintf(fd, "    #TcInterval\t\t%0.2f\n", in->cnf->tc_params.emission_interval);
666           if(in->cnf->tc_params.validity_time != TOP_HOLD_TIME)
667             fprintf(fd, "    TcValidityTime\t%0.2f\n", in->cnf->tc_params.validity_time);
668           else
669             fprintf(fd, "    #TcValidityTime\t%0.2f\n", in->cnf->tc_params.validity_time);
670           if(in->cnf->mid_params.emission_interval != MID_INTERVAL)
671             fprintf(fd, "    MidInterval\t\t%0.2f\n", in->cnf->mid_params.emission_interval);
672           else
673             fprintf(fd, "    #MidInterval\t%0.2f\n", in->cnf->mid_params.emission_interval);
674           if(in->cnf->mid_params.validity_time != MID_HOLD_TIME)
675             fprintf(fd, "    MidValidityTime\t%0.2f\n", in->cnf->mid_params.validity_time);
676           else
677             fprintf(fd, "    #MidValidityTime\t%0.2f\n", in->cnf->mid_params.validity_time);
678           if(in->cnf->hna_params.emission_interval != HNA_INTERVAL)
679             fprintf(fd, "    HnaInterval\t\t%0.2f\n", in->cnf->hna_params.emission_interval);
680           else
681             fprintf(fd, "    #HnaInterval\t%0.2f\n", in->cnf->hna_params.emission_interval);
682           if(in->cnf->hna_params.validity_time != HNA_HOLD_TIME)
683             fprintf(fd, "    HnaValidityTime\t%0.2f\n", in->cnf->hna_params.validity_time);       
684           else
685             fprintf(fd, "    #HnaValidityTime\t%0.2f\n", in->cnf->hna_params.validity_time);      
686           
687           
688           
689           fprintf(fd, "}\n\n");
690           in = in->next;
691         }
692
693     }
694
695
696   fprintf(fd, "\n# END AUTOGENERATED CONFIG\n");
697
698   fclose(fd);
699   printf("DONE\n");
700
701   return 1;
702 }
703
704
705
706
707
708 void
709 olsrd_print_cnf(struct olsrd_config *cnf)
710 {
711   struct hna4_entry        *h4 = cnf->hna4_entries;
712   struct hna6_entry        *h6 = cnf->hna6_entries;
713   struct olsr_if           *in = cnf->interfaces;
714   struct plugin_entry      *pe = cnf->plugins;
715   struct ipc_host          *ih = cnf->ipc_hosts;
716   struct ipc_net           *ie = cnf->ipc_nets;
717   char ipv6_buf[100];             /* buffer for IPv6 inet_htop */
718   struct in_addr in4;
719
720   printf(" *** olsrd configuration ***\n");
721
722   printf("Debug Level      : %d\n", cnf->debug_level);
723   if(cnf->ip_version == AF_INET6)
724     printf("IpVersion        : 6\n");
725   else
726     printf("IpVersion        : 4\n");
727   if(cnf->allow_no_interfaces)
728     printf("No interfaces    : ALLOWED\n");
729   else
730     printf("No interfaces    : NOT ALLOWED\n");
731   printf("TOS              : 0x%02x\n", cnf->tos);
732   if(cnf->willingness_auto)
733     printf("Willingness      : AUTO\n");
734   else
735     printf("Willingness      : %d\n", cnf->willingness);
736
737   printf("IPC connections  : %d\n", cnf->ipc_connections);
738
739   while(ih)
740     {
741       in4.s_addr = ih->host.v4;
742       printf("\tHost %s\n", inet_ntoa(in4));
743       ih = ih->next;
744     }
745   
746   while(ie)
747     {
748       in4.s_addr = ie->net.v4;
749       printf("\tNet %s/", inet_ntoa(in4));
750       in4.s_addr = ie->mask.v4;
751       printf("%s\n", inet_ntoa(in4));
752       ie = ie->next;
753     }
754
755
756   printf("Pollrate         : %0.2f\n", cnf->pollrate);
757
758   printf("TC redundancy    : %d\n", cnf->tc_redundancy);
759
760   printf("MPR coverage     : %d\n", cnf->mpr_coverage);
761    
762   printf("LQ level         : %d\n", cnf->lq_level);
763
764   printf("LQ window size   : %d\n", cnf->lq_wsize);
765
766   printf("Clear screen     : %s\n", cnf->clear_screen ? "yes" : "no");
767
768   /* Interfaces */
769   if(in)
770     {
771       printf("Interfaces:\n");
772       while(in)
773         {
774           printf(" dev: \"%s\"\n", in->name);
775           
776           if(in->cnf->ipv4_broadcast.v4)
777             {
778               in4.s_addr = in->cnf->ipv4_broadcast.v4;
779               printf("\tIPv4 broadcast        : %s\n", inet_ntoa(in4));
780             }
781           else
782             {
783               printf("\tIPv4 broadcast        : AUTO\n");
784             }
785           
786           if(in->cnf->ipv6_addrtype)
787             printf("\tIPv6 addrtype         : site-local\n");
788           else
789             printf("\tIPv6 addrtype         : global\n");
790           
791           //union olsr_ip_addr       ipv6_multi_site;
792           //union olsr_ip_addr       ipv6_multi_glbl;
793           printf("\tIPv6 multicast site   : %s\n", (char *)inet_ntop(AF_INET6, &in->cnf->ipv6_multi_site.v6, ipv6_buf, sizeof(ipv6_buf)));
794           printf("\tIPv6 multicast global : %s\n", (char *)inet_ntop(AF_INET6, &in->cnf->ipv6_multi_glbl.v6, ipv6_buf, sizeof(ipv6_buf)));
795           
796           printf("\tHELLO emission int    : %0.2f\n", in->cnf->hello_params.emission_interval);
797           printf("\tHELLO validity time   : %0.2f\n", in->cnf->hello_params.validity_time);
798           printf("\tTC emission int       : %0.2f\n", in->cnf->tc_params.emission_interval);
799           printf("\tTC validity time      : %0.2f\n", in->cnf->tc_params.validity_time);
800           printf("\tMID emission int      : %0.2f\n", in->cnf->mid_params.emission_interval);
801           printf("\tMID validity time     : %0.2f\n", in->cnf->mid_params.validity_time);
802           printf("\tHNA emission int      : %0.2f\n", in->cnf->hna_params.emission_interval);
803           printf("\tHNA validity time     : %0.2f\n", in->cnf->hna_params.validity_time);
804           
805           
806           
807           in = in->next;
808
809         }
810     }
811
812
813
814
815   /* Plugins */
816   if(pe)
817     {
818       printf("Plugins:\n");
819
820       while(pe)
821         {
822           printf("\tName: \"%s\"\n", pe->name);
823           pe = pe->next;
824         }
825     }
826
827   /* Hysteresis */
828   if(cnf->use_hysteresis)
829     {
830       printf("Using hysteresis:\n");
831       printf("\tScaling : %0.2f\n", cnf->hysteresis_param.scaling);
832       printf("\tThr high: %0.2f\n", cnf->hysteresis_param.thr_high);
833       printf("\tThr low : %0.2f\n", cnf->hysteresis_param.thr_low);
834     }
835   else
836     printf("Not using hysteresis\n");
837
838   /* HNA IPv4 */
839   if(h4)
840     {
841
842       printf("HNA4 entries:\n");
843       while(h4)
844         {
845           in4.s_addr = h4->net.v4;
846           printf("\t%s/", inet_ntoa(in4));
847           in4.s_addr = h4->netmask.v4;
848           printf("%s\n", inet_ntoa(in4));
849
850           h4 = h4->next;
851         }
852     }
853
854   /* HNA IPv6 */
855   if(h6)
856     {
857       printf("HNA6 entries:\n");
858       while(h6)
859         {
860           printf("\t%s/%d\n", (char *)inet_ntop(AF_INET6, &h6->net.v6, ipv6_buf, sizeof(ipv6_buf)), h6->prefix_len);
861           h6 = h6->next;
862         }
863     }
864 }
865
866 void *olsrd_cnf_malloc(unsigned int len)
867 {
868   return malloc(len);
869 }
870
871 void olsrd_cnf_free(void *addr)
872 {
873   free(addr);
874 }