10 #include "interface.h"
11 #include "interface-ip.h"
27 static const struct blobmsg_policy route_attr
[__ROUTE_MAX
] = {
28 [ROUTE_INTERFACE
] = { .name
= "interface", .type
= BLOBMSG_TYPE_STRING
},
29 [ROUTE_TARGET
] = { .name
= "target", .type
= BLOBMSG_TYPE_STRING
},
30 [ROUTE_MASK
] = { .name
= "netmask", .type
= BLOBMSG_TYPE_STRING
},
31 [ROUTE_GATEWAY
] = { .name
= "gateway", .type
= BLOBMSG_TYPE_STRING
},
32 [ROUTE_DEVICE
] = { .name
= "device", .type
= BLOBMSG_TYPE_STRING
},
33 [ROUTE_METRIC
] = { .name
= "metric", .type
= BLOBMSG_TYPE_INT32
},
34 [ROUTE_MTU
] = { .name
= "mtu", .type
= BLOBMSG_TYPE_INT32
},
37 const struct config_param_list route_attr_list
= {
38 .n_params
= __ROUTE_MAX
,
43 interface_ip_add_route(struct interface
*iface
, struct blob_attr
*attr
, bool v6
)
45 struct interface_ip_settings
*ip
;
46 struct blob_attr
*tb
[__ROUTE_MAX
], *cur
;
47 struct device_route
*route
;
48 int af
= v6
? AF_INET6
: AF_INET
;
51 blobmsg_parse(route_attr
, __ROUTE_MAX
, tb
, blobmsg_data(attr
), blobmsg_data_len(attr
));
53 if (!tb
[ROUTE_GATEWAY
] && !tb
[ROUTE_DEVICE
])
57 if ((cur
= tb
[ROUTE_INTERFACE
]) == NULL
)
60 iface
= vlist_find(&interfaces
, blobmsg_data(cur
), iface
, node
);
64 ip
= &iface
->config_ip
;
67 ip
= &iface
->proto_ip
;
70 route
= calloc(1, sizeof(*route
));
74 route
->mask
= v6
? 128 : 32;
75 if ((cur
= tb
[ROUTE_MASK
]) != NULL
) {
76 route
->mask
= parse_netmask_string(blobmsg_data(cur
), v6
);
77 if (route
->mask
> (v6
? 128 : 32))
81 if ((cur
= tb
[ROUTE_TARGET
]) != NULL
) {
82 if (!inet_pton(af
, blobmsg_data(cur
), &route
->addr
)) {
83 DPRINTF("Failed to parse route target: %s\n", (char *) blobmsg_data(cur
));
88 if ((cur
= tb
[ROUTE_GATEWAY
]) != NULL
) {
89 if (!inet_pton(af
, blobmsg_data(cur
), &route
->nexthop
)) {
90 DPRINTF("Failed to parse route gateway: %s\n", (char *) blobmsg_data(cur
));
95 if ((cur
= tb
[ROUTE_METRIC
]) != NULL
)
96 route
->metric
= blobmsg_get_u32(cur
);
98 if ((cur
= tb
[ROUTE_MTU
]) != NULL
)
99 route
->mtu
= blobmsg_get_u32(cur
);
101 if (!config
&& (cur
= tb
[ROUTE_DEVICE
]) != NULL
)
102 route
->device
= device_get(blobmsg_data(cur
), true);
104 vlist_add(&ip
->route
, &route
->node
);
112 addr_cmp(const void *k1
, const void *k2
, void *ptr
)
114 return memcmp(k1
, k2
, sizeof(struct device_addr
) -
115 offsetof(struct device_addr
, mask
));
119 route_cmp(const void *k1
, const void *k2
, void *ptr
)
121 return memcmp(k1
, k2
, sizeof(struct device_route
) -
122 offsetof(struct device_route
, mask
));
126 interface_update_proto_addr(struct vlist_tree
*tree
,
127 struct vlist_node
*node_new
,
128 struct vlist_node
*node_old
)
130 struct interface_ip_settings
*ip
;
131 struct interface
*iface
;
133 struct device_addr
*addr
;
136 ip
= container_of(tree
, struct interface_ip_settings
, addr
);
138 dev
= iface
->l3_dev
->dev
;
140 if (node_old
&& node_new
)
144 addr
= container_of(node_old
, struct device_addr
, node
);
145 if (!(addr
->flags
& DEVADDR_EXTERNAL
) && addr
->enabled
&& !keep
)
146 system_del_address(dev
, addr
);
151 addr
= container_of(node_new
, struct device_addr
, node
);
152 if (!(addr
->flags
& DEVADDR_EXTERNAL
) && !keep
)
153 system_add_address(dev
, addr
);
154 addr
->enabled
= true;
159 interface_update_proto_route(struct vlist_tree
*tree
,
160 struct vlist_node
*node_new
,
161 struct vlist_node
*node_old
)
163 struct interface_ip_settings
*ip
;
164 struct interface
*iface
;
166 struct device_route
*route_old
, *route_new
;
169 ip
= container_of(tree
, struct interface_ip_settings
, route
);
171 dev
= iface
->l3_dev
->dev
;
173 route_old
= container_of(node_old
, struct device_route
, node
);
174 route_new
= container_of(node_new
, struct device_route
, node
);
176 if (node_old
&& node_new
)
177 keep
= !memcmp(&route_old
->nexthop
, &route_new
->nexthop
, sizeof(route_old
->nexthop
));
180 if (!(route_old
->flags
& DEVADDR_EXTERNAL
) && route_old
->enabled
&& !keep
)
181 system_del_route(dev
, route_old
);
186 if (!(route_new
->flags
& DEVADDR_EXTERNAL
) && !keep
)
187 system_add_route(dev
, route_new
);
188 route_new
->enabled
= true;
193 interface_add_dns_server(struct interface_ip_settings
*ip
, const char *str
)
195 struct dns_server
*s
;
197 s
= calloc(1, sizeof(*s
));
199 if (inet_pton(s
->af
, str
, &s
->addr
.in
))
203 if (inet_pton(s
->af
, str
, &s
->addr
.in
))
210 D(INTERFACE
, "Add IPv%c DNS server: %s\n",
211 s
->af
== AF_INET6
? '6' : '4', str
);
212 list_add_tail(&s
->list
, &ip
->dns_servers
);
216 interface_add_dns_server_list(struct interface_ip_settings
*ip
, struct blob_attr
*list
)
218 struct blob_attr
*cur
;
221 blobmsg_for_each_attr(cur
, list
, rem
) {
222 if (blobmsg_type(cur
) != BLOBMSG_TYPE_STRING
)
225 if (!blobmsg_check_attr(cur
, NULL
))
228 interface_add_dns_server(ip
, blobmsg_data(cur
));
233 interface_add_dns_search_domain(struct interface_ip_settings
*ip
, const char *str
)
235 struct dns_search_domain
*s
;
236 int len
= strlen(str
);
238 s
= calloc(1, sizeof(*s
) + len
+ 1);
242 D(INTERFACE
, "Add DNS search domain: %s\n", str
);
243 memcpy(s
->name
, str
, len
);
244 list_add_tail(&s
->list
, &ip
->dns_search
);
248 interface_add_dns_search_list(struct interface_ip_settings
*ip
, struct blob_attr
*list
)
250 struct blob_attr
*cur
;
253 blobmsg_for_each_attr(cur
, list
, rem
) {
254 if (blobmsg_type(cur
) != BLOBMSG_TYPE_STRING
)
257 if (!blobmsg_check_attr(cur
, NULL
))
260 interface_add_dns_search_domain(ip
, blobmsg_data(cur
));
265 interface_clear_dns_servers(struct interface_ip_settings
*ip
)
267 struct dns_server
*s
, *tmp
;
269 list_for_each_entry_safe(s
, tmp
, &ip
->dns_servers
, list
) {
276 interface_clear_dns_search(struct interface_ip_settings
*ip
)
278 struct dns_search_domain
*s
, *tmp
;
280 list_for_each_entry_safe(s
, tmp
, &ip
->dns_search
, list
) {
287 interface_write_resolv_conf(void)
289 struct interface
*iface
;
290 struct dns_server
*s
;
291 struct dns_search_domain
*d
;
292 char *path
= alloca(strlen(resolv_conf
) + 5);
297 sprintf(path
, "%s.tmp", resolv_conf
);
299 f
= fopen(path
, "w");
301 D(INTERFACE
, "Failed to open %s for writing\n", path
);
305 vlist_for_each_element(&interfaces
, iface
, node
) {
306 if (iface
->state
!= IFS_UP
)
309 if (list_empty(&iface
->proto_ip
.dns_search
) &&
310 list_empty(&iface
->proto_ip
.dns_servers
))
313 fprintf(f
, "# Interface %s\n", iface
->name
);
314 list_for_each_entry(s
, &iface
->proto_ip
.dns_servers
, list
) {
315 str
= inet_ntop(s
->af
, &s
->addr
, buf
, sizeof(buf
));
319 fprintf(f
, "nameserver %s\n", str
);
322 list_for_each_entry(d
, &iface
->proto_ip
.dns_search
, list
) {
323 fprintf(f
, "search %s\n", d
->name
);
327 if (rename(path
, resolv_conf
) < 0) {
328 D(INTERFACE
, "Failed to replace %s\n", resolv_conf
);
333 void interface_ip_set_enabled(struct interface_ip_settings
*ip
, bool enabled
)
335 struct device_addr
*addr
;
336 struct device_route
*route
;
339 ip
->enabled
= enabled
;
340 dev
= ip
->iface
->l3_dev
->dev
;
344 vlist_for_each_element(&ip
->addr
, addr
, node
) {
345 if (addr
->enabled
== enabled
)
349 system_add_address(dev
, addr
);
351 system_del_address(dev
, addr
);
352 addr
->enabled
= enabled
;
355 vlist_for_each_element(&ip
->route
, route
, node
) {
356 if (route
->enabled
== enabled
)
360 system_add_route(dev
, route
);
362 system_del_route(dev
, route
);
363 route
->enabled
= enabled
;
368 interface_ip_update_start(struct interface_ip_settings
*ip
)
370 interface_clear_dns_servers(ip
);
371 interface_clear_dns_search(ip
);
372 vlist_update(&ip
->route
);
373 vlist_update(&ip
->addr
);
377 interface_ip_update_complete(struct interface_ip_settings
*ip
)
379 vlist_flush(&ip
->route
);
380 vlist_flush(&ip
->addr
);
384 interface_ip_flush(struct interface_ip_settings
*ip
)
386 interface_clear_dns_servers(ip
);
387 interface_clear_dns_search(ip
);
388 vlist_flush_all(&ip
->route
);
389 vlist_flush_all(&ip
->addr
);
393 interface_ip_init(struct interface_ip_settings
*ip
, struct interface
*iface
)
397 INIT_LIST_HEAD(&ip
->dns_search
);
398 INIT_LIST_HEAD(&ip
->dns_servers
);
399 vlist_init(&ip
->route
, route_cmp
, interface_update_proto_route
,
400 struct device_route
, node
, mask
);
401 vlist_init(&ip
->addr
, addr_cmp
, interface_update_proto_addr
,
402 struct device_addr
, node
, mask
);