sgw: fix a shadowing warning
[olsrd.git] / src / egressFile.c
1 #ifdef __linux__
2
3 #include "egressFile.h"
4
5 /* Plugin includes */
6
7 /* OLSRD includes */
8 #include "olsr_cfg.h"
9 #include "gateway_costs.h"
10 #include "gateway.h"
11 #include "scheduler.h"
12 #include "ipcalc.h"
13 #include "log.h"
14
15 /* System includes */
16 #include <unistd.h>
17 #include <fcntl.h>
18 #include <regex.h>
19 #include <sys/stat.h>
20 #include <assert.h>
21 #include <net/if.h>
22
23 /** the weights for the cost calculation */
24 static struct costs_weights gw_costs_weights_storage;
25 static struct costs_weights * gw_costs_weights = NULL;
26
27 /** regular expression describing a comment */
28 static const char * regexComment = "^([[:space:]]*|[[:space:]#]+.*)$";
29
30 /**
31  * regular expression describing an egress line.
32  *
33  * # interface=uplink (Kbps),downlink (Kbps),path cost,gateway
34  */
35 static const char * regexEgress = "^[[:space:]]*" //
36         "([^[:space:]=]+)"                     /* 01: interface, mandatory, can NOT be empty */
37         "[[:space:]]*=[[:space:]]*"            /* --: field separator */
38         "([[:digit:]]*)"                       /* 02: uplink, mandatory, can be empty */
39         "[[:space:]]*,[[:space:]]*"            /* --: field separator */
40         "([[:digit:]]*)"                       /* 03: downlink, mandatory, can be empty */
41         "("                                    /* 04: (the rest is optional) */
42         "[[:space:]]*,[[:space:]]*"            /* --: field separator */
43         "([[:digit:]]*)"                       /* 05: path cost, optional, can be empty */
44         "("                                    /* 06: (the rest is optional) */
45         "[[:space:]]*,[[:space:]]*"            /* --: field separator */
46         "(|([[:digit:]\\.:]+)/([[:digit:]]+))" /* 07: network, optional, can be empty, 07=ip/x 08=ip 09=x */
47         "("                                    /* 10: (the rest is optional) */
48         "[[:space:]]*,[[:space:]]*"            /* --: field separator */
49         "([[:digit:]\\.:]*)"                   /* 11: gateway, optional, can be empty */
50         ")?"                                   /* 10 */
51         ")?"                                   /* 06 */
52         ")?"                                   /* 04 */
53         "[[:space:]]*$";
54
55 /** the number of matches in regexEgress */
56 #define REGEX_EGRESS_LINE_MATCH_COUNT (1 /* 00 */ + 11)
57
58 /** the compiled regular expression describing a comment */
59 static regex_t compiledRegexComment;
60
61 /** the compiled regular expression describing an egress line */
62 static regex_t compiledRegexEgress;
63
64 /** true when the file reader has been started */
65 static bool started = false;
66
67 /** type to hold the cached stat result */
68 typedef struct _CachedStat {
69 #if defined(__linux__) && !defined(__ANDROID__)
70   struct timespec timeStamp; /* Time of last modification (full resolution) */
71 #else
72   time_t timeStamp; /* Time of last modification (second resolution) */
73 #endif
74 } CachedStat;
75
76 /** the cached stat result */
77 static CachedStat cachedStat;
78 static CachedStat cachedStatClear;
79
80 /** the malloc-ed buffer in which to store a line read from the file */
81 static char * line = NULL;
82
83 /** the maximum length of a line that is read from the file */
84 static size_t line_length = 256;
85
86 /* forward declaration */
87 static bool readEgressFile(const char * fileName);
88
89 /*
90  * Error Reporting
91  */
92
93 /** the maximum length of an error report */
94 #define ERROR_LENGTH 1024
95
96 /** true when errors have been reported, used to reduce error reports */
97 static bool reportedErrors = false;
98
99 /**
100  * Report an error.
101  *
102  * @param useErrno
103  * when true then errno is used in the error message; the error reason is also
104  * reported.
105  * @param lineNo
106  * the line number of the caller
107  * @param format
108  * a pointer to the format string
109  * @param ...
110  * arguments to the format string
111  */
112 __attribute__ ((format(printf, 3, 4)))
113 static void egressFileError(bool useErrno, int lineNo, const char *format, ...) {
114   char str[ERROR_LENGTH];
115   char *strErr = NULL;
116
117   if (reportedErrors) {
118     return;
119   }
120
121   if (useErrno) {
122     strErr = strerror(errno);
123   }
124
125   if ((format == NULL ) || (*format == '\0')) {
126     olsr_syslog(OLSR_LOG_ERR, "%s@%d: %s\n", __FILE__, lineNo, useErrno ? strErr : "Unknown error");
127   } else {
128     va_list arglist;
129
130     va_start(arglist, format);
131     vsnprintf(str, sizeof(str), format, arglist);
132     va_end(arglist);
133
134     str[sizeof(str) - 1] = '\0'; /* Ensures null termination */
135
136     if (useErrno) {
137       olsr_syslog(OLSR_LOG_ERR, "%s@%d: %s: %s\n", __FILE__, lineNo, str, strErr);
138     } else {
139       olsr_syslog(OLSR_LOG_ERR, "%s@%d: %s\n", __FILE__, lineNo, str);
140     }
141   }
142 }
143
144 /*
145  * Helpers
146  */
147
148 #ifdef __ANDROID__
149 static ssize_t getline(char **lineptr, size_t *n, FILE *stream)
150 {
151     char *ptr;
152     size_t len;
153
154     ptr = fgetln(stream, n);
155
156     if (ptr == NULL) {
157         return -1;
158     }
159
160     /* Free the original ptr */
161     if (*lineptr != NULL) free(*lineptr);
162
163     /* Add one more space for '\0' */
164     len = n[0] + 1;
165
166     /* Update the length */
167     n[0] = len;
168
169     /* Allocate a new buffer */
170     *lineptr = malloc(len);
171
172     /* Copy over the string */
173     memcpy(*lineptr, ptr, len-1);
174
175     /* Write the NULL character */
176     (*lineptr)[len-1] = '\0';
177
178     /* Return the length of the new buffer */
179     return len;
180 }
181 #endif
182
183 /**
184  * Read an (olsr_ip_addr) IP address from a string:
185  * First tries to parse the value as an IPv4 address, and if not successful
186  * tries to parse it as an IPv6 address.
187  *
188  * @param str
189  * The string to convert to an (olsr_ip_addr) IP address
190  * @param dst
191  * A pointer to the location where to store the (olsr_ip_addr) IP address upon
192  * successful conversion. Not touched when errors are reported.
193  * @param dstSet
194  * A pointer to the location where to store the flag that signals whether the
195  * IP address is set. Not touched when errors are reported.
196  * @param dstIpVersion
197  * A pointer to the location where to store the IP version of the IP address.
198  * Not touched when errors are reported.
199  *
200  * @return
201  * - true on success
202  * - false otherwise
203  */
204 static bool readIPAddress(const char * str, union olsr_ip_addr * dst, bool * dstSet, int * dstIpVersion) {
205   int conversion;
206   union olsr_ip_addr ip;
207   int ip_version;
208
209   assert(str);
210   assert(dst);
211   assert(dstSet);
212   assert(dstIpVersion);
213
214   /* try IPv4 first */
215   ip_version = AF_INET;
216   memset(&ip, 0, sizeof(ip));
217   conversion = inet_pton(ip_version, str, &ip.v4);
218
219   if (conversion != 1) {
220     /* now try IPv6: IPv4 conversion was not successful */
221     ip_version = AF_INET6;
222     memset(&ip, 0, sizeof(ip));
223     conversion = inet_pton(ip_version, str, &ip.v6);
224   }
225
226   if (conversion != 1) {
227     return false;
228   }
229
230   *dst = ip;
231   *dstSet = true;
232   *dstIpVersion = ip_version;
233   return true;
234 }
235
236 /**
237  * Read an unsigned long long number from a value string.
238  * An empty string results in a value of zero.
239  *
240  * @param value
241  * The string to convert to a number
242  * @param valueNumber
243  * A pointer to the location where to store the number upon successful conversion.
244  * Not touched when errors are reported.
245  *
246  * @return
247  * - true on success
248  * - false otherwise
249  */
250 static bool readULL(const char * value, unsigned long long * valueNumber) {
251   char * endPtr = NULL;
252   unsigned long valueNew;
253
254   assert(value);
255   assert(valueNumber);
256
257   if (!value || !strlen(value)) {
258     *valueNumber = 0;
259     return true;
260   }
261
262   errno = 0;
263   valueNew = strtoull(value, &endPtr, 10);
264
265   if (!((endPtr != value) && (*value != '\0') && (*endPtr == '\0')) || (errno == ERANGE)) {
266     /* invalid conversion */
267     return false;
268   }
269
270   *valueNumber = valueNew;
271   return true;
272 }
273
274 /**
275  * Strip EOL characters from the end of a string
276  *
277  * @param str the string to strip
278  * @param length the length of the string
279  */
280 static void stripEols(char * str, ssize_t length) {
281   ssize_t len = length;
282   while ((len > 0) && ((str[len - 1] == '\n') || (str[len - 1] == '\r'))) {
283     len--;
284   }
285   str[len] = '\0';
286 }
287
288 /**
289  * Find an egress interface in the configuration
290  *
291  * @param name the name of the egress interface
292  * @return the pointer to the egress interface, NULL when not found
293  */
294 struct sgw_egress_if * findEgressInterface(char * name) {
295   if (name && (name[0] != '\0')) {
296     struct sgw_egress_if * egress_if = olsr_cnf->smart_gw_egress_interfaces;
297     while (egress_if) {
298       if (!strcmp(egress_if->name, name)) {
299         return egress_if;
300       }
301       egress_if = egress_if->next;
302     }
303   }
304
305   return NULL ;
306 }
307
308 /**
309  * Find an egress interface in the configuration by if_index
310  *
311  * @param if_index the index of the egress interface
312  * @return the pointer to the egress interface, NULL when not found
313  */
314 struct sgw_egress_if * findEgressInterfaceByIndex(int if_index) {
315   if (if_index > 0) {
316     struct sgw_egress_if * egress_if = olsr_cnf->smart_gw_egress_interfaces;
317     while (egress_if) {
318       if (egress_if->if_index == if_index) {
319         return egress_if;
320       }
321       egress_if = egress_if->next;
322     }
323   }
324
325   return NULL ;
326 }
327
328 /**
329  * Calculate the costs from the bandwidth parameters
330  *
331  * @param bw the bandwidth parameters
332  * @param up true when the interface is up
333  * @return true when the costs changed
334  */
335 bool egressBwCalculateCosts(struct egress_if_bw * bw, bool up) {
336   if (!gw_costs_weights) {
337     gw_costs_weights_storage.WexitU = olsr_cnf->smart_gw_weight_exitlink_up;
338     gw_costs_weights_storage.WexitD = olsr_cnf->smart_gw_weight_exitlink_down;
339     gw_costs_weights_storage.Wetx = olsr_cnf->smart_gw_weight_etx;
340     gw_costs_weights_storage.Detx = olsr_cnf->smart_gw_divider_etx;
341     gw_costs_weights = &gw_costs_weights_storage;
342   }
343
344   {
345     int64_t costsPrevious = bw->costs;
346     bw->costs = gw_costs_weigh(up, gw_costs_weights_storage, bw->path_cost, bw->egressUk, bw->egressDk);
347     return (costsPrevious != bw->costs);
348   }
349 }
350
351 /**
352  * Clear the bandwidth parameters
353  * @param bw the bandwidth parameters
354  * @param up true when the interface is up
355  */
356 void egressBwClear(struct egress_if_bw * bw, bool up) {
357   bw->egressUk = 0;
358   bw->egressDk = 0;
359   bw->path_cost = 0;
360   memset(&bw->network, 0, sizeof(bw->network));
361   memset(&bw->gateway, 0, sizeof(bw->gateway));
362
363   bw->networkSet = false;
364   bw->gatewaySet = false;
365
366   egressBwCalculateCosts(bw, up);
367 }
368
369 /*
370  * Timer
371  */
372
373 /** the timer for polling the egress file for changes */
374 static struct timer_entry *egress_file_timer;
375
376 /**
377  * Timer callback to read the egress file
378  *
379  * @param unused unused
380  */
381 static void egress_file_timer_callback(void *unused __attribute__ ((unused))) {
382   if (readEgressFile(olsr_cnf->smart_gw_egress_file)) {
383     doRoutesMultiGw(true, false, GW_MULTI_CHANGE_PHASE_RUNTIME);
384   }
385 }
386
387 /*
388  * Life Cycle
389  */
390
391 /**
392  * Initialises the egress file reader
393  *
394  * @return
395  * - true upon success
396  * - false otherwise
397  */
398 bool startEgressFile(void) {
399   int r;
400
401   if (started) {
402     return true;
403   }
404
405   line = malloc(line_length);
406   if (!line) {
407     egressFileError(false, __LINE__, "Could not allocate a line buffer");
408     return false;
409   }
410   *line = '\0';
411
412   r = regcomp(&compiledRegexComment, regexComment, REG_EXTENDED);
413   if (r) {
414     regerror(r, &compiledRegexComment, line, line_length);
415     egressFileError(false, __LINE__, "Could not compile regex \"%s\" (%d = %s)", regexComment, r, line);
416
417     free(line);
418     line = NULL;
419     return false;
420   }
421
422   r = regcomp(&compiledRegexEgress, regexEgress, REG_EXTENDED);
423   if (r) {
424     regerror(r, &compiledRegexEgress, line, line_length);
425     egressFileError(false, __LINE__, "Could not compile regex \"%s\" (%d = %s)", regexEgress, r, line);
426
427     regfree(&compiledRegexComment);
428     free(line);
429     line = NULL;
430     return false;
431   }
432
433   memset(&cachedStat.timeStamp, 0, sizeof(cachedStat.timeStamp));
434   memset(&cachedStatClear.timeStamp, 0, sizeof(cachedStatClear.timeStamp));
435
436   readEgressFile(olsr_cnf->smart_gw_egress_file);
437
438   olsr_set_timer(&egress_file_timer, olsr_cnf->smart_gw_egress_file_period, 0, true, &egress_file_timer_callback, NULL, NULL);
439
440   started = true;
441   return true;
442 }
443
444 /**
445  * Cleans up the egress file reader.
446  */
447 void stopEgressFile(void) {
448   if (started) {
449     olsr_stop_timer(egress_file_timer);
450     egress_file_timer = NULL;
451
452     regfree(&compiledRegexEgress);
453     regfree(&compiledRegexComment);
454     free(line);
455
456     started = false;
457   }
458 }
459
460 /*
461  * File Reader
462  */
463
464 /** the buffer with regex matches */
465 static regmatch_t pmatch[REGEX_EGRESS_LINE_MATCH_COUNT];
466
467 static void readEgressFileClear(void) {
468   struct sgw_egress_if * egress_if = olsr_cnf->smart_gw_egress_interfaces;
469   while (egress_if) {
470     egress_if->bwPrevious = egress_if->bwCurrent;
471     egress_if->bwCostsChanged = false;
472     egress_if->bwNetworkChanged = false;
473     egress_if->bwGatewayChanged = false;
474     egress_if->bwChanged = false;
475
476     egress_if->inEgressFile = false;
477
478     egress_if = egress_if->next;
479   }
480 }
481
482 /**
483  * Read the egress file
484  *
485  * @param fileName the filename
486  * @return true to indicate changes (any egress_if->bwChanged is true)
487  */
488 static bool readEgressFile(const char * fileName) {
489   bool changed = false;
490
491   FILE * fp = NULL;
492   struct stat statBuf;
493   unsigned int lineNumber = 0;
494   ssize_t length = -1;
495   bool reportedErrorsLocal = false;
496   const char * filepath = !fileName ? DEF_GW_EGRESS_FILE : fileName;
497   void * mtim;
498
499   if (memcmp(&cachedStat.timeStamp, &cachedStatClear.timeStamp, sizeof(cachedStat.timeStamp))) {
500     /* read the file before */
501
502     if (stat(filepath, &statBuf)) {
503       /* could not stat the file */
504       memset(&cachedStat.timeStamp, 0, sizeof(cachedStat.timeStamp));
505       readEgressFileClear();
506       goto outerror;
507     }
508
509 #if defined(__linux__) && !defined(__ANDROID__)
510     mtim = &statBuf.st_mtim;
511 #else
512     mtim = &statBuf.st_mtime;
513 #endif
514     if (!memcmp(&cachedStat.timeStamp, mtim, sizeof(cachedStat.timeStamp))) {
515       /* file did not change since last read */
516       return false;
517     }
518   }
519
520   fp = fopen(filepath, "r");
521   if (!fp) {
522     /* could not open the file */
523     memset(&cachedStat.timeStamp, 0, sizeof(cachedStat.timeStamp));
524     readEgressFileClear();
525     goto outerror;
526   }
527
528   /* copy 'current' egress interfaces into 'previous' field */
529   readEgressFileClear();
530
531   while ((length = getline(&line, &line_length, fp)) != -1) {
532     struct sgw_egress_if * egress_if = NULL;
533     unsigned long long uplink = DEF_EGRESS_UPLINK_KBPS;
534     unsigned long long downlink = DEF_EGRESS_DOWNLINK_KBPS;
535     unsigned long long pathCosts = DEF_EGRESS_PATH_COSTS;
536     struct olsr_ip_prefix network;
537     union olsr_ip_addr gateway;
538     bool networkSet = false;
539     bool gatewaySet = false;
540     int networkIpVersion = AF_INET;
541     int gatewayIpVersion = AF_INET;
542
543     lineNumber++;
544
545     if (!regexec(&compiledRegexComment, line, 0, NULL, 0)) {
546       /* the line is a comment */
547       continue;
548     }
549
550     memset(&network, 0, sizeof(network));
551     memset(&gateway, 0, sizeof(gateway));
552
553     stripEols(line, length);
554
555     memset(pmatch, 0, sizeof(pmatch));
556     if (regexec(&compiledRegexEgress, line, REGEX_EGRESS_LINE_MATCH_COUNT, pmatch, 0)) {
557       egressFileError(false, __LINE__, "Egress speed file line %d uses invalid syntax: line is ignored (%s)", lineNumber, line);
558       reportedErrorsLocal = true;
559       continue;
560     }
561
562     /* iface: mandatory presence, guaranteed through regex match */
563     {
564       regoff_t len = pmatch[1].rm_eo - pmatch[1].rm_so;
565       char * ifaceString = &line[pmatch[1].rm_so];
566       line[pmatch[1].rm_eo] = '\0';
567
568       if (len > IFNAMSIZ) {
569         /* interface name is too long */
570         egressFileError(false, __LINE__, "Egress speed file line %d: interface \"%s\" is too long: line is ignored", lineNumber, ifaceString);
571         reportedErrorsLocal = true;
572         continue;
573       }
574
575       egress_if = findEgressInterface(ifaceString);
576       if (!egress_if) {
577         /* not a known egress interface */
578         egressFileError(false, __LINE__, "Egress speed file line %d: interface \"%s\" is not a configured egress interface: line is ignored", lineNumber,
579             ifaceString);
580         reportedErrorsLocal = true;
581         continue;
582       }
583     }
584     assert(egress_if);
585
586     /* uplink: mandatory presence, guaranteed through regex match */
587     {
588       regoff_t len = pmatch[2].rm_eo - pmatch[2].rm_so;
589       char * uplinkString = &line[pmatch[2].rm_so];
590       line[pmatch[2].rm_eo] = '\0';
591
592       if ((len > 0) && !readULL(uplinkString, &uplink)) {
593         egressFileError(false, __LINE__, "Egress speed file line %d: uplink bandwidth \"%s\" is not a valid number: line is ignored", lineNumber, uplinkString);
594         reportedErrorsLocal = true;
595         continue;
596       }
597     }
598     uplink = MIN(uplink, MAX_SMARTGW_SPEED);
599
600     /* downlink: mandatory presence, guaranteed through regex match */
601     {
602       regoff_t len = pmatch[3].rm_eo - pmatch[3].rm_so;
603       char * downlinkString = &line[pmatch[3].rm_so];
604       line[pmatch[3].rm_eo] = '\0';
605
606       if ((len > 0) && !readULL(downlinkString, &downlink)) {
607         egressFileError(false, __LINE__, "Egress speed file line %d: downlink bandwidth \"%s\" is not a valid number: line is ignored", lineNumber,
608             downlinkString);
609         reportedErrorsLocal = true;
610         continue;
611       }
612     }
613     downlink = MIN(downlink, MAX_SMARTGW_SPEED);
614
615     /* path costs: optional presence */
616     if (pmatch[5].rm_so != -1) {
617       regoff_t len = pmatch[5].rm_eo - pmatch[5].rm_so;
618       char * pathCostsString = &line[pmatch[5].rm_so];
619       line[pmatch[5].rm_eo] = '\0';
620
621       if ((len > 0) && !readULL(pathCostsString, &pathCosts)) {
622         egressFileError(false, __LINE__, "Egress speed file line %d: path costs \"%s\" is not a valid number: line is ignored", lineNumber, pathCostsString);
623         reportedErrorsLocal = true;
624         continue;
625       }
626     }
627     pathCosts = MIN(pathCosts, UINT32_MAX);
628
629     /* network: optional presence */
630     if ((pmatch[7].rm_so != -1) && ((pmatch[7].rm_eo - pmatch[7].rm_so) > 0)) {
631       /* network is present: guarantees IP and prefix presence */
632       unsigned long long prefix_len;
633       char * networkString = &line[pmatch[8].rm_so];
634       char * prefixlenString = &line[pmatch[9].rm_so];
635       line[pmatch[8].rm_eo] = '\0';
636       line[pmatch[9].rm_eo] = '\0';
637
638       if (!readIPAddress(networkString, &network.prefix, &networkSet, &networkIpVersion)) {
639         egressFileError(false, __LINE__, "Egress speed file line %d: network IP address \"%s\" is not a valid IP address: line is ignored", lineNumber,
640             networkString);
641         reportedErrorsLocal = true;
642         continue;
643       }
644
645       if (!readULL(prefixlenString, &prefix_len)) {
646         egressFileError(false, __LINE__, "Egress speed file line %d: network prefix \"%s\" is not a valid number: line is ignored", lineNumber,
647             prefixlenString);
648         reportedErrorsLocal = true;
649         continue;
650       }
651
652       if (prefix_len > ((networkIpVersion == AF_INET) ? 32 : 128)) {
653         egressFileError(false, __LINE__, "Egress speed file line %d: network prefix \"%s\" is not in the range [0, %d]: line is ignored", lineNumber,
654             prefixlenString, ((networkIpVersion == AF_INET) ? 32 : 128));
655         reportedErrorsLocal = true;
656         continue;
657       }
658
659       network.prefix_len = prefix_len;
660     }
661
662     /* gateway: optional presence */
663     if (pmatch[11].rm_so != -1) {
664       regoff_t len = pmatch[11].rm_eo - pmatch[11].rm_so;
665       char * gatewayString = &line[pmatch[11].rm_so];
666       line[pmatch[11].rm_eo] = '\0';
667
668       if ((len > 0) && !readIPAddress(gatewayString, &gateway, &gatewaySet, &gatewayIpVersion)) {
669         egressFileError(false, __LINE__, "Egress speed file line %d: gateway IP address \"%s\" is not a valid IP address: line is ignored", lineNumber,
670             gatewayString);
671         reportedErrorsLocal = true;
672         continue;
673       }
674     }
675
676     /* check all IP versions are the same */
677     if ((networkSet && gatewaySet) && (networkIpVersion != gatewayIpVersion)) {
678       egressFileError(false, __LINE__, "Egress speed file line %d: network and gateway IP addresses must be of the same IP version: line is ignored",
679           lineNumber);
680       reportedErrorsLocal = true;
681       continue;
682     }
683
684     /* check no IPv6 */
685     if ((networkSet && networkIpVersion == AF_INET6) || //
686         (gatewaySet && gatewayIpVersion == AF_INET6)) {
687       egressFileError(false, __LINE__, "Egress speed file line %d: network and gateway IP addresses must not be IPv6 addresses: line is ignored", lineNumber);
688       reportedErrorsLocal = true;
689       continue;
690     }
691
692     /* ensure network is masked by netmask */
693     if (networkSet) {
694       /* assumes IPv4 */
695       in_addr_t mask = (network.prefix_len == 0) ? 0 : (~0U << (32 - network.prefix_len));
696       uint32_t masked = ntohl(network.prefix.v4.s_addr) & mask;
697       network.prefix.v4.s_addr = htonl(masked);
698     }
699
700     if (!uplink || !downlink) {
701       egressBwClear(&egress_if->bwCurrent, egress_if->upCurrent);
702     } else {
703       egress_if->bwCurrent.egressUk = uplink;
704       egress_if->bwCurrent.egressDk = downlink;
705       egress_if->bwCurrent.path_cost = pathCosts;
706       egress_if->bwCurrent.network = network;
707       egress_if->bwCurrent.gateway = gateway;
708
709       egress_if->bwCurrent.networkSet = networkSet;
710       egress_if->bwCurrent.gatewaySet = gatewaySet;
711
712       egressBwCalculateCosts(&egress_if->bwCurrent, egress_if->upCurrent);
713     }
714
715     egress_if->inEgressFile = true;
716   }
717
718   fclose(fp);
719   fp = NULL;
720
721 #if defined(__linux__) && !defined(__ANDROID__)
722     mtim = &statBuf.st_mtim;
723 #else
724     mtim = &statBuf.st_mtime;
725 #endif
726   memcpy(&cachedStat.timeStamp, mtim, sizeof(cachedStat.timeStamp));
727
728   reportedErrors = reportedErrorsLocal;
729
730   outerror:
731
732   /* clear absent egress interfaces and setup 'changed' status */
733   {
734     struct sgw_egress_if * egress_if = olsr_cnf->smart_gw_egress_interfaces;
735     while (egress_if) {
736       if (!egress_if->inEgressFile) {
737         egressBwClear(&egress_if->bwCurrent, egress_if->upCurrent);
738       }
739
740       egress_if->bwCostsChanged = egressBwCostsChanged(egress_if);
741       egress_if->bwNetworkChanged = egressBwNetworkChanged(egress_if);
742       egress_if->bwGatewayChanged = egressBwGatewayChanged(egress_if);
743       egress_if->bwChanged = egressBwChanged(egress_if);
744       if (egress_if->bwChanged) {
745         changed = true;
746       }
747
748       egress_if = egress_if->next;
749     }
750   }
751
752   return changed;
753 }
754
755 #endif /* __linux__ */