2017-03-23 17:48:53 +01:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Wireless daemon for Linux
|
|
|
|
*
|
2020-02-27 00:32:24 +01:00
|
|
|
* Copyright (C) 2017-2020 Intel Corporation. All rights reserved.
|
2017-03-23 17:48:53 +01:00
|
|
|
*
|
|
|
|
* 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
|
|
|
|
|
2017-04-13 18:56:13 +02:00
|
|
|
#include <stdio.h>
|
2017-03-23 17:48:53 +01:00
|
|
|
#include <ell/ell.h>
|
|
|
|
|
2019-12-13 09:02:42 +01:00
|
|
|
#include "client/agent-manager.h"
|
|
|
|
#include "client/dbus-proxy.h"
|
|
|
|
#include "client/display.h"
|
|
|
|
#include "client/command.h"
|
|
|
|
#include "client/properties.h"
|
2017-03-23 17:48:53 +01:00
|
|
|
|
2017-03-24 00:32:09 +01:00
|
|
|
#define IWD_SERVICE "net.connman.iwd"
|
|
|
|
#define IWD_ROOT_PATH "/"
|
|
|
|
|
2017-04-07 23:55:31 +02:00
|
|
|
struct proxy_interface {
|
|
|
|
void *data;
|
|
|
|
char *path;
|
|
|
|
const struct proxy_interface_type *type;
|
|
|
|
};
|
|
|
|
|
2017-03-24 00:32:09 +01:00
|
|
|
static struct l_dbus *dbus;
|
|
|
|
|
|
|
|
static struct l_queue *proxy_interfaces;
|
|
|
|
static struct l_queue *proxy_interface_types;
|
|
|
|
|
2024-08-12 17:46:08 +02:00
|
|
|
void proxy_properties_display_inline(const struct proxy_interface *proxy,
|
|
|
|
const char *margin,
|
|
|
|
unsigned int name_column_width,
|
|
|
|
unsigned int value_column_width)
|
2017-04-27 01:08:41 +02:00
|
|
|
{
|
|
|
|
const void *data;
|
|
|
|
const struct proxy_interface_property *properties;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
if (!proxy->type->properties)
|
|
|
|
return;
|
|
|
|
|
|
|
|
data = proxy_interface_get_data(proxy);
|
|
|
|
properties = proxy->type->properties;
|
|
|
|
|
|
|
|
for (i = 0; properties[i].name; i++) {
|
2022-06-29 21:27:08 +02:00
|
|
|
const char *str;
|
|
|
|
|
2017-04-27 01:08:41 +02:00
|
|
|
if (!properties[i].tostr)
|
|
|
|
continue;
|
|
|
|
|
2022-06-29 21:27:08 +02:00
|
|
|
str = properties[i].tostr(data);
|
|
|
|
|
2022-07-07 19:55:00 +02:00
|
|
|
display_table_row(MARGIN, 3, 8, properties[i].is_read_write ?
|
2022-07-07 19:55:05 +02:00
|
|
|
COLOR_BOLDGRAY(" *") : "",
|
2022-07-07 19:55:00 +02:00
|
|
|
name_column_width, properties[i].name,
|
|
|
|
value_column_width, str ? : "");
|
2017-04-27 01:08:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-12 17:46:08 +02:00
|
|
|
void proxy_properties_display_header(const char *caption, const char *margin,
|
|
|
|
unsigned int name_column_width,
|
|
|
|
unsigned int value_column_width)
|
|
|
|
{
|
|
|
|
display_table_header(caption, "%s%-*s %-*s %-*s", margin,
|
|
|
|
8, "Settable",
|
|
|
|
name_column_width, "Property",
|
|
|
|
value_column_width, "Value");
|
|
|
|
}
|
|
|
|
|
|
|
|
void proxy_properties_display(const struct proxy_interface *proxy,
|
|
|
|
const char *caption, const char *margin,
|
|
|
|
unsigned int name_column_width,
|
|
|
|
unsigned int value_column_width)
|
|
|
|
{
|
|
|
|
if (!proxy->type->properties)
|
|
|
|
return;
|
|
|
|
|
|
|
|
proxy_properties_display_header(caption, margin, name_column_width,
|
|
|
|
value_column_width);
|
|
|
|
|
|
|
|
proxy_properties_display_inline(proxy, margin, name_column_width,
|
|
|
|
value_column_width);
|
|
|
|
}
|
|
|
|
|
2017-04-27 01:08:37 +02:00
|
|
|
static const void *proxy_interface_property_tostr(
|
|
|
|
const struct proxy_interface *proxy,
|
|
|
|
const char *name)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
const struct proxy_interface_property *property_table =
|
|
|
|
proxy->type->properties;
|
|
|
|
|
|
|
|
for (i = 0; property_table[i].name; i++) {
|
|
|
|
if (strcmp(property_table[i].name, name))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (!property_table[i].tostr)
|
|
|
|
break;
|
|
|
|
|
|
|
|
return property_table[i].tostr(proxy->data);
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2018-07-25 01:17:17 +02:00
|
|
|
static void proxy_interface_property_update(struct proxy_interface *proxy,
|
2017-04-07 23:55:34 +02:00
|
|
|
const char *name,
|
|
|
|
struct l_dbus_message_iter *variant)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
const struct proxy_interface_property *property_table =
|
|
|
|
proxy->type->properties;
|
|
|
|
|
2022-06-28 23:57:39 +02:00
|
|
|
if (!property_table)
|
|
|
|
return;
|
|
|
|
|
2017-04-07 23:55:34 +02:00
|
|
|
for (i = 0; property_table[i].name; i++) {
|
|
|
|
if (strcmp(property_table[i].name, name))
|
|
|
|
continue;
|
|
|
|
|
2018-07-25 01:17:17 +02:00
|
|
|
if (!property_table[i].update)
|
2017-04-07 23:55:34 +02:00
|
|
|
return;
|
|
|
|
|
2018-07-25 01:17:17 +02:00
|
|
|
property_table[i].update(proxy->data, variant);
|
2017-04-07 23:55:34 +02:00
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
l_debug("Unknown property name: %s for interface %s", name,
|
|
|
|
proxy->type->interface);
|
|
|
|
}
|
|
|
|
|
2017-04-07 23:55:32 +02:00
|
|
|
static void interface_update_properties(struct proxy_interface *proxy,
|
|
|
|
struct l_dbus_message_iter *changed,
|
|
|
|
struct l_dbus_message_iter *invalidated)
|
|
|
|
{
|
2017-04-07 23:55:34 +02:00
|
|
|
const char *name;
|
|
|
|
struct l_dbus_message_iter variant;
|
|
|
|
|
|
|
|
while (l_dbus_message_iter_next_entry(changed, &name, &variant))
|
2018-07-25 01:17:17 +02:00
|
|
|
proxy_interface_property_update(proxy, name, &variant);
|
2017-04-07 23:55:34 +02:00
|
|
|
|
|
|
|
if (!invalidated)
|
|
|
|
return;
|
|
|
|
|
|
|
|
while (l_dbus_message_iter_next_entry(invalidated, &name))
|
2018-07-25 01:17:17 +02:00
|
|
|
proxy_interface_property_update(proxy, name, NULL);
|
2017-04-07 23:55:32 +02:00
|
|
|
}
|
|
|
|
|
2019-07-24 01:21:35 +02:00
|
|
|
static char *strdup_quoted_for_spaces(const char *str)
|
|
|
|
{
|
|
|
|
const char *p;
|
|
|
|
|
|
|
|
for (p = str; *p; p++) {
|
|
|
|
if (*p != ' ')
|
|
|
|
continue;
|
|
|
|
|
|
|
|
return l_strdup_printf("\"%s\"", str);
|
|
|
|
}
|
|
|
|
|
|
|
|
return l_strdup(str);
|
|
|
|
}
|
|
|
|
|
2017-04-27 01:08:37 +02:00
|
|
|
char *proxy_property_str_completion(const struct proxy_interface_type *type,
|
|
|
|
proxy_property_match_func_t function,
|
|
|
|
const char *property_name,
|
2018-09-14 10:45:14 +02:00
|
|
|
const void *value, int state,
|
|
|
|
const char *extra_interface)
|
2017-04-27 01:08:37 +02:00
|
|
|
{
|
|
|
|
static struct l_queue *match;
|
|
|
|
static const struct l_queue_entry *entry;
|
|
|
|
|
|
|
|
if (!state) {
|
|
|
|
match = proxy_interface_find_all(type->interface, function,
|
|
|
|
value);
|
|
|
|
if (!match)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
entry = l_queue_get_entries(match);
|
|
|
|
}
|
|
|
|
|
|
|
|
while (entry) {
|
|
|
|
const struct proxy_interface *proxy = entry->data;
|
|
|
|
const char *str;
|
|
|
|
|
|
|
|
entry = entry->next;
|
|
|
|
|
2018-09-14 10:45:14 +02:00
|
|
|
if (extra_interface) {
|
|
|
|
const char *path = proxy_interface_get_path(proxy);
|
|
|
|
const struct proxy_interface *extra =
|
|
|
|
proxy_interface_find(extra_interface, path);
|
|
|
|
|
|
|
|
if (!extra)
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-04-27 01:08:37 +02:00
|
|
|
str = proxy_interface_property_tostr(proxy, property_name);
|
|
|
|
if (!str)
|
2019-04-05 19:47:00 +02:00
|
|
|
goto done;
|
2017-04-27 01:08:37 +02:00
|
|
|
|
2019-07-24 01:21:35 +02:00
|
|
|
return strdup_quoted_for_spaces(str);
|
2017-04-27 01:08:37 +02:00
|
|
|
}
|
2019-04-05 19:47:00 +02:00
|
|
|
done:
|
2017-04-27 01:08:37 +02:00
|
|
|
l_queue_destroy(match, NULL);
|
|
|
|
match = NULL;
|
|
|
|
entry = NULL;
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2019-04-11 00:58:19 +02:00
|
|
|
static char *proxy_property_completion_value_options(const char **options,
|
|
|
|
const char *text,
|
|
|
|
int state)
|
2018-07-25 21:54:42 +02:00
|
|
|
{
|
|
|
|
static int index;
|
|
|
|
static int len;
|
|
|
|
const char *opt;
|
|
|
|
|
|
|
|
if (!state) {
|
|
|
|
index = 0;
|
|
|
|
len = strlen(text);
|
|
|
|
}
|
|
|
|
|
2019-04-11 00:58:19 +02:00
|
|
|
while ((opt = options[index++])) {
|
2018-07-25 21:54:42 +02:00
|
|
|
if (strncmp(opt, text, len))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
return l_strdup(opt);
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
char *proxy_property_completion(
|
|
|
|
const struct proxy_interface_property *properties,
|
|
|
|
const char *text, int state)
|
|
|
|
{
|
|
|
|
static size_t i;
|
|
|
|
static size_t j;
|
|
|
|
static size_t len;
|
|
|
|
static bool first_pass;
|
|
|
|
const char *name;
|
|
|
|
|
|
|
|
if (!state) {
|
|
|
|
j = 0;
|
|
|
|
first_pass = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (first_pass && (name = properties[j].name)) {
|
|
|
|
if (!properties[j].is_read_write)
|
|
|
|
goto next;
|
|
|
|
|
|
|
|
if (!command_line_find_token(name, 2))
|
|
|
|
goto next;
|
|
|
|
|
|
|
|
if (!properties[j].options)
|
|
|
|
goto next;
|
|
|
|
|
|
|
|
return proxy_property_completion_value_options(
|
|
|
|
properties[j].options, text,
|
|
|
|
state);
|
|
|
|
next:
|
|
|
|
j++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (first_pass) {
|
|
|
|
i = 0;
|
|
|
|
first_pass = false;
|
|
|
|
len = strlen(text);
|
|
|
|
}
|
|
|
|
|
|
|
|
while ((name = properties[i].name)) {
|
|
|
|
if (!properties[i++].is_read_write)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (strncmp(name, text, len))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
return l_strdup(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2018-07-25 21:54:39 +02:00
|
|
|
static const struct proxy_interface_property *proxy_property_find(
|
|
|
|
const struct proxy_interface_property *types,
|
|
|
|
const char *name)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; types[i].name; i++) {
|
|
|
|
if (strcmp(types[i].name, name))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
return &types[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2018-08-03 23:35:40 +02:00
|
|
|
struct proxy_callback_data {
|
|
|
|
l_dbus_message_func_t callback;
|
|
|
|
void *user_data;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void proxy_callback(struct l_dbus_message *message, void *user_data)
|
|
|
|
{
|
|
|
|
struct proxy_callback_data *callback_data = user_data;
|
2019-08-15 20:40:42 +02:00
|
|
|
const struct proxy_interface *proxy;
|
2018-08-03 23:35:40 +02:00
|
|
|
const char *name;
|
|
|
|
const char *text;
|
|
|
|
|
|
|
|
if (callback_data->callback)
|
|
|
|
callback_data->callback(message, callback_data->user_data);
|
|
|
|
|
|
|
|
if (command_is_interactive_mode())
|
|
|
|
return;
|
|
|
|
|
2019-08-15 20:40:42 +02:00
|
|
|
if (l_dbus_message_get_error(message, &name, &text)) {
|
2018-08-03 23:35:40 +02:00
|
|
|
command_set_exit_status(EXIT_FAILURE);
|
2019-08-15 20:40:42 +02:00
|
|
|
goto quit;
|
|
|
|
}
|
|
|
|
|
|
|
|
proxy = callback_data->user_data;
|
2021-11-02 20:57:49 +01:00
|
|
|
if (!strcmp(proxy->type->interface, IWD_AGENT_MANAGER_INTERFACE) ||
|
|
|
|
!strcmp(proxy->type->interface, IWD_DAEMON_INTERFACE))
|
2019-08-15 20:40:42 +02:00
|
|
|
return;
|
2018-08-03 23:35:40 +02:00
|
|
|
|
2019-08-15 20:40:42 +02:00
|
|
|
quit:
|
2018-08-03 23:35:40 +02:00
|
|
|
l_main_quit();
|
|
|
|
}
|
|
|
|
|
2018-07-25 21:54:39 +02:00
|
|
|
bool proxy_property_set(const struct proxy_interface *proxy, const char *name,
|
|
|
|
const char *value_str, l_dbus_message_func_t callback)
|
|
|
|
{
|
|
|
|
struct l_dbus_message_builder *builder;
|
|
|
|
struct l_dbus_message *msg;
|
|
|
|
const struct proxy_interface_property *property;
|
2018-08-03 23:35:40 +02:00
|
|
|
struct proxy_callback_data *callback_data;
|
2018-07-25 21:54:39 +02:00
|
|
|
|
|
|
|
if (!proxy || !name)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
property = proxy_property_find(proxy->type->properties, name);
|
|
|
|
if (!property)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (!property->is_read_write)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (!property->append)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
msg = l_dbus_message_new_method_call(dbus, IWD_SERVICE, proxy->path,
|
|
|
|
L_DBUS_INTERFACE_PROPERTIES,
|
|
|
|
"Set");
|
|
|
|
if (!msg)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
|
|
builder = l_dbus_message_builder_new(msg);
|
|
|
|
if (!builder) {
|
|
|
|
l_dbus_message_unref(msg);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
l_dbus_message_builder_append_basic(builder, 's',
|
|
|
|
proxy->type->interface);
|
|
|
|
l_dbus_message_builder_append_basic(builder, 's', property->name);
|
|
|
|
l_dbus_message_builder_enter_variant(builder, property->type);
|
|
|
|
|
|
|
|
if (!property->append(builder, value_str)) {
|
2018-08-03 03:01:52 +02:00
|
|
|
l_dbus_message_builder_destroy(builder);
|
2018-07-25 21:54:39 +02:00
|
|
|
l_dbus_message_unref(msg);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
l_dbus_message_builder_leave_variant(builder);
|
|
|
|
l_dbus_message_builder_finalize(builder);
|
|
|
|
l_dbus_message_builder_destroy(builder);
|
|
|
|
|
2018-08-03 23:35:40 +02:00
|
|
|
callback_data = l_new(struct proxy_callback_data, 1);
|
|
|
|
callback_data->callback = callback;
|
|
|
|
callback_data->user_data = (void *) proxy;
|
|
|
|
|
|
|
|
l_dbus_send_with_reply(dbus, msg, proxy_callback, callback_data,
|
|
|
|
l_free);
|
2018-07-25 21:54:39 +02:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-04-19 00:04:15 +02:00
|
|
|
bool dbus_message_has_error(struct l_dbus_message *message)
|
2017-04-06 20:59:37 +02:00
|
|
|
{
|
|
|
|
const char *name;
|
|
|
|
const char *text;
|
|
|
|
|
|
|
|
if (l_dbus_message_get_error(message, &name, &text)) {
|
|
|
|
display_error(text);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-04-07 23:55:32 +02:00
|
|
|
static bool interface_match_by_type_name(const void *a, const void *b)
|
|
|
|
{
|
|
|
|
const struct proxy_interface_type *type = a;
|
|
|
|
const char *interface = b;
|
|
|
|
|
|
|
|
return !strcmp(type->interface, interface);
|
|
|
|
}
|
|
|
|
|
2017-04-07 23:55:31 +02:00
|
|
|
struct proxy_interface *proxy_interface_find(const char *interface,
|
|
|
|
const char *path)
|
|
|
|
{
|
|
|
|
const struct l_queue_entry *entry;
|
|
|
|
|
|
|
|
if (!interface || !path)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
for (entry = l_queue_get_entries(proxy_interfaces); entry;
|
|
|
|
entry = entry->next) {
|
|
|
|
struct proxy_interface *proxy = entry->data;
|
|
|
|
|
|
|
|
if (strcmp(proxy->path, path))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (strcmp(proxy->type->interface, interface))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
return proxy;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2017-04-17 18:50:10 +02:00
|
|
|
struct l_queue *proxy_interface_find_all(const char *interface,
|
|
|
|
proxy_property_match_func_t function,
|
|
|
|
const void *value)
|
|
|
|
{
|
|
|
|
const struct l_queue_entry *entry;
|
|
|
|
struct l_queue *match = NULL;
|
|
|
|
|
2018-09-14 09:55:49 +02:00
|
|
|
if (!interface)
|
2017-04-17 18:50:10 +02:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
for (entry = l_queue_get_entries(proxy_interfaces); entry;
|
|
|
|
entry = entry->next) {
|
|
|
|
struct proxy_interface *proxy = entry->data;
|
|
|
|
|
|
|
|
if (!interface_match_by_type_name(proxy->type, interface))
|
|
|
|
continue;
|
|
|
|
|
2018-09-14 09:55:49 +02:00
|
|
|
if (function && !function(proxy->data, value))
|
2017-04-17 18:50:10 +02:00
|
|
|
continue;
|
|
|
|
|
|
|
|
if (!match)
|
|
|
|
match = l_queue_new();
|
|
|
|
|
|
|
|
l_queue_push_tail(match, proxy);
|
|
|
|
}
|
|
|
|
|
|
|
|
return match;
|
|
|
|
}
|
|
|
|
|
2018-06-01 00:01:31 +02:00
|
|
|
bool proxy_interface_is_same(const struct proxy_interface *a,
|
|
|
|
const struct proxy_interface *b)
|
|
|
|
{
|
|
|
|
return !strcmp(a->path, b->path);
|
|
|
|
}
|
|
|
|
|
2017-05-01 19:58:24 +02:00
|
|
|
static void properties_changed_callback(struct l_dbus_message *message,
|
|
|
|
void *data)
|
|
|
|
{
|
|
|
|
struct proxy_interface *proxy;
|
|
|
|
const char *path;
|
|
|
|
const char *interface;
|
|
|
|
struct l_dbus_message_iter changed;
|
|
|
|
struct l_dbus_message_iter invalidated;
|
|
|
|
|
|
|
|
if (dbus_message_has_error(message))
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!l_dbus_message_get_arguments(message, "sa{sv}as", &interface,
|
|
|
|
&changed, &invalidated)) {
|
|
|
|
l_debug("Failed to parse properties changed callback message");
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
path = l_dbus_message_get_path(message);
|
|
|
|
if (!path)
|
|
|
|
return;
|
|
|
|
|
|
|
|
proxy = proxy_interface_find(interface, path);
|
|
|
|
if (!proxy)
|
|
|
|
return;
|
|
|
|
|
|
|
|
interface_update_properties(proxy, &changed, &invalidated);
|
|
|
|
}
|
|
|
|
|
2017-04-07 23:55:32 +02:00
|
|
|
static bool is_ignorable(const char *interface)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
static const struct {
|
|
|
|
const char *interface;
|
|
|
|
} interfaces_to_ignore[] = {
|
|
|
|
{ L_DBUS_INTERFACE_OBJECT_MANAGER },
|
|
|
|
{ L_DBUS_INTERFACE_INTROSPECTABLE },
|
2017-05-01 22:52:53 +02:00
|
|
|
{ L_DBUS_INTERFACE_PROPERTIES },
|
2017-04-07 23:55:32 +02:00
|
|
|
{ }
|
|
|
|
};
|
|
|
|
|
|
|
|
for (i = 0; interfaces_to_ignore[i].interface; i++)
|
|
|
|
if (!strcmp(interfaces_to_ignore[i].interface, interface))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-02-07 22:11:22 +01:00
|
|
|
static void proxy_interfaces_update_properties(const char *path,
|
|
|
|
struct l_dbus_message_iter *interfaces)
|
|
|
|
{
|
|
|
|
const char *interface;
|
|
|
|
struct l_dbus_message_iter properties;
|
|
|
|
struct proxy_interface *proxy;
|
|
|
|
struct proxy_interface_type *interface_type;
|
|
|
|
|
|
|
|
if (!path)
|
|
|
|
return;
|
|
|
|
|
|
|
|
while (l_dbus_message_iter_next_entry(interfaces, &interface,
|
|
|
|
&properties)) {
|
|
|
|
interface_type = l_queue_find(proxy_interface_types,
|
|
|
|
interface_match_by_type_name,
|
|
|
|
interface);
|
|
|
|
if (!interface_type)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
proxy = proxy_interface_find(interface_type->interface, path);
|
|
|
|
if (!proxy)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
interface_update_properties(proxy, &properties, NULL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-06 20:59:37 +02:00
|
|
|
static void proxy_interface_create(const char *path,
|
|
|
|
struct l_dbus_message_iter *interfaces)
|
|
|
|
{
|
2017-04-07 23:55:32 +02:00
|
|
|
const char *interface;
|
|
|
|
struct l_dbus_message_iter properties;
|
|
|
|
struct proxy_interface *proxy;
|
|
|
|
struct proxy_interface_type *interface_type;
|
|
|
|
|
|
|
|
if (!path)
|
|
|
|
return;
|
|
|
|
|
|
|
|
while (l_dbus_message_iter_next_entry(interfaces, &interface,
|
|
|
|
&properties)) {
|
|
|
|
interface_type = l_queue_find(proxy_interface_types,
|
|
|
|
interface_match_by_type_name,
|
|
|
|
interface);
|
|
|
|
|
|
|
|
if (!interface_type) {
|
|
|
|
if (!is_ignorable(interface))
|
|
|
|
l_debug("Unknown DBus interface type %s",
|
|
|
|
interface);
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
proxy = proxy_interface_find(interface_type->interface, path);
|
|
|
|
|
2020-02-07 22:11:22 +01:00
|
|
|
if (proxy)
|
2017-04-07 23:55:32 +02:00
|
|
|
continue;
|
|
|
|
|
|
|
|
proxy = l_new(struct proxy_interface, 1);
|
|
|
|
proxy->path = l_strdup(path);
|
|
|
|
proxy->type = interface_type;
|
|
|
|
|
2020-02-27 00:32:23 +01:00
|
|
|
l_queue_push_tail(proxy_interfaces, proxy);
|
|
|
|
|
2020-02-07 22:11:22 +01:00
|
|
|
if (interface_type->ops && interface_type->ops->create)
|
2017-04-07 23:55:32 +02:00
|
|
|
proxy->data = interface_type->ops->create();
|
|
|
|
}
|
2017-04-06 20:59:37 +02:00
|
|
|
}
|
|
|
|
|
2017-04-07 23:55:33 +02:00
|
|
|
static void proxy_interface_destroy(void *data)
|
|
|
|
{
|
|
|
|
struct proxy_interface *proxy = data;
|
|
|
|
|
|
|
|
l_free(proxy->path);
|
|
|
|
|
2017-04-27 01:08:43 +02:00
|
|
|
if (proxy->type->ops && proxy->type->ops->destroy)
|
2017-04-07 23:55:33 +02:00
|
|
|
proxy->type->ops->destroy(proxy->data);
|
|
|
|
|
|
|
|
proxy->type = NULL;
|
|
|
|
|
|
|
|
l_free(proxy);
|
|
|
|
}
|
|
|
|
|
2017-04-18 23:42:45 +02:00
|
|
|
bool proxy_interface_method_call(const struct proxy_interface *proxy,
|
|
|
|
const char *name, const char *signature,
|
|
|
|
l_dbus_message_func_t callback, ...)
|
|
|
|
{
|
2018-08-03 23:35:40 +02:00
|
|
|
struct proxy_callback_data *callback_data;
|
2017-04-18 23:42:45 +02:00
|
|
|
struct l_dbus_message *call;
|
|
|
|
va_list args;
|
|
|
|
|
|
|
|
if (!proxy || !name)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
call = l_dbus_message_new_method_call(dbus, IWD_SERVICE, proxy->path,
|
|
|
|
proxy->type->interface, name);
|
|
|
|
|
|
|
|
va_start(args, callback);
|
|
|
|
l_dbus_message_set_arguments_valist(call, signature, args);
|
|
|
|
va_end(args);
|
|
|
|
|
2018-08-03 23:35:40 +02:00
|
|
|
callback_data = l_new(struct proxy_callback_data, 1);
|
|
|
|
callback_data->callback = callback;
|
|
|
|
callback_data->user_data = (void *) proxy;
|
|
|
|
|
|
|
|
l_dbus_send_with_reply(dbus, call, proxy_callback, callback_data,
|
|
|
|
l_free);
|
2017-04-18 23:42:45 +02:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-04-14 19:52:42 +02:00
|
|
|
void *proxy_interface_get_data(const struct proxy_interface *proxy)
|
|
|
|
{
|
|
|
|
return proxy->data;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *proxy_interface_get_interface(const struct proxy_interface *proxy)
|
|
|
|
{
|
|
|
|
return proxy->type->interface;
|
|
|
|
}
|
|
|
|
|
2018-09-14 09:53:36 +02:00
|
|
|
const char *proxy_interface_get_path(const struct proxy_interface *proxy)
|
|
|
|
{
|
|
|
|
return proxy->path;
|
|
|
|
}
|
|
|
|
|
2017-04-14 19:52:42 +02:00
|
|
|
const char *proxy_interface_get_identity_str(
|
|
|
|
const struct proxy_interface *proxy)
|
|
|
|
{
|
|
|
|
if (proxy->type->ops && proxy->type->ops->identity)
|
|
|
|
return proxy->type->ops->identity(proxy->data);
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2017-04-19 00:04:15 +02:00
|
|
|
void proxy_interface_display_list(const char *interface)
|
|
|
|
{
|
|
|
|
const struct l_queue_entry *entry;
|
|
|
|
|
|
|
|
for (entry = l_queue_get_entries(proxy_interfaces); entry;
|
|
|
|
entry = entry->next) {
|
|
|
|
const struct proxy_interface *proxy = entry->data;
|
|
|
|
|
|
|
|
if (!interface_match_by_type_name(proxy->type, interface))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (!proxy->type->ops || !proxy->type->ops->display)
|
|
|
|
break;
|
|
|
|
|
|
|
|
proxy->type->ops->display(MARGIN, proxy->data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-24 00:32:09 +01:00
|
|
|
static void interfaces_added_callback(struct l_dbus_message *message,
|
|
|
|
void *user_data)
|
|
|
|
{
|
2017-04-07 23:55:31 +02:00
|
|
|
const char *path;
|
|
|
|
struct l_dbus_message_iter object;
|
|
|
|
|
|
|
|
if (dbus_message_has_error(message))
|
|
|
|
return;
|
|
|
|
|
2019-10-23 22:23:57 +02:00
|
|
|
if (!l_dbus_message_get_arguments(message, "oa{sa{sv}}", &path,
|
|
|
|
&object))
|
|
|
|
return;
|
2017-04-07 23:55:31 +02:00
|
|
|
|
|
|
|
proxy_interface_create(path, &object);
|
2020-02-07 22:11:22 +01:00
|
|
|
|
|
|
|
if (!l_dbus_message_get_arguments(message, "oa{sa{sv}}", &path,
|
|
|
|
&object))
|
|
|
|
return;
|
|
|
|
|
|
|
|
proxy_interfaces_update_properties(path, &object);
|
2017-03-24 00:32:09 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void interfaces_removed_callback(struct l_dbus_message *message,
|
|
|
|
void *user_data)
|
|
|
|
{
|
2017-04-07 23:55:31 +02:00
|
|
|
const char *interface;
|
|
|
|
const char *path;
|
|
|
|
struct l_dbus_message_iter interfaces;
|
|
|
|
struct proxy_interface *proxy;
|
|
|
|
|
|
|
|
if (dbus_message_has_error(message))
|
|
|
|
return;
|
|
|
|
|
2019-10-23 22:23:57 +02:00
|
|
|
if (!l_dbus_message_get_arguments(message, "oas", &path, &interfaces))
|
|
|
|
return;
|
2017-04-07 23:55:31 +02:00
|
|
|
|
|
|
|
while (l_dbus_message_iter_next_entry(&interfaces, &interface)) {
|
|
|
|
proxy = proxy_interface_find(interface, path);
|
|
|
|
|
|
|
|
if (!proxy)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
l_queue_remove(proxy_interfaces, proxy);
|
2017-05-12 21:15:37 +02:00
|
|
|
|
|
|
|
proxy_interface_destroy(proxy);
|
2017-04-07 23:55:31 +02:00
|
|
|
}
|
2017-03-24 00:32:09 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void get_managed_objects_callback(struct l_dbus_message *message,
|
|
|
|
void *user_data)
|
|
|
|
{
|
2017-04-06 20:59:37 +02:00
|
|
|
struct l_dbus_message_iter objects;
|
|
|
|
struct l_dbus_message_iter object;
|
|
|
|
const char *path;
|
|
|
|
|
|
|
|
if (dbus_message_has_error(message)) {
|
2020-02-07 22:11:22 +01:00
|
|
|
display_error("Failed to retrieve IWD dbus objects, "
|
|
|
|
"quitting...\n");
|
2018-08-03 23:35:40 +02:00
|
|
|
|
2020-02-07 22:11:22 +01:00
|
|
|
goto error;
|
2017-04-06 20:59:37 +02:00
|
|
|
}
|
|
|
|
|
2019-10-23 22:23:56 +02:00
|
|
|
if (!l_dbus_message_get_arguments(message, "a{oa{sa{sv}}}", &objects)) {
|
2020-02-07 22:11:22 +01:00
|
|
|
display_error("Failed to parse IWD dbus objects, "
|
|
|
|
"quitting...\n");
|
2019-10-23 22:23:56 +02:00
|
|
|
|
2020-02-07 22:11:22 +01:00
|
|
|
goto error;
|
2019-10-23 22:23:56 +02:00
|
|
|
}
|
2017-04-06 20:59:37 +02:00
|
|
|
|
|
|
|
while (l_dbus_message_iter_next_entry(&objects, &path, &object))
|
|
|
|
proxy_interface_create(path, &object);
|
|
|
|
|
2020-02-07 22:11:22 +01:00
|
|
|
if (!l_dbus_message_get_arguments(message, "a{oa{sa{sv}}}", &objects))
|
|
|
|
/*
|
|
|
|
* Shouldn't happen since we parsed it above, but check return
|
|
|
|
* anyway.
|
|
|
|
*/
|
|
|
|
goto error;
|
2019-08-31 01:46:12 +02:00
|
|
|
|
2020-02-07 22:11:22 +01:00
|
|
|
while (l_dbus_message_iter_next_entry(&objects, &path, &object))
|
|
|
|
proxy_interfaces_update_properties(path, &object);
|
2019-08-31 01:46:12 +02:00
|
|
|
|
2020-02-07 22:11:22 +01:00
|
|
|
if (command_is_interactive_mode())
|
|
|
|
display_enable_cmd_prompt();
|
|
|
|
else
|
2019-08-15 20:40:40 +02:00
|
|
|
command_noninteractive_trigger();
|
2018-05-03 22:57:50 +02:00
|
|
|
|
2020-02-07 22:11:22 +01:00
|
|
|
return;
|
2018-05-03 22:57:50 +02:00
|
|
|
|
2020-02-07 22:11:22 +01:00
|
|
|
error:
|
|
|
|
if (!command_is_interactive_mode())
|
|
|
|
command_set_exit_status(EXIT_FAILURE);
|
|
|
|
|
|
|
|
l_main_quit();
|
2017-03-24 00:32:09 +01:00
|
|
|
}
|
|
|
|
|
2019-11-08 20:32:23 +01:00
|
|
|
static void get_managed_objects(void)
|
2017-03-24 00:32:09 +01:00
|
|
|
{
|
2019-11-08 20:32:23 +01:00
|
|
|
l_dbus_method_call(dbus, IWD_SERVICE, IWD_ROOT_PATH,
|
|
|
|
L_DBUS_INTERFACE_OBJECT_MANAGER,
|
|
|
|
"GetManagedObjects", NULL,
|
|
|
|
get_managed_objects_callback,
|
|
|
|
NULL, NULL);
|
|
|
|
}
|
2018-08-03 23:35:40 +02:00
|
|
|
|
2019-11-08 20:32:23 +01:00
|
|
|
static void service_appeared_callback(struct l_dbus *dbus, void *user_data)
|
|
|
|
{
|
2017-03-24 00:32:09 +01:00
|
|
|
l_dbus_add_signal_watch(dbus, IWD_SERVICE, IWD_ROOT_PATH,
|
|
|
|
L_DBUS_INTERFACE_OBJECT_MANAGER,
|
|
|
|
"InterfacesAdded", L_DBUS_MATCH_NONE,
|
|
|
|
interfaces_added_callback, NULL);
|
|
|
|
|
|
|
|
l_dbus_add_signal_watch(dbus, IWD_SERVICE, IWD_ROOT_PATH,
|
|
|
|
L_DBUS_INTERFACE_OBJECT_MANAGER,
|
|
|
|
"InterfacesRemoved", L_DBUS_MATCH_NONE,
|
|
|
|
interfaces_removed_callback, NULL);
|
|
|
|
|
2017-05-01 19:58:24 +02:00
|
|
|
l_dbus_add_signal_watch(dbus, IWD_SERVICE, NULL,
|
|
|
|
L_DBUS_INTERFACE_PROPERTIES,
|
|
|
|
"PropertiesChanged", L_DBUS_MATCH_NONE,
|
|
|
|
properties_changed_callback, NULL);
|
2019-11-08 20:32:23 +01:00
|
|
|
|
|
|
|
get_managed_objects();
|
2017-03-24 00:32:09 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void service_disappeared_callback(struct l_dbus *dbus,
|
|
|
|
void *user_data)
|
|
|
|
{
|
2018-08-03 23:35:40 +02:00
|
|
|
if (!command_is_interactive_mode()) {
|
|
|
|
command_set_exit_status(EXIT_FAILURE);
|
|
|
|
l_main_quit();
|
|
|
|
}
|
|
|
|
|
2017-04-07 23:55:33 +02:00
|
|
|
l_queue_clear(proxy_interfaces, proxy_interface_destroy);
|
|
|
|
|
|
|
|
display_disable_cmd_prompt();
|
2017-03-24 00:32:09 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void dbus_disconnect_callback(void *user_data)
|
|
|
|
{
|
2018-08-03 23:35:40 +02:00
|
|
|
if (!command_is_interactive_mode())
|
|
|
|
return;
|
|
|
|
|
2017-03-24 00:32:09 +01:00
|
|
|
l_main_quit();
|
|
|
|
}
|
|
|
|
|
2017-04-10 22:16:46 +02:00
|
|
|
void proxy_interface_type_register(
|
|
|
|
const struct proxy_interface_type *interface_type)
|
|
|
|
{
|
|
|
|
l_queue_push_tail(proxy_interface_types, (void *) interface_type);
|
|
|
|
}
|
|
|
|
|
|
|
|
void proxy_interface_type_unregister(
|
|
|
|
const struct proxy_interface_type *interface_type)
|
|
|
|
{
|
|
|
|
l_queue_remove(proxy_interface_types, (void *) interface_type);
|
|
|
|
}
|
|
|
|
|
2018-05-03 22:57:51 +02:00
|
|
|
struct l_dbus *dbus_get_bus(void)
|
|
|
|
{
|
|
|
|
return dbus;
|
|
|
|
}
|
|
|
|
|
2017-04-13 18:56:13 +02:00
|
|
|
extern struct interface_type_desc __start___interface[];
|
|
|
|
extern struct interface_type_desc __stop___interface[];
|
|
|
|
|
2017-03-23 17:48:53 +01:00
|
|
|
bool dbus_proxy_init(void)
|
|
|
|
{
|
2017-04-13 18:56:13 +02:00
|
|
|
struct interface_type_desc *desc;
|
|
|
|
|
2017-03-24 00:32:09 +01:00
|
|
|
if (dbus)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
dbus = l_dbus_new_default(L_DBUS_SYSTEM_BUS);
|
|
|
|
if (!dbus)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
proxy_interface_types = l_queue_new();
|
|
|
|
proxy_interfaces = l_queue_new();
|
|
|
|
|
2017-04-13 18:56:13 +02:00
|
|
|
for (desc = __start___interface; desc < __stop___interface; desc++) {
|
|
|
|
if (!desc->init)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
desc->init();
|
|
|
|
}
|
|
|
|
|
2017-03-24 00:32:09 +01:00
|
|
|
l_dbus_set_disconnect_handler(dbus, dbus_disconnect_callback, NULL,
|
|
|
|
NULL);
|
|
|
|
|
2019-11-08 20:32:23 +01:00
|
|
|
if (command_is_interactive_mode())
|
|
|
|
l_dbus_add_service_watch(dbus, IWD_SERVICE,
|
|
|
|
service_appeared_callback,
|
2017-03-24 00:32:09 +01:00
|
|
|
service_disappeared_callback,
|
|
|
|
NULL, NULL);
|
2019-11-08 20:32:23 +01:00
|
|
|
else
|
|
|
|
get_managed_objects();
|
2017-03-24 00:32:09 +01:00
|
|
|
|
2017-03-23 17:48:53 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool dbus_proxy_exit(void)
|
|
|
|
{
|
2017-04-13 18:56:13 +02:00
|
|
|
struct interface_type_desc *desc;
|
|
|
|
|
2020-02-27 00:32:24 +01:00
|
|
|
agent_manager_unregister_agent();
|
2018-05-03 22:57:50 +02:00
|
|
|
|
2017-04-13 18:56:13 +02:00
|
|
|
for (desc = __start___interface; desc < __stop___interface; desc++) {
|
|
|
|
if (!desc->exit)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
desc->exit();
|
|
|
|
}
|
|
|
|
|
2017-03-24 00:32:09 +01:00
|
|
|
l_queue_destroy(proxy_interface_types, NULL);
|
|
|
|
proxy_interface_types = NULL;
|
|
|
|
|
|
|
|
l_queue_destroy(proxy_interfaces, proxy_interface_destroy);
|
|
|
|
proxy_interfaces = NULL;
|
|
|
|
|
|
|
|
l_dbus_destroy(dbus);
|
|
|
|
dbus = NULL;
|
|
|
|
|
2017-03-23 17:48:53 +01:00
|
|
|
return true;
|
|
|
|
}
|