diff options
Diffstat (limited to 'tui/nmt-device-entry.c')
-rw-r--r-- | tui/nmt-device-entry.c | 589 |
1 files changed, 589 insertions, 0 deletions
diff --git a/tui/nmt-device-entry.c b/tui/nmt-device-entry.c new file mode 100644 index 000000000..7ce6d7d42 --- /dev/null +++ b/tui/nmt-device-entry.c @@ -0,0 +1,589 @@ +/* -*- Mode: C; tab-width: 4; indent-tabs-mode: t; c-basic-offset: 4 -*- */ +/* + * 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, see <http://www.gnu.org/licenses/>. + * + * Copyright 2013 Red Hat, Inc. + */ + +/** + * SECTION:nmt-device-entry + * @short_description: #NmtNewtEntry for identifying a device + * + * #NmtDeviceEntry provides a widget for identifying a device, either + * by interface name or by hardware address. The user can enter either + * value, and the entry's #NmtDeviceEntry:interface-name or + * #NmtDeviceEntry:mac-address property will be set accordingly. If + * the entry recognizes the interface name or mac address typed in as + * matching a known #NMDevice, then it will also display the other + * property in parentheses. + * + * FIXME: #NmtDeviceEntry is currently an #NmtPageGrid object, so that + * we can possibly eventually add a button to its "extra" field, that + * would pop up a form for selecting a device. But if we're not going + * to implement that then we should make it just an #NmtNewtEntry. + */ + +#include "config.h" + +#include <string.h> +#include <sys/socket.h> +#include <linux/if_arp.h> + +#include <glib/gi18n-lib.h> +#include <nm-device.h> +#include <nm-device-infiniband.h> +#include <nm-utils.h> + +#include "nmtui.h" +#include "nmt-device-entry.h" + +G_DEFINE_TYPE (NmtDeviceEntry, nmt_device_entry, NMT_TYPE_PAGE_GRID) + +#define NMT_DEVICE_ENTRY_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE ((o), NMT_TYPE_DEVICE_ENTRY, NmtDeviceEntryPrivate)) + +typedef struct { + GType hardware_type; + NmtDeviceEntryDeviceFilter device_filter; + gpointer device_filter_data; + int arptype; + + char *interface_name; + GByteArray *mac_address; + + char *label; + NmtNewtEntry *entry; + NmtNewtWidget *button; + + gboolean updating; +} NmtDeviceEntryPrivate; + +enum { + PROP_0, + PROP_LABEL, + PROP_WIDTH, + PROP_HARDWARE_TYPE, + PROP_INTERFACE_NAME, + PROP_MAC_ADDRESS, + + LAST_PROP +}; + +/** + * nmt_device_entry_new: + * @label: the label for the entry + * @width: the width of the entry + * @hardware_type: the type of #NMDevice to be selected, or + * %G_TYPE_NONE if this is for a virtual device type. + * + * Creates a new #NmtDeviceEntry, for identifying a device of type + * @hardware_type. If @hardware_type is %G_TYPE_NONE (and you do not + * set a #NmtDeviceEntryDeviceFilter), then this will only allow + * specifying an interface name, not a hardware address. + * + * Returns: a new #NmtDeviceEntry. + */ +NmtNewtWidget * +nmt_device_entry_new (const char *label, + int width, + GType hardware_type) +{ + return g_object_new (NMT_TYPE_DEVICE_ENTRY, + "label", label, + "width", width, + "hardware-type", hardware_type, + NULL); +} + +static gboolean +device_entry_parse (NmtDeviceEntry *deventry, + const char *text, + char **interface_name, + char **mac_address) +{ + NmtDeviceEntryPrivate *priv = NMT_DEVICE_ENTRY_GET_PRIVATE (deventry); + guint8 buf[NM_UTILS_HWADDR_LEN_MAX]; + char **words; + int len; + + *interface_name = *mac_address = NULL; + if (!*text) + return TRUE; + + if (priv->hardware_type == G_TYPE_NONE && !priv->device_filter) { + if (nm_utils_iface_valid_name (text)) { + *interface_name = g_strdup (text); + return TRUE; + } else + return FALSE; + } + + words = g_strsplit (text, " ", -1); + if (g_strv_length (words) > 2) { + g_strfreev (words); + return FALSE; + } + + if (words[1]) { + len = strlen (words[1]); + if (len < 3 || words[1][0] != '(' || words[1][len - 1] != ')') + goto fail; + + memmove (words[1], words[1] + 1, len - 2); + words[1][len - 2] = '\0'; + } + + if ( nm_utils_hwaddr_aton (words[0], priv->arptype, buf) + && (!words[1] || nm_utils_iface_valid_name (words[1]))) { + *mac_address = words[0]; + *interface_name = NULL; + g_free (words); + return TRUE; + } else if ( nm_utils_iface_valid_name (words[0]) + && (!words[1] || nm_utils_hwaddr_aton (words[1], priv->arptype, buf))) { + *interface_name = words[0]; + *mac_address = NULL; + g_free (words); + return TRUE; + } + + fail: + g_strfreev (words); + return FALSE; +} + +static gboolean +device_entry_validate (NmtNewtEntry *entry, + const char *text, + gpointer user_data) +{ + NmtDeviceEntry *deventry = user_data; + char *ifname, *mac; + + if (!device_entry_parse (deventry, text, &ifname, &mac)) + return FALSE; + + g_free (ifname); + g_free (mac); + return TRUE; +} + +static NMDevice * +find_device_by_interface_name (NmtDeviceEntry *deventry, + const char *interface_name) +{ + NmtDeviceEntryPrivate *priv = NMT_DEVICE_ENTRY_GET_PRIVATE (deventry); + const GPtrArray *devices; + NMDevice *device = NULL; + int i; + + devices = nm_client_get_devices (nm_client); + if (!devices) + return NULL; + + for (i = 0; i < devices->len && !device; i++) { + NMDevice *candidate = devices->pdata[i]; + + if ( priv->hardware_type != G_TYPE_NONE + && !G_TYPE_CHECK_INSTANCE_TYPE (candidate, priv->hardware_type)) + continue; + + if ( priv->device_filter + && !priv->device_filter (deventry, candidate, priv->device_filter_data)) + continue; + + if (!g_strcmp0 (interface_name, nm_device_get_iface (candidate))) + device = candidate; + } + + return device; +} + +static NMDevice * +find_device_by_mac_address (NmtDeviceEntry *deventry, + const char *mac_address) +{ + NmtDeviceEntryPrivate *priv = NMT_DEVICE_ENTRY_GET_PRIVATE (deventry); + const GPtrArray *devices; + NMDevice *device = NULL; + int i; + + devices = nm_client_get_devices (nm_client); + if (!devices) + return NULL; + + for (i = 0; i < devices->len && !device; i++) { + NMDevice *candidate = devices->pdata[i]; + char *hwaddr; + + if ( priv->hardware_type != G_TYPE_NONE + && !G_TYPE_CHECK_INSTANCE_TYPE (candidate, priv->hardware_type)) + continue; + + if ( priv->device_filter + && !priv->device_filter (deventry, candidate, priv->device_filter_data)) + continue; + + g_object_get (G_OBJECT (candidate), "hw-address", &hwaddr, NULL); + if (hwaddr && !g_ascii_strcasecmp (mac_address, hwaddr)) + device = candidate; + g_free (hwaddr); + } + + return device; +} + +static void +update_entry (NmtDeviceEntry *deventry) +{ + NmtDeviceEntryPrivate *priv = NMT_DEVICE_ENTRY_GET_PRIVATE (deventry); + const char *ifname; + char *mac, *text; + NMDevice *ifname_device, *mac_device; + + if (priv->interface_name) { + ifname = priv->interface_name; + ifname_device = find_device_by_interface_name (deventry, priv->interface_name); + } else { + ifname = NULL; + ifname_device = NULL; + } + + if (priv->mac_address) { + mac = nm_utils_hwaddr_ntoa (priv->mac_address->data, priv->arptype); + mac_device = find_device_by_mac_address (deventry, mac); + } else { + mac = NULL; + mac_device = NULL; + } + + if (!ifname && mac_device) + ifname = nm_device_get_iface (mac_device); + if (!mac && ifname_device && (priv->hardware_type != G_TYPE_NONE)) + g_object_get (G_OBJECT (ifname_device), "hw-address", &mac, NULL); + + if (ifname_device && mac_device && ifname_device != mac_device) { + /* Mismatch! */ + text = g_strdup_printf ("%s != %s", priv->interface_name, mac); + } else if (ifname && mac) { + if (ifname_device) + text = g_strdup_printf ("%s (%s)", ifname, mac); + else + text = g_strdup_printf ("%s (%s)", mac, ifname); + } else if (ifname) + text = g_strdup (ifname); + else if (mac) + text = g_strdup (mac); + else + text = g_strdup (""); + + priv->updating = TRUE; + g_object_set (G_OBJECT (priv->entry), "text", text, NULL); + priv->updating = FALSE; + g_free (text); + + g_free (mac); +} + +static gboolean +nmt_device_entry_set_interface_name (NmtDeviceEntry *deventry, + const char *interface_name) +{ + NmtDeviceEntryPrivate *priv = NMT_DEVICE_ENTRY_GET_PRIVATE (deventry); + + if (g_strcmp0 (interface_name, priv->interface_name) != 0) { + g_free (priv->interface_name); + priv->interface_name = g_strdup (interface_name); + + g_object_notify (G_OBJECT (deventry), "interface-name"); + return TRUE; + } else + return FALSE; +} + +static gboolean +nmt_device_entry_set_mac_address (NmtDeviceEntry *deventry, + GByteArray *mac_address) +{ + NmtDeviceEntryPrivate *priv = NMT_DEVICE_ENTRY_GET_PRIVATE (deventry); + gboolean changed; + + if (mac_address) + g_return_val_if_fail (mac_address->len == nm_utils_hwaddr_len (priv->arptype), FALSE); + + if (mac_address && !priv->mac_address) { + priv->mac_address = g_boxed_copy (DBUS_TYPE_G_UCHAR_ARRAY, mac_address); + changed = TRUE; + } else if (!mac_address && priv->mac_address) { + g_clear_pointer (&priv->mac_address, g_byte_array_unref); + changed = TRUE; + } else if ( mac_address && priv->mac_address + && memcmp (mac_address->data, priv->mac_address->data, mac_address->len) != 0) { + g_byte_array_unref (priv->mac_address); + priv->mac_address = g_boxed_copy (DBUS_TYPE_G_UCHAR_ARRAY, mac_address); + changed = TRUE; + } else + changed = FALSE; + + if (changed) + g_object_notify (G_OBJECT (deventry), "mac-address"); + return changed; +} + +static void +entry_text_changed (GObject *object, + GParamSpec *pspec, + gpointer deventry) +{ + NmtDeviceEntryPrivate *priv = NMT_DEVICE_ENTRY_GET_PRIVATE (deventry); + const char *text; + char *ifname, *mac; + + if (priv->updating) + return; + + text = nmt_newt_entry_get_text (priv->entry); + if (!device_entry_parse (deventry, text, &ifname, &mac)) + return; + + if (ifname) { + nmt_device_entry_set_interface_name (deventry, ifname); + g_free (ifname); + } else + nmt_device_entry_set_interface_name (deventry, NULL); + + if (mac) { + GByteArray *mac_address; + + mac_address = nm_utils_hwaddr_atoba (mac, priv->arptype); + nmt_device_entry_set_mac_address (deventry, mac_address); + g_byte_array_unref (mac_address); + g_free (mac); + } else + nmt_device_entry_set_mac_address (deventry, NULL); +} + +static void +nmt_device_entry_init (NmtDeviceEntry *deventry) +{ + NmtDeviceEntryPrivate *priv = NMT_DEVICE_ENTRY_GET_PRIVATE (deventry); + NmtNewtWidget *entry; + + priv->hardware_type = G_TYPE_NONE; + + entry = nmt_newt_entry_new (-1, 0); + priv->entry = NMT_NEWT_ENTRY (entry); + nmt_newt_entry_set_validator (priv->entry, device_entry_validate, deventry); + g_signal_connect (priv->entry, "notify::text", + G_CALLBACK (entry_text_changed), deventry); + +#if 0 + priv->button = nmt_newt_button_new (_("Select...")); + g_signal_connect (priv->button, "clicked", + G_CALLBACK (do_select_dialog), deventry); +#endif +} + +static void +nmt_device_entry_constructed (GObject *object) +{ + NmtDeviceEntryPrivate *priv = NMT_DEVICE_ENTRY_GET_PRIVATE (object); + + nmt_page_grid_append (NMT_PAGE_GRID (object), priv->label, NMT_NEWT_WIDGET (priv->entry), NULL); + + G_OBJECT_CLASS (nmt_device_entry_parent_class)->constructed (object); +} + +static void +nmt_device_entry_finalize (GObject *object) +{ + NmtDeviceEntryPrivate *priv = NMT_DEVICE_ENTRY_GET_PRIVATE (object); + + g_free (priv->interface_name); + if (priv->mac_address) + g_byte_array_unref (priv->mac_address); + + G_OBJECT_CLASS (nmt_device_entry_parent_class)->finalize (object); +} + +/** + * NmtDeviceEntryDeviceFilter: + * @deventry: the #NmtDeviceEntry + * @device: an #NMDevice + * @user_data: user data + * + * Filter function for determining which devices can be specified + * on an entry. + * + * Returns: %TRUE if @device is acceptable for @deventry + */ + +/** + * nmt_device_entry_set_device_filter: + * @deventry: the #NmtDeviceEntry + * @filter: the filter + * @user_data: data for @filter + * + * Sets a device filter on @deventry. Only devices that pass @filter + * will be recognized by @deventry. + * + * If the entry's #NmtDeviceEntry:hardware-type is not %G_TYPE_NONE, + * then only devices that both match the hardware type and are + * accepted by the filter will be allowed. + */ +void +nmt_device_entry_set_device_filter (NmtDeviceEntry *deventry, + NmtDeviceEntryDeviceFilter filter, + gpointer user_data) +{ + NmtDeviceEntryPrivate *priv = NMT_DEVICE_ENTRY_GET_PRIVATE (deventry); + + priv->device_filter = filter; + priv->device_filter_data = user_data; +} + +static void +nmt_device_entry_set_property (GObject *object, + guint prop_id, + const GValue *value, + GParamSpec *pspec) +{ + NmtDeviceEntry *deventry = NMT_DEVICE_ENTRY (object); + NmtDeviceEntryPrivate *priv = NMT_DEVICE_ENTRY_GET_PRIVATE (deventry); + const char *interface_name; + GByteArray *mac_address; + + switch (prop_id) { + case PROP_LABEL: + priv->label = g_value_dup_string (value); + break; + case PROP_WIDTH: + nmt_newt_entry_set_width (priv->entry, g_value_get_int (value)); + break; + case PROP_HARDWARE_TYPE: + priv->hardware_type = g_value_get_gtype (value); + priv->arptype = (priv->hardware_type == NM_TYPE_DEVICE_INFINIBAND) ? ARPHRD_INFINIBAND : ARPHRD_ETHER; + break; + case PROP_INTERFACE_NAME: + interface_name = g_value_get_string (value); + if (nmt_device_entry_set_interface_name (deventry, interface_name)) + update_entry (deventry); + break; + case PROP_MAC_ADDRESS: + mac_address = g_value_get_boxed (value); + if (nmt_device_entry_set_mac_address (deventry, mac_address)) + update_entry (deventry); + break; + default: + G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); + break; + } +} + +static void +nmt_device_entry_get_property (GObject *object, + guint prop_id, + GValue *value, + GParamSpec *pspec) +{ + NmtDeviceEntryPrivate *priv = NMT_DEVICE_ENTRY_GET_PRIVATE (object); + + switch (prop_id) { + case PROP_LABEL: + g_value_set_string (value, priv->label); + break; + case PROP_WIDTH: + g_value_set_int (value, nmt_newt_entry_get_width (priv->entry)); + break; + case PROP_HARDWARE_TYPE: + g_value_set_gtype (value, priv->hardware_type); + break; + case PROP_INTERFACE_NAME: + g_value_set_string (value, priv->interface_name); + break; + case PROP_MAC_ADDRESS: + g_value_set_boxed (value, priv->mac_address); + break; + default: + G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); + break; + } +} + +static void +nmt_device_entry_class_init (NmtDeviceEntryClass *deventry_class) +{ + GObjectClass *object_class = G_OBJECT_CLASS (deventry_class); + + g_type_class_add_private (deventry_class, sizeof (NmtDeviceEntryPrivate)); + + /* virtual methods */ + object_class->constructed = nmt_device_entry_constructed; + object_class->set_property = nmt_device_entry_set_property; + object_class->get_property = nmt_device_entry_get_property; + object_class->finalize = nmt_device_entry_finalize; + + /** + * NmtDeviceEntry:label: + * + * The entry's label + */ + g_object_class_install_property (object_class, PROP_LABEL, + g_param_spec_string ("label", "", "", + NULL, + G_PARAM_READWRITE | + G_PARAM_CONSTRUCT_ONLY | + G_PARAM_STATIC_STRINGS)); + /** + * NmtDeviceEntry:width: + * + * The entry's width in characters + */ + g_object_class_install_property (object_class, PROP_WIDTH, + g_param_spec_int ("width", "", "", + -1, 80, -1, + G_PARAM_READWRITE | + G_PARAM_STATIC_STRINGS)); + /** + * NmtDeviceEntry:hardware-type: + * + * The type of #NMDevice to limit the entry to, or %G_TYPE_NONE + * if the entry is for a virtual device and should not accept + * hardware addresses. + */ + g_object_class_install_property (object_class, PROP_HARDWARE_TYPE, + g_param_spec_gtype ("hardware-type", "", "", + G_TYPE_NONE, + G_PARAM_READWRITE | + G_PARAM_STATIC_STRINGS)); + /** + * NmtDeviceEntry:interface-name: + * + * The interface name of the device identified by the entry. + */ + g_object_class_install_property (object_class, PROP_INTERFACE_NAME, + g_param_spec_string ("interface-name", "", "", + NULL, + G_PARAM_READWRITE | + G_PARAM_STATIC_STRINGS)); + /** + * NmtDeviceEntry:mac-address: + * + * The hardware address of the device identified by the entry. + */ + g_object_class_install_property (object_class, PROP_MAC_ADDRESS, + g_param_spec_boxed ("mac-address", "", "", + DBUS_TYPE_G_UCHAR_ARRAY, + G_PARAM_READWRITE | + G_PARAM_STATIC_STRINGS)); +} |