mirror of
				https://git.kernel.org/pub/scm/network/wireless/iwd.git
				synced 2025-11-04 08:57:29 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			360 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			360 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 *
 | 
						|
 *  Wireless daemon for Linux
 | 
						|
 *
 | 
						|
 *  Copyright (C) 2017  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 <ell/ell.h>
 | 
						|
 | 
						|
#include "dbus-proxy.h"
 | 
						|
#include "display.h"
 | 
						|
#include "network.h"
 | 
						|
 | 
						|
struct network {
 | 
						|
	bool connected;
 | 
						|
	char *identity;
 | 
						|
	char *name;
 | 
						|
	char *type;
 | 
						|
	const struct proxy_interface *device;
 | 
						|
};
 | 
						|
 | 
						|
static void check_errors_method_callback(struct l_dbus_message *message,
 | 
						|
								void *user_data)
 | 
						|
{
 | 
						|
	dbus_message_has_error(message);
 | 
						|
}
 | 
						|
 | 
						|
bool network_is_connected(const char *path)
 | 
						|
{
 | 
						|
	const struct network *network;
 | 
						|
	const struct proxy_interface *proxy =
 | 
						|
			proxy_interface_find(IWD_NETWORK_INTERFACE, path);
 | 
						|
 | 
						|
	network = proxy_interface_get_data(proxy);
 | 
						|
 | 
						|
	return network->connected;
 | 
						|
}
 | 
						|
 | 
						|
void network_connect(const struct proxy_interface *proxy)
 | 
						|
{
 | 
						|
	if (!proxy)
 | 
						|
		return;
 | 
						|
 | 
						|
	proxy_interface_method_call(proxy, "Connect", "",
 | 
						|
						check_errors_method_callback);
 | 
						|
}
 | 
						|
 | 
						|
static const char *skip_spaces(const char *p)
 | 
						|
{
 | 
						|
	while (*p == ' ')
 | 
						|
		p++;
 | 
						|
 | 
						|
	return p;
 | 
						|
}
 | 
						|
 | 
						|
struct network_args *network_parse_args(const char *args)
 | 
						|
{
 | 
						|
	struct network_args *network_args;
 | 
						|
	char **arg_arr;
 | 
						|
	size_t quoted_len = 0;
 | 
						|
	size_t i;
 | 
						|
	const char *p;
 | 
						|
 | 
						|
	if (unlikely(!args))
 | 
						|
		return NULL;
 | 
						|
 | 
						|
	if (args[0] == '\0')
 | 
						|
		return NULL;
 | 
						|
 | 
						|
	args = skip_spaces(args);
 | 
						|
	p = args;
 | 
						|
 | 
						|
	network_args = l_new(struct network_args, 1);
 | 
						|
 | 
						|
	if (*p == '"') {
 | 
						|
		for (++p, i = 0; *p; p++) {
 | 
						|
			i++;
 | 
						|
 | 
						|
			if (*p != '"')
 | 
						|
				continue;
 | 
						|
 | 
						|
			quoted_len = i;
 | 
						|
		}
 | 
						|
 | 
						|
		if (!quoted_len) {
 | 
						|
			p = args;
 | 
						|
			goto split;
 | 
						|
		}
 | 
						|
 | 
						|
		network_args->name = l_strndup(args + 1, quoted_len - 1);
 | 
						|
		p = args + quoted_len + 1;
 | 
						|
		p = skip_spaces(p);
 | 
						|
	}
 | 
						|
split:
 | 
						|
 | 
						|
	arg_arr = l_strsplit(p, ' ');
 | 
						|
	if (!arg_arr || !arg_arr[0])
 | 
						|
		goto done;
 | 
						|
 | 
						|
	if (quoted_len) {
 | 
						|
		network_args->type = l_strdup(arg_arr[0]);
 | 
						|
	} else {
 | 
						|
		network_args->name = l_strdup(arg_arr[0]);
 | 
						|
 | 
						|
		i = 1;
 | 
						|
 | 
						|
		while (arg_arr[i] && *arg_arr[i] == '\0')
 | 
						|
			i++;
 | 
						|
 | 
						|
		network_args->type = l_strdup(arg_arr[i]);
 | 
						|
	}
 | 
						|
 | 
						|
done:
 | 
						|
	l_strfreev(arg_arr);
 | 
						|
 | 
						|
	return network_args;
 | 
						|
}
 | 
						|
 | 
						|
void network_args_destroy(struct network_args *network_args)
 | 
						|
{
 | 
						|
	if (unlikely(!network_args))
 | 
						|
		return;
 | 
						|
 | 
						|
	l_free(network_args->name);
 | 
						|
	l_free(network_args->type);
 | 
						|
	l_free(network_args);
 | 
						|
}
 | 
						|
 | 
						|
static const char *get_name(const void *data)
 | 
						|
{
 | 
						|
	const struct network *network = data;
 | 
						|
 | 
						|
	return network->name;
 | 
						|
}
 | 
						|
 | 
						|
static void set_name(void *data, struct l_dbus_message_iter *variant)
 | 
						|
{
 | 
						|
	struct network *network = data;
 | 
						|
	const char *value;
 | 
						|
 | 
						|
	l_free(network->name);
 | 
						|
 | 
						|
	if (!l_dbus_message_iter_get_variant(variant, "s", &value)) {
 | 
						|
		network->name = NULL;
 | 
						|
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
	network->name = l_strdup(value);
 | 
						|
}
 | 
						|
 | 
						|
static void set_connected(void *data, struct l_dbus_message_iter *variant)
 | 
						|
{
 | 
						|
	struct network *network = data;
 | 
						|
	bool value;
 | 
						|
 | 
						|
	if (!l_dbus_message_iter_get_variant(variant, "b", &value)) {
 | 
						|
		network->connected = false;
 | 
						|
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
	network->connected = value;
 | 
						|
}
 | 
						|
 | 
						|
static void set_device(void *data, struct l_dbus_message_iter *variant)
 | 
						|
{
 | 
						|
	struct network *network = data;
 | 
						|
	const char *path;
 | 
						|
 | 
						|
	if (!l_dbus_message_iter_get_variant(variant, "o", &path)) {
 | 
						|
		network->device = NULL;
 | 
						|
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
	network->device = proxy_interface_find(IWD_DEVICE_INTERFACE, path);
 | 
						|
}
 | 
						|
 | 
						|
static void set_type(void *data, struct l_dbus_message_iter *variant)
 | 
						|
{
 | 
						|
	struct network *network = data;
 | 
						|
	const char *value;
 | 
						|
 | 
						|
	l_free(network->type);
 | 
						|
 | 
						|
	if (!l_dbus_message_iter_get_variant(variant, "s", &value)) {
 | 
						|
		network->type = NULL;
 | 
						|
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
	network->type = l_strdup(value);
 | 
						|
}
 | 
						|
 | 
						|
static const struct proxy_interface_property network_properties[] = {
 | 
						|
	{ "Name",       "s", set_name, get_name },
 | 
						|
	{ "Connected",  "b", set_connected},
 | 
						|
	{ "Device",     "o", set_device},
 | 
						|
	{ "Type",       "s", set_type},
 | 
						|
	{ }
 | 
						|
};
 | 
						|
 | 
						|
static const char *network_identity(void *data)
 | 
						|
{
 | 
						|
	struct network *network = data;
 | 
						|
 | 
						|
	if (!network->identity)
 | 
						|
		network->identity =
 | 
						|
			l_strdup_printf("%s %s", network->name, network->type);
 | 
						|
 | 
						|
	return network->identity;
 | 
						|
}
 | 
						|
 | 
						|
static void network_display_inline(const char *margin, const void *data)
 | 
						|
{
 | 
						|
	const struct network *network = data;
 | 
						|
 | 
						|
	display("%s%s %s %s\n", margin, network->name ? network->name : "",
 | 
						|
			network->type ? network->type : "",
 | 
						|
			network->connected ? "connected" : "diconnected");
 | 
						|
}
 | 
						|
 | 
						|
static void *network_create(void)
 | 
						|
{
 | 
						|
	return l_new(struct network, 1);
 | 
						|
}
 | 
						|
 | 
						|
static void network_destroy(void *data)
 | 
						|
{
 | 
						|
	struct network *network = data;
 | 
						|
 | 
						|
	l_free(network->name);
 | 
						|
	l_free(network->type);
 | 
						|
	l_free(network->identity);
 | 
						|
 | 
						|
	network->device = NULL;
 | 
						|
 | 
						|
	l_free(network);
 | 
						|
}
 | 
						|
 | 
						|
static const struct proxy_interface_type_ops ops = {
 | 
						|
	.create = network_create,
 | 
						|
	.destroy = network_destroy,
 | 
						|
	.display = network_display_inline,
 | 
						|
	.identity = network_identity,
 | 
						|
};
 | 
						|
 | 
						|
static struct proxy_interface_type network_interface_type = {
 | 
						|
	.interface = IWD_NETWORK_INTERFACE,
 | 
						|
	.properties = network_properties,
 | 
						|
	.ops = &ops,
 | 
						|
};
 | 
						|
 | 
						|
struct completion_search_parameters {
 | 
						|
	const char *text;
 | 
						|
	const struct proxy_interface *device;
 | 
						|
};
 | 
						|
 | 
						|
static bool match_by_partial_name(const void *a, const void *b)
 | 
						|
{
 | 
						|
	const struct network *network = a;
 | 
						|
	const struct completion_search_parameters *params = b;
 | 
						|
	const char *name;
 | 
						|
	const char *text;
 | 
						|
 | 
						|
	if (!proxy_interface_is_same(network->device, params->device))
 | 
						|
		return false;
 | 
						|
 | 
						|
	for (text = params->text, name = network->name; *text && *name;
 | 
						|
							name++, text++) {
 | 
						|
		if (*name == *text)
 | 
						|
			continue;
 | 
						|
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
char *network_name_completion(const struct proxy_interface *device,
 | 
						|
						const char *text, int state)
 | 
						|
{
 | 
						|
	const struct completion_search_parameters params = {
 | 
						|
		.text = text, .device = device,
 | 
						|
	};
 | 
						|
 | 
						|
	return proxy_property_str_completion(&network_interface_type,
 | 
						|
						match_by_partial_name, "Name",
 | 
						|
						¶ms, state);
 | 
						|
}
 | 
						|
 | 
						|
struct network_search_parameters {
 | 
						|
	const struct network_args *args;
 | 
						|
	const struct proxy_interface *device;
 | 
						|
};
 | 
						|
 | 
						|
static bool match_by_device_and_args(const void *a, const void *b)
 | 
						|
{
 | 
						|
	const struct network *network = a;
 | 
						|
	const struct network_search_parameters *params = b;
 | 
						|
 | 
						|
	if (!proxy_interface_is_same(network->device, params->device))
 | 
						|
		return false;
 | 
						|
 | 
						|
	if (strcmp(network->name, params->args->name))
 | 
						|
		return false;
 | 
						|
 | 
						|
	if (params->args->type && strcmp(network->type, params->args->type))
 | 
						|
		return false;
 | 
						|
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
struct l_queue *network_match_by_device_and_args(
 | 
						|
					const struct proxy_interface *device,
 | 
						|
					const struct network_args *args)
 | 
						|
{
 | 
						|
	struct network_search_parameters params = {
 | 
						|
		.args = args, .device = device
 | 
						|
	};
 | 
						|
 | 
						|
	return proxy_interface_find_all(network_interface_type.interface,
 | 
						|
					match_by_device_and_args, ¶ms);
 | 
						|
}
 | 
						|
 | 
						|
static int network_interface_init(void)
 | 
						|
{
 | 
						|
	proxy_interface_type_register(&network_interface_type);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void network_interface_exit(void)
 | 
						|
{
 | 
						|
	proxy_interface_type_unregister(&network_interface_type);
 | 
						|
}
 | 
						|
 | 
						|
INTERFACE_TYPE(network_interface_type, network_interface_init,
 | 
						|
						network_interface_exit)
 |