Hysteresis and LQ cannot be active at the same time, added config check
[olsrd.git] / src / cfgparser / olsrd_conf.c
1 /*
2  * The olsr.org Optimized Link-State Routing daemon(olsrd)
3  * Copyright (c) 2004, Andreas T√łnnesen(andreto@olsr.org)
4  * All rights reserved.
5  *
6  * Redistribution and use in source and binary forms, with or without 
7  * modification, are permitted provided that the following conditions 
8  * are met:
9  *
10  * * Redistributions of source code must retain the above copyright 
11  *   notice, this list of conditions and the following disclaimer.
12  * * Redistributions in binary form must reproduce the above copyright 
13  *   notice, this list of conditions and the following disclaimer in 
14  *   the documentation and/or other materials provided with the 
15  *   distribution.
16  * * Neither the name of olsr.org, olsrd nor the names of its 
17  *   contributors may be used to endorse or promote products derived 
18  *   from this software without specific prior written permission.
19  *
20  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 
21  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 
22  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS 
23  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE 
24  * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, 
25  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, 
26  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; 
27  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER 
28  * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 
29  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN 
30  * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 
31  * POSSIBILITY OF SUCH DAMAGE.
32  *
33  * Visit http://www.olsr.org for more information.
34  *
35  * If you find this software useful feel free to make a donation
36  * to the project. For more information see the website or contact
37  * the copyright holders.
38  *
39  * $Id: olsrd_conf.c,v 1.30 2004/12/05 13:21:02 kattemat Exp $
40  */
41
42
43 #include <stdio.h>
44 #include <string.h>
45 #include <errno.h>
46 #include <stdlib.h>
47 #include <sys/types.h>
48 #include <sys/socket.h>
49 #include <netinet/in.h>
50 #include <arpa/inet.h>
51
52 #include "olsrd_conf.h"
53
54
55 extern FILE *yyin;
56 extern int yyparse(void);
57
58 static char copyright_string[] = "The olsr.org Optimized Link-State Routing daemon(olsrd) Copyright (c) 2004, Andreas T√łnnesen(andreto@olsr.org) All rights reserved.";
59
60 #ifdef MAKELIB
61
62 /* Build as DLL */
63
64 void __attribute__ ((constructor)) 
65 my_init(void);
66
67 void __attribute__ ((destructor)) 
68 my_fini(void);
69
70
71 /**
72  *Constructor
73  */
74 void
75 my_init()
76 {
77   /* Print plugin info to stdout */
78   printf("olsrd config file parser %s loaded\n", PARSER_VERSION);
79
80   return;
81 }
82
83 /**
84  *Destructor
85  */
86 void
87 my_fini()
88 {
89   printf("See you around!\n");
90   return;
91 }
92
93 #else
94
95 #ifdef MAKEBIN
96
97 /* Build as standalone binary */
98 int 
99 main(int argc, char *argv[])
100 {
101   struct olsrd_config *cnf;
102
103   if(argc == 1)
104     {
105       fprintf(stderr, "Usage: olsrd_cfgparser [filename] -print\n\n");
106       exit(EXIT_FAILURE);
107     }
108
109   if((cnf = olsrd_parse_cnf(argv[1])) != NULL)
110     {
111       if((argc > 2) && (!strcmp(argv[2], "-print")))
112         {
113           olsrd_print_cnf(cnf);  
114           olsrd_write_cnf(cnf, "./out.conf");
115         }
116       else
117         printf("Use -print to view parsed values\n");
118       printf("Configfile parsed OK\n");
119     }
120   else
121     {
122       printf("Failed parsing \"%s\"\n", argv[1]);
123     }
124
125   return 0;
126 }
127
128 #else
129
130 /* Build as part of olsrd */
131
132
133 #endif
134
135 #endif
136
137 struct olsrd_config *
138 olsrd_parse_cnf(const char *filename)
139 {
140   struct olsr_if *in;
141
142   /* Stop the compiler from complaining */
143   strlen(copyright_string);
144
145   cnf = malloc(sizeof(struct olsrd_config));
146   if (cnf == NULL)
147     {
148       fprintf(stderr, "Out of memory %s\n", __func__);
149       return NULL;
150   }
151
152   set_default_cnf(cnf);
153
154   printf("Parsing file: \"%s\"\n", filename);
155
156   yyin = fopen(filename, "r");
157   
158   if (yyin == NULL)
159     {
160       fprintf(stderr, "Cannot open configuration file '%s': %s.\n",
161               filename, strerror(errno));
162       free(cnf);
163       return NULL;
164   }
165
166   current_line = 1;
167
168   if (yyparse() != 0)
169     {
170       fclose(yyin);
171       olsrd_free_cnf(cnf);
172       return NULL;
173     }
174   
175   fclose(yyin);
176
177
178   in = cnf->interfaces;
179
180   while(in)
181     {
182       /* set various stuff */
183       in->index = cnf->ifcnt++;
184       in->configured = OLSR_FALSE;
185       in->interf = NULL;
186       in = in->next;
187     }
188
189
190   return cnf;
191 }
192
193
194 int
195 olsrd_sanity_check_cnf(struct olsrd_config *cnf)
196 {
197   struct olsr_if           *in = cnf->interfaces;
198   struct if_config_options *io;
199
200   /* Debug level */
201   if(cnf->debug_level < MIN_DEBUGLVL ||
202      cnf->debug_level > MAX_DEBUGLVL)
203     {
204       fprintf(stderr, "Debuglevel %d is not allowed\n", cnf->debug_level);
205       return -1;
206     }
207
208   /* IP version */
209   if(cnf->ip_version != AF_INET &&
210      cnf->ip_version != AF_INET6)
211     {
212       fprintf(stderr, "Ipversion %d not allowed!\n", cnf->ip_version);
213       return -1;
214     }
215
216   /* TOS */
217   if(//cnf->tos < MIN_TOS ||
218      cnf->tos > MAX_TOS)
219     {
220       fprintf(stderr, "TOS %d is not allowed\n", cnf->tos);
221       return -1;
222     }
223
224   if(cnf->willingness_auto == OLSR_FALSE &&
225      (cnf->willingness < MIN_WILLINGNESS ||
226       cnf->willingness > MAX_WILLINGNESS))
227     {
228       fprintf(stderr, "Willingness %d is not allowed\n", cnf->willingness);
229       return -1;
230     }
231
232   /* Hysteresis */
233   if(cnf->use_hysteresis == OLSR_TRUE)
234     {
235       if(cnf->hysteresis_param.scaling < MIN_HYST_PARAM ||
236          cnf->hysteresis_param.scaling > MAX_HYST_PARAM)
237         {
238           fprintf(stderr, "Hyst scaling %0.2f is not allowed\n", cnf->hysteresis_param.scaling);
239           return -1;
240         }
241
242       if(cnf->hysteresis_param.thr_high <= cnf->hysteresis_param.thr_low)
243         {
244           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);
245           return -1;
246         }
247
248       if(cnf->hysteresis_param.thr_high < MIN_HYST_PARAM ||
249          cnf->hysteresis_param.thr_high > MAX_HYST_PARAM)
250         {
251           fprintf(stderr, "Hyst upper thr %0.2f is not allowed\n", cnf->hysteresis_param.thr_high);
252           return -1;
253         }
254
255       if(cnf->hysteresis_param.thr_low < MIN_HYST_PARAM ||
256          cnf->hysteresis_param.thr_low > MAX_HYST_PARAM)
257         {
258           fprintf(stderr, "Hyst lower thr %0.2f is not allowed\n", cnf->hysteresis_param.thr_low);
259           return -1;
260         }
261     }
262
263   /* Pollrate */
264
265   if(cnf->pollrate < MIN_POLLRATE ||
266      cnf->pollrate > MAX_POLLRATE)
267     {
268       fprintf(stderr, "Pollrate %0.2f is not allowed\n", cnf->pollrate);
269       return -1;
270     }
271
272   /* TC redundancy */
273
274   if(//cnf->tc_redundancy < MIN_TC_REDUNDANCY ||
275      cnf->tc_redundancy > MAX_TC_REDUNDANCY)
276     {
277       fprintf(stderr, "TC redundancy %d is not allowed\n", cnf->tc_redundancy);
278       return -1;
279     }
280
281   /* MPR coverage */
282   if(cnf->mpr_coverage < MIN_MPR_COVERAGE ||
283      cnf->mpr_coverage > MAX_MPR_COVERAGE)
284     {
285       fprintf(stderr, "MPR coverage %d is not allowed\n", cnf->mpr_coverage);
286       return -1;
287     }
288
289   /* Link Q and hysteresis cannot be activated at the same time */
290   if(cnf->use_hysteresis == OLSR_TRUE && cnf->lq_level)
291     {
292       fprintf(stderr, "Hysteresis and LinkQuality cannot both be active! Deactivate one of them.\n");
293       return -1;
294     }
295
296   /* Link quality level */
297
298   if(cnf->lq_level > MAX_LQ_LEVEL)
299     {
300       fprintf(stderr, "LQ level %d is not allowed\n", cnf->lq_level);
301       return -1;
302     }
303
304   /* Link quality window size */
305   if(cnf->lq_level && (cnf->lq_wsize < MIN_LQ_WSIZE || cnf->lq_wsize > MAX_LQ_WSIZE))
306     {
307       fprintf(stderr, "LQ window size %d is not allowed\n", cnf->lq_wsize);
308       return -1;
309     }
310
311   if(in == NULL)
312     {
313       fprintf(stderr, "No interfaces configured!\n");
314       return -1;
315     }
316
317   /* Interfaces */
318   while(in)
319     {
320       io = in->cnf;
321
322       if(in->name == NULL || !strlen(in->name))
323         {
324           fprintf(stderr, "Interface has no name!\n");
325           return -1;
326         }
327
328       if(io == NULL)
329         {
330           fprintf(stderr, "Interface %s has no configuration!\n", in->name);
331           return -1;
332         }
333
334       /* HELLO interval */
335       if(io->hello_params.emission_interval < cnf->pollrate ||
336          io->hello_params.emission_interval > io->hello_params.validity_time)
337         {
338           fprintf(stderr, "Bad HELLO parameters! (em: %0.2f, vt: %0.2f)\n", io->hello_params.emission_interval, io->hello_params.validity_time);
339           return -1;
340         }
341
342       /* TC interval */
343       if(io->tc_params.emission_interval < cnf->pollrate ||
344          io->tc_params.emission_interval > io->tc_params.validity_time)
345         {
346           fprintf(stderr, "Bad TC parameters! (em: %0.2f, vt: %0.2f)\n", io->tc_params.emission_interval, io->tc_params.validity_time);
347           return -1;
348         }
349
350       /* MID interval */
351       if(io->mid_params.emission_interval < cnf->pollrate ||
352          io->mid_params.emission_interval > io->mid_params.validity_time)
353         {
354           fprintf(stderr, "Bad MID parameters! (em: %0.2f, vt: %0.2f)\n", io->mid_params.emission_interval, io->mid_params.validity_time);
355           return -1;
356         }
357
358       /* HNA interval */
359       if(io->hna_params.emission_interval < cnf->pollrate ||
360          io->hna_params.emission_interval > io->hna_params.validity_time)
361         {
362           fprintf(stderr, "Bad HNA parameters! (em: %0.2f, vt: %0.2f)\n", io->hna_params.emission_interval, io->hna_params.validity_time);
363           return -1;
364         }
365
366       in = in->next;
367     }
368
369   return 0;
370 }
371
372
373 void
374 olsrd_free_cnf(struct olsrd_config *cnf)
375 {
376   struct hna4_entry        *h4d, *h4 = cnf->hna4_entries;
377   struct hna6_entry        *h6d, *h6 = cnf->hna6_entries;
378   struct olsr_if           *ind, *in = cnf->interfaces;
379   struct plugin_entry      *ped, *pe = cnf->plugins;
380   
381   while(h4)
382     {
383       h4d = h4;
384       h4 = h4->next;
385       free(h4d);
386     }
387
388   while(h6)
389     {
390       h6d = h6;
391       h6 = h6->next;
392       free(h6d);
393     }
394
395   while(in)
396     {
397       free(in->cnf);
398       ind = in;
399       in = in->next;
400       free(ind->name);
401       free(ind->config);
402       free(ind);
403     }
404
405   while(pe)
406     {
407       ped = pe;
408       pe = pe->next;
409       free(ped->name);
410       free(ped);
411     }
412
413   return;
414 }
415
416
417
418 struct olsrd_config *
419 olsrd_get_default_cnf()
420 {
421   cnf = malloc(sizeof(struct olsrd_config));
422   if (cnf == NULL)
423     {
424       fprintf(stderr, "Out of memory %s\n", __func__);
425       return NULL;
426   }
427
428   set_default_cnf(cnf);
429
430   return cnf;
431 }
432
433
434
435
436 void
437 set_default_cnf(struct olsrd_config *cnf)
438 {
439     memset(cnf, 0, sizeof(struct olsrd_config));
440     
441     cnf->debug_level = DEF_DEBUGLVL;
442     cnf->ip_version  = AF_INET;
443     cnf->allow_no_interfaces = DEF_ALLOW_NO_INTS;
444     cnf->tos = DEF_TOS;
445     cnf->willingness_auto = DEF_WILL_AUTO;
446     cnf->ipc_connections = DEF_IPC_CONNECTIONS;
447     cnf->open_ipc = cnf->ipc_connections ? OLSR_TRUE : OLSR_FALSE;
448
449     cnf->use_hysteresis = DEF_USE_HYST;
450     cnf->hysteresis_param.scaling = HYST_SCALING;
451     cnf->hysteresis_param.thr_high = HYST_THRESHOLD_HIGH;
452     cnf->hysteresis_param.thr_low = HYST_THRESHOLD_LOW;
453
454     cnf->pollrate = DEF_POLLRATE;
455
456     cnf->tc_redundancy = TC_REDUNDANCY;
457     cnf->mpr_coverage = MPR_COVERAGE;
458     cnf->lq_level = DEF_LQ_LEVEL;
459     cnf->lq_wsize = DEF_LQ_WSIZE;
460     cnf->clear_screen = DEF_CLEAR_SCREEN;
461 }
462
463
464
465
466 struct if_config_options *
467 get_default_if_config()
468 {
469   struct if_config_options *io = malloc(sizeof(struct if_config_options));
470   struct in6_addr in6;
471  
472   memset(io, 0, sizeof(struct if_config_options));
473
474   io->ipv6_addrtype = 1; /* XXX - FixMe */
475
476   if(inet_pton(AF_INET6, OLSR_IPV6_MCAST_SITE_LOCAL, &in6) < 0)
477     {
478       fprintf(stderr, "Failed converting IP address %s\n", OLSR_IPV6_MCAST_SITE_LOCAL);
479       return NULL;
480     }
481   memcpy(&io->ipv6_multi_site.v6, &in6, sizeof(struct in6_addr));
482
483   if(inet_pton(AF_INET6, OLSR_IPV6_MCAST_GLOBAL, &in6) < 0)
484     {
485       fprintf(stderr, "Failed converting IP address %s\n", OLSR_IPV6_MCAST_GLOBAL);
486       return NULL;
487     }
488   memcpy(&io->ipv6_multi_glbl.v6, &in6, sizeof(struct in6_addr));
489
490
491   io->hello_params.emission_interval = HELLO_INTERVAL;
492   io->hello_params.validity_time = NEIGHB_HOLD_TIME;
493   io->tc_params.emission_interval = TC_INTERVAL;
494   io->tc_params.validity_time = TOP_HOLD_TIME;
495   io->mid_params.emission_interval = MID_INTERVAL;
496   io->mid_params.validity_time = MID_HOLD_TIME;
497   io->hna_params.emission_interval = HNA_INTERVAL;
498   io->hna_params.validity_time = HNA_HOLD_TIME;
499
500   return io;
501
502 }
503
504
505
506
507
508 int
509 olsrd_write_cnf(struct olsrd_config *cnf, const char *fname)
510 {
511   struct hna4_entry        *h4 = cnf->hna4_entries;
512   struct hna6_entry        *h6 = cnf->hna6_entries;
513   struct olsr_if           *in = cnf->interfaces;
514   struct plugin_entry      *pe = cnf->plugins;
515   struct plugin_param      *pp;
516   struct ipc_host          *ih = cnf->ipc_hosts;
517   struct ipc_net           *ie = cnf->ipc_nets;
518
519   char ipv6_buf[100];             /* buffer for IPv6 inet_htop */
520   struct in_addr in4;
521
522   FILE *fd;
523
524   fd = fopen(fname, "w");
525
526   if(fd == NULL)
527     {
528       fprintf(stderr, "Could not open file %s for writing\n%s\n", fname, strerror(errno));
529       return -1;
530     }
531
532   printf("Writing config to file \"%s\".... ", fname);
533
534   fprintf(fd, "#\n# Configuration file for olsr.org olsrd\n# automatically generated by olsrd-cnf %s\n#\n\n\n", PARSER_VERSION);
535
536   /* Debug level */
537   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);
538
539   /* IP version */
540   if(cnf->ip_version == AF_INET6)
541     fprintf(fd, "# IP version to use (4 or 6)\n\nIpVersion\t6\n\n");
542   else
543     fprintf(fd, "# IP version to use (4 or 6)\n\nIpVersion\t4\n\n");
544
545
546   /* HNA IPv4 */
547   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");
548   while(h4)
549     {
550       in4.s_addr = h4->net.v4;
551       fprintf(fd, "    %s ", inet_ntoa(in4));
552       in4.s_addr = h4->netmask.v4;
553       fprintf(fd, "%s\n", inet_ntoa(in4));
554       h4 = h4->next;
555     }
556   fprintf(fd, "}\n\n");
557
558
559   /* HNA IPv6 */
560   fprintf(fd, "# HNA IPv6 routes\n# syntax: netaddr prefix\n# Example Internet gateway:\nHna6\n{\n");
561   while(h6)
562     {
563       fprintf(fd, "    %s/%d\n", (char *)inet_ntop(AF_INET6, &h6->net.v6, ipv6_buf, sizeof(ipv6_buf)), h6->prefix_len);
564       h6 = h6->next;
565     }
566
567   fprintf(fd, "}\n\n");
568
569   /* No interfaces */
570   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");
571   if(cnf->allow_no_interfaces)
572     fprintf(fd, "yes\n\n");
573   else
574     fprintf(fd, "no\n\n");
575
576   /* TOS */
577   fprintf(fd, "# TOS(type of service) value for\n# the IP header of control traffic.\n# default is 16\n\n");
578   fprintf(fd, "TosValue\t%d\n\n", cnf->tos);
579
580   /* Willingness */
581   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");
582   if(cnf->willingness_auto)
583     fprintf(fd, "#Willingness\t4\n\n");
584   else
585     fprintf(fd, "Willingness%d\n\n", cnf->willingness);
586
587   /* IPC */
588   fprintf(fd, "# Allow processes like the GUI front-end\n# to connect to the daemon.\n\n");
589   fprintf(fd, "IpcConnect\n{\n");
590   fprintf(fd, "   MaxConnections  %d\n\n", cnf->ipc_connections);
591
592   while(ih)
593     {
594       in4.s_addr = ih->host.v4;
595       fprintf(fd, "   Host          %s\n", inet_ntoa(in4));
596       ih = ih->next;
597     }
598   fprintf(fd, "\n");
599   while(ie)
600     {
601       in4.s_addr = ie->net.v4;
602       fprintf(fd, "   Net           %s ", inet_ntoa(in4));
603       in4.s_addr = ie->mask.v4;
604       fprintf(fd, "%s\n", inet_ntoa(in4));
605       ie = ie->next;
606     }
607
608   fprintf(fd, "}\n\n");
609
610
611
612   /* Hysteresis */
613   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");
614
615   if(cnf->use_hysteresis)
616     {
617       fprintf(fd, "UseHysteresis\tyes\n\n");
618       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");
619       fprintf(fd, "HystScaling\t%0.2f\n", cnf->hysteresis_param.scaling);
620       fprintf(fd, "HystThrHigh\t%0.2f\n", cnf->hysteresis_param.thr_high);
621       fprintf(fd, "HystThrLow\t%0.2f\n", cnf->hysteresis_param.thr_low);
622     }
623   else
624     fprintf(fd, "UseHysteresis\tno\n\n");
625
626   fprintf(fd, "\n\n");
627
628   /* Pollrate */
629   fprintf(fd, "# Polling rate in seconds(float).\n# Auto uses default value 0.05 sec\n\n");
630   fprintf(fd, "Pollrate\t%0.2f\n", cnf->pollrate);
631
632   /* TC redundancy */
633   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");
634   fprintf(fd, "TcRedundancy\t%d\n\n", cnf->tc_redundancy);
635
636   /* MPR coverage */
637   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");
638
639   fprintf(fd, "MprCoverage\t%d\n\n", cnf->mpr_coverage);
640
641   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");
642   fprintf(fd, "LinkQualityLevel\t%d\n\n", cnf->lq_level);
643
644   fprintf(fd, "# Link quality window size\n\n");
645   fprintf(fd, "LinkQualityWinSize\t%d\n\n", cnf->lq_wsize);
646
647   fprintf(fd, "# Clear screen when printing debug output?\n\n");
648   fprintf(fd, "ClearScreen\t%s\n\n", cnf->clear_screen ? "yes" : "no");
649
650   /* Plugins */
651   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");
652   if(pe)
653     {
654       while(pe)
655         {
656           fprintf(fd, "LoadPlugin \"%s\"\n{\n", pe->name);
657           pp = pe->params;
658           while(pp)
659             {
660               fprintf(fd, "    PlParam \"%s\" \"%s\"\n", pp->key, pp->value);
661               pp = pp->next;
662             }
663           fprintf(fd, "}\n");
664           pe = pe->next;
665         }
666     }
667   fprintf(fd, "\n");
668
669   
670   
671
672   /* Interfaces */
673   fprintf(fd, "# Interfaces\n\n");
674   /* Interfaces */
675   if(in)
676     {
677       while(in)
678         {
679           fprintf(fd, "Interface \"%s\"\n{\n", in->name);
680           fprintf(fd, "\n");
681       
682           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");
683
684           if(in->cnf->ipv4_broadcast.v4)
685             {
686               in4.s_addr = in->cnf->ipv4_broadcast.v4;
687               fprintf(fd, "    Ip4Broadcast\t %s\n\n", inet_ntoa(in4));
688             }
689           else
690             {
691               fprintf(fd, "    #Ip4Broadcast\t255.255.255.255\n\n");
692             }
693           
694           
695           fprintf(fd, "    # IPv6 address scope to use.\n    # Must be 'site-local' or 'global'\n\n");
696           if(in->cnf->ipv6_addrtype)
697             fprintf(fd, "    Ip6AddrType \tsite-local\n\n");
698           else
699             fprintf(fd, "    Ip6AddrType \tglobal\n\n");
700           
701           fprintf(fd, "    # IPv6 multicast address to use when\n    # using site-local addresses.\n    # If not defined, ff05::15 is used\n");
702           fprintf(fd, "    Ip6MulticastSite\t%s\n\n", (char *)inet_ntop(AF_INET6, &in->cnf->ipv6_multi_site.v6, ipv6_buf, sizeof(ipv6_buf)));
703           fprintf(fd, "    # IPv6 multicast address to use when\n    # using global addresses\n    # If not defined, ff0e::1 is used\n");
704           fprintf(fd, "    Ip6MulticastGlobal\t%s\n\n", (char *)inet_ntop(AF_INET6, &in->cnf->ipv6_multi_glbl.v6, ipv6_buf, sizeof(ipv6_buf)));
705           
706           
707           
708           fprintf(fd, "    # Emission and validity intervals.\n    # If not defined, RFC proposed values will\n    # in most cases be used.\n\n");
709           
710           
711           if(in->cnf->hello_params.emission_interval != HELLO_INTERVAL)
712             fprintf(fd, "    HelloInterval\t%0.2f\n", in->cnf->hello_params.emission_interval);
713           else
714             fprintf(fd, "    #HelloInterval\t%0.2f\n", in->cnf->hello_params.emission_interval);
715           if(in->cnf->hello_params.validity_time != NEIGHB_HOLD_TIME)
716             fprintf(fd, "    HelloValidityTime\t%0.2f\n", in->cnf->hello_params.validity_time);
717           else
718             fprintf(fd, "    #HelloValidityTime\t%0.2f\n", in->cnf->hello_params.validity_time);
719           if(in->cnf->tc_params.emission_interval != TC_INTERVAL)
720             fprintf(fd, "    TcInterval\t\t%0.2f\n", in->cnf->tc_params.emission_interval);
721           else
722             fprintf(fd, "    #TcInterval\t\t%0.2f\n", in->cnf->tc_params.emission_interval);
723           if(in->cnf->tc_params.validity_time != TOP_HOLD_TIME)
724             fprintf(fd, "    TcValidityTime\t%0.2f\n", in->cnf->tc_params.validity_time);
725           else
726             fprintf(fd, "    #TcValidityTime\t%0.2f\n", in->cnf->tc_params.validity_time);
727           if(in->cnf->mid_params.emission_interval != MID_INTERVAL)
728             fprintf(fd, "    MidInterval\t\t%0.2f\n", in->cnf->mid_params.emission_interval);
729           else
730             fprintf(fd, "    #MidInterval\t%0.2f\n", in->cnf->mid_params.emission_interval);
731           if(in->cnf->mid_params.validity_time != MID_HOLD_TIME)
732             fprintf(fd, "    MidValidityTime\t%0.2f\n", in->cnf->mid_params.validity_time);
733           else
734             fprintf(fd, "    #MidValidityTime\t%0.2f\n", in->cnf->mid_params.validity_time);
735           if(in->cnf->hna_params.emission_interval != HNA_INTERVAL)
736             fprintf(fd, "    HnaInterval\t\t%0.2f\n", in->cnf->hna_params.emission_interval);
737           else
738             fprintf(fd, "    #HnaInterval\t%0.2f\n", in->cnf->hna_params.emission_interval);
739           if(in->cnf->hna_params.validity_time != HNA_HOLD_TIME)
740             fprintf(fd, "    HnaValidityTime\t%0.2f\n", in->cnf->hna_params.validity_time);       
741           else
742             fprintf(fd, "    #HnaValidityTime\t%0.2f\n", in->cnf->hna_params.validity_time);      
743           
744           
745           
746           fprintf(fd, "}\n\n");
747           in = in->next;
748         }
749
750     }
751
752
753   fprintf(fd, "\n# END AUTOGENERATED CONFIG\n");
754
755   fclose(fd);
756   printf("DONE\n");
757
758   return 1;
759 }
760
761
762
763
764
765 void
766 olsrd_print_cnf(struct olsrd_config *cnf)
767 {
768   struct hna4_entry        *h4 = cnf->hna4_entries;
769   struct hna6_entry        *h6 = cnf->hna6_entries;
770   struct olsr_if           *in = cnf->interfaces;
771   struct plugin_entry      *pe = cnf->plugins;
772   struct ipc_host          *ih = cnf->ipc_hosts;
773   struct ipc_net           *ie = cnf->ipc_nets;
774   char ipv6_buf[100];             /* buffer for IPv6 inet_htop */
775   struct in_addr in4;
776
777   printf(" *** olsrd configuration ***\n");
778
779   printf("Debug Level      : %d\n", cnf->debug_level);
780   if(cnf->ip_version == AF_INET6)
781     printf("IpVersion        : 6\n");
782   else
783     printf("IpVersion        : 4\n");
784   if(cnf->allow_no_interfaces)
785     printf("No interfaces    : ALLOWED\n");
786   else
787     printf("No interfaces    : NOT ALLOWED\n");
788   printf("TOS              : 0x%02x\n", cnf->tos);
789   if(cnf->willingness_auto)
790     printf("Willingness      : AUTO\n");
791   else
792     printf("Willingness      : %d\n", cnf->willingness);
793
794   printf("IPC connections  : %d\n", cnf->ipc_connections);
795
796   while(ih)
797     {
798       in4.s_addr = ih->host.v4;
799       printf("\tHost %s\n", inet_ntoa(in4));
800       ih = ih->next;
801     }
802   
803   while(ie)
804     {
805       in4.s_addr = ie->net.v4;
806       printf("\tNet %s/", inet_ntoa(in4));
807       in4.s_addr = ie->mask.v4;
808       printf("%s\n", inet_ntoa(in4));
809       ie = ie->next;
810     }
811
812
813   printf("Pollrate         : %0.2f\n", cnf->pollrate);
814
815   printf("TC redundancy    : %d\n", cnf->tc_redundancy);
816
817   printf("MPR coverage     : %d\n", cnf->mpr_coverage);
818    
819   printf("LQ level         : %d\n", cnf->lq_level);
820
821   printf("LQ window size   : %d\n", cnf->lq_wsize);
822
823   printf("Clear screen     : %s\n", cnf->clear_screen ? "yes" : "no");
824
825   /* Interfaces */
826   if(in)
827     {
828       printf("Interfaces:\n");
829       while(in)
830         {
831           printf(" dev: \"%s\"\n", in->name);
832           
833           if(in->cnf->ipv4_broadcast.v4)
834             {
835               in4.s_addr = in->cnf->ipv4_broadcast.v4;
836               printf("\tIPv4 broadcast           : %s\n", inet_ntoa(in4));
837             }
838           else
839             {
840               printf("\tIPv4 broadcast           : AUTO\n");
841             }
842           
843           if(in->cnf->ipv6_addrtype)
844             printf("\tIPv6 addrtype            : %s\n", in->cnf->ipv6_addrtype ? "site-local" : "global");
845           
846           //union olsr_ip_addr       ipv6_multi_site;
847           //union olsr_ip_addr       ipv6_multi_glbl;
848           printf("\tIPv6 multicast site/glbl : %s", (char *)inet_ntop(AF_INET6, &in->cnf->ipv6_multi_site.v6, ipv6_buf, sizeof(ipv6_buf)));
849           printf("/%s\n", (char *)inet_ntop(AF_INET6, &in->cnf->ipv6_multi_glbl.v6, ipv6_buf, sizeof(ipv6_buf)));
850           
851           printf("\tHELLO emission/validity  : %0.2f/%0.2f\n", in->cnf->hello_params.emission_interval, in->cnf->hello_params.validity_time);
852           printf("\tTC emission/validity     : %0.2f/%0.2f\n", in->cnf->tc_params.emission_interval, in->cnf->tc_params.validity_time);
853           printf("\tMID emission/validity    : %0.2f/%0.2f\n", in->cnf->mid_params.emission_interval, in->cnf->mid_params.validity_time);
854           printf("\tHNA emission/validity    : %0.2f/%0.2f\n", in->cnf->hna_params.emission_interval, in->cnf->hna_params.validity_time);
855           
856           
857           
858           in = in->next;
859
860         }
861     }
862
863
864
865
866   /* Plugins */
867   if(pe)
868     {
869       printf("Plugins:\n");
870
871       while(pe)
872         {
873           printf("\tName: \"%s\"\n", pe->name);
874           pe = pe->next;
875         }
876     }
877
878   /* Hysteresis */
879   if(cnf->use_hysteresis)
880     {
881       printf("Using hysteresis:\n");
882       printf("\tScaling      : %0.2f\n", cnf->hysteresis_param.scaling);
883       printf("\tThr high/low : %0.2f/%0.2f\n", cnf->hysteresis_param.thr_high, cnf->hysteresis_param.thr_low);
884     }
885   else
886     printf("Not using hysteresis\n");
887
888   /* HNA IPv4 */
889   if(h4)
890     {
891
892       printf("HNA4 entries:\n");
893       while(h4)
894         {
895           in4.s_addr = h4->net.v4;
896           printf("\t%s/", inet_ntoa(in4));
897           in4.s_addr = h4->netmask.v4;
898           printf("%s\n", inet_ntoa(in4));
899
900           h4 = h4->next;
901         }
902     }
903
904   /* HNA IPv6 */
905   if(h6)
906     {
907       printf("HNA6 entries:\n");
908       while(h6)
909         {
910           printf("\t%s/%d\n", (char *)inet_ntop(AF_INET6, &h6->net.v6, ipv6_buf, sizeof(ipv6_buf)), h6->prefix_len);
911           h6 = h6->next;
912         }
913     }
914 }
915
916 void *olsrd_cnf_malloc(unsigned int len)
917 {
918   return malloc(len);
919 }
920
921 void olsrd_cnf_free(void *addr)
922 {
923   free(addr);
924 }
925
926 #if defined WIN32_STDIO_HACK
927 struct ioinfo
928 {
929         unsigned int handle;
930         unsigned char attr;
931         char buff;
932         int flag;
933         CRITICAL_SECTION lock;
934 };
935
936 void win32_stdio_hack(unsigned int handle)
937 {
938   HMODULE lib;
939   struct ioinfo **info;
940
941   lib = LoadLibrary("msvcrt.dll");
942
943   info = (struct ioinfo **)GetProcAddress(lib, "__pioinfo");
944
945   // (*info)[1].handle = handle;
946   // (*info)[1].attr = 0x89; // FOPEN | FTEXT | FPIPE;
947
948   (*info)[2].handle = handle;
949   (*info)[2].attr = 0x89;
950
951   // stdout->_file = 1;
952   stderr->_file = 2;
953
954   // setbuf(stdout, NULL);
955   setbuf(stderr, NULL);
956 }
957 #else
958 void win32_stdio_hack(unsigned int handle) {}
959 #endif