Implemented LinkQualityMult per-interface configuration directive.
[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.32 2005/02/15 17:17:44 tlopatic 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, *new_ifqueue, *in_tmp;
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   /* Turn the if queue "around" (added by user request) */
178   in = cnf->interfaces;
179   new_ifqueue = NULL;
180
181   while(in)
182     {
183       in_tmp = in; 
184       in = in->next;
185
186       in_tmp->next = new_ifqueue;
187       new_ifqueue = in_tmp;
188     }
189
190   cnf->interfaces = new_ifqueue;
191
192   in = cnf->interfaces;
193
194   while(in)
195     {
196       /* set various stuff */
197       in->index = cnf->ifcnt++;
198       in->configured = OLSR_FALSE;
199       in->interf = NULL;
200       in = in->next;
201     }
202
203
204   return cnf;
205 }
206
207
208 int
209 olsrd_sanity_check_cnf(struct olsrd_config *cnf)
210 {
211   struct olsr_if           *in = cnf->interfaces;
212   struct if_config_options *io;
213
214   /* Debug level */
215   if(cnf->debug_level < MIN_DEBUGLVL ||
216      cnf->debug_level > MAX_DEBUGLVL)
217     {
218       fprintf(stderr, "Debuglevel %d is not allowed\n", cnf->debug_level);
219       return -1;
220     }
221
222   /* IP version */
223   if(cnf->ip_version != AF_INET &&
224      cnf->ip_version != AF_INET6)
225     {
226       fprintf(stderr, "Ipversion %d not allowed!\n", cnf->ip_version);
227       return -1;
228     }
229
230   /* TOS */
231   if(//cnf->tos < MIN_TOS ||
232      cnf->tos > MAX_TOS)
233     {
234       fprintf(stderr, "TOS %d is not allowed\n", cnf->tos);
235       return -1;
236     }
237
238   if(cnf->willingness_auto == OLSR_FALSE &&
239      (cnf->willingness < MIN_WILLINGNESS ||
240       cnf->willingness > MAX_WILLINGNESS))
241     {
242       fprintf(stderr, "Willingness %d is not allowed\n", cnf->willingness);
243       return -1;
244     }
245
246   /* Hysteresis */
247   if(cnf->use_hysteresis == OLSR_TRUE)
248     {
249       if(cnf->hysteresis_param.scaling < MIN_HYST_PARAM ||
250          cnf->hysteresis_param.scaling > MAX_HYST_PARAM)
251         {
252           fprintf(stderr, "Hyst scaling %0.2f is not allowed\n", cnf->hysteresis_param.scaling);
253           return -1;
254         }
255
256       if(cnf->hysteresis_param.thr_high <= cnf->hysteresis_param.thr_low)
257         {
258           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);
259           return -1;
260         }
261
262       if(cnf->hysteresis_param.thr_high < MIN_HYST_PARAM ||
263          cnf->hysteresis_param.thr_high > MAX_HYST_PARAM)
264         {
265           fprintf(stderr, "Hyst upper thr %0.2f is not allowed\n", cnf->hysteresis_param.thr_high);
266           return -1;
267         }
268
269       if(cnf->hysteresis_param.thr_low < MIN_HYST_PARAM ||
270          cnf->hysteresis_param.thr_low > MAX_HYST_PARAM)
271         {
272           fprintf(stderr, "Hyst lower thr %0.2f is not allowed\n", cnf->hysteresis_param.thr_low);
273           return -1;
274         }
275     }
276
277   /* Pollrate */
278
279   if(cnf->pollrate < MIN_POLLRATE ||
280      cnf->pollrate > MAX_POLLRATE)
281     {
282       fprintf(stderr, "Pollrate %0.2f is not allowed\n", cnf->pollrate);
283       return -1;
284     }
285
286   /* TC redundancy */
287
288   if(//cnf->tc_redundancy < MIN_TC_REDUNDANCY ||
289      cnf->tc_redundancy > MAX_TC_REDUNDANCY)
290     {
291       fprintf(stderr, "TC redundancy %d is not allowed\n", cnf->tc_redundancy);
292       return -1;
293     }
294
295   /* MPR coverage */
296   if(cnf->mpr_coverage < MIN_MPR_COVERAGE ||
297      cnf->mpr_coverage > MAX_MPR_COVERAGE)
298     {
299       fprintf(stderr, "MPR coverage %d is not allowed\n", cnf->mpr_coverage);
300       return -1;
301     }
302
303   /* Link Q and hysteresis cannot be activated at the same time */
304   if(cnf->use_hysteresis == OLSR_TRUE && cnf->lq_level)
305     {
306       fprintf(stderr, "Hysteresis and LinkQuality cannot both be active! Deactivate one of them.\n");
307       return -1;
308     }
309
310   /* Link quality level */
311
312   if(cnf->lq_level > MAX_LQ_LEVEL)
313     {
314       fprintf(stderr, "LQ level %d is not allowed\n", cnf->lq_level);
315       return -1;
316     }
317
318   /* Link quality window size */
319   if(cnf->lq_level && (cnf->lq_wsize < MIN_LQ_WSIZE || cnf->lq_wsize > MAX_LQ_WSIZE))
320     {
321       fprintf(stderr, "LQ window size %d is not allowed\n", cnf->lq_wsize);
322       return -1;
323     }
324
325   if(in == NULL)
326     {
327       fprintf(stderr, "No interfaces configured!\n");
328       return -1;
329     }
330
331   /* Interfaces */
332   while(in)
333     {
334       io = in->cnf;
335
336       if(in->name == NULL || !strlen(in->name))
337         {
338           fprintf(stderr, "Interface has no name!\n");
339           return -1;
340         }
341
342       if(io == NULL)
343         {
344           fprintf(stderr, "Interface %s has no configuration!\n", in->name);
345           return -1;
346         }
347
348       /* HELLO interval */
349       if(io->hello_params.emission_interval < cnf->pollrate ||
350          io->hello_params.emission_interval > io->hello_params.validity_time)
351         {
352           fprintf(stderr, "Bad HELLO parameters! (em: %0.2f, vt: %0.2f)\n", io->hello_params.emission_interval, io->hello_params.validity_time);
353           return -1;
354         }
355
356       /* TC interval */
357       if(io->tc_params.emission_interval < cnf->pollrate ||
358          io->tc_params.emission_interval > io->tc_params.validity_time)
359         {
360           fprintf(stderr, "Bad TC parameters! (em: %0.2f, vt: %0.2f)\n", io->tc_params.emission_interval, io->tc_params.validity_time);
361           return -1;
362         }
363
364       /* MID interval */
365       if(io->mid_params.emission_interval < cnf->pollrate ||
366          io->mid_params.emission_interval > io->mid_params.validity_time)
367         {
368           fprintf(stderr, "Bad MID parameters! (em: %0.2f, vt: %0.2f)\n", io->mid_params.emission_interval, io->mid_params.validity_time);
369           return -1;
370         }
371
372       /* HNA interval */
373       if(io->hna_params.emission_interval < cnf->pollrate ||
374          io->hna_params.emission_interval > io->hna_params.validity_time)
375         {
376           fprintf(stderr, "Bad HNA parameters! (em: %0.2f, vt: %0.2f)\n", io->hna_params.emission_interval, io->hna_params.validity_time);
377           return -1;
378         }
379
380       in = in->next;
381     }
382
383   return 0;
384 }
385
386
387 void
388 olsrd_free_cnf(struct olsrd_config *cnf)
389 {
390   struct hna4_entry        *h4d, *h4 = cnf->hna4_entries;
391   struct hna6_entry        *h6d, *h6 = cnf->hna6_entries;
392   struct olsr_if           *ind, *in = cnf->interfaces;
393   struct plugin_entry      *ped, *pe = cnf->plugins;
394   struct olsr_lq_mult      *mult, *next_mult;
395   
396   while(h4)
397     {
398       h4d = h4;
399       h4 = h4->next;
400       free(h4d);
401     }
402
403   while(h6)
404     {
405       h6d = h6;
406       h6 = h6->next;
407       free(h6d);
408     }
409
410   while(in)
411     {
412       for (mult = in->cnf->lq_mult; mult != NULL; mult = next_mult)
413       {
414         next_mult = mult->next;
415         free(mult);
416       }
417
418       free(in->cnf);
419       ind = in;
420       in = in->next;
421       free(ind->name);
422       free(ind->config);
423       free(ind);
424     }
425
426   while(pe)
427     {
428       ped = pe;
429       pe = pe->next;
430       free(ped->name);
431       free(ped);
432     }
433
434   return;
435 }
436
437
438
439 struct olsrd_config *
440 olsrd_get_default_cnf()
441 {
442   cnf = malloc(sizeof(struct olsrd_config));
443   if (cnf == NULL)
444     {
445       fprintf(stderr, "Out of memory %s\n", __func__);
446       return NULL;
447   }
448
449   set_default_cnf(cnf);
450
451   return cnf;
452 }
453
454
455
456
457 void
458 set_default_cnf(struct olsrd_config *cnf)
459 {
460     memset(cnf, 0, sizeof(struct olsrd_config));
461     
462     cnf->debug_level = DEF_DEBUGLVL;
463     cnf->ip_version  = AF_INET;
464     cnf->allow_no_interfaces = DEF_ALLOW_NO_INTS;
465     cnf->tos = DEF_TOS;
466     cnf->willingness_auto = DEF_WILL_AUTO;
467     cnf->ipc_connections = DEF_IPC_CONNECTIONS;
468     cnf->open_ipc = cnf->ipc_connections ? OLSR_TRUE : OLSR_FALSE;
469
470     cnf->use_hysteresis = DEF_USE_HYST;
471     cnf->hysteresis_param.scaling = HYST_SCALING;
472     cnf->hysteresis_param.thr_high = HYST_THRESHOLD_HIGH;
473     cnf->hysteresis_param.thr_low = HYST_THRESHOLD_LOW;
474
475     cnf->pollrate = DEF_POLLRATE;
476
477     cnf->tc_redundancy = TC_REDUNDANCY;
478     cnf->mpr_coverage = MPR_COVERAGE;
479     cnf->lq_level = DEF_LQ_LEVEL;
480     cnf->lq_wsize = DEF_LQ_WSIZE;
481     cnf->clear_screen = DEF_CLEAR_SCREEN;
482 }
483
484
485
486
487 struct if_config_options *
488 get_default_if_config()
489 {
490   struct if_config_options *io = malloc(sizeof(struct if_config_options));
491   struct in6_addr in6;
492  
493   memset(io, 0, sizeof(struct if_config_options));
494
495   io->ipv6_addrtype = 1; /* XXX - FixMe */
496
497   if(inet_pton(AF_INET6, OLSR_IPV6_MCAST_SITE_LOCAL, &in6) < 0)
498     {
499       fprintf(stderr, "Failed converting IP address %s\n", OLSR_IPV6_MCAST_SITE_LOCAL);
500       return NULL;
501     }
502   memcpy(&io->ipv6_multi_site.v6, &in6, sizeof(struct in6_addr));
503
504   if(inet_pton(AF_INET6, OLSR_IPV6_MCAST_GLOBAL, &in6) < 0)
505     {
506       fprintf(stderr, "Failed converting IP address %s\n", OLSR_IPV6_MCAST_GLOBAL);
507       return NULL;
508     }
509   memcpy(&io->ipv6_multi_glbl.v6, &in6, sizeof(struct in6_addr));
510
511   io->lq_mult = NULL;
512
513   io->hello_params.emission_interval = HELLO_INTERVAL;
514   io->hello_params.validity_time = NEIGHB_HOLD_TIME;
515   io->tc_params.emission_interval = TC_INTERVAL;
516   io->tc_params.validity_time = TOP_HOLD_TIME;
517   io->mid_params.emission_interval = MID_INTERVAL;
518   io->mid_params.validity_time = MID_HOLD_TIME;
519   io->hna_params.emission_interval = HNA_INTERVAL;
520   io->hna_params.validity_time = HNA_HOLD_TIME;
521
522   return io;
523
524 }
525
526
527
528
529
530 int
531 olsrd_write_cnf(struct olsrd_config *cnf, const char *fname)
532 {
533   struct hna4_entry        *h4 = cnf->hna4_entries;
534   struct hna6_entry        *h6 = cnf->hna6_entries;
535   struct olsr_if           *in = cnf->interfaces;
536   struct plugin_entry      *pe = cnf->plugins;
537   struct plugin_param      *pp;
538   struct ipc_host          *ih = cnf->ipc_hosts;
539   struct ipc_net           *ie = cnf->ipc_nets;
540   struct olsr_lq_mult      *mult;
541
542   char ipv6_buf[100];             /* buffer for IPv6 inet_htop */
543   struct in_addr in4;
544
545   FILE *fd;
546
547   fd = fopen(fname, "w");
548
549   if(fd == NULL)
550     {
551       fprintf(stderr, "Could not open file %s for writing\n%s\n", fname, strerror(errno));
552       return -1;
553     }
554
555   printf("Writing config to file \"%s\".... ", fname);
556
557   fprintf(fd, "#\n# Configuration file for olsr.org olsrd\n# automatically generated by olsrd-cnf %s\n#\n\n\n", PARSER_VERSION);
558
559   /* Debug level */
560   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);
561
562   /* IP version */
563   if(cnf->ip_version == AF_INET6)
564     fprintf(fd, "# IP version to use (4 or 6)\n\nIpVersion\t6\n\n");
565   else
566     fprintf(fd, "# IP version to use (4 or 6)\n\nIpVersion\t4\n\n");
567
568
569   /* HNA IPv4 */
570   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");
571   while(h4)
572     {
573       in4.s_addr = h4->net.v4;
574       fprintf(fd, "    %s ", inet_ntoa(in4));
575       in4.s_addr = h4->netmask.v4;
576       fprintf(fd, "%s\n", inet_ntoa(in4));
577       h4 = h4->next;
578     }
579   fprintf(fd, "}\n\n");
580
581
582   /* HNA IPv6 */
583   fprintf(fd, "# HNA IPv6 routes\n# syntax: netaddr prefix\n# Example Internet gateway:\nHna6\n{\n");
584   while(h6)
585     {
586       fprintf(fd, "    %s/%d\n", (char *)inet_ntop(AF_INET6, &h6->net.v6, ipv6_buf, sizeof(ipv6_buf)), h6->prefix_len);
587       h6 = h6->next;
588     }
589
590   fprintf(fd, "}\n\n");
591
592   /* No interfaces */
593   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");
594   if(cnf->allow_no_interfaces)
595     fprintf(fd, "yes\n\n");
596   else
597     fprintf(fd, "no\n\n");
598
599   /* TOS */
600   fprintf(fd, "# TOS(type of service) value for\n# the IP header of control traffic.\n# default is 16\n\n");
601   fprintf(fd, "TosValue\t%d\n\n", cnf->tos);
602
603   /* Willingness */
604   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");
605   if(cnf->willingness_auto)
606     fprintf(fd, "#Willingness\t4\n\n");
607   else
608     fprintf(fd, "Willingness%d\n\n", cnf->willingness);
609
610   /* IPC */
611   fprintf(fd, "# Allow processes like the GUI front-end\n# to connect to the daemon.\n\n");
612   fprintf(fd, "IpcConnect\n{\n");
613   fprintf(fd, "   MaxConnections  %d\n\n", cnf->ipc_connections);
614
615   while(ih)
616     {
617       in4.s_addr = ih->host.v4;
618       fprintf(fd, "   Host          %s\n", inet_ntoa(in4));
619       ih = ih->next;
620     }
621   fprintf(fd, "\n");
622   while(ie)
623     {
624       in4.s_addr = ie->net.v4;
625       fprintf(fd, "   Net           %s ", inet_ntoa(in4));
626       in4.s_addr = ie->mask.v4;
627       fprintf(fd, "%s\n", inet_ntoa(in4));
628       ie = ie->next;
629     }
630
631   fprintf(fd, "}\n\n");
632
633
634
635   /* Hysteresis */
636   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");
637
638   if(cnf->use_hysteresis)
639     {
640       fprintf(fd, "UseHysteresis\tyes\n\n");
641       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");
642       fprintf(fd, "HystScaling\t%0.2f\n", cnf->hysteresis_param.scaling);
643       fprintf(fd, "HystThrHigh\t%0.2f\n", cnf->hysteresis_param.thr_high);
644       fprintf(fd, "HystThrLow\t%0.2f\n", cnf->hysteresis_param.thr_low);
645     }
646   else
647     fprintf(fd, "UseHysteresis\tno\n\n");
648
649   fprintf(fd, "\n\n");
650
651   /* Pollrate */
652   fprintf(fd, "# Polling rate in seconds(float).\n# Auto uses default value 0.05 sec\n\n");
653   fprintf(fd, "Pollrate\t%0.2f\n", cnf->pollrate);
654
655   /* TC redundancy */
656   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");
657   fprintf(fd, "TcRedundancy\t%d\n\n", cnf->tc_redundancy);
658
659   /* MPR coverage */
660   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");
661
662   fprintf(fd, "MprCoverage\t%d\n\n", cnf->mpr_coverage);
663
664   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");
665   fprintf(fd, "LinkQualityLevel\t%d\n\n", cnf->lq_level);
666
667   fprintf(fd, "# Link quality window size\n\n");
668   fprintf(fd, "LinkQualityWinSize\t%d\n\n", cnf->lq_wsize);
669
670   fprintf(fd, "# Clear screen when printing debug output?\n\n");
671   fprintf(fd, "ClearScreen\t%s\n\n", cnf->clear_screen ? "yes" : "no");
672
673   /* Plugins */
674   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");
675   if(pe)
676     {
677       while(pe)
678         {
679           fprintf(fd, "LoadPlugin \"%s\"\n{\n", pe->name);
680           pp = pe->params;
681           while(pp)
682             {
683               fprintf(fd, "    PlParam \"%s\" \"%s\"\n", pp->key, pp->value);
684               pp = pp->next;
685             }
686           fprintf(fd, "}\n");
687           pe = pe->next;
688         }
689     }
690   fprintf(fd, "\n");
691
692   
693   
694
695   /* Interfaces */
696   fprintf(fd, "# Interfaces\n\n");
697   /* Interfaces */
698   if(in)
699     {
700       while(in)
701         {
702           fprintf(fd, "Interface \"%s\"\n{\n", in->name);
703           fprintf(fd, "\n");
704       
705           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");
706
707           if(in->cnf->ipv4_broadcast.v4)
708             {
709               in4.s_addr = in->cnf->ipv4_broadcast.v4;
710               fprintf(fd, "    Ip4Broadcast\t %s\n\n", inet_ntoa(in4));
711             }
712           else
713             {
714               fprintf(fd, "    #Ip4Broadcast\t255.255.255.255\n\n");
715             }
716           
717           
718           fprintf(fd, "    # IPv6 address scope to use.\n    # Must be 'site-local' or 'global'\n\n");
719           if(in->cnf->ipv6_addrtype)
720             fprintf(fd, "    Ip6AddrType \tsite-local\n\n");
721           else
722             fprintf(fd, "    Ip6AddrType \tglobal\n\n");
723           
724           fprintf(fd, "    # IPv6 multicast address to use when\n    # using site-local addresses.\n    # If not defined, ff05::15 is used\n");
725           fprintf(fd, "    Ip6MulticastSite\t%s\n\n", (char *)inet_ntop(AF_INET6, &in->cnf->ipv6_multi_site.v6, ipv6_buf, sizeof(ipv6_buf)));
726           fprintf(fd, "    # IPv6 multicast address to use when\n    # using global addresses\n    # If not defined, ff0e::1 is used\n");
727           fprintf(fd, "    Ip6MulticastGlobal\t%s\n\n", (char *)inet_ntop(AF_INET6, &in->cnf->ipv6_multi_glbl.v6, ipv6_buf, sizeof(ipv6_buf)));
728           
729           
730           
731           fprintf(fd, "    # Emission and validity intervals.\n    # If not defined, RFC proposed values will\n    # in most cases be used.\n\n");
732           
733           
734           if(in->cnf->hello_params.emission_interval != HELLO_INTERVAL)
735             fprintf(fd, "    HelloInterval\t%0.2f\n", in->cnf->hello_params.emission_interval);
736           else
737             fprintf(fd, "    #HelloInterval\t%0.2f\n", in->cnf->hello_params.emission_interval);
738           if(in->cnf->hello_params.validity_time != NEIGHB_HOLD_TIME)
739             fprintf(fd, "    HelloValidityTime\t%0.2f\n", in->cnf->hello_params.validity_time);
740           else
741             fprintf(fd, "    #HelloValidityTime\t%0.2f\n", in->cnf->hello_params.validity_time);
742           if(in->cnf->tc_params.emission_interval != TC_INTERVAL)
743             fprintf(fd, "    TcInterval\t\t%0.2f\n", in->cnf->tc_params.emission_interval);
744           else
745             fprintf(fd, "    #TcInterval\t\t%0.2f\n", in->cnf->tc_params.emission_interval);
746           if(in->cnf->tc_params.validity_time != TOP_HOLD_TIME)
747             fprintf(fd, "    TcValidityTime\t%0.2f\n", in->cnf->tc_params.validity_time);
748           else
749             fprintf(fd, "    #TcValidityTime\t%0.2f\n", in->cnf->tc_params.validity_time);
750           if(in->cnf->mid_params.emission_interval != MID_INTERVAL)
751             fprintf(fd, "    MidInterval\t\t%0.2f\n", in->cnf->mid_params.emission_interval);
752           else
753             fprintf(fd, "    #MidInterval\t%0.2f\n", in->cnf->mid_params.emission_interval);
754           if(in->cnf->mid_params.validity_time != MID_HOLD_TIME)
755             fprintf(fd, "    MidValidityTime\t%0.2f\n", in->cnf->mid_params.validity_time);
756           else
757             fprintf(fd, "    #MidValidityTime\t%0.2f\n", in->cnf->mid_params.validity_time);
758           if(in->cnf->hna_params.emission_interval != HNA_INTERVAL)
759             fprintf(fd, "    HnaInterval\t\t%0.2f\n", in->cnf->hna_params.emission_interval);
760           else
761             fprintf(fd, "    #HnaInterval\t%0.2f\n", in->cnf->hna_params.emission_interval);
762           if(in->cnf->hna_params.validity_time != HNA_HOLD_TIME)
763             fprintf(fd, "    HnaValidityTime\t%0.2f\n", in->cnf->hna_params.validity_time);       
764           else
765             fprintf(fd, "    #HnaValidityTime\t%0.2f\n", in->cnf->hna_params.validity_time);      
766           
767           mult = in->cnf->lq_mult;
768
769           if (mult == NULL)
770             fprintf(fd, "    #LinkQualityMult\tdefault 1.0\n");
771
772           else
773           {
774             while (mult != NULL)
775             {
776               inet_ntop(cnf->ip_version, &mult->addr, ipv6_buf,
777                         sizeof (ipv6_buf));
778
779               fprintf(fd, "    LinkQualityMult\t%s %0.2f\n",
780                       ipv6_buf, mult->val);
781
782               mult = mult->next;
783             }
784           }
785           
786           fprintf(fd, "}\n\n");
787           in = in->next;
788         }
789
790     }
791
792
793   fprintf(fd, "\n# END AUTOGENERATED CONFIG\n");
794
795   fclose(fd);
796   printf("DONE\n");
797
798   return 1;
799 }
800
801
802
803
804
805 void
806 olsrd_print_cnf(struct olsrd_config *cnf)
807 {
808   struct hna4_entry        *h4 = cnf->hna4_entries;
809   struct hna6_entry        *h6 = cnf->hna6_entries;
810   struct olsr_if           *in = cnf->interfaces;
811   struct plugin_entry      *pe = cnf->plugins;
812   struct ipc_host          *ih = cnf->ipc_hosts;
813   struct ipc_net           *ie = cnf->ipc_nets;
814   struct olsr_lq_mult      *mult;
815   char ipv6_buf[100];             /* buffer for IPv6 inet_htop */
816   struct in_addr in4;
817
818   printf(" *** olsrd configuration ***\n");
819
820   printf("Debug Level      : %d\n", cnf->debug_level);
821   if(cnf->ip_version == AF_INET6)
822     printf("IpVersion        : 6\n");
823   else
824     printf("IpVersion        : 4\n");
825   if(cnf->allow_no_interfaces)
826     printf("No interfaces    : ALLOWED\n");
827   else
828     printf("No interfaces    : NOT ALLOWED\n");
829   printf("TOS              : 0x%02x\n", cnf->tos);
830   if(cnf->willingness_auto)
831     printf("Willingness      : AUTO\n");
832   else
833     printf("Willingness      : %d\n", cnf->willingness);
834
835   printf("IPC connections  : %d\n", cnf->ipc_connections);
836
837   while(ih)
838     {
839       in4.s_addr = ih->host.v4;
840       printf("\tHost %s\n", inet_ntoa(in4));
841       ih = ih->next;
842     }
843   
844   while(ie)
845     {
846       in4.s_addr = ie->net.v4;
847       printf("\tNet %s/", inet_ntoa(in4));
848       in4.s_addr = ie->mask.v4;
849       printf("%s\n", inet_ntoa(in4));
850       ie = ie->next;
851     }
852
853
854   printf("Pollrate         : %0.2f\n", cnf->pollrate);
855
856   printf("TC redundancy    : %d\n", cnf->tc_redundancy);
857
858   printf("MPR coverage     : %d\n", cnf->mpr_coverage);
859    
860   printf("LQ level         : %d\n", cnf->lq_level);
861
862   printf("LQ window size   : %d\n", cnf->lq_wsize);
863
864   printf("Clear screen     : %s\n", cnf->clear_screen ? "yes" : "no");
865
866   /* Interfaces */
867   if(in)
868     {
869       printf("Interfaces:\n");
870       while(in)
871         {
872           printf(" dev: \"%s\"\n", in->name);
873           
874           if(in->cnf->ipv4_broadcast.v4)
875             {
876               in4.s_addr = in->cnf->ipv4_broadcast.v4;
877               printf("\tIPv4 broadcast           : %s\n", inet_ntoa(in4));
878             }
879           else
880             {
881               printf("\tIPv4 broadcast           : AUTO\n");
882             }
883           
884           if(in->cnf->ipv6_addrtype)
885             printf("\tIPv6 addrtype            : %s\n", in->cnf->ipv6_addrtype ? "site-local" : "global");
886           
887           //union olsr_ip_addr       ipv6_multi_site;
888           //union olsr_ip_addr       ipv6_multi_glbl;
889           printf("\tIPv6 multicast site/glbl : %s", (char *)inet_ntop(AF_INET6, &in->cnf->ipv6_multi_site.v6, ipv6_buf, sizeof(ipv6_buf)));
890           printf("/%s\n", (char *)inet_ntop(AF_INET6, &in->cnf->ipv6_multi_glbl.v6, ipv6_buf, sizeof(ipv6_buf)));
891           
892           printf("\tHELLO emission/validity  : %0.2f/%0.2f\n", in->cnf->hello_params.emission_interval, in->cnf->hello_params.validity_time);
893           printf("\tTC emission/validity     : %0.2f/%0.2f\n", in->cnf->tc_params.emission_interval, in->cnf->tc_params.validity_time);
894           printf("\tMID emission/validity    : %0.2f/%0.2f\n", in->cnf->mid_params.emission_interval, in->cnf->mid_params.validity_time);
895           printf("\tHNA emission/validity    : %0.2f/%0.2f\n", in->cnf->hna_params.emission_interval, in->cnf->hna_params.validity_time);
896           
897           for (mult = in->cnf->lq_mult; mult != NULL; mult = mult->next)
898           {
899             inet_ntop(cnf->ip_version, &mult->addr, ipv6_buf,
900                       sizeof (ipv6_buf));
901
902             printf("\tLinkQualityMult          : %s %0.2f\n",
903                    ipv6_buf, mult->val);
904           }
905           
906           in = in->next;
907         }
908     }
909
910
911
912
913   /* Plugins */
914   if(pe)
915     {
916       printf("Plugins:\n");
917
918       while(pe)
919         {
920           printf("\tName: \"%s\"\n", pe->name);
921           pe = pe->next;
922         }
923     }
924
925   /* Hysteresis */
926   if(cnf->use_hysteresis)
927     {
928       printf("Using hysteresis:\n");
929       printf("\tScaling      : %0.2f\n", cnf->hysteresis_param.scaling);
930       printf("\tThr high/low : %0.2f/%0.2f\n", cnf->hysteresis_param.thr_high, cnf->hysteresis_param.thr_low);
931     }
932   else
933     printf("Not using hysteresis\n");
934
935   /* HNA IPv4 */
936   if(h4)
937     {
938
939       printf("HNA4 entries:\n");
940       while(h4)
941         {
942           in4.s_addr = h4->net.v4;
943           printf("\t%s/", inet_ntoa(in4));
944           in4.s_addr = h4->netmask.v4;
945           printf("%s\n", inet_ntoa(in4));
946
947           h4 = h4->next;
948         }
949     }
950
951   /* HNA IPv6 */
952   if(h6)
953     {
954       printf("HNA6 entries:\n");
955       while(h6)
956         {
957           printf("\t%s/%d\n", (char *)inet_ntop(AF_INET6, &h6->net.v6, ipv6_buf, sizeof(ipv6_buf)), h6->prefix_len);
958           h6 = h6->next;
959         }
960     }
961 }
962
963 void *olsrd_cnf_malloc(unsigned int len)
964 {
965   return malloc(len);
966 }
967
968 void olsrd_cnf_free(void *addr)
969 {
970   free(addr);
971 }
972
973 #if defined WIN32_STDIO_HACK
974 struct ioinfo
975 {
976         unsigned int handle;
977         unsigned char attr;
978         char buff;
979         int flag;
980         CRITICAL_SECTION lock;
981 };
982
983 void win32_stdio_hack(unsigned int handle)
984 {
985   HMODULE lib;
986   struct ioinfo **info;
987
988   lib = LoadLibrary("msvcrt.dll");
989
990   info = (struct ioinfo **)GetProcAddress(lib, "__pioinfo");
991
992   // (*info)[1].handle = handle;
993   // (*info)[1].attr = 0x89; // FOPEN | FTEXT | FPIPE;
994
995   (*info)[2].handle = handle;
996   (*info)[2].attr = 0x89;
997
998   // stdout->_file = 1;
999   stderr->_file = 2;
1000
1001   // setbuf(stdout, NULL);
1002   setbuf(stderr, NULL);
1003 }
1004 #else
1005 void win32_stdio_hack(unsigned int handle) {}
1006 #endif