resolved-link.c revision 6bedfcbb2970e06a4d3280c8fb62083d252ede73
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering This file is part of systemd.
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering Copyright 2014 Lennart Poettering
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering systemd is free software; you can redistribute it and/or modify it
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering under the terms of the GNU Lesser General Public License as published by
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering the Free Software Foundation; either version 2.1 of the License, or
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering (at your option) any later version.
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering systemd is distributed in the hope that it will be useful, but
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering WITHOUT ANY WARRANTY; without even the implied warranty of
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering Lesser General Public License for more details.
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering You should have received a copy of the GNU Lesser General Public License
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering along with systemd; If not, see <http://www.gnu.org/licenses/>.
9df3ba6c6cb65eecec06f39dfe85a3596cedac4eTom Gundersenint link_new(Manager *m, Link **ret, int ifindex) {
be808ea083fa07271116b4519c3c27fd20c5f077Tom Gundersen r = hashmap_ensure_allocated(&m->links, NULL);
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering r = hashmap_put(m->links, INT_TO_PTR(ifindex), l);
be808ea083fa07271116b4519c3c27fd20c5f077Tom Gundersen hashmap_remove(l->manager->links, INT_TO_PTR(l->ifindex));
0b58db658b5c3f586ac3a837427f1f7fec2abb2eLennart Poetteringstatic void link_allocate_scopes(Link *l) {
eed857b71702f8551b46b66b31fa0d08583cf23cLennart Poettering r = dns_scope_new(l->manager, &l->unicast_scope, l, DNS_PROTOCOL_DNS, AF_UNSPEC);
0b58db658b5c3f586ac3a837427f1f7fec2abb2eLennart Poettering log_warning_errno(r, "Failed to allocate DNS scope: %m");
0b58db658b5c3f586ac3a837427f1f7fec2abb2eLennart Poettering l->unicast_scope = dns_scope_free(l->unicast_scope);
4e945a6f7971fd7d1f6b2c62ee3afdaff3c95ce4Lennart Poettering l->manager->llmnr_support != SUPPORT_NO) {
4e945a6f7971fd7d1f6b2c62ee3afdaff3c95ce4Lennart Poettering r = dns_scope_new(l->manager, &l->llmnr_ipv4_scope, l, DNS_PROTOCOL_LLMNR, AF_INET);
3e684349c2cead2e6fd2f816c34eb17daba23a49Lennart Poettering log_warning_errno(r, "Failed to allocate LLMNR IPv4 scope: %m");
4e945a6f7971fd7d1f6b2c62ee3afdaff3c95ce4Lennart Poettering l->llmnr_ipv4_scope = dns_scope_free(l->llmnr_ipv4_scope);
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering l->manager->llmnr_support != SUPPORT_NO &&
91b14d6ff362b938a72db17b095ee9903d07381bTom Gundersen r = dns_scope_new(l->manager, &l->llmnr_ipv6_scope, l, DNS_PROTOCOL_LLMNR, AF_INET6);
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering log_warning_errno(r, "Failed to allocate LLMNR IPv6 scope: %m");
91b14d6ff362b938a72db17b095ee9903d07381bTom Gundersen l->llmnr_ipv6_scope = dns_scope_free(l->llmnr_ipv6_scope);
91b14d6ff362b938a72db17b095ee9903d07381bTom Gundersenvoid link_add_rrs(Link *l, bool force_remove) {
0eac462399c8e87bcce252cf058eba9f2678f2bdLennart Poetteringint link_update_rtnl(Link *l, sd_netlink_message *m) {
91b14d6ff362b938a72db17b095ee9903d07381bTom Gundersen const char *n = NULL;
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering r = sd_rtnl_message_link_get_flags(m, &l->flags);
0eac462399c8e87bcce252cf058eba9f2678f2bdLennart Poettering sd_netlink_message_read_u32(m, IFLA_MTU, &l->mtu);
0eac462399c8e87bcce252cf058eba9f2678f2bdLennart Poettering if (sd_netlink_message_read_string(m, IFLA_IFNAME, &n) >= 0) {
eed857b71702f8551b46b66b31fa0d08583cf23cLennart Poetteringstatic int link_update_dns_servers(Link *l) {
0eac462399c8e87bcce252cf058eba9f2678f2bdLennart Poettering _cleanup_strv_free_ char **nameservers = NULL;
0eac462399c8e87bcce252cf058eba9f2678f2bdLennart Poettering r = sd_network_link_get_dns(l->ifindex, &nameservers);
0eac462399c8e87bcce252cf058eba9f2678f2bdLennart Poettering r = in_addr_from_string_auto(*nameserver, &family, &a);
91b14d6ff362b938a72db17b095ee9903d07381bTom Gundersen r = dns_server_new(l->manager, NULL, DNS_SERVER_LINK, l, family, &a);
0b58db658b5c3f586ac3a837427f1f7fec2abb2eLennart Poettering LIST_FOREACH_SAFE(servers, s, nx, l->dns_servers)
0b58db658b5c3f586ac3a837427f1f7fec2abb2eLennart Poetteringstatic int link_update_llmnr_support(Link *l) {
0b58db658b5c3f586ac3a837427f1f7fec2abb2eLennart Poettering r = sd_network_link_get_llmnr(l->ifindex, &b);
0b58db658b5c3f586ac3a837427f1f7fec2abb2eLennart Poettering } else if (r > 0)
be808ea083fa07271116b4519c3c27fd20c5f077Tom Gundersen l->unicast_scope->domains = strv_free(l->unicast_scope->domains);
9df3ba6c6cb65eecec06f39dfe85a3596cedac4eTom Gundersen /* A link is relevant if it isn't a loopback or pointopoint
4e0b8b17a7465653f4e7b819dad5f8e30d64c0c4Tom Gundersen * device, has a link beat, can do multicast and has at least
4e0b8b17a7465653f4e7b819dad5f8e30d64c0c4Tom Gundersen * one relevant IP address */
4e0b8b17a7465653f4e7b819dad5f8e30d64c0c4Tom Gundersen if (l->flags & (IFF_LOOPBACK|IFF_POINTOPOINT|IFF_DORMANT))
4e0b8b17a7465653f4e7b819dad5f8e30d64c0c4Tom Gundersen return false;
4e0b8b17a7465653f4e7b819dad5f8e30d64c0c4Tom Gundersen if ((l->flags & (IFF_UP|IFF_LOWER_UP|IFF_MULTICAST)) != (IFF_UP|IFF_LOWER_UP|IFF_MULTICAST))
4e0b8b17a7465653f4e7b819dad5f8e30d64c0c4Tom Gundersen return false;
4e0b8b17a7465653f4e7b819dad5f8e30d64c0c4Tom Gundersen sd_network_link_get_operational_state(l->ifindex, &state);
b652d4a2099d1c167584dcc1d179d47c58dc38a2Lennart Poettering if (state && !STR_IN_SET(state, "unknown", "degraded", "routable"))
b652d4a2099d1c167584dcc1d179d47c58dc38a2Lennart Poettering if (a->family == family && link_address_relevant(a))
b652d4a2099d1c167584dcc1d179d47c58dc38a2Lennart PoetteringLinkAddress *link_find_address(Link *l, int family, const union in_addr_union *in_addr) {
be808ea083fa07271116b4519c3c27fd20c5f077Tom Gundersen if (a->family == family && in_addr_equal(family, &a->in_addr, in_addr))
be808ea083fa07271116b4519c3c27fd20c5f077Tom GundersenDnsServer* link_find_dns_server(Link *l, int family, const union in_addr_union *in_addr) {
be808ea083fa07271116b4519c3c27fd20c5f077Tom Gundersen if (s->family == family && in_addr_equal(family, &s->address, in_addr))
be808ea083fa07271116b4519c3c27fd20c5f077Tom GundersenDnsServer* link_set_dns_server(Link *l, DnsServer *s) {
be808ea083fa07271116b4519c3c27fd20c5f077Tom Gundersen in_addr_to_string(s->family, &s->address, &ip);
be808ea083fa07271116b4519c3c27fd20c5f077Tom Gundersen log_info("Switching to DNS server %s for interface %s.", strna(ip), l->name);
519ef04651b07a547f010d6462603669d7fde4e5Lennart Poettering if (l->current_dns_server->servers_next) {
519ef04651b07a547f010d6462603669d7fde4e5Lennart Poettering link_set_dns_server(l, l->current_dns_server->servers_next);
519ef04651b07a547f010d6462603669d7fde4e5Lennart Poetteringint link_address_new(Link *l, LinkAddress **ret, int family, const union in_addr_union *in_addr) {
87f5a19343acf8ba697acc5a62bdb1a2b8c9eda3Lennart Poettering LIST_PREPEND(addresses, l->addresses, a);
d5099efc47d4e6ac60816b5381a5f607ab03f06eMichal SchmidtLinkAddress *link_address_free(LinkAddress *a) {
87f5a19343acf8ba697acc5a62bdb1a2b8c9eda3Lennart Poettering LIST_REMOVE(addresses, a->link->addresses, a);
87f5a19343acf8ba697acc5a62bdb1a2b8c9eda3Lennart Poettering if (a->family == AF_INET && a->link->llmnr_ipv4_scope)
87f5a19343acf8ba697acc5a62bdb1a2b8c9eda3Lennart Poettering dns_zone_remove_rr(&a->link->llmnr_ipv4_scope->zone, a->llmnr_address_rr);
d5099efc47d4e6ac60816b5381a5f607ab03f06eMichal Schmidt else if (a->family == AF_INET6 && a->link->llmnr_ipv6_scope)
d5099efc47d4e6ac60816b5381a5f607ab03f06eMichal Schmidt dns_zone_remove_rr(&a->link->llmnr_ipv6_scope->zone, a->llmnr_address_rr);
636e813dc98ea40c58c6c85bc5e7e3c9f0904ea2Lennart Poettering if (a->family == AF_INET && a->link->llmnr_ipv4_scope)
4b95f1798f22c1bb75295f448188560cb6ec9eceLennart Poettering dns_zone_remove_rr(&a->link->llmnr_ipv4_scope->zone, a->llmnr_ptr_rr);
4b95f1798f22c1bb75295f448188560cb6ec9eceLennart Poettering else if (a->family == AF_INET6 && a->link->llmnr_ipv6_scope)
0eac462399c8e87bcce252cf058eba9f2678f2bdLennart Poettering dns_zone_remove_rr(&a->link->llmnr_ipv6_scope->zone, a->llmnr_ptr_rr);
4b95f1798f22c1bb75295f448188560cb6ec9eceLennart Poettering dns_resource_record_unref(a->llmnr_address_rr);
4b95f1798f22c1bb75295f448188560cb6ec9eceLennart Poettering dns_resource_record_unref(a->llmnr_ptr_rr);
4b95f1798f22c1bb75295f448188560cb6ec9eceLennart Poetteringvoid link_address_add_rrs(LinkAddress *a, bool force_remove) {
4b95f1798f22c1bb75295f448188560cb6ec9eceLennart Poettering a->link->manager->llmnr_support == SUPPORT_YES) {
4b95f1798f22c1bb75295f448188560cb6ec9eceLennart Poettering if (!a->link->manager->llmnr_host_ipv4_key) {
4b95f1798f22c1bb75295f448188560cb6ec9eceLennart Poettering a->link->manager->llmnr_host_ipv4_key = dns_resource_key_new(DNS_CLASS_IN, DNS_TYPE_A, a->link->manager->llmnr_hostname);
4b95f1798f22c1bb75295f448188560cb6ec9eceLennart Poettering if (!a->link->manager->llmnr_host_ipv4_key) {
4b95f1798f22c1bb75295f448188560cb6ec9eceLennart Poettering a->llmnr_address_rr = dns_resource_record_new(a->link->manager->llmnr_host_ipv4_key);
4b95f1798f22c1bb75295f448188560cb6ec9eceLennart Poettering a->llmnr_address_rr->a.in_addr = a->in_addr.in;
4b95f1798f22c1bb75295f448188560cb6ec9eceLennart Poettering a->llmnr_address_rr->ttl = LLMNR_DEFAULT_TTL;
f2f1dbe50fea13abadc9c1e845a29031b90b40f3Lennart Poettering r = dns_resource_record_new_reverse(&a->llmnr_ptr_rr, a->family, &a->in_addr, a->link->manager->llmnr_hostname);
4b95f1798f22c1bb75295f448188560cb6ec9eceLennart Poettering a->llmnr_ptr_rr->ttl = LLMNR_DEFAULT_TTL;
4b95f1798f22c1bb75295f448188560cb6ec9eceLennart Poettering r = dns_zone_put(&a->link->llmnr_ipv4_scope->zone, a->link->llmnr_ipv4_scope, a->llmnr_address_rr, true);
4b95f1798f22c1bb75295f448188560cb6ec9eceLennart Poettering log_warning_errno(r, "Failed to add A record to LLMNR zone: %m");
4b95f1798f22c1bb75295f448188560cb6ec9eceLennart Poettering r = dns_zone_put(&a->link->llmnr_ipv4_scope->zone, a->link->llmnr_ipv4_scope, a->llmnr_ptr_rr, false);
f2f1dbe50fea13abadc9c1e845a29031b90b40f3Lennart Poettering log_warning_errno(r, "Failed to add IPv6 PTR record to LLMNR zone: %m");
f2f1dbe50fea13abadc9c1e845a29031b90b40f3Lennart Poettering dns_zone_remove_rr(&a->link->llmnr_ipv4_scope->zone, a->llmnr_address_rr);
f2f1dbe50fea13abadc9c1e845a29031b90b40f3Lennart Poettering a->llmnr_address_rr = dns_resource_record_unref(a->llmnr_address_rr);
f2f1dbe50fea13abadc9c1e845a29031b90b40f3Lennart Poettering dns_zone_remove_rr(&a->link->llmnr_ipv4_scope->zone, a->llmnr_ptr_rr);
f2f1dbe50fea13abadc9c1e845a29031b90b40f3Lennart Poettering a->llmnr_ptr_rr = dns_resource_record_unref(a->llmnr_ptr_rr);
f2f1dbe50fea13abadc9c1e845a29031b90b40f3Lennart Poettering a->link->manager->llmnr_support == SUPPORT_YES) {
f2f1dbe50fea13abadc9c1e845a29031b90b40f3Lennart Poettering if (!a->link->manager->llmnr_host_ipv6_key) {
f2f1dbe50fea13abadc9c1e845a29031b90b40f3Lennart Poettering a->link->manager->llmnr_host_ipv6_key = dns_resource_key_new(DNS_CLASS_IN, DNS_TYPE_AAAA, a->link->manager->llmnr_hostname);
f2f1dbe50fea13abadc9c1e845a29031b90b40f3Lennart Poettering if (!a->link->manager->llmnr_host_ipv6_key) {
f2f1dbe50fea13abadc9c1e845a29031b90b40f3Lennart Poettering a->llmnr_address_rr = dns_resource_record_new(a->link->manager->llmnr_host_ipv6_key);
f2f1dbe50fea13abadc9c1e845a29031b90b40f3Lennart Poettering a->llmnr_address_rr->aaaa.in6_addr = a->in_addr.in6;
f2f1dbe50fea13abadc9c1e845a29031b90b40f3Lennart Poettering a->llmnr_address_rr->ttl = LLMNR_DEFAULT_TTL;
f2f1dbe50fea13abadc9c1e845a29031b90b40f3Lennart Poettering r = dns_resource_record_new_reverse(&a->llmnr_ptr_rr, a->family, &a->in_addr, a->link->manager->llmnr_hostname);
f2f1dbe50fea13abadc9c1e845a29031b90b40f3Lennart Poettering a->llmnr_ptr_rr->ttl = LLMNR_DEFAULT_TTL;
f2f1dbe50fea13abadc9c1e845a29031b90b40f3Lennart Poettering r = dns_zone_put(&a->link->llmnr_ipv6_scope->zone, a->link->llmnr_ipv6_scope, a->llmnr_address_rr, true);
f2f1dbe50fea13abadc9c1e845a29031b90b40f3Lennart Poettering log_warning_errno(r, "Failed to add AAAA record to LLMNR zone: %m");
f2f1dbe50fea13abadc9c1e845a29031b90b40f3Lennart Poettering r = dns_zone_put(&a->link->llmnr_ipv6_scope->zone, a->link->llmnr_ipv6_scope, a->llmnr_ptr_rr, false);
f2f1dbe50fea13abadc9c1e845a29031b90b40f3Lennart Poettering log_warning_errno(r, "Failed to add IPv6 PTR record to LLMNR zone: %m");
f2f1dbe50fea13abadc9c1e845a29031b90b40f3Lennart Poettering dns_zone_remove_rr(&a->link->llmnr_ipv6_scope->zone, a->llmnr_address_rr);
f2f1dbe50fea13abadc9c1e845a29031b90b40f3Lennart Poettering a->llmnr_address_rr = dns_resource_record_unref(a->llmnr_address_rr);
f2f1dbe50fea13abadc9c1e845a29031b90b40f3Lennart Poettering dns_zone_remove_rr(&a->link->llmnr_ipv6_scope->zone, a->llmnr_ptr_rr);
f2f1dbe50fea13abadc9c1e845a29031b90b40f3Lennart Poettering a->llmnr_ptr_rr = dns_resource_record_unref(a->llmnr_ptr_rr);
f2f1dbe50fea13abadc9c1e845a29031b90b40f3Lennart Poettering log_debug_errno(r, "Failed to update address RRs: %m");
be808ea083fa07271116b4519c3c27fd20c5f077Tom Gundersenint link_address_update_rtnl(LinkAddress *a, sd_netlink_message *m) {
be808ea083fa07271116b4519c3c27fd20c5f077Tom Gundersen r = sd_rtnl_message_addr_get_flags(m, &a->flags);
assert(a);