NetworkManager/src/nm-firewall-manager.c

643 lines
19 KiB
C

// SPDX-License-Identifier: GPL-2.0+
/*
* Copyright (C) 2011 - 2015 Red Hat, Inc.
*/
#include "nm-default.h"
#include "nm-firewall-manager.h"
#include "nm-glib-aux/nm-dbus-aux.h"
#include "c-list/src/c-list.h"
#include "NetworkManagerUtils.h"
#include "nm-dbus-manager.h"
#define FIREWALL_DBUS_SERVICE "org.fedoraproject.FirewallD1"
#define FIREWALL_DBUS_PATH "/org/fedoraproject/FirewallD1"
#define FIREWALL_DBUS_INTERFACE_ZONE "org.fedoraproject.FirewallD1.zone"
/*****************************************************************************/
enum {
STATE_CHANGED,
LAST_SIGNAL
};
static guint signals[LAST_SIGNAL] = { 0 };
typedef struct {
GDBusConnection *dbus_connection;
GCancellable *get_name_owner_cancellable;
CList pending_calls;
guint name_owner_changed_id;
bool dbus_inited:1;
bool running:1;
} NMFirewallManagerPrivate;
struct _NMFirewallManager {
GObject parent;
NMFirewallManagerPrivate _priv;
};
struct _NMFirewallManagerClass {
GObjectClass parent;
};
G_DEFINE_TYPE (NMFirewallManager, nm_firewall_manager, G_TYPE_OBJECT)
#define NM_FIREWALL_MANAGER_GET_PRIVATE(self) _NM_GET_PRIVATE (self, NMFirewallManager, NM_IS_FIREWALL_MANAGER)
/*****************************************************************************/
NM_DEFINE_SINGLETON_GETTER (NMFirewallManager, nm_firewall_manager_get, NM_TYPE_FIREWALL_MANAGER);
/*****************************************************************************/
typedef enum {
OPS_TYPE_ADD = 1,
OPS_TYPE_CHANGE,
OPS_TYPE_REMOVE,
} OpsType;
struct _NMFirewallManagerCallId {
CList lst;
NMFirewallManager *self;
char *iface;
NMFirewallManagerAddRemoveCallback callback;
gpointer user_data;
union {
struct {
GCancellable *cancellable;
GVariant *arg;
} dbus;
struct {
guint id;
} idle;
};
OpsType ops_type;
bool is_idle:1;
};
/*****************************************************************************/
static const char *
_ops_type_to_string (OpsType ops_type)
{
switch (ops_type) {
case OPS_TYPE_ADD: return "add";
case OPS_TYPE_REMOVE: return "remove";
case OPS_TYPE_CHANGE: return "change";
}
nm_assert_not_reached ();
return NULL;
}
#define _NMLOG_DOMAIN LOGD_FIREWALL
#define _NMLOG_PREFIX_NAME "firewall"
#define _NMLOG(level, call_id, ...) \
G_STMT_START { \
if (nm_logging_enabled ((level), (_NMLOG_DOMAIN))) { \
NMFirewallManagerCallId *_call_id = (call_id); \
char _prefix_name[30]; \
char _prefix_info[100]; \
\
_nm_log ((level), (_NMLOG_DOMAIN), 0, NULL, NULL, \
"%s: %s" _NM_UTILS_MACRO_FIRST(__VA_ARGS__), \
(self) != singleton_instance \
? ({ \
g_snprintf (_prefix_name, \
sizeof (_prefix_name), \
"%s["NM_HASH_OBFUSCATE_PTR_FMT"]", \
""_NMLOG_PREFIX_NAME,\
NM_HASH_OBFUSCATE_PTR (self)); \
_prefix_name; \
}) \
: _NMLOG_PREFIX_NAME, \
_call_id \
? ({ \
g_snprintf (_prefix_info, \
sizeof (_prefix_info), \
"["NM_HASH_OBFUSCATE_PTR_FMT",%s%s:%s%s%s]: ", \
NM_HASH_OBFUSCATE_PTR (_call_id), \
_ops_type_to_string (_call_id->ops_type), \
_call_id->is_idle ? "*" : "", \
NM_PRINT_FMT_QUOTE_STRING (_call_id->iface)); \
_prefix_info; \
}) \
: "" \
_NM_UTILS_MACRO_REST(__VA_ARGS__)); \
} \
} G_STMT_END
/*****************************************************************************/
static gboolean
_get_running (NMFirewallManagerPrivate *priv)
{
/* when starting, we need to asynchronously check whether there is
* a name owner. During that time we optimistically assume that the
* service is indeed running. That is the time when we queue the
* requests, and they will be started once the get-name-owner call
* returns. */
return priv->running
|| ( priv->dbus_connection
&& !priv->dbus_inited);
}
gboolean
nm_firewall_manager_get_running (NMFirewallManager *self)
{
g_return_val_if_fail (NM_IS_FIREWALL_MANAGER (self), FALSE);
return _get_running (NM_FIREWALL_MANAGER_GET_PRIVATE (self));
}
/*****************************************************************************/
static NMFirewallManagerCallId *
_cb_info_create (NMFirewallManager *self,
OpsType ops_type,
const char *iface,
const char *zone,
NMFirewallManagerAddRemoveCallback callback,
gpointer user_data)
{
NMFirewallManagerPrivate *priv = NM_FIREWALL_MANAGER_GET_PRIVATE (self);
NMFirewallManagerCallId *call_id;
call_id = g_slice_new0 (NMFirewallManagerCallId);
call_id->self = g_object_ref (self);
call_id->ops_type = ops_type;
call_id->iface = g_strdup (iface);
call_id->callback = callback;
call_id->user_data = user_data;
if (_get_running (priv)) {
call_id->is_idle = FALSE;
call_id->dbus.arg = g_variant_new ("(ss)", zone ?: "", iface);
} else
call_id->is_idle = TRUE;
c_list_link_tail (&priv->pending_calls, &call_id->lst);
return call_id;
}
static void
_cb_info_complete (NMFirewallManagerCallId *call_id,
GError *error)
{
c_list_unlink (&call_id->lst);
if (call_id->callback)
call_id->callback (call_id->self, call_id, error, call_id->user_data);
if (call_id->is_idle)
nm_clear_g_source (&call_id->idle.id);
else {
nm_g_variant_unref (call_id->dbus.arg);
nm_clear_g_cancellable (&call_id->dbus.cancellable);
}
g_free (call_id->iface);
g_object_unref (call_id->self);
nm_g_slice_free (call_id);
}
static gboolean
_handle_idle_cb (gpointer user_data)
{
NMFirewallManager *self;
NMFirewallManagerCallId *call_id = user_data;
nm_assert (call_id);
nm_assert (NM_IS_FIREWALL_MANAGER (call_id->self));
nm_assert (call_id->is_idle);
nm_assert (c_list_contains (&NM_FIREWALL_MANAGER_GET_PRIVATE (call_id->self)->pending_calls, &call_id->lst));
self = call_id->self;
_LOGD (call_id, "complete: fake success");
call_id->idle.id = 0;
_cb_info_complete (call_id, NULL);
return G_SOURCE_REMOVE;
}
static gboolean
_handle_idle_start (NMFirewallManager *self,
NMFirewallManagerCallId *call_id)
{
if (!call_id->callback) {
/* if the user did not provide a callback and firewalld is not running,
* there is no point in scheduling an idle-request to fake success. Just
* return right away. */
_LOGD (call_id, "complete: drop request simulating success");
_cb_info_complete (call_id, NULL);
return FALSE;
}
call_id->idle.id = g_idle_add (_handle_idle_cb, call_id);
return TRUE;
}
static void
_handle_dbus_cb (GObject *source,
GAsyncResult *result,
gpointer user_data)
{
NMFirewallManager *self;
NMFirewallManagerCallId *call_id;
gs_free_error GError *error = NULL;
gs_unref_variant GVariant *ret = NULL;
ret = g_dbus_connection_call_finish (G_DBUS_CONNECTION (source), result, &error);
if ( !ret
&& nm_utils_error_is_cancelled (error))
return;
call_id = user_data;
nm_assert (call_id);
nm_assert (NM_IS_FIREWALL_MANAGER (call_id->self));
nm_assert (!call_id->is_idle);
nm_assert (c_list_contains (&NM_FIREWALL_MANAGER_GET_PRIVATE (call_id->self)->pending_calls, &call_id->lst));
self = call_id->self;
if (error) {
const char *non_error = NULL;
g_dbus_error_strip_remote_error (error);
switch (call_id->ops_type) {
case OPS_TYPE_ADD:
case OPS_TYPE_CHANGE:
non_error = "ZONE_ALREADY_SET";
break;
case OPS_TYPE_REMOVE:
non_error = "UNKNOWN_INTERFACE";
break;
}
if ( error->message
&& non_error
&& g_str_has_prefix (error->message, non_error)
&& NM_IN_SET (error->message[strlen (non_error)], '\0', ':')) {
_LOGD (call_id, "complete: request failed with a non-error (%s)", error->message);
/* The operation failed with an error reason that we don't want
* to propagate. Instead, signal success. */
g_clear_error (&error);
} else
_LOGW (call_id, "complete: request failed (%s)", error->message);
} else
_LOGD (call_id, "complete: success");
g_clear_object (&call_id->dbus.cancellable);
_cb_info_complete (call_id, error);
}
static void
_handle_dbus_start (NMFirewallManager *self,
NMFirewallManagerCallId *call_id)
{
NMFirewallManagerPrivate *priv = NM_FIREWALL_MANAGER_GET_PRIVATE (self);
const char *dbus_method = NULL;
GVariant *arg;
nm_assert (call_id);
nm_assert (priv->running);
nm_assert (!call_id->is_idle);
nm_assert (c_list_contains (&priv->pending_calls, &call_id->lst));
switch (call_id->ops_type) {
case OPS_TYPE_ADD:
dbus_method = "addInterface";
break;
case OPS_TYPE_CHANGE:
dbus_method = "changeZone";
break;
case OPS_TYPE_REMOVE:
dbus_method = "removeInterface";
break;
}
nm_assert (dbus_method);
arg = g_steal_pointer (&call_id->dbus.arg);
nm_assert (arg && g_variant_is_floating (arg));
nm_assert (!call_id->dbus.cancellable);
call_id->dbus.cancellable = g_cancellable_new ();
g_dbus_connection_call (priv->dbus_connection,
FIREWALL_DBUS_SERVICE,
FIREWALL_DBUS_PATH,
FIREWALL_DBUS_INTERFACE_ZONE,
dbus_method,
arg,
NULL,
G_DBUS_CALL_FLAGS_NONE,
10000,
call_id->dbus.cancellable,
_handle_dbus_cb,
call_id);
}
static NMFirewallManagerCallId *
_start_request (NMFirewallManager *self,
OpsType ops_type,
const char *iface,
const char *zone,
NMFirewallManagerAddRemoveCallback callback,
gpointer user_data)
{
NMFirewallManagerPrivate *priv;
NMFirewallManagerCallId *call_id;
g_return_val_if_fail (NM_IS_FIREWALL_MANAGER (self), NULL);
g_return_val_if_fail (iface && *iface, NULL);
priv = NM_FIREWALL_MANAGER_GET_PRIVATE (self);
call_id = _cb_info_create (self, ops_type, iface, zone, callback, user_data);
_LOGD (call_id, "firewall zone %s %s:%s%s%s%s",
_ops_type_to_string (call_id->ops_type),
iface,
NM_PRINT_FMT_QUOTED (zone, "\"", zone, "\"", "default"),
call_id->is_idle
? " (not running, simulate success)"
: (!priv->running
? " (waiting to initialize)"
: ""));
if (!call_id->is_idle) {
if (priv->running)
_handle_dbus_start (self, call_id);
if (!call_id->callback) {
/* if the user did not provide a callback, the call_id is useless.
* Especially, the user cannot use the call-id to cancel the request,
* because he cannot know whether the request is still pending.
*
* Hence, returning %NULL doesn't mean that the request could not be started
* (this function never fails and always starts a request). */
return NULL;
}
} else {
if (!_handle_idle_start (self, call_id)) {
/* if the user did not provide a callback and firewalld is not running,
* there is no point in scheduling an idle-request to fake success. Just
* return right away. */
return NULL;
}
}
return call_id;
}
NMFirewallManagerCallId *
nm_firewall_manager_add_or_change_zone (NMFirewallManager *self,
const char *iface,
const char *zone,
gboolean add, /* TRUE == add, FALSE == change */
NMFirewallManagerAddRemoveCallback callback,
gpointer user_data)
{
return _start_request (self,
add ? OPS_TYPE_ADD : OPS_TYPE_CHANGE,
iface,
zone,
callback,
user_data);
}
NMFirewallManagerCallId *
nm_firewall_manager_remove_from_zone (NMFirewallManager *self,
const char *iface,
const char *zone,
NMFirewallManagerAddRemoveCallback callback,
gpointer user_data)
{
return _start_request (self,
OPS_TYPE_REMOVE,
iface,
zone,
callback,
user_data);
}
void
nm_firewall_manager_cancel_call (NMFirewallManagerCallId *call_id)
{
NMFirewallManager *self;
NMFirewallManagerPrivate *priv;
gs_free_error GError *error = NULL;
g_return_if_fail (call_id);
g_return_if_fail (NM_IS_FIREWALL_MANAGER (call_id->self));
g_return_if_fail (!c_list_is_empty (&call_id->lst));
self = call_id->self;
priv = NM_FIREWALL_MANAGER_GET_PRIVATE (self);
nm_assert (c_list_contains (&priv->pending_calls, &call_id->lst));
nm_utils_error_set_cancelled (&error, FALSE, "NMFirewallManager");
_LOGD (call_id, "complete: cancel (%s)", error->message);
_cb_info_complete (call_id, error);
}
/*****************************************************************************/
static void
name_owner_changed (NMFirewallManager *self,
const char *owner)
{
_nm_unused gs_unref_object NMFirewallManager *self_keep_alive = g_object_ref (self);
NMFirewallManagerPrivate *priv = NM_FIREWALL_MANAGER_GET_PRIVATE (self);
gboolean was_running;
gboolean now_running;
gboolean just_initied;
owner = nm_str_not_empty (owner);
if (!owner)
_LOGT (NULL, "D-Bus name for firewalld has no owner (firewall stopped)");
else
_LOGT (NULL, "D-Bus name for firewalld has owner %s (firewall started)", owner);
was_running = _get_running (priv);
just_initied = !priv->dbus_inited;
priv->dbus_inited = TRUE;
priv->running = !!owner;
now_running = _get_running (priv);
if (just_initied) {
NMFirewallManagerCallId *call_id_safe;
NMFirewallManagerCallId *call_id;
/* We kick of the requests that we have pending. Note that this is
* entirely asynchronous and also we don't invoke any callbacks for
* the user.
* Even _handle_idle_start() just schedules an idle handler. That is,
* because we don't want to callback to the user before emitting the
* DISCONNECTED signal below. Also, emitting callbacks means the user
* can call back to modify the list of pending-calls and we'd have
* to handle reentrancy. */
c_list_for_each_entry_safe (call_id, call_id_safe, &priv->pending_calls, lst) {
nm_assert (!call_id->is_idle);
nm_assert (call_id->dbus.arg);
if (priv->running) {
_LOGD (call_id, "initalizing: make D-Bus call");
_handle_dbus_start (self, call_id);
} else {
/* we don't want to invoke callbacks to the user right away. That is because
* the user might schedule/cancel more calls, which messes up the order.
*
* Instead, convert the pending calls to idle requests... */
nm_clear_pointer (&call_id->dbus.arg, g_variant_unref);
call_id->is_idle = TRUE;
_LOGD (call_id, "initializing: fake success on idle");
_handle_idle_start (self, call_id);
}
}
}
if (was_running != now_running)
g_signal_emit (self, signals[STATE_CHANGED], 0, FALSE);
}
static void
name_owner_changed_cb (GDBusConnection *connection,
const char *sender_name,
const char *object_path,
const char *interface_name,
const char *signal_name,
GVariant *parameters,
gpointer user_data)
{
NMFirewallManager *self = user_data;
const char *new_owner;
if (!g_variant_is_of_type (parameters, G_VARIANT_TYPE ("(sss)")))
return;
g_variant_get (parameters,
"(&s&s&s)",
NULL,
NULL,
&new_owner);
name_owner_changed (self, new_owner);
}
static void
get_name_owner_cb (const char *name_owner,
GError *error,
gpointer user_data)
{
NMFirewallManager *self;
NMFirewallManagerPrivate *priv;
if ( !name_owner
&& g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
return;
self = user_data;
priv = NM_FIREWALL_MANAGER_GET_PRIVATE (self);
g_clear_object (&priv->get_name_owner_cancellable);
name_owner_changed (self, name_owner);
}
/*****************************************************************************/
static void
nm_firewall_manager_init (NMFirewallManager *self)
{
NMFirewallManagerPrivate *priv = NM_FIREWALL_MANAGER_GET_PRIVATE (self);
c_list_init (&priv->pending_calls);
priv->dbus_connection = nm_g_object_ref (NM_MAIN_DBUS_CONNECTION_GET);
if (!priv->dbus_connection) {
_LOGD (NULL, "no D-Bus connection");
return;
}
priv->name_owner_changed_id = nm_dbus_connection_signal_subscribe_name_owner_changed (priv->dbus_connection,
FIREWALL_DBUS_SERVICE,
name_owner_changed_cb,
self,
NULL);
priv->get_name_owner_cancellable = g_cancellable_new ();
nm_dbus_connection_call_get_name_owner (priv->dbus_connection,
FIREWALL_DBUS_SERVICE,
-1,
priv->get_name_owner_cancellable,
get_name_owner_cb,
self);
}
static void
dispose (GObject *object)
{
NMFirewallManager *self = NM_FIREWALL_MANAGER (object);
NMFirewallManagerPrivate *priv = NM_FIREWALL_MANAGER_GET_PRIVATE (self);
/* as every pending operation takes a reference to the manager,
* we don't expect pending operations at this point. */
nm_assert (c_list_is_empty (&priv->pending_calls));
nm_clear_g_dbus_connection_signal (priv->dbus_connection,
&priv->name_owner_changed_id);
nm_clear_g_cancellable (&priv->get_name_owner_cancellable);
G_OBJECT_CLASS (nm_firewall_manager_parent_class)->dispose (object);
g_clear_object (&priv->dbus_connection);
}
static void
nm_firewall_manager_class_init (NMFirewallManagerClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS (klass);
object_class->dispose = dispose;
signals[STATE_CHANGED] =
g_signal_new (NM_FIREWALL_MANAGER_STATE_CHANGED,
G_OBJECT_CLASS_TYPE (object_class),
G_SIGNAL_RUN_FIRST,
0,
NULL, NULL,
g_cclosure_marshal_VOID__BOOLEAN,
G_TYPE_NONE, 1,
G_TYPE_BOOLEAN /* initialized_now */);
}