mirror of
				https://git.kernel.org/pub/scm/network/wireless/iwd.git
				synced 2025-11-04 00:37:22 +01:00 
			
		
		
		
	Add ability to populate search domains for resolvconf based systems. Search domains are added using the 'search' directive and added using the <ifname>.domain key into resolvconf.
		
			
				
	
	
		
			586 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			586 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 *
 | 
						|
 *  Wireless daemon for Linux
 | 
						|
 *
 | 
						|
 *  Copyright (C) 2019  Intel Corporation. All rights reserved.
 | 
						|
 *
 | 
						|
 *  This library is free software; you can redistribute it and/or
 | 
						|
 *  modify it under the terms of the GNU Lesser General Public
 | 
						|
 *  License as published by the Free Software Foundation; either
 | 
						|
 *  version 2.1 of the License, or (at your option) any later version.
 | 
						|
 *
 | 
						|
 *  This library is distributed in the hope that it will be useful,
 | 
						|
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
						|
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 | 
						|
 *  Lesser General Public License for more details.
 | 
						|
 *
 | 
						|
 *  You should have received a copy of the GNU Lesser General Public
 | 
						|
 *  License along with this library; if not, write to the Free Software
 | 
						|
 *  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
 | 
						|
 *
 | 
						|
 */
 | 
						|
 | 
						|
#ifdef HAVE_CONFIG_H
 | 
						|
#include <config.h>
 | 
						|
#endif
 | 
						|
 | 
						|
#include <errno.h>
 | 
						|
#include <arpa/inet.h>
 | 
						|
#include <sys/stat.h>
 | 
						|
#include <stdio.h>
 | 
						|
#include <unistd.h>
 | 
						|
 | 
						|
#include <ell/ell.h>
 | 
						|
 | 
						|
#include "src/iwd.h"
 | 
						|
#include "src/module.h"
 | 
						|
#include "src/dbus.h"
 | 
						|
#include "src/resolve.h"
 | 
						|
 | 
						|
struct resolve_ops {
 | 
						|
	void (*add_dns)(struct resolve *resolve,
 | 
						|
					uint8_t type, char **dns_list);
 | 
						|
	void (*add_domain_name)(struct resolve *resolve,
 | 
						|
					const char *domain_name);
 | 
						|
	void (*revert)(struct resolve *resolve);
 | 
						|
	void (*destroy)(struct resolve *resolve);
 | 
						|
};
 | 
						|
 | 
						|
struct resolve {
 | 
						|
	const struct resolve_ops *ops;
 | 
						|
	uint32_t ifindex;
 | 
						|
};
 | 
						|
 | 
						|
static inline void _resolve_init(struct resolve *resolve, uint32_t ifindex,
 | 
						|
					const struct resolve_ops *ops)
 | 
						|
{
 | 
						|
	resolve->ifindex = ifindex;
 | 
						|
	resolve->ops = ops;
 | 
						|
}
 | 
						|
 | 
						|
void resolve_add_dns(struct resolve *resolve, uint8_t type, char **dns_list)
 | 
						|
{
 | 
						|
	if (!dns_list || !*dns_list)
 | 
						|
		return;
 | 
						|
 | 
						|
	if (!resolve->ops->add_dns)
 | 
						|
		return;
 | 
						|
 | 
						|
	resolve->ops->add_dns(resolve, type, dns_list);
 | 
						|
}
 | 
						|
 | 
						|
void resolve_add_domain_name(struct resolve *resolve, const char *domain_name)
 | 
						|
{
 | 
						|
	if (!domain_name)
 | 
						|
		return;
 | 
						|
 | 
						|
	if (!resolve->ops->add_domain_name)
 | 
						|
		return;
 | 
						|
 | 
						|
	resolve->ops->add_domain_name(resolve, domain_name);
 | 
						|
}
 | 
						|
 | 
						|
void resolve_revert(struct resolve *resolve)
 | 
						|
{
 | 
						|
	if (!resolve->ops->revert)
 | 
						|
		return;
 | 
						|
 | 
						|
	resolve->ops->revert(resolve);
 | 
						|
}
 | 
						|
 | 
						|
void resolve_free(struct resolve *resolve)
 | 
						|
{
 | 
						|
	resolve->ops->destroy(resolve);
 | 
						|
}
 | 
						|
 | 
						|
struct resolve_method_ops {
 | 
						|
	int (*init)(void);
 | 
						|
	void (*exit)(void);
 | 
						|
	struct resolve *(*alloc)(uint32_t ifindex);
 | 
						|
};
 | 
						|
 | 
						|
#define SYSTEMD_RESOLVED_SERVICE           "org.freedesktop.resolve1"
 | 
						|
#define SYSTEMD_RESOLVED_MANAGER_PATH      "/org/freedesktop/resolve1"
 | 
						|
#define SYSTEMD_RESOLVED_MANAGER_INTERFACE "org.freedesktop.resolve1.Manager"
 | 
						|
 | 
						|
struct systemd_method_state {
 | 
						|
	uint32_t service_watch;
 | 
						|
	bool is_ready:1;
 | 
						|
};
 | 
						|
 | 
						|
struct systemd_method_state systemd_state;
 | 
						|
 | 
						|
struct systemd {
 | 
						|
	struct resolve super;
 | 
						|
};
 | 
						|
 | 
						|
static void systemd_link_dns_reply(struct l_dbus_message *message,
 | 
						|
								void *user_data)
 | 
						|
{
 | 
						|
	const char *name;
 | 
						|
	const char *text;
 | 
						|
 | 
						|
	if (!l_dbus_message_is_error(message))
 | 
						|
		return;
 | 
						|
 | 
						|
	l_dbus_message_get_error(message, &name, &text);
 | 
						|
 | 
						|
	l_error("resolve-systemd: Failed to modify the DNS entries. %s: %s",
 | 
						|
								name, text);
 | 
						|
}
 | 
						|
 | 
						|
static bool systemd_builder_add_dns(struct l_dbus_message_builder *builder,
 | 
						|
						uint8_t type, const char *dns)
 | 
						|
{
 | 
						|
	uint8_t buf[16];
 | 
						|
	uint8_t buf_size;
 | 
						|
	uint8_t i;
 | 
						|
	int t = (int) type;
 | 
						|
 | 
						|
	l_debug("installing DNS: %s %u", dns, type);
 | 
						|
 | 
						|
	l_dbus_message_builder_append_basic(builder, 'i', &t);
 | 
						|
	l_dbus_message_builder_enter_array(builder, "y");
 | 
						|
 | 
						|
	switch (type) {
 | 
						|
	case AF_INET:
 | 
						|
		if (inet_pton(AF_INET, dns, buf) < 1)
 | 
						|
			return false;
 | 
						|
 | 
						|
		buf_size = 4;
 | 
						|
 | 
						|
		break;
 | 
						|
	case AF_INET6:
 | 
						|
		if (inet_pton(AF_INET6, dns, buf) < 1)
 | 
						|
			return false;
 | 
						|
 | 
						|
		buf_size = 16;
 | 
						|
 | 
						|
		break;
 | 
						|
	default:
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
 | 
						|
	for (i = 0; i < buf_size; i++)
 | 
						|
		l_dbus_message_builder_append_basic(builder, 'y', &buf[i]);
 | 
						|
 | 
						|
	l_dbus_message_builder_leave_array(builder);
 | 
						|
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
static void resolve_systemd_add_dns(struct resolve *resolve,
 | 
						|
					uint8_t type, char **dns_list)
 | 
						|
{
 | 
						|
	struct l_dbus_message_builder *builder;
 | 
						|
	struct l_dbus_message *message;
 | 
						|
 | 
						|
	l_debug("ifindex: %u", resolve->ifindex);
 | 
						|
 | 
						|
	if (L_WARN_ON(!systemd_state.is_ready))
 | 
						|
		return;
 | 
						|
 | 
						|
	message = l_dbus_message_new_method_call(dbus_get_bus(),
 | 
						|
					SYSTEMD_RESOLVED_SERVICE,
 | 
						|
					SYSTEMD_RESOLVED_MANAGER_PATH,
 | 
						|
					SYSTEMD_RESOLVED_MANAGER_INTERFACE,
 | 
						|
					"SetLinkDNS");
 | 
						|
 | 
						|
	if (!message)
 | 
						|
		return;
 | 
						|
 | 
						|
	builder = l_dbus_message_builder_new(message);
 | 
						|
	if (!builder) {
 | 
						|
		l_dbus_message_unref(message);
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
	l_dbus_message_builder_append_basic(builder, 'i', &resolve->ifindex);
 | 
						|
 | 
						|
	l_dbus_message_builder_enter_array(builder, "(iay)");
 | 
						|
 | 
						|
	for (; *dns_list; dns_list++) {
 | 
						|
		l_dbus_message_builder_enter_struct(builder, "iay");
 | 
						|
 | 
						|
		if (systemd_builder_add_dns(builder, type, *dns_list)) {
 | 
						|
			l_dbus_message_builder_leave_struct(builder);
 | 
						|
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
 | 
						|
		l_dbus_message_builder_destroy(builder);
 | 
						|
		l_dbus_message_unref(message);
 | 
						|
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
	l_dbus_message_builder_leave_array(builder);
 | 
						|
 | 
						|
	l_dbus_message_builder_finalize(builder);
 | 
						|
	l_dbus_message_builder_destroy(builder);
 | 
						|
 | 
						|
	l_dbus_send_with_reply(dbus_get_bus(), message, systemd_link_dns_reply,
 | 
						|
								NULL, NULL);
 | 
						|
}
 | 
						|
 | 
						|
static void systemd_link_add_domains_reply(struct l_dbus_message *message,
 | 
						|
								void *user_data)
 | 
						|
{
 | 
						|
	const char *name;
 | 
						|
	const char *text;
 | 
						|
 | 
						|
	if (!l_dbus_message_is_error(message))
 | 
						|
		return;
 | 
						|
 | 
						|
	l_dbus_message_get_error(message, &name, &text);
 | 
						|
 | 
						|
	l_error("resolve-systemd: Failed to modify the domain entries. %s: %s",
 | 
						|
								name, text);
 | 
						|
}
 | 
						|
 | 
						|
static void resolve_systemd_add_domain_name(struct resolve *resolve,
 | 
						|
						const char *domain_name)
 | 
						|
{
 | 
						|
	struct l_dbus_message *message;
 | 
						|
 | 
						|
	l_debug("ifindex: %u", resolve->ifindex);
 | 
						|
 | 
						|
	if (L_WARN_ON(!systemd_state.is_ready))
 | 
						|
		return;
 | 
						|
 | 
						|
	message = l_dbus_message_new_method_call(dbus_get_bus(),
 | 
						|
					SYSTEMD_RESOLVED_SERVICE,
 | 
						|
					SYSTEMD_RESOLVED_MANAGER_PATH,
 | 
						|
					SYSTEMD_RESOLVED_MANAGER_INTERFACE,
 | 
						|
					"SetLinkDomains");
 | 
						|
 | 
						|
	if (!message)
 | 
						|
		return;
 | 
						|
 | 
						|
	l_dbus_message_set_arguments(message, "ia(sb)", resolve->ifindex,
 | 
						|
						1, domain_name, false);
 | 
						|
 | 
						|
	l_dbus_send_with_reply(dbus_get_bus(), message,
 | 
						|
				systemd_link_add_domains_reply, NULL, NULL);
 | 
						|
}
 | 
						|
 | 
						|
static void resolve_systemd_revert(struct resolve *resolve)
 | 
						|
{
 | 
						|
	struct l_dbus_message *message;
 | 
						|
 | 
						|
	l_debug("ifindex: %u", resolve->ifindex);
 | 
						|
 | 
						|
	if (L_WARN_ON(!systemd_state.is_ready))
 | 
						|
		return;
 | 
						|
 | 
						|
	message = l_dbus_message_new_method_call(dbus_get_bus(),
 | 
						|
					SYSTEMD_RESOLVED_SERVICE,
 | 
						|
					SYSTEMD_RESOLVED_MANAGER_PATH,
 | 
						|
					SYSTEMD_RESOLVED_MANAGER_INTERFACE,
 | 
						|
					"RevertLink");
 | 
						|
	if (!message)
 | 
						|
		return;
 | 
						|
 | 
						|
	l_dbus_message_set_arguments(message, "i", resolve->ifindex);
 | 
						|
	l_dbus_send_with_reply(dbus_get_bus(), message, systemd_link_dns_reply,
 | 
						|
								NULL, NULL);
 | 
						|
}
 | 
						|
 | 
						|
static void resolve_systemd_destroy(struct resolve *resolve)
 | 
						|
{
 | 
						|
	struct systemd *sd = l_container_of(resolve, struct systemd, super);
 | 
						|
 | 
						|
	l_free(sd);
 | 
						|
}
 | 
						|
 | 
						|
static const struct resolve_ops systemd_ops = {
 | 
						|
	.add_dns = resolve_systemd_add_dns,
 | 
						|
	.add_domain_name = resolve_systemd_add_domain_name,
 | 
						|
	.revert = resolve_systemd_revert,
 | 
						|
	.destroy = resolve_systemd_destroy,
 | 
						|
};
 | 
						|
 | 
						|
static void systemd_appeared(struct l_dbus *dbus, void *user_data)
 | 
						|
{
 | 
						|
	systemd_state.is_ready = true;
 | 
						|
}
 | 
						|
 | 
						|
static void systemd_disappeared(struct l_dbus *dbus, void *user_data)
 | 
						|
{
 | 
						|
	systemd_state.is_ready = false;
 | 
						|
}
 | 
						|
 | 
						|
static int resolve_systemd_init(void)
 | 
						|
{
 | 
						|
	systemd_state.service_watch =
 | 
						|
				l_dbus_add_service_watch(dbus_get_bus(),
 | 
						|
						SYSTEMD_RESOLVED_SERVICE,
 | 
						|
						systemd_appeared,
 | 
						|
						systemd_disappeared,
 | 
						|
						NULL, NULL);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void resolve_systemd_exit(void)
 | 
						|
{
 | 
						|
	l_dbus_remove_watch(dbus_get_bus(), systemd_state.service_watch);
 | 
						|
	memset(&systemd_state, 0, sizeof(systemd_state));
 | 
						|
}
 | 
						|
 | 
						|
static struct resolve *resolve_systemd_alloc(uint32_t ifindex)
 | 
						|
{
 | 
						|
	struct systemd *sd = l_new(struct systemd, 1);
 | 
						|
 | 
						|
	_resolve_init(&sd->super, ifindex, &systemd_ops);
 | 
						|
 | 
						|
	return &sd->super;
 | 
						|
}
 | 
						|
 | 
						|
static const struct resolve_method_ops resolve_method_systemd_ops = {
 | 
						|
	.init = resolve_systemd_init,
 | 
						|
	.exit = resolve_systemd_exit,
 | 
						|
	.alloc = resolve_systemd_alloc,
 | 
						|
};
 | 
						|
 | 
						|
char *resolvconf_path;
 | 
						|
 | 
						|
static bool resolvconf_invoke(const char *ifname, const char *type,
 | 
						|
						const char *content)
 | 
						|
{
 | 
						|
	FILE *resolvconf;
 | 
						|
	L_AUTO_FREE_VAR(char *, cmd) = NULL;
 | 
						|
	int error;
 | 
						|
 | 
						|
	if (content) {
 | 
						|
		cmd = l_strdup_printf("%s -a %s.%s", resolvconf_path,
 | 
						|
						ifname, type);
 | 
						|
		resolvconf = popen(cmd, "w");
 | 
						|
	} else {
 | 
						|
		cmd = l_strdup_printf("%s -d %s.%s", resolvconf_path,
 | 
						|
						ifname, type);
 | 
						|
		resolvconf = popen(cmd, "r");
 | 
						|
	}
 | 
						|
 | 
						|
	if (!resolvconf) {
 | 
						|
		l_error("resolve: Failed to start %s (%s).", resolvconf_path,
 | 
						|
							strerror(errno));
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
 | 
						|
	if (content && fprintf(resolvconf, "%s", content) < 0)
 | 
						|
		l_error("resolve: Failed to print into %s stdin.",
 | 
						|
							resolvconf_path);
 | 
						|
 | 
						|
	error = pclose(resolvconf);
 | 
						|
	if (error < 0)
 | 
						|
		l_error("resolve: Failed to close pipe to %s (%s).",
 | 
						|
					resolvconf_path, strerror(errno));
 | 
						|
	else if (error > 0)
 | 
						|
		l_info("resolve: %s exited with status (%d).", resolvconf_path,
 | 
						|
									error);
 | 
						|
 | 
						|
	return !error;
 | 
						|
}
 | 
						|
 | 
						|
struct resolvconf {
 | 
						|
	struct resolve super;
 | 
						|
	bool have_domain : 1;
 | 
						|
	bool have_dns : 1;
 | 
						|
	char *ifname;
 | 
						|
};
 | 
						|
 | 
						|
static void resolve_resolvconf_add_dns(struct resolve *resolve,
 | 
						|
					uint8_t type, char **dns_list)
 | 
						|
{
 | 
						|
	struct resolvconf *rc =
 | 
						|
			l_container_of(resolve, struct resolvconf, super);
 | 
						|
	struct l_string *content;
 | 
						|
	L_AUTO_FREE_VAR(char *, str) = NULL;
 | 
						|
 | 
						|
	if (L_WARN_ON(!resolvconf_path))
 | 
						|
		return;
 | 
						|
 | 
						|
	content = l_string_new(0);
 | 
						|
 | 
						|
	for (; *dns_list; dns_list++)
 | 
						|
		l_string_append_printf(content, "nameserver %s\n", *dns_list);
 | 
						|
 | 
						|
	str = l_string_unwrap(content);
 | 
						|
 | 
						|
	if (resolvconf_invoke(rc->ifname, "dns", str))
 | 
						|
		rc->have_dns = true;
 | 
						|
}
 | 
						|
 | 
						|
static void resolve_resolvconf_add_domain_name(struct resolve *resolve,
 | 
						|
							const char *domain_name)
 | 
						|
{
 | 
						|
	struct resolvconf *rc =
 | 
						|
			l_container_of(resolve, struct resolvconf, super);
 | 
						|
	L_AUTO_FREE_VAR(char *, domain_str) = NULL;
 | 
						|
 | 
						|
	if (L_WARN_ON(!resolvconf_path))
 | 
						|
		return;
 | 
						|
 | 
						|
	domain_str = l_strdup_printf("search %s\n", domain_name);
 | 
						|
 | 
						|
	if (resolvconf_invoke(rc->ifname, "domain", domain_str))
 | 
						|
		rc->have_domain = true;
 | 
						|
}
 | 
						|
 | 
						|
static void resolve_resolvconf_revert(struct resolve *resolve)
 | 
						|
{
 | 
						|
	struct resolvconf *rc =
 | 
						|
			l_container_of(resolve, struct resolvconf, super);
 | 
						|
 | 
						|
	if (rc->have_dns)
 | 
						|
		resolvconf_invoke(rc->ifname, "dns", NULL);
 | 
						|
 | 
						|
	if (rc->have_domain)
 | 
						|
		resolvconf_invoke(rc->ifname, "domain", NULL);
 | 
						|
 | 
						|
	rc->have_dns = false;
 | 
						|
	rc->have_domain = false;
 | 
						|
}
 | 
						|
 | 
						|
static void resolve_resolvconf_destroy(struct resolve *resolve)
 | 
						|
{
 | 
						|
	struct resolvconf *rc =
 | 
						|
			l_container_of(resolve, struct resolvconf, super);
 | 
						|
 | 
						|
	l_free(rc->ifname);
 | 
						|
	l_free(rc);
 | 
						|
}
 | 
						|
 | 
						|
static struct resolve_ops resolvconf_ops = {
 | 
						|
	.add_dns = resolve_resolvconf_add_dns,
 | 
						|
	.add_domain_name = resolve_resolvconf_add_domain_name,
 | 
						|
	.revert = resolve_resolvconf_revert,
 | 
						|
	.destroy = resolve_resolvconf_destroy,
 | 
						|
};
 | 
						|
 | 
						|
static int resolve_resolvconf_init(void)
 | 
						|
{
 | 
						|
	static const char *default_path = "/sbin:/usr/sbin";
 | 
						|
	const char *path;
 | 
						|
 | 
						|
	l_debug("Trying to find resolvconf in $PATH");
 | 
						|
 | 
						|
	path = getenv("PATH");
 | 
						|
	if (path)
 | 
						|
		resolvconf_path = l_path_find("resolvconf", path, X_OK);
 | 
						|
 | 
						|
	if (!resolvconf_path) {
 | 
						|
		l_debug("Trying to find resolvconf in default paths");
 | 
						|
		resolvconf_path = l_path_find("resolvconf", default_path, X_OK);
 | 
						|
	}
 | 
						|
 | 
						|
	if (!resolvconf_path) {
 | 
						|
		l_error("No usable resolvconf found on system");
 | 
						|
		return -ENOENT;
 | 
						|
	}
 | 
						|
 | 
						|
	l_debug("resolvconf found as: %s", resolvconf_path);
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void resolve_resolvconf_exit(void)
 | 
						|
{
 | 
						|
	l_free(resolvconf_path);
 | 
						|
	resolvconf_path = NULL;
 | 
						|
}
 | 
						|
 | 
						|
static struct resolve *resolve_resolvconf_alloc(uint32_t ifindex)
 | 
						|
{
 | 
						|
	struct resolvconf *rc = l_new(struct resolvconf, 1);
 | 
						|
 | 
						|
	_resolve_init(&rc->super, ifindex, &resolvconf_ops);
 | 
						|
 | 
						|
	rc->ifname = l_net_get_name(ifindex);
 | 
						|
	if (!rc->ifname)
 | 
						|
		rc->ifname = l_strdup_printf("%u", ifindex);
 | 
						|
 | 
						|
	return &rc->super;
 | 
						|
}
 | 
						|
 | 
						|
static const struct resolve_method_ops resolve_method_resolvconf_ops = {
 | 
						|
	.init = resolve_resolvconf_init,
 | 
						|
	.exit = resolve_resolvconf_exit,
 | 
						|
	.alloc = resolve_resolvconf_alloc,
 | 
						|
};
 | 
						|
 | 
						|
static const struct resolve_method_ops *configured_method;
 | 
						|
 | 
						|
struct resolve *resolve_new(uint32_t ifindex)
 | 
						|
{
 | 
						|
	if (L_WARN_ON(!configured_method))
 | 
						|
		return NULL;
 | 
						|
 | 
						|
	return configured_method->alloc(ifindex);
 | 
						|
}
 | 
						|
 | 
						|
static const struct {
 | 
						|
	const char *name;
 | 
						|
	const struct resolve_method_ops *method_ops;
 | 
						|
} resolve_method_ops_list[] = {
 | 
						|
	{ "systemd", &resolve_method_systemd_ops },
 | 
						|
	{ "resolvconf", &resolve_method_resolvconf_ops },
 | 
						|
	{ }
 | 
						|
};
 | 
						|
 | 
						|
static int resolve_init(void)
 | 
						|
{
 | 
						|
	const char *method_name;
 | 
						|
	bool enabled;
 | 
						|
	uint8_t i;
 | 
						|
 | 
						|
	if (!l_settings_get_bool(iwd_get_config(), "General",
 | 
						|
					"EnableNetworkConfiguration",
 | 
						|
					&enabled)) {
 | 
						|
		if (!l_settings_get_bool(iwd_get_config(), "General",
 | 
						|
					"enable_network_config", &enabled))
 | 
						|
			enabled = false;
 | 
						|
	}
 | 
						|
 | 
						|
	if (!enabled)
 | 
						|
		return 0;
 | 
						|
 | 
						|
	method_name = l_settings_get_value(iwd_get_config(), "Network",
 | 
						|
						"NameResolvingService");
 | 
						|
	if (!method_name) {
 | 
						|
		method_name = l_settings_get_value(iwd_get_config(), "General",
 | 
						|
							"dns_resolve_method");
 | 
						|
		if (method_name)
 | 
						|
			l_warn("[General].dns_resolve_method is deprecated, "
 | 
						|
				"use [Network].NameResolvingService");
 | 
						|
		else /* Default to systemd-resolved service. */
 | 
						|
			method_name = "systemd";
 | 
						|
	}
 | 
						|
 | 
						|
	for (i = 0; resolve_method_ops_list[i].name; i++) {
 | 
						|
		if (strcmp(resolve_method_ops_list[i].name, method_name))
 | 
						|
			continue;
 | 
						|
 | 
						|
		configured_method = resolve_method_ops_list[i].method_ops;
 | 
						|
		break;
 | 
						|
	}
 | 
						|
 | 
						|
	if (!configured_method) {
 | 
						|
		l_error("Unknown resolution method: %s", method_name);
 | 
						|
		return -EINVAL;
 | 
						|
	}
 | 
						|
 | 
						|
	return configured_method->init();
 | 
						|
}
 | 
						|
 | 
						|
static void resolve_exit(void)
 | 
						|
{
 | 
						|
	if (!configured_method)
 | 
						|
		return;
 | 
						|
 | 
						|
	configured_method->exit();
 | 
						|
	configured_method = NULL;
 | 
						|
}
 | 
						|
 | 
						|
IWD_MODULE(resolve, resolve_init, resolve_exit)
 |