// SPDX-License-Identifier: LGPL-2.1+ /* * Copyright (C) 2007 - 2008 Novell, Inc. * Copyright (C) 2007 - 2012 Red Hat, Inc. */ #include "nm-default.h" #include "nm-vpn-connection.h" #include "nm-dbus-interface.h" #include "nm-utils.h" #include "nm-object-private.h" #include "nm-active-connection.h" #include "nm-dbus-helpers.h" /*****************************************************************************/ NM_GOBJECT_PROPERTIES_DEFINE (NMVpnConnection, PROP_VPN_STATE, PROP_BANNER, ); enum { VPN_STATE_CHANGED, LAST_SIGNAL }; static guint signals[LAST_SIGNAL] = { 0 }; typedef struct { char *banner; guint32 vpn_state; guint32 reason; } NMVpnConnectionPrivate; struct _NMVpnConnection { NMActiveConnection parent; NMVpnConnectionPrivate _priv; }; struct _NMVpnConnectionClass { NMActiveConnectionClass parent; }; G_DEFINE_TYPE (NMVpnConnection, nm_vpn_connection, NM_TYPE_ACTIVE_CONNECTION) #define NM_VPN_CONNECTION_GET_PRIVATE(self) _NM_GET_PRIVATE(self, NMVpnConnection, NM_IS_VPN_CONNECTION, NMObject, NMActiveConnection) G_STATIC_ASSERT (sizeof (NMVpnConnectionStateReason) == sizeof (NMActiveConnectionStateReason)); /*****************************************************************************/ /** * nm_vpn_connection_get_banner: * @vpn: a #NMVpnConnection * * Gets the VPN login banner of the active #NMVpnConnection. * * Returns: the VPN login banner of the VPN connection. This is the internal * string used by the connection, and must not be modified. **/ const char * nm_vpn_connection_get_banner (NMVpnConnection *vpn) { g_return_val_if_fail (NM_IS_VPN_CONNECTION (vpn), NULL); return _nml_coerce_property_str_not_empty (NM_VPN_CONNECTION_GET_PRIVATE (vpn)->banner); } /** * nm_vpn_connection_get_vpn_state: * @vpn: a #NMVpnConnection * * Gets the current #NMVpnConnection state. * * Returns: the VPN state of the active VPN connection. **/ NMVpnConnectionState nm_vpn_connection_get_vpn_state (NMVpnConnection *vpn) { g_return_val_if_fail (NM_IS_VPN_CONNECTION (vpn), NM_VPN_CONNECTION_STATE_UNKNOWN); return NM_VPN_CONNECTION_GET_PRIVATE (vpn)->vpn_state; } /*****************************************************************************/ static void _notify_event_state_changed (NMClient *client, NMClientNotifyEventWithPtr *notify_event) { gs_unref_object NMVpnConnection *self = notify_event->user_data; NMVpnConnectionPrivate *priv = NM_VPN_CONNECTION_GET_PRIVATE (self); /* we expose here the value cache in @priv. In practice, this is the same * value as we received from the signal. In the unexpected case where they * differ, the cached value of the current instance would still be more correct. */ g_signal_emit (self, signals[VPN_STATE_CHANGED], 0, (guint) priv->vpn_state, (guint) priv->reason); } void _nm_vpn_connection_state_changed_commit (NMVpnConnection *self, guint32 state, guint32 reason) { NMClient *client; NMVpnConnectionPrivate *priv = NM_VPN_CONNECTION_GET_PRIVATE (self); client = _nm_object_get_client (self); if (priv->vpn_state != state) { priv->vpn_state = state; _nm_client_queue_notify_object (client, self, obj_properties[PROP_VPN_STATE]); } priv->reason = reason; _nm_client_notify_event_queue_with_ptr (client, NM_CLIENT_NOTIFY_EVENT_PRIO_GPROP + 1, _notify_event_state_changed, g_object_ref (self)); } /*****************************************************************************/ static void nm_vpn_connection_init (NMVpnConnection *connection) { } static void finalize (GObject *object) { NMVpnConnectionPrivate *priv = NM_VPN_CONNECTION_GET_PRIVATE (object); g_free (priv->banner); G_OBJECT_CLASS (nm_vpn_connection_parent_class)->finalize (object); } static void get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { NMVpnConnection *self = NM_VPN_CONNECTION (object); switch (prop_id) { case PROP_VPN_STATE: g_value_set_enum (value, nm_vpn_connection_get_vpn_state (self)); break; case PROP_BANNER: g_value_set_string (value, nm_vpn_connection_get_banner (self)); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } const NMLDBusMetaIface _nml_dbus_meta_iface_nm_vpn_connection = NML_DBUS_META_IFACE_INIT_PROP ( NM_DBUS_INTERFACE_VPN_CONNECTION, nm_vpn_connection_get_type, NML_DBUS_META_INTERFACE_PRIO_INSTANTIATE_HIGH, NML_DBUS_META_IFACE_DBUS_PROPERTIES ( NML_DBUS_META_PROPERTY_INIT_S ("Banner", PROP_BANNER, NMVpnConnection, _priv.banner ), NML_DBUS_META_PROPERTY_INIT_U ("VpnState", PROP_VPN_STATE, NMVpnConnection, _priv.vpn_state ), ), ); static void nm_vpn_connection_class_init (NMVpnConnectionClass *connection_class) { GObjectClass *object_class = G_OBJECT_CLASS (connection_class); object_class->get_property = get_property; object_class->finalize = finalize; /** * NMVpnConnection:vpn-state: * * The VPN state of the active VPN connection. **/ obj_properties[PROP_VPN_STATE] = g_param_spec_enum (NM_VPN_CONNECTION_VPN_STATE, "", "", NM_TYPE_VPN_CONNECTION_STATE, NM_VPN_CONNECTION_STATE_UNKNOWN, G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); /** * NMVpnConnection:banner: * * The VPN login banner of the active VPN connection. **/ obj_properties[PROP_BANNER] = g_param_spec_string (NM_VPN_CONNECTION_BANNER, "", "", NULL, G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); _nml_dbus_meta_class_init_with_properties (object_class, &_nml_dbus_meta_iface_nm_vpn_connection); /* TODO: the state reason should also be exposed as a property in libnm's NMVpnConnection, * like done for NMDevice's state reason. */ /* TODO: the D-Bus API should also expose the state-reason as a property instead of * a "VpnStateChanged" signal. Like done for Device's "StateReason". */ G_GNUC_BEGIN_IGNORE_DEPRECATIONS signals[VPN_STATE_CHANGED] = g_signal_new ("vpn-state-changed", G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_FIRST, 0, NULL, NULL, NULL, G_TYPE_NONE, 2, G_TYPE_UINT, G_TYPE_UINT); G_GNUC_END_IGNORE_DEPRECATIONS }