
These are (most likely) only warnings and not severe bugs. Some of these changes are mostly made to get a clean run of Coverity without any warnings. Error found by running Coverity scan https://bugzilla.redhat.com/show_bug.cgi?id=1025894 Co-Authored-By: Jiří Klimeš <jklimes@redhat.com> Signed-off-by: Thomas Haller <thaller@redhat.com>
410 lines
13 KiB
C
410 lines
13 KiB
C
/* -*- Mode: C; tab-width: 4; indent-tabs-mode: t; c-basic-offset: 4 -*- */
|
|
/* NetworkManager -- Network link manager
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program 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 General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along
|
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*
|
|
* Copyright (C) 2004 - 2012 Red Hat, Inc.
|
|
* Copyright (C) 2005 - 2008 Novell, Inc.
|
|
*/
|
|
|
|
#include <glib.h>
|
|
#include <string.h>
|
|
|
|
#include "nm-dispatcher.h"
|
|
#include "nm-dispatcher-action.h"
|
|
#include "NetworkManagerUtils.h"
|
|
#include "nm-utils.h"
|
|
#include "nm-logging.h"
|
|
#include "nm-dbus-manager.h"
|
|
#include "nm-dbus-glib-types.h"
|
|
#include "nm-glib-compat.h"
|
|
|
|
static GSList *requests = NULL;
|
|
|
|
static void
|
|
dump_object_to_props (GObject *object, GHashTable *hash)
|
|
{
|
|
GParamSpec **pspecs;
|
|
guint len = 0, i;
|
|
|
|
pspecs = g_object_class_list_properties (G_OBJECT_GET_CLASS (object), &len);
|
|
for (i = 0; i < len; i++) {
|
|
value_hash_add_object_property (hash,
|
|
pspecs[i]->name,
|
|
object,
|
|
pspecs[i]->name,
|
|
pspecs[i]->value_type);
|
|
}
|
|
g_free (pspecs);
|
|
}
|
|
|
|
static void
|
|
dump_dhcp4_to_props (NMDHCP4Config *config, GHashTable *hash)
|
|
{
|
|
GSList *options, *iter;
|
|
|
|
options = nm_dhcp4_config_list_options (config);
|
|
for (iter = options; iter; iter = g_slist_next (iter)) {
|
|
const char *option = (const char *) iter->data;
|
|
const char *val;
|
|
|
|
val = nm_dhcp4_config_get_option (config, option);
|
|
value_hash_add_str (hash, option, val);
|
|
}
|
|
g_slist_free (options);
|
|
}
|
|
|
|
static void
|
|
dump_dhcp6_to_props (NMDHCP6Config *config, GHashTable *hash)
|
|
{
|
|
GSList *options, *iter;
|
|
|
|
options = nm_dhcp6_config_list_options (config);
|
|
for (iter = options; iter; iter = g_slist_next (iter)) {
|
|
const char *option = (const char *) iter->data;
|
|
const char *val;
|
|
|
|
val = nm_dhcp6_config_get_option (config, option);
|
|
value_hash_add_str (hash, option, val);
|
|
}
|
|
g_slist_free (options);
|
|
}
|
|
|
|
static void
|
|
fill_device_props (NMDevice *device,
|
|
GHashTable *dev_hash,
|
|
GHashTable *ip4_hash,
|
|
GHashTable *ip6_hash,
|
|
GHashTable *dhcp4_hash,
|
|
GHashTable *dhcp6_hash)
|
|
{
|
|
NMIP4Config *ip4_config;
|
|
NMIP6Config *ip6_config;
|
|
NMDHCP4Config *dhcp4_config;
|
|
NMDHCP6Config *dhcp6_config;
|
|
|
|
/* If the action is for a VPN, send the VPN's IP interface instead of the device's */
|
|
value_hash_add_str (dev_hash, NMD_DEVICE_PROPS_IP_INTERFACE, nm_device_get_ip_iface (device));
|
|
value_hash_add_str (dev_hash, NMD_DEVICE_PROPS_INTERFACE, nm_device_get_iface (device));
|
|
value_hash_add_uint (dev_hash, NMD_DEVICE_PROPS_TYPE, nm_device_get_device_type (device));
|
|
value_hash_add_uint (dev_hash, NMD_DEVICE_PROPS_STATE, nm_device_get_state (device));
|
|
value_hash_add_object_path (dev_hash, NMD_DEVICE_PROPS_PATH, nm_device_get_path (device));
|
|
|
|
ip4_config = nm_device_get_ip4_config (device);
|
|
if (ip4_config)
|
|
dump_object_to_props (G_OBJECT (ip4_config), ip4_hash);
|
|
|
|
ip6_config = nm_device_get_ip6_config (device);
|
|
if (ip6_config)
|
|
dump_object_to_props (G_OBJECT (ip6_config), ip6_hash);
|
|
|
|
dhcp4_config = nm_device_get_dhcp4_config (device);
|
|
if (dhcp4_config)
|
|
dump_dhcp4_to_props (dhcp4_config, dhcp4_hash);
|
|
|
|
dhcp6_config = nm_device_get_dhcp6_config (device);
|
|
if (dhcp6_config)
|
|
dump_dhcp6_to_props (dhcp6_config, dhcp6_hash);
|
|
}
|
|
|
|
static void
|
|
fill_vpn_props (NMIP4Config *ip4_config,
|
|
NMIP6Config *ip6_config,
|
|
GHashTable *ip4_hash,
|
|
GHashTable *ip6_hash)
|
|
{
|
|
if (ip4_config)
|
|
dump_object_to_props (G_OBJECT (ip4_config), ip4_hash);
|
|
if (ip6_config)
|
|
dump_object_to_props (G_OBJECT (ip6_config), ip6_hash);
|
|
}
|
|
|
|
typedef struct {
|
|
DispatcherFunc callback;
|
|
gpointer user_data;
|
|
} DispatchInfo;
|
|
|
|
static void
|
|
dispatcher_info_free (DispatchInfo *info)
|
|
{
|
|
requests = g_slist_remove (requests, info);
|
|
g_free (info);
|
|
}
|
|
|
|
static const char *
|
|
dispatch_result_to_string (DispatchResult result)
|
|
{
|
|
switch (result) {
|
|
case DISPATCH_RESULT_UNKNOWN:
|
|
return "unknown";
|
|
case DISPATCH_RESULT_SUCCESS:
|
|
return "success";
|
|
case DISPATCH_RESULT_EXEC_FAILED:
|
|
return "exec failed";
|
|
case DISPATCH_RESULT_FAILED:
|
|
return "failed";
|
|
case DISPATCH_RESULT_TIMEOUT:
|
|
return "timed out";
|
|
}
|
|
g_assert_not_reached ();
|
|
}
|
|
|
|
static void
|
|
dispatcher_done_cb (DBusGProxy *proxy, DBusGProxyCall *call, gpointer user_data)
|
|
{
|
|
DispatchInfo *info = user_data;
|
|
GError *error = NULL;
|
|
GPtrArray *results = NULL;
|
|
guint i;
|
|
|
|
if (dbus_g_proxy_end_call (proxy, call, &error,
|
|
DISPATCHER_TYPE_RESULT_ARRAY, &results,
|
|
G_TYPE_INVALID)) {
|
|
for (i = 0; results && (i < results->len); i++) {
|
|
GValueArray *item = g_ptr_array_index (results, i);
|
|
GValue *tmp;
|
|
const char *script, *err;
|
|
DispatchResult result;
|
|
|
|
if (item->n_values != 3) {
|
|
nm_log_dbg (LOGD_CORE, "Unexpected number of items in "
|
|
"dispatcher result (got %d, expectd 3)",
|
|
item->n_values);
|
|
goto next;
|
|
}
|
|
|
|
/* Script */
|
|
tmp = g_value_array_get_nth (item, 0);
|
|
if (G_VALUE_TYPE (tmp) != G_TYPE_STRING) {
|
|
nm_log_dbg (LOGD_CORE, "Dispatcher result %d element 0 invalid type %s",
|
|
i, G_VALUE_TYPE_NAME (tmp));
|
|
goto next;
|
|
}
|
|
script = g_value_get_string (tmp);
|
|
|
|
/* Result */
|
|
tmp = g_value_array_get_nth (item, 1);
|
|
if (G_VALUE_TYPE (tmp) != G_TYPE_UINT) {
|
|
nm_log_dbg (LOGD_CORE, "Dispatcher result %d element 1 invalid type %s",
|
|
i, G_VALUE_TYPE_NAME (tmp));
|
|
goto next;
|
|
}
|
|
result = g_value_get_uint (tmp);
|
|
|
|
/* Error */
|
|
tmp = g_value_array_get_nth (item, 2);
|
|
if (G_VALUE_TYPE (tmp) != G_TYPE_STRING) {
|
|
nm_log_dbg (LOGD_CORE, "Dispatcher result %d element 2 invalid type %s",
|
|
i, G_VALUE_TYPE_NAME (tmp));
|
|
goto next;
|
|
}
|
|
err = g_value_get_string (tmp);
|
|
|
|
if (result != DISPATCH_RESULT_SUCCESS) {
|
|
nm_log_warn (LOGD_CORE, "Dispatcher script \"%s\" failed with %s: %s",
|
|
script, dispatch_result_to_string (result), err);
|
|
}
|
|
|
|
next:
|
|
g_value_array_free (item);
|
|
}
|
|
g_ptr_array_free (results, TRUE);
|
|
} else {
|
|
g_assert (error);
|
|
nm_log_warn (LOGD_CORE, "Dispatcher failed: (%d) %s", error->code, error->message);
|
|
}
|
|
|
|
if (info->callback)
|
|
info->callback (info, info->user_data);
|
|
|
|
g_clear_error (&error);
|
|
g_object_unref (proxy);
|
|
}
|
|
|
|
static const char *
|
|
action_to_string (DispatcherAction action)
|
|
{
|
|
switch (action) {
|
|
case DISPATCHER_ACTION_HOSTNAME:
|
|
return "hostname";
|
|
case DISPATCHER_ACTION_UP:
|
|
return "up";
|
|
case DISPATCHER_ACTION_PRE_DOWN:
|
|
return "pre-down";
|
|
case DISPATCHER_ACTION_DOWN:
|
|
return "down";
|
|
case DISPATCHER_ACTION_VPN_UP:
|
|
return "vpn-up";
|
|
case DISPATCHER_ACTION_VPN_PRE_DOWN:
|
|
return "vpn-pre-down";
|
|
case DISPATCHER_ACTION_VPN_DOWN:
|
|
return "vpn-down";
|
|
case DISPATCHER_ACTION_DHCP4_CHANGE:
|
|
return "dhcp4-change";
|
|
case DISPATCHER_ACTION_DHCP6_CHANGE:
|
|
return "dhcp6-change";
|
|
default:
|
|
break;
|
|
}
|
|
g_assert_not_reached ();
|
|
}
|
|
|
|
static gconstpointer
|
|
_dispatcher_call (DispatcherAction action,
|
|
NMConnection *connection,
|
|
NMDevice *device,
|
|
const char *vpn_iface,
|
|
NMIP4Config *vpn_ip4_config,
|
|
NMIP6Config *vpn_ip6_config,
|
|
DispatcherFunc callback,
|
|
gpointer user_data)
|
|
{
|
|
DBusGProxy *proxy;
|
|
DBusGConnection *g_connection;
|
|
GHashTable *connection_hash;
|
|
GHashTable *connection_props;
|
|
GHashTable *device_props;
|
|
GHashTable *device_ip4_props;
|
|
GHashTable *device_ip6_props;
|
|
GHashTable *device_dhcp4_props;
|
|
GHashTable *device_dhcp6_props;
|
|
GHashTable *vpn_ip4_props;
|
|
GHashTable *vpn_ip6_props;
|
|
DispatchInfo *info;
|
|
|
|
/* All actions except 'hostname' require a device */
|
|
if (action != DISPATCHER_ACTION_HOSTNAME)
|
|
g_return_val_if_fail (NM_IS_DEVICE (device), NULL);
|
|
/* VPN actions require at least an IPv4 config (for now) */
|
|
if (action == DISPATCHER_ACTION_VPN_UP)
|
|
g_return_val_if_fail (vpn_ip4_config != NULL, NULL);
|
|
|
|
g_connection = nm_dbus_manager_get_connection (nm_dbus_manager_get ());
|
|
proxy = dbus_g_proxy_new_for_name (g_connection,
|
|
NM_DISPATCHER_DBUS_SERVICE,
|
|
NM_DISPATCHER_DBUS_PATH,
|
|
NM_DISPATCHER_DBUS_IFACE);
|
|
if (!proxy) {
|
|
nm_log_err (LOGD_CORE, "could not get dispatcher proxy!");
|
|
return NULL;
|
|
}
|
|
|
|
if (connection) {
|
|
connection_hash = nm_connection_to_hash (connection, NM_SETTING_HASH_FLAG_NO_SECRETS);
|
|
|
|
connection_props = value_hash_create ();
|
|
value_hash_add_object_path (connection_props,
|
|
NMD_CONNECTION_PROPS_PATH,
|
|
nm_connection_get_path (connection));
|
|
} else {
|
|
connection_hash = value_hash_create ();
|
|
connection_props = value_hash_create ();
|
|
}
|
|
|
|
device_props = value_hash_create ();
|
|
device_ip4_props = value_hash_create ();
|
|
device_ip6_props = value_hash_create ();
|
|
device_dhcp4_props = value_hash_create ();
|
|
device_dhcp6_props = value_hash_create ();
|
|
vpn_ip4_props = value_hash_create ();
|
|
vpn_ip6_props = value_hash_create ();
|
|
|
|
/* hostname actions only send the hostname */
|
|
if (action != DISPATCHER_ACTION_HOSTNAME) {
|
|
fill_device_props (device,
|
|
device_props,
|
|
device_ip4_props,
|
|
device_ip6_props,
|
|
device_dhcp4_props,
|
|
device_dhcp6_props);
|
|
if (vpn_iface)
|
|
fill_vpn_props (vpn_ip4_config, vpn_ip6_config, vpn_ip4_props, vpn_ip6_props);
|
|
}
|
|
|
|
info = g_malloc0 (sizeof (*info));
|
|
info->callback = callback;
|
|
info->user_data = user_data;
|
|
|
|
/* Send the action to the dispatcher */
|
|
dbus_g_proxy_begin_call_with_timeout (proxy, "Action",
|
|
dispatcher_done_cb,
|
|
info,
|
|
(GDestroyNotify) dispatcher_info_free,
|
|
15000,
|
|
G_TYPE_STRING, action_to_string (action),
|
|
DBUS_TYPE_G_MAP_OF_MAP_OF_VARIANT, connection_hash,
|
|
DBUS_TYPE_G_MAP_OF_VARIANT, connection_props,
|
|
DBUS_TYPE_G_MAP_OF_VARIANT, device_props,
|
|
DBUS_TYPE_G_MAP_OF_VARIANT, device_ip4_props,
|
|
DBUS_TYPE_G_MAP_OF_VARIANT, device_ip6_props,
|
|
DBUS_TYPE_G_MAP_OF_VARIANT, device_dhcp4_props,
|
|
DBUS_TYPE_G_MAP_OF_VARIANT, device_dhcp6_props,
|
|
G_TYPE_STRING, vpn_iface ? vpn_iface : "",
|
|
DBUS_TYPE_G_MAP_OF_VARIANT, vpn_ip4_props,
|
|
DBUS_TYPE_G_MAP_OF_VARIANT, vpn_ip6_props,
|
|
G_TYPE_INVALID);
|
|
g_hash_table_destroy (connection_hash);
|
|
g_hash_table_destroy (connection_props);
|
|
g_hash_table_destroy (device_props);
|
|
g_hash_table_destroy (device_ip4_props);
|
|
g_hash_table_destroy (device_ip6_props);
|
|
g_hash_table_destroy (device_dhcp4_props);
|
|
g_hash_table_destroy (device_dhcp6_props);
|
|
g_hash_table_destroy (vpn_ip4_props);
|
|
g_hash_table_destroy (vpn_ip6_props);
|
|
|
|
/* Track the request in case of cancelation */
|
|
requests = g_slist_append (requests, info);
|
|
|
|
return info;
|
|
}
|
|
|
|
gconstpointer
|
|
nm_dispatcher_call (DispatcherAction action,
|
|
NMConnection *connection,
|
|
NMDevice *device,
|
|
DispatcherFunc callback,
|
|
gpointer user_data)
|
|
{
|
|
return _dispatcher_call (action, connection, device, NULL, NULL, NULL, callback, user_data);
|
|
}
|
|
|
|
gconstpointer
|
|
nm_dispatcher_call_vpn (DispatcherAction action,
|
|
NMConnection *connection,
|
|
NMDevice *device,
|
|
const char *vpn_iface,
|
|
NMIP4Config *vpn_ip4_config,
|
|
NMIP6Config *vpn_ip6_config,
|
|
DispatcherFunc callback,
|
|
gpointer user_data)
|
|
{
|
|
return _dispatcher_call (action, connection, device, vpn_iface, vpn_ip4_config, vpn_ip6_config, callback, user_data);
|
|
}
|
|
|
|
void
|
|
nm_dispatcher_call_cancel (gconstpointer call)
|
|
{
|
|
/* 'call' is really a DispatchInfo pointer, just opaque to callers.
|
|
* Look it up in our requests list, but don't access it directly before
|
|
* we've made sure it's a valid request,since it may have long since been
|
|
* freed. Canceling just means the callback doesn't get called, so set
|
|
* the DispatcherInfo's callback to NULL.
|
|
*/
|
|
if (g_slist_find (requests, call))
|
|
((DispatchInfo *) call)->callback = NULL;
|
|
}
|
|
|