CMake: bump the minimum required CMake version to 3.5
[project/netifd.git] / config.c
1 /*
2 * netifd - network interface daemon
3 * Copyright (C) 2012 Felix Fietkau <nbd@openwrt.org>
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License version 2
7 * as published by the Free Software Foundation
8 *
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
13 */
14 #define _GNU_SOURCE
15 #include <string.h>
16 #include <stdlib.h>
17 #include <stdio.h>
18
19 #include <uci.h>
20
21 #include <libubox/blobmsg_json.h>
22
23 #include "netifd.h"
24 #include "interface.h"
25 #include "interface-ip.h"
26 #include "iprule.h"
27 #include "proto.h"
28 #include "wireless.h"
29 #include "config.h"
30
31 bool config_init = false;
32
33 static struct uci_context *uci_ctx;
34 static struct uci_package *uci_network;
35 static struct uci_package *uci_wireless;
36 static struct blob_attr *board_netdevs;
37 static struct blob_buf b;
38
39 static int
40 config_section_idx(struct uci_section *s)
41 {
42 struct uci_element *e;
43 int idx = 0;
44
45 uci_foreach_element(&uci_wireless->sections, e) {
46 struct uci_section *cur = uci_to_section(e);
47
48 if (s == cur)
49 return idx;
50
51 if (!strcmp(cur->type, s->type))
52 idx++;
53 }
54
55 return -1;
56 }
57
58 static bool
59 config_bridge_has_vlans(const char *br_name)
60 {
61 struct uci_element *e;
62
63 uci_foreach_element(&uci_network->sections, e) {
64 struct uci_section *s = uci_to_section(e);
65 const char *name;
66
67 if (strcmp(s->type, "bridge-vlan") != 0)
68 continue;
69
70 name = uci_lookup_option_string(uci_ctx, s, "device");
71 if (!name)
72 continue;
73
74 if (!strcmp(name, br_name))
75 return true;
76 }
77
78 return false;
79 }
80
81 static void
82 config_fixup_bridge_var(struct uci_section *s, const char *name, const char *val)
83 {
84 struct uci_ptr ptr = {
85 .p = s->package,
86 .s = s,
87 .option = name,
88 .value = val,
89 };
90
91 uci_lookup_ptr(uci_ctx, &ptr, NULL, false);
92 if (ptr.o)
93 return;
94
95 uci_set(uci_ctx, &ptr);
96 }
97
98 /**
99 * config_fixup_bridge_ports - translate deprecated configs
100 *
101 * Old configs used "ifname" option for specifying bridge ports. For backward
102 * compatibility translate it into the new "ports" option.
103 */
104 static void config_fixup_bridge_ports(struct uci_section *s)
105 {
106 struct uci_ptr ptr = {
107 .p = s->package,
108 .s = s,
109 .option = "ifname",
110 };
111
112 if (uci_lookup_option(uci_ctx, s, "ports"))
113 return;
114
115 uci_lookup_ptr(uci_ctx, &ptr, NULL, false);
116 if (!ptr.o)
117 return;
118
119 ptr.value = "ports";
120 uci_rename(uci_ctx, &ptr);
121 }
122
123 static void
124 config_fixup_bridge_vlan_filtering(struct uci_section *s, const char *name)
125 {
126 bool has_vlans = config_bridge_has_vlans(name);
127
128 config_fixup_bridge_var(s, "__has_vlans", has_vlans ? "1" : "0");
129
130 if (!has_vlans)
131 return;
132
133 config_fixup_bridge_var(s, "vlan_filtering", "1");
134 }
135
136 static int
137 config_parse_bridge_interface(struct uci_section *s, struct device_type *devtype)
138 {
139 char *name;
140
141 name = alloca(strlen(s->e.name) + strlen(devtype->name_prefix) + 2);
142 sprintf(name, "%s-%s", devtype->name_prefix, s->e.name);
143 blobmsg_add_string(&b, "name", name);
144
145 config_fixup_bridge_ports(s);
146 config_fixup_bridge_vlan_filtering(s, name);
147 uci_to_blob(&b, s, devtype->config_params);
148 if (!device_create(name, devtype, b.head)) {
149 D(INTERFACE, "Failed to create '%s' device for interface '%s'",
150 devtype->name, s->e.name);
151 }
152
153 blob_buf_init(&b, 0);
154 blobmsg_add_string(&b, "ifname", name);
155 return 0;
156 }
157
158 static void
159 config_parse_interface(struct uci_section *s, bool alias)
160 {
161 struct interface *iface;
162 const char *type = NULL, *disabled;
163 struct blob_attr *config;
164 bool bridge = false;
165 struct device_type *devtype = NULL;
166
167 disabled = uci_lookup_option_string(uci_ctx, s, "disabled");
168 if (disabled && !strcmp(disabled, "1"))
169 return;
170
171 blob_buf_init(&b, 0);
172
173 if (!alias)
174 type = uci_lookup_option_string(uci_ctx, s, "type");
175
176 if (type)
177 devtype = device_type_get(type);
178
179 if (devtype && devtype->bridge_capability) {
180 if (config_parse_bridge_interface(s, devtype))
181 return;
182
183 bridge = true;
184 }
185
186 uci_to_blob(&b, s, &interface_attr_list);
187
188 iface = interface_alloc(s->e.name, b.head, false);
189 if (!iface)
190 return;
191
192 if (iface->proto_handler && iface->proto_handler->config_params)
193 uci_to_blob(&b, s, iface->proto_handler->config_params);
194
195 if (!bridge && uci_to_blob(&b, s, simple_device_type.config_params))
196 iface->device_config = true;
197
198 config = blob_memdup(b.head);
199 if (!config)
200 goto error;
201
202 if (alias) {
203 if (!interface_add_alias(iface, config))
204 goto error_free_config;
205 } else {
206 if (!interface_add(iface, config))
207 goto error_free_config;
208 }
209 return;
210
211 error_free_config:
212 free(config);
213 error:
214 free(iface);
215 }
216
217 static void
218 config_parse_route(struct uci_section *s, bool v6)
219 {
220 void *route;
221
222 blob_buf_init(&b, 0);
223 route = blobmsg_open_array(&b, "route");
224 uci_to_blob(&b, s, &route_attr_list);
225 blobmsg_close_array(&b, route);
226 interface_ip_add_route(NULL, blob_data(b.head), v6);
227 }
228
229 static void
230 config_parse_neighbor(struct uci_section *s, bool v6)
231 {
232 void *neighbor;
233 blob_buf_init(&b,0);
234 neighbor = blobmsg_open_array(&b, "neighbor");
235 uci_to_blob(&b,s, &neighbor_attr_list);
236 blobmsg_close_array(&b, neighbor);
237 interface_ip_add_neighbor(NULL, blob_data(b.head), v6);
238 }
239
240 static void
241 config_parse_rule(struct uci_section *s, bool v6)
242 {
243 void *rule;
244
245 blob_buf_init(&b, 0);
246 rule = blobmsg_open_array(&b, "rule");
247 uci_to_blob(&b, s, &rule_attr_list);
248 blobmsg_close_array(&b, rule);
249 iprule_add(blob_data(b.head), v6);
250 }
251
252 static void
253 config_init_devices(bool bridge)
254 {
255 struct uci_element *e;
256
257 uci_foreach_element(&uci_network->sections, e) {
258 const struct uci_blob_param_list *params = NULL;
259 struct uci_section *s = uci_to_section(e);
260 struct device_type *devtype = NULL;
261 struct device *dev;
262 const char *type, *name;
263
264 if (strcmp(s->type, "device") != 0)
265 continue;
266
267 name = uci_lookup_option_string(uci_ctx, s, "name");
268 if (!name)
269 continue;
270
271 type = uci_lookup_option_string(uci_ctx, s, "type");
272 if (type)
273 devtype = device_type_get(type);
274
275 if (bridge != (devtype && devtype->bridge_capability))
276 continue;
277
278 if (devtype)
279 params = devtype->config_params;
280 if (!params)
281 params = simple_device_type.config_params;
282
283 if (devtype && devtype->bridge_capability) {
284 config_fixup_bridge_ports(s);
285 config_fixup_bridge_vlan_filtering(s, name);
286 }
287
288 blob_buf_init(&b, 0);
289 uci_to_blob(&b, s, params);
290 if (devtype) {
291 dev = device_create(name, devtype, b.head);
292 if (!dev)
293 continue;
294 } else {
295 dev = device_get(name, 1);
296 if (!dev)
297 continue;
298
299 dev->current_config = true;
300 device_apply_config(dev, dev->type, b.head);
301 }
302 dev->default_config = false;
303 }
304 }
305
306 static void
307 config_parse_vlan(struct device *dev, struct uci_section *s)
308 {
309 enum {
310 BRVLAN_ATTR_VID,
311 BRVLAN_ATTR_LOCAL,
312 BRVLAN_ATTR_PORTS,
313 BRVLAN_ATTR_ALIAS,
314 __BRVLAN_ATTR_MAX,
315 };
316 static const struct blobmsg_policy vlan_attrs[__BRVLAN_ATTR_MAX] = {
317 [BRVLAN_ATTR_VID] = { "vlan", BLOBMSG_TYPE_INT32 },
318 [BRVLAN_ATTR_LOCAL] = { "local", BLOBMSG_TYPE_BOOL },
319 [BRVLAN_ATTR_PORTS] = { "ports", BLOBMSG_TYPE_ARRAY },
320 [BRVLAN_ATTR_ALIAS] = { "alias", BLOBMSG_TYPE_ARRAY },
321 };
322 static const struct uci_blob_param_info vlan_attr_info[__BRVLAN_ATTR_MAX] = {
323 [BRVLAN_ATTR_PORTS] = { .type = BLOBMSG_TYPE_STRING },
324 [BRVLAN_ATTR_ALIAS] = { .type = BLOBMSG_TYPE_STRING },
325 };
326 static const struct uci_blob_param_list vlan_attr_list = {
327 .n_params = __BRVLAN_ATTR_MAX,
328 .params = vlan_attrs,
329 .info = vlan_attr_info,
330 };
331 struct blob_attr *tb[__BRVLAN_ATTR_MAX];
332 struct blob_attr *cur;
333 struct bridge_vlan_port *port;
334 struct bridge_vlan *vlan;
335 unsigned int vid;
336 const char *val;
337 char *name_buf;
338 int name_len = 0;
339 int n_ports = 0;
340 size_t rem;
341
342 val = uci_lookup_option_string(uci_ctx, s, "vlan");
343 if (!val)
344 return;
345
346 blob_buf_init(&b, 0);
347 uci_to_blob(&b, s, &vlan_attr_list);
348 blobmsg_parse(vlan_attrs, __BRVLAN_ATTR_MAX, tb, blob_data(b.head), blob_len(b.head));
349
350 if (!tb[BRVLAN_ATTR_VID])
351 return;
352
353 vid = blobmsg_get_u32(tb[BRVLAN_ATTR_VID]);
354 if (!vid || vid > 4095)
355 return;
356
357 blobmsg_for_each_attr(cur, tb[BRVLAN_ATTR_PORTS], rem) {
358 name_len += strlen(blobmsg_get_string(cur)) + 1;
359 n_ports++;
360 }
361
362 vlan = calloc(1, sizeof(*vlan) + n_ports * sizeof(*port) + name_len);
363 if (!vlan)
364 return;
365
366 vlan->vid = vid;
367 vlan->local = true;
368 if (tb[BRVLAN_ATTR_LOCAL])
369 vlan->local = blobmsg_get_bool(tb[BRVLAN_ATTR_LOCAL]);
370
371 vlan->n_ports = n_ports;
372 vlan->ports = port = (struct bridge_vlan_port *)&vlan[1];
373 INIT_LIST_HEAD(&vlan->hotplug_ports);
374 name_buf = (char *)&port[n_ports];
375
376 blobmsg_for_each_attr(cur, tb[BRVLAN_ATTR_PORTS], rem) {
377 char *sep;
378
379 port->ifname = name_buf;
380 port->flags = BRVLAN_F_UNTAGGED;
381 strcpy(name_buf, blobmsg_get_string(cur));
382
383 sep = strchr(name_buf, ':');
384 if (sep) {
385 for (*sep = 0, sep++; *sep; sep++)
386 switch (*sep) {
387 case '*':
388 port->flags |= BRVLAN_F_PVID;
389 break;
390 case 't':
391 port->flags &= ~BRVLAN_F_UNTAGGED;
392 break;
393 }
394 }
395
396 name_buf += strlen(name_buf) + 1;
397 port++;
398 }
399
400 blobmsg_for_each_attr(cur, tb[BRVLAN_ATTR_ALIAS], rem)
401 kvlist_set(&dev->vlan_aliases, blobmsg_get_string(cur), &vid);
402
403 vlist_add(&dev->vlans, &vlan->node, &vlan->vid);
404 }
405
406
407 static void
408 config_init_vlans(void)
409 {
410 struct uci_element *e;
411 struct device *dev;
412
413 device_vlan_update(false);
414 uci_foreach_element(&uci_network->sections, e) {
415 struct uci_section *s = uci_to_section(e);
416 const char *name;
417
418 if (strcmp(s->type, "bridge-vlan") != 0)
419 continue;
420
421 name = uci_lookup_option_string(uci_ctx, s, "device");
422 if (!name)
423 continue;
424
425 dev = device_get(name, 0);
426 if (!dev || !dev->vlans.update)
427 continue;
428
429 config_parse_vlan(dev, s);
430 }
431 device_vlan_update(true);
432 }
433
434 static struct uci_package *
435 config_init_package(const char *config)
436 {
437 struct uci_context *ctx = uci_ctx;
438 struct uci_package *p = NULL;
439
440 if (!ctx) {
441 ctx = uci_alloc_context();
442 uci_ctx = ctx;
443
444 ctx->flags &= ~UCI_FLAG_STRICT;
445 if (config_path)
446 uci_set_confdir(ctx, config_path);
447
448 #ifdef DUMMY_MODE
449 uci_set_savedir(ctx, "./tmp");
450 #endif
451 } else {
452 p = uci_lookup_package(ctx, config);
453 if (p)
454 uci_unload(ctx, p);
455 }
456
457 if (uci_load(ctx, config, &p))
458 return NULL;
459
460 return p;
461 }
462
463 static void
464 config_init_interfaces(void)
465 {
466 struct uci_element *e;
467
468 uci_foreach_element(&uci_network->sections, e) {
469 struct uci_section *s = uci_to_section(e);
470
471 if (!strcmp(s->type, "interface"))
472 config_parse_interface(s, false);
473 }
474
475 uci_foreach_element(&uci_network->sections, e) {
476 struct uci_section *s = uci_to_section(e);
477
478 if (!strcmp(s->type, "alias"))
479 config_parse_interface(s, true);
480 }
481 }
482
483 static void
484 config_init_ip(void)
485 {
486 struct interface *iface;
487 struct uci_element *e;
488
489 vlist_for_each_element(&interfaces, iface, node)
490 interface_ip_update_start(&iface->config_ip);
491
492 uci_foreach_element(&uci_network->sections, e) {
493 struct uci_section *s = uci_to_section(e);
494
495 if (!strcmp(s->type, "route"))
496 config_parse_route(s, false);
497 else if (!strcmp(s->type, "route6"))
498 config_parse_route(s, true);
499 if (!strcmp(s->type, "neighbor"))
500 config_parse_neighbor(s, false);
501 else if (!strcmp(s->type, "neighbor6"))
502 config_parse_neighbor(s, true);
503 }
504
505 vlist_for_each_element(&interfaces, iface, node)
506 interface_ip_update_complete(&iface->config_ip);
507 }
508
509 static void
510 config_init_rules(void)
511 {
512 struct uci_element *e;
513
514 iprule_update_start();
515
516 uci_foreach_element(&uci_network->sections, e) {
517 struct uci_section *s = uci_to_section(e);
518
519 if (!strcmp(s->type, "rule"))
520 config_parse_rule(s, false);
521 else if (!strcmp(s->type, "rule6"))
522 config_parse_rule(s, true);
523 }
524
525 iprule_update_complete();
526 }
527
528 static void
529 config_init_globals(void)
530 {
531 struct uci_section *globals = uci_lookup_section(
532 uci_ctx, uci_network, "globals");
533 if (!globals)
534 return;
535
536 const char *ula_prefix = uci_lookup_option_string(
537 uci_ctx, globals, "ula_prefix");
538 interface_ip_set_ula_prefix(ula_prefix);
539 }
540
541 static void
542 config_parse_wireless_device(struct uci_section *s)
543 {
544 struct wireless_driver *drv;
545 const char *driver_name;
546
547 driver_name = uci_lookup_option_string(uci_ctx, s, "type");
548 if (!driver_name)
549 return;
550
551 drv = avl_find_element(&wireless_drivers, driver_name, drv, node);
552 if (!drv)
553 return;
554
555 blob_buf_init(&b, 0);
556 uci_to_blob(&b, s, drv->device.config);
557 wireless_device_create(drv, s->e.name, b.head);
558 }
559
560 static void
561 config_parse_wireless_vlan(struct wireless_interface *vif, struct uci_section *s)
562 {
563 char *name;
564
565 name = alloca(strlen(s->type) + 16);
566 sprintf(name, "@%s[%d]", s->type, config_section_idx(s));
567
568 blob_buf_init(&b, 0);
569 uci_to_blob(&b, s, vif->wdev->drv->vlan.config);
570 wireless_vlan_create(vif, b.head, s->anonymous ? name : s->e.name);
571 }
572
573 static void
574 config_parse_wireless_station(struct wireless_interface *vif, struct uci_section *s)
575 {
576 char *name;
577
578 name = alloca(strlen(s->type) + 16);
579 sprintf(name, "@%s[%d]", s->type, config_section_idx(s));
580
581 blob_buf_init(&b, 0);
582 uci_to_blob(&b, s, vif->wdev->drv->station.config);
583 wireless_station_create(vif, b.head, s->anonymous ? name : s->e.name);
584 }
585
586 static void
587 config_parse_wireless_interface(struct wireless_device *wdev, struct uci_section *s)
588 {
589 struct wireless_interface *vif;
590 struct uci_element *f;
591 char *name;
592
593 name = alloca(strlen(s->type) + 16);
594 sprintf(name, "@%s[%d]", s->type, config_section_idx(s));
595
596 blob_buf_init(&b, 0);
597 uci_to_blob(&b, s, wdev->drv->interface.config);
598 vif = wireless_interface_create(wdev, b.head, s->anonymous ? name : s->e.name);
599 if (!vif)
600 return;
601
602 vif->vlan_idx = vif->sta_idx = 0;
603 vlist_update(&vif->vlans);
604 vlist_update(&vif->stations);
605
606 if (s->anonymous)
607 goto out;
608
609 uci_foreach_element(&uci_wireless->sections, f) {
610 struct uci_section *cur = uci_to_section(f);
611 const char *vif_name;
612
613 if (strcmp(cur->type, "wifi-vlan") != 0)
614 continue;
615
616 vif_name = uci_lookup_option_string(uci_ctx, cur, "iface");
617 if (vif_name && strcmp(s->e.name, vif_name))
618 continue;
619 config_parse_wireless_vlan(vif, cur);
620 }
621
622 uci_foreach_element(&uci_wireless->sections, f) {
623 struct uci_section *cur = uci_to_section(f);
624 const char *vif_name;
625
626 if (strcmp(cur->type, "wifi-station") != 0)
627 continue;
628
629 vif_name = uci_lookup_option_string(uci_ctx, cur, "iface");
630 if (vif_name && strcmp(s->e.name, vif_name))
631 continue;
632 config_parse_wireless_station(vif, cur);
633 }
634
635 out:
636 vlist_flush(&vif->vlans);
637 vlist_flush(&vif->stations);
638 }
639
640 static void
641 config_init_wireless(void)
642 {
643 struct wireless_device *wdev;
644 struct uci_element *e;
645 const char *dev_name;
646
647 if (!uci_wireless) {
648 D(WIRELESS, "No wireless configuration found");
649 return;
650 }
651
652 vlist_update(&wireless_devices);
653
654 uci_foreach_element(&uci_wireless->sections, e) {
655 struct uci_section *s = uci_to_section(e);
656 if (strcmp(s->type, "wifi-device") != 0)
657 continue;
658
659 config_parse_wireless_device(s);
660 }
661
662 vlist_flush(&wireless_devices);
663
664 vlist_for_each_element(&wireless_devices, wdev, node) {
665 wdev->vif_idx = 0;
666 vlist_update(&wdev->interfaces);
667 }
668
669 uci_foreach_element(&uci_wireless->sections, e) {
670 struct uci_section *s = uci_to_section(e);
671
672 if (strcmp(s->type, "wifi-iface") != 0)
673 continue;
674
675 dev_name = uci_lookup_option_string(uci_ctx, s, "device");
676 if (!dev_name)
677 continue;
678
679 wdev = vlist_find(&wireless_devices, dev_name, wdev, node);
680 if (!wdev) {
681 D(WIRELESS, "device %s not found!", dev_name);
682 continue;
683 }
684
685 config_parse_wireless_interface(wdev, s);
686 }
687
688 vlist_for_each_element(&wireless_devices, wdev, node)
689 vlist_flush(&wdev->interfaces);
690 }
691
692
693 static struct blob_attr *
694 config_find_blobmsg_attr(struct blob_attr *attr, const char *name, int type)
695 {
696 struct blobmsg_policy policy = { .name = name, .type = type };
697 struct blob_attr *cur;
698
699 blobmsg_parse(&policy, 1, &cur, blobmsg_data(attr), blobmsg_len(attr));
700
701 return cur;
702 }
703
704 struct ether_addr *config_get_default_macaddr(const char *ifname)
705 {
706 struct blob_attr *cur;
707
708 if (!board_netdevs)
709 return NULL;
710
711 cur = config_find_blobmsg_attr(board_netdevs, ifname, BLOBMSG_TYPE_TABLE);
712 if (!cur)
713 return NULL;
714
715 cur = config_find_blobmsg_attr(cur, "macaddr", BLOBMSG_TYPE_STRING);
716 if (!cur)
717 return NULL;
718
719 return ether_aton(blobmsg_get_string(cur));
720 }
721
722 int config_get_default_gro(const char *ifname)
723 {
724 struct blob_attr *cur;
725
726 if (!board_netdevs)
727 return -1;
728
729 cur = config_find_blobmsg_attr(board_netdevs, ifname, BLOBMSG_TYPE_TABLE);
730 if (!cur)
731 return -1;
732
733 cur = config_find_blobmsg_attr(cur, "gro", BLOBMSG_TYPE_BOOL);
734 if (!cur)
735 return -1;
736
737 return blobmsg_get_bool(cur);
738 }
739
740 const char *config_get_default_conduit(const char *ifname)
741 {
742 struct blob_attr *cur;
743
744 if (!board_netdevs)
745 return NULL;
746
747 cur = config_find_blobmsg_attr(board_netdevs, ifname, BLOBMSG_TYPE_TABLE);
748 if (!cur)
749 return NULL;
750
751 cur = config_find_blobmsg_attr(cur, "conduit", BLOBMSG_TYPE_STRING);
752 if (!cur)
753 return NULL;
754
755 return blobmsg_get_string(cur);
756 }
757
758 static void
759 config_init_board(void)
760 {
761 struct blob_attr *cur;
762
763 blob_buf_init(&b, 0);
764
765 if (!blobmsg_add_json_from_file(&b, DEFAULT_BOARD_JSON))
766 return;
767
768 free(board_netdevs);
769 board_netdevs = NULL;
770
771 cur = config_find_blobmsg_attr(b.head, "network_device",
772 BLOBMSG_TYPE_TABLE);
773 if (!cur)
774 return;
775
776 board_netdevs = blob_memdup(cur);
777 }
778
779 int
780 config_init_all(void)
781 {
782 int ret = 0;
783 char *err;
784
785 uci_network = config_init_package("network");
786 if (!uci_network) {
787 uci_get_errorstr(uci_ctx, &err, NULL);
788 netifd_log_message(L_CRIT, "Failed to load network config (%s)\n", err);
789 free(err);
790 return -1;
791 }
792
793 uci_wireless = config_init_package("wireless");
794 if (!uci_wireless && uci_ctx->err != UCI_ERR_NOTFOUND) {
795 uci_get_errorstr(uci_ctx, &err, NULL);
796 netifd_log_message(L_CRIT, "Failed to load wireless config (%s)\n", err);
797 free(err);
798 ret = -1;
799 }
800
801 config_init_board();
802
803 vlist_update(&interfaces);
804 config_init = true;
805
806 device_reset_config();
807 config_init_devices(true);
808 config_init_vlans();
809 config_init_devices(false);
810 config_init_interfaces();
811 config_init_ip();
812 config_init_rules();
813 config_init_globals();
814 config_init_wireless();
815
816 config_init = false;
817
818 device_reset_old();
819 device_init_pending();
820 vlist_flush(&interfaces);
821 interface_refresh_assignments(false);
822 interface_start_pending();
823 wireless_start_pending(0);
824
825 return ret;
826 }