Blame src/wireless-security/ws-wpa-psk.c

Packit Service 639700
// SPDX-License-Identifier: GPL-2.0+
Packit fabffb
/* NetworkManager Applet -- allow user control over networking
Packit fabffb
 *
Packit fabffb
 * Dan Williams <dcbw@redhat.com>
Packit fabffb
 *
Packit fabffb
 * Copyright 2007 - 2014 Red Hat, Inc.
Packit fabffb
 */
Packit fabffb
Packit fabffb
#include "nm-default.h"
Packit fabffb
Packit fabffb
#include <ctype.h>
Packit fabffb
#include <string.h>
Packit fabffb
Packit fabffb
#include "wireless-security.h"
Packit fabffb
#include "helpers.h"
Packit fabffb
#include "nma-ui-utils.h"
Packit fabffb
#include "utils.h"
Packit fabffb
Packit fabffb
#define WPA_PMK_LEN 32
Packit fabffb
Packit fabffb
struct _WirelessSecurityWPAPSK {
Packit fabffb
	WirelessSecurity parent;
Packit fabffb
Packit fabffb
	gboolean editing_connection;
Packit fabffb
	const char *password_flags_name;
Packit fabffb
};
Packit fabffb
Packit fabffb
static void
Packit fabffb
show_toggled_cb (GtkCheckButton *button, WirelessSecurity *sec)
Packit fabffb
{
Packit fabffb
	GtkWidget *widget;
Packit fabffb
	gboolean visible;
Packit fabffb
Packit fabffb
	widget = GTK_WIDGET (gtk_builder_get_object (sec->builder, "wpa_psk_entry"));
Packit fabffb
	g_assert (widget);
Packit fabffb
Packit fabffb
	visible = gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (button));
Packit fabffb
	gtk_entry_set_visibility (GTK_ENTRY (widget), visible);
Packit fabffb
}
Packit fabffb
Packit fabffb
static gboolean
Packit fabffb
validate (WirelessSecurity *parent, GError **error)
Packit fabffb
{
Packit fabffb
	GtkWidget *entry;
Packit fabffb
	NMSettingSecretFlags secret_flags;
Packit fabffb
	const char *key;
Packit fabffb
	gsize len;
Packit fabffb
	int i;
Packit fabffb
Packit fabffb
	entry = GTK_WIDGET (gtk_builder_get_object (parent->builder, "wpa_psk_entry"));
Packit fabffb
	g_assert (entry);
Packit fabffb
Packit fabffb
	secret_flags = nma_utils_menu_to_secret_flags (entry);
Packit fabffb
	key = gtk_entry_get_text (GTK_ENTRY (entry));
Packit fabffb
	len = key ? strlen (key) : 0;
Packit fabffb
Packit fabffb
        if (   secret_flags & NM_SETTING_SECRET_FLAG_NOT_SAVED
Packit fabffb
            || secret_flags & NM_SETTING_SECRET_FLAG_NOT_REQUIRED) {
Packit fabffb
		/* All good. */
Packit fabffb
	} else if ((len < 8) || (len > 64)) {
Packit fabffb
		widget_set_error (entry);
Packit fabffb
		g_set_error (error, NMA_ERROR, NMA_ERROR_GENERIC, _("invalid wpa-psk: invalid key-length %zu. Must be [8,63] bytes or 64 hex digits"), len);
Packit fabffb
		return FALSE;
Packit fabffb
	} else if (len == 64) {
Packit fabffb
		/* Hex PSK */
Packit fabffb
		for (i = 0; i < len; i++) {
Packit fabffb
			if (!isxdigit (key[i])) {
Packit fabffb
				widget_set_error (entry);
Packit fabffb
				g_set_error_literal (error, NMA_ERROR, NMA_ERROR_GENERIC, _("invalid wpa-psk: cannot interpret key with 64 bytes as hex"));
Packit fabffb
				return FALSE;
Packit fabffb
			}
Packit fabffb
		}
Packit fabffb
	}
Packit fabffb
	widget_unset_error (entry);
Packit fabffb
Packit fabffb
	/* passphrase can be between 8 and 63 characters inclusive */
Packit fabffb
Packit fabffb
	return TRUE;
Packit fabffb
}
Packit fabffb
Packit fabffb
static void
Packit fabffb
add_to_size_group (WirelessSecurity *parent, GtkSizeGroup *group)
Packit fabffb
{
Packit fabffb
	GtkWidget *widget;
Packit fabffb
Packit fabffb
	widget = GTK_WIDGET (gtk_builder_get_object (parent->builder, "wpa_psk_type_label"));
Packit fabffb
	gtk_size_group_add_widget (group, widget);
Packit fabffb
Packit fabffb
	widget = GTK_WIDGET (gtk_builder_get_object (parent->builder, "wpa_psk_label"));
Packit fabffb
	gtk_size_group_add_widget (group, widget);
Packit fabffb
}
Packit fabffb
Packit fabffb
static void
Packit fabffb
fill_connection (WirelessSecurity *parent, NMConnection *connection)
Packit fabffb
{
Packit fabffb
	WirelessSecurityWPAPSK *wpa_psk = (WirelessSecurityWPAPSK *) parent;
Packit fabffb
	GtkWidget *widget, *passwd_entry;
Packit fabffb
	const char *key;
Packit fabffb
	NMSettingWireless *s_wireless;
Packit fabffb
	NMSettingWirelessSecurity *s_wireless_sec;
Packit fabffb
	NMSettingSecretFlags secret_flags;
Packit fabffb
	const char *mode;
Packit fabffb
	gboolean is_adhoc = FALSE;
Packit fabffb
Packit fabffb
	s_wireless = nm_connection_get_setting_wireless (connection);
Packit fabffb
	g_assert (s_wireless);
Packit fabffb
Packit fabffb
	mode = nm_setting_wireless_get_mode (s_wireless);
Packit fabffb
	if (mode && !strcmp (mode, "adhoc"))
Packit fabffb
		is_adhoc = TRUE;
Packit fabffb
Packit fabffb
	/* Blow away the old security setting by adding a clear one */
Packit fabffb
	s_wireless_sec = (NMSettingWirelessSecurity *) nm_setting_wireless_security_new ();
Packit fabffb
	nm_connection_add_setting (connection, (NMSetting *) s_wireless_sec);
Packit fabffb
Packit fabffb
	widget = GTK_WIDGET (gtk_builder_get_object (parent->builder, "wpa_psk_entry"));
Packit fabffb
	passwd_entry = widget;
Packit fabffb
	key = gtk_entry_get_text (GTK_ENTRY (widget));
Packit fabffb
	g_object_set (s_wireless_sec, NM_SETTING_WIRELESS_SECURITY_PSK, key, NULL);
Packit fabffb
Packit fabffb
	/* Save PSK_FLAGS to the connection */
Packit fabffb
	secret_flags = nma_utils_menu_to_secret_flags (passwd_entry);
Packit fabffb
	nm_setting_set_secret_flags (NM_SETTING (s_wireless_sec), NM_SETTING_WIRELESS_SECURITY_PSK,
Packit fabffb
	                             secret_flags, NULL);
Packit fabffb
Packit fabffb
	/* Update secret flags and popup when editing the connection */
Packit fabffb
	if (wpa_psk->editing_connection)
Packit fabffb
		nma_utils_update_password_storage (passwd_entry, secret_flags,
Packit fabffb
		                                   NM_SETTING (s_wireless_sec), wpa_psk->password_flags_name);
Packit fabffb
Packit fabffb
	wireless_security_clear_ciphers (connection);
Packit fabffb
	if (is_adhoc) {
Packit fabffb
		/* Ad-Hoc settings as specified by the supplicant */
Packit Service 639700
		g_object_set (s_wireless_sec, NM_SETTING_WIRELESS_SECURITY_KEY_MGMT, "wpa-psk", NULL);
Packit Service 639700
		nm_setting_wireless_security_add_proto (s_wireless_sec, "rsn");
Packit Service 639700
		nm_setting_wireless_security_add_pairwise (s_wireless_sec, "ccmp");
Packit Service 639700
		nm_setting_wireless_security_add_group (s_wireless_sec, "ccmp");
Packit fabffb
	} else {
Packit fabffb
		g_object_set (s_wireless_sec, NM_SETTING_WIRELESS_SECURITY_KEY_MGMT, "wpa-psk", NULL);
Packit fabffb
Packit fabffb
		/* Just leave ciphers and protocol empty, the supplicant will
Packit fabffb
		 * figure that out magically based on the AP IEs and card capabilities.
Packit fabffb
		 */
Packit fabffb
	}
Packit fabffb
}
Packit fabffb
Packit fabffb
static void
Packit fabffb
update_secrets (WirelessSecurity *parent, NMConnection *connection)
Packit fabffb
{
Packit fabffb
	helper_fill_secret_entry (connection,
Packit fabffb
	                          parent->builder,
Packit fabffb
	                          "wpa_psk_entry",
Packit fabffb
	                          NM_TYPE_SETTING_WIRELESS_SECURITY,
Packit fabffb
	                          (HelperSecretFunc) nm_setting_wireless_security_get_psk);
Packit fabffb
}
Packit fabffb
Packit fabffb
WirelessSecurityWPAPSK *
Packit fabffb
ws_wpa_psk_new (NMConnection *connection, gboolean secrets_only)
Packit fabffb
{
Packit fabffb
	WirelessSecurity *parent;
Packit fabffb
	WirelessSecurityWPAPSK *sec;
Packit fabffb
	NMSetting *setting = NULL;
Packit fabffb
	GtkWidget *widget;
Packit fabffb
Packit fabffb
	parent = wireless_security_init (sizeof (WirelessSecurityWPAPSK),
Packit fabffb
	                                 validate,
Packit fabffb
	                                 add_to_size_group,
Packit fabffb
	                                 fill_connection,
Packit fabffb
	                                 update_secrets,
Packit fabffb
	                                 NULL,
Packit fabffb
	                                 "/org/freedesktop/network-manager-applet/ws-wpa-psk.ui",
Packit fabffb
	                                 "wpa_psk_notebook",
Packit fabffb
	                                 "wpa_psk_entry");
Packit fabffb
	if (!parent)
Packit fabffb
		return NULL;
Packit fabffb
Packit Service 639700
	parent->adhoc_compatible = TRUE;
Packit fabffb
	sec = (WirelessSecurityWPAPSK *) parent;
Packit fabffb
	sec->editing_connection = secrets_only ? FALSE : TRUE;
Packit fabffb
	sec->password_flags_name = NM_SETTING_WIRELESS_SECURITY_PSK;
Packit fabffb
Packit fabffb
	widget = GTK_WIDGET (gtk_builder_get_object (parent->builder, "wpa_psk_entry"));
Packit fabffb
	g_assert (widget);
Packit fabffb
	g_signal_connect (G_OBJECT (widget), "changed",
Packit fabffb
	                  (GCallback) wireless_security_changed_cb,
Packit fabffb
	                  sec);
Packit fabffb
	gtk_entry_set_width_chars (GTK_ENTRY (widget), 28);
Packit fabffb
Packit fabffb
	/* Create password-storage popup menu for password entry under entry's secondary icon */
Packit fabffb
	if (connection)
Packit fabffb
		setting = (NMSetting *) nm_connection_get_setting_wireless_security (connection);
Packit fabffb
	nma_utils_setup_password_storage (widget, 0, setting, sec->password_flags_name,
Packit fabffb
	                                  FALSE, secrets_only);
Packit fabffb
Packit fabffb
	/* Fill secrets, if any */
Packit fabffb
	if (connection)
Packit fabffb
		update_secrets (WIRELESS_SECURITY (sec), connection);
Packit fabffb
Packit fabffb
	widget = GTK_WIDGET (gtk_builder_get_object (parent->builder, "show_checkbutton_wpa"));
Packit fabffb
	g_assert (widget);
Packit fabffb
	g_signal_connect (G_OBJECT (widget), "toggled",
Packit fabffb
	                  (GCallback) show_toggled_cb,
Packit fabffb
	                  sec);
Packit fabffb
Packit fabffb
	/* Hide WPA/RSN for now since this can be autodetected by NM and the
Packit fabffb
	 * supplicant when connecting to the AP.
Packit fabffb
	 */
Packit fabffb
Packit fabffb
	widget = GTK_WIDGET (gtk_builder_get_object (parent->builder, "wpa_psk_type_combo"));
Packit fabffb
	g_assert (widget);
Packit fabffb
	gtk_widget_hide (widget);
Packit fabffb
Packit fabffb
	widget = GTK_WIDGET (gtk_builder_get_object (parent->builder, "wpa_psk_type_label"));
Packit fabffb
	g_assert (widget);
Packit fabffb
	gtk_widget_hide (widget);
Packit fabffb
Packit fabffb
	return sec;
Packit fabffb
}
Packit fabffb