2019-07-03 01:14:50 +02:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* 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>
|
2019-07-03 01:14:53 +02:00
|
|
|
#include <arpa/inet.h>
|
2019-07-03 01:14:50 +02:00
|
|
|
|
|
|
|
#include <ell/ell.h>
|
|
|
|
|
|
|
|
#include "src/iwd.h"
|
2019-07-03 01:14:52 +02:00
|
|
|
#include "src/dbus.h"
|
2019-07-03 01:14:50 +02:00
|
|
|
#include "src/resolve.h"
|
|
|
|
|
2019-07-03 01:14:51 +02:00
|
|
|
struct resolve_method_ops {
|
|
|
|
void *(*init)(void);
|
|
|
|
void (*exit)(void *data);
|
|
|
|
void (*add_dns)(uint32_t ifindex, uint8_t type, char **dns_list,
|
|
|
|
void *data);
|
|
|
|
void (*remove)(uint32_t ifindex, void *data);
|
|
|
|
};
|
|
|
|
|
|
|
|
struct resolve_method {
|
|
|
|
void *data;
|
|
|
|
const struct resolve_method_ops *ops;
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct resolve_method method;
|
|
|
|
|
2019-07-03 01:14:52 +02:00
|
|
|
#define SYSTEMD_RESOLVED_SERVICE "org.freedesktop.resolve1"
|
2019-07-03 01:14:53 +02:00
|
|
|
#define SYSTEMD_RESOLVED_MANAGER_PATH "/org/freedesktop/resolve1"
|
|
|
|
#define SYSTEMD_RESOLVED_MANAGER_INTERFACE "org.freedesktop.resolve1.Manager"
|
2019-07-03 01:14:52 +02:00
|
|
|
|
|
|
|
struct systemd_state {
|
|
|
|
uint32_t service_watch;
|
|
|
|
bool is_ready:1;
|
|
|
|
};
|
|
|
|
|
2019-07-03 01:14:53 +02:00
|
|
|
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;
|
|
|
|
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(uint32_t ifindex, uint8_t type,
|
|
|
|
char **dns_list, void *data)
|
2019-07-03 01:14:52 +02:00
|
|
|
{
|
|
|
|
struct systemd_state *state = data;
|
2019-07-03 01:14:53 +02:00
|
|
|
struct l_dbus_message_builder *builder;
|
|
|
|
struct l_dbus_message *message;
|
2019-07-03 01:14:52 +02:00
|
|
|
|
|
|
|
l_debug("ifindex: %u", ifindex);
|
|
|
|
|
|
|
|
if (!state->is_ready) {
|
|
|
|
l_error("resolve-systemd: Failed to add DNS entries. "
|
|
|
|
"Is 'systemd-resolved' service running?");
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-07-03 01:14:53 +02:00
|
|
|
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', &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,
|
|
|
|
state, NULL);
|
2019-07-03 01:14:52 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void resolve_systemd_remove(uint32_t ifindex, void *data)
|
|
|
|
{
|
|
|
|
struct systemd_state *state = data;
|
|
|
|
|
|
|
|
l_debug("ifindex: %u", ifindex);
|
|
|
|
|
|
|
|
if (!state->is_ready) {
|
|
|
|
l_error("resolve-systemd: Failed to remove DNS entries. "
|
|
|
|
"Is 'systemd-resolved' service running?");
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* TODO */
|
|
|
|
}
|
|
|
|
|
|
|
|
static void systemd_appeared(struct l_dbus *dbus, void *user_data)
|
|
|
|
{
|
|
|
|
struct systemd_state *state = user_data;
|
|
|
|
|
|
|
|
state->is_ready = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void systemd_disappeared(struct l_dbus *dbus, void *user_data)
|
|
|
|
{
|
|
|
|
struct systemd_state *state = user_data;
|
|
|
|
|
|
|
|
state->is_ready = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void *resolve_systemd_init(void)
|
|
|
|
{
|
|
|
|
struct systemd_state *state;
|
|
|
|
|
|
|
|
state = l_new(struct systemd_state, 1);
|
|
|
|
|
|
|
|
state->service_watch =
|
|
|
|
l_dbus_add_service_watch(dbus_get_bus(),
|
|
|
|
SYSTEMD_RESOLVED_SERVICE,
|
|
|
|
systemd_appeared,
|
|
|
|
systemd_disappeared,
|
|
|
|
state, NULL);
|
|
|
|
|
|
|
|
return state;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void resolve_systemd_exit(void *data)
|
|
|
|
{
|
|
|
|
struct systemd_state *state = data;
|
|
|
|
|
|
|
|
l_dbus_remove_watch(dbus_get_bus(), state->service_watch);
|
|
|
|
|
|
|
|
l_free(state);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct resolve_method_ops resolve_method_systemd = {
|
|
|
|
.init = resolve_systemd_init,
|
|
|
|
.exit = resolve_systemd_exit,
|
|
|
|
.add_dns = resolve_systemd_add_dns,
|
|
|
|
.remove = resolve_systemd_remove,
|
|
|
|
};
|
|
|
|
|
2019-07-03 01:14:50 +02:00
|
|
|
void resolve_add_dns(uint32_t ifindex, uint8_t type, char **dns_list)
|
|
|
|
{
|
2019-07-03 01:14:51 +02:00
|
|
|
if (!dns_list || !*dns_list)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!method.ops || !method.ops->add_dns)
|
|
|
|
return;
|
|
|
|
|
|
|
|
method.ops->add_dns(ifindex, type, dns_list, method.data);
|
2019-07-03 01:14:50 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void resolve_remove(uint32_t ifindex)
|
|
|
|
{
|
2019-07-03 01:14:51 +02:00
|
|
|
if (!method.ops || !method.ops->remove)
|
|
|
|
return;
|
|
|
|
|
|
|
|
method.ops->remove(ifindex, method.data);
|
2019-07-03 01:14:50 +02:00
|
|
|
}
|
|
|
|
|
2019-07-03 01:14:51 +02:00
|
|
|
static const struct {
|
|
|
|
const char *name;
|
|
|
|
const struct resolve_method_ops *method_ops;
|
|
|
|
} resolve_method_ops_list[] = {
|
2019-07-03 01:14:52 +02:00
|
|
|
{ "systemd", &resolve_method_systemd },
|
2019-07-03 01:14:51 +02:00
|
|
|
{ }
|
|
|
|
};
|
|
|
|
|
2019-07-03 01:14:50 +02:00
|
|
|
static int resolve_init(void)
|
|
|
|
{
|
2019-07-03 01:14:51 +02:00
|
|
|
const char *method_name;
|
2019-07-03 01:14:50 +02:00
|
|
|
bool enabled;
|
2019-07-03 01:14:51 +02:00
|
|
|
uint8_t i;
|
2019-07-03 01:14:50 +02:00
|
|
|
|
|
|
|
if (!l_settings_get_bool(iwd_get_config(), "General",
|
|
|
|
"enable_network_config", &enabled) ||
|
|
|
|
!enabled)
|
|
|
|
return 0;
|
|
|
|
|
2019-07-03 01:14:51 +02:00
|
|
|
method_name = l_settings_get_value(iwd_get_config(), "General",
|
|
|
|
"dns_resolve_method");
|
|
|
|
|
|
|
|
if (!method_name)
|
|
|
|
/* 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;
|
|
|
|
|
|
|
|
method.ops = resolve_method_ops_list[i].method_ops;
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!method.ops)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
if (method.ops->init)
|
|
|
|
method.data = method.ops->init();
|
|
|
|
|
2019-07-03 01:14:50 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void resolve_exit(void)
|
|
|
|
{
|
2019-07-03 19:42:09 +02:00
|
|
|
if (!method.ops || !method.ops->exit)
|
2019-07-03 01:14:51 +02:00
|
|
|
return;
|
|
|
|
|
|
|
|
method.ops->exit(method.data);
|
2019-07-03 01:14:50 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
IWD_MODULE(resolve, resolve_init, resolve_exit)
|