/* -*- mode: c; c-basic-offset: 2; indent-tabs-mode: nil; -*- */ /* cc-qr-code.c * * Copyright 2019 Purism SPC * * 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 3 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 . * * Author(s): * Mohammed Sadiq * * SPDX-License-Identifier: GPL-3.0-or-later */ #undef G_LOG_DOMAIN #define G_LOG_DOMAIN "cc-qr-code" #ifdef HAVE_CONFIG_H #include "config.h" #endif #include "cc-qr-code.h" #include "qrcodegen.c" /** * SECTION: cc-qr_code * @title: CcQrCode * @short_description: A Simple QR Code wrapper around libqrencode * @include: "cc-qr-code.h" * * Generate a QR image from a given text. */ #define BYTES_PER_R8G8B8 3 struct _CcQrCode { GObject parent_instance; gchar *text; GdkTexture *texture; gint size; }; G_DEFINE_TYPE (CcQrCode, cc_qr_code, G_TYPE_OBJECT) static void cc_qr_code_finalize (GObject *object) { CcQrCode *self = (CcQrCode *) object; g_clear_object (&self->texture); g_clear_pointer (&self->text, g_free); G_OBJECT_CLASS (cc_qr_code_parent_class)->finalize (object); } static void cc_qr_code_class_init (CcQrCodeClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); object_class->finalize = cc_qr_code_finalize; } static void cc_qr_code_init (CcQrCode *self) { } CcQrCode * cc_qr_code_new (void) { return g_object_new (CC_TYPE_QR_CODE, NULL); } gboolean cc_qr_code_set_text (CcQrCode *self, const gchar *text) { g_return_val_if_fail (CC_IS_QR_CODE (self), FALSE); g_return_val_if_fail (!text || *text, FALSE); if (g_strcmp0 (text, self->text) == 0) return FALSE; g_clear_object (&self->texture); g_free (self->text); self->text = g_strdup (text); return TRUE; } static void cc_fill_pixel (GByteArray *array, guint8 value, int pixel_size) { guint i; for (i = 0; i < pixel_size; i++) { g_byte_array_append (array, &value, 1); /* R */ g_byte_array_append (array, &value, 1); /* G */ g_byte_array_append (array, &value, 1); /* B */ } } GdkPaintable * cc_qr_code_get_paintable (CcQrCode *self, gint size) { uint8_t qr_code[qrcodegen_BUFFER_LEN_FOR_VERSION (qrcodegen_VERSION_MAX)]; uint8_t temp_buf[qrcodegen_BUFFER_LEN_FOR_VERSION (qrcodegen_VERSION_MAX)]; g_autoptr (GBytes) bytes = NULL; GByteArray *qr_matrix; gint pixel_size, qr_size, total_size; gint column, row, i; gboolean success = FALSE; g_return_val_if_fail (CC_IS_QR_CODE (self), NULL); g_return_val_if_fail (size > 0, NULL); if (!self->text || !*self->text) { g_warn_if_reached (); cc_qr_code_set_text (self, "invalid text"); } if (self->texture && self->size == size) return GDK_PAINTABLE (self->texture); self->size = size; success = qrcodegen_encodeText (self->text, temp_buf, qr_code, qrcodegen_Ecc_LOW, qrcodegen_VERSION_MIN, qrcodegen_VERSION_MAX, qrcodegen_Mask_AUTO, FALSE); if (!success) return NULL; qr_size = qrcodegen_getSize (qr_code); pixel_size = MAX (1, size / (qr_size)); total_size = qr_size * pixel_size; qr_matrix = g_byte_array_sized_new (total_size * total_size * pixel_size * BYTES_PER_R8G8B8); for (column = 0; column < total_size; column++) { for (i = 0; i < pixel_size; i++) { for (row = 0; row < total_size / pixel_size; row++) { if (qrcodegen_getModule (qr_code, column, row)) cc_fill_pixel (qr_matrix, 0x00, pixel_size); else cc_fill_pixel (qr_matrix, 0xff, pixel_size); } } } bytes = g_byte_array_free_to_bytes (qr_matrix); g_clear_object (&self->texture); self->texture = gdk_memory_texture_new (total_size, total_size, GDK_MEMORY_R8G8B8, bytes, total_size * BYTES_PER_R8G8B8); return GDK_PAINTABLE (self->texture); } static gchar * escape_string (const gchar *str, gboolean quote) { GString *string; const char *next; if (!str) return NULL; string = g_string_new (""); if (quote) g_string_append_c (string, '"'); while ((next = strpbrk (str, "\\;,:\""))) { g_string_append_len (string, str, next - str); g_string_append_c (string, '\\'); g_string_append_c (string, *next); str = next + 1; } g_string_append (string, str); if (quote) g_string_append_c (string, '"'); return g_string_free (string, FALSE); } static const gchar * get_connection_security_type (NMConnection *c) { NMSettingWirelessSecurity *setting; const char *key_mgmt; g_return_val_if_fail (c, "nopass"); setting = nm_connection_get_setting_wireless_security (c); if (!setting) return "nopass"; key_mgmt = nm_setting_wireless_security_get_key_mgmt (setting); /* No IEEE 802.1x */ if (g_strcmp0 (key_mgmt, "none") == 0) return "WEP"; if (g_strcmp0 (key_mgmt, "wpa-psk") == 0) return "WPA"; if (g_strcmp0 (key_mgmt, "sae") == 0) return "SAE"; return "nopass"; } gboolean is_qr_code_supported (NMConnection *c) { NMSettingWirelessSecurity *setting; const char *key_mgmt; NMSettingConnection *s_con; guint64 timestamp; g_return_val_if_fail (c, TRUE); s_con = nm_connection_get_setting_connection (c); timestamp = nm_setting_connection_get_timestamp (s_con); /* Check timestamp to determine if connection was successful in the past */ if (timestamp == 0) return FALSE; setting = nm_connection_get_setting_wireless_security (c); if (!setting) return TRUE; key_mgmt = nm_setting_wireless_security_get_key_mgmt (setting); if (g_str_equal (key_mgmt, "none") || g_str_equal (key_mgmt, "wpa-psk") || g_str_equal (key_mgmt, "sae")) return TRUE; return FALSE; } gchar * get_wifi_password (NMConnection *c) { NMSettingWirelessSecurity *setting; const gchar *sec_type, *password; gint wep_index; sec_type = get_connection_security_type (c); setting = nm_connection_get_setting_wireless_security (c); if (g_str_equal (sec_type, "nopass")) return NULL; if (g_str_equal (sec_type, "WEP")) { wep_index = nm_setting_wireless_security_get_wep_tx_keyidx (setting); password = nm_setting_wireless_security_get_wep_key (setting, wep_index); } else { password = nm_setting_wireless_security_get_psk (setting); } return g_strdup (password); } /* Generate a string representing the connection * An example generated text: * WIFI:S:ssid;T:WPA;P:my-valid-pass;H:true; * Where, * S = ssid, T = security, P = password, H = hidden (Optional) * * See https://github.com/zxing/zxing/wiki/Barcode-Contents#wi-fi-network-config-android-ios-11 */ gchar * get_qr_string_for_connection (NMConnection *c) { NMSettingWireless *setting; g_autofree char *ssid_text = NULL; g_autofree char *escaped_ssid = NULL; g_autofree char *password_str = NULL; g_autofree char *escaped_password = NULL; GString *string; GBytes *ssid; gboolean hidden; setting = nm_connection_get_setting_wireless (c); ssid = nm_setting_wireless_get_ssid (setting); if (!ssid) return NULL; string = g_string_new ("WIFI:S:"); /* SSID */ ssid_text = nm_utils_ssid_to_utf8 (g_bytes_get_data (ssid, NULL), g_bytes_get_size (ssid)); escaped_ssid = escape_string (ssid_text, FALSE); g_string_append (string, escaped_ssid); g_string_append_c (string, ';'); /* Security type */ g_string_append (string, "T:"); g_string_append (string, get_connection_security_type (c)); g_string_append_c (string, ';'); /* Password */ g_string_append (string, "P:"); password_str = get_wifi_password (c); escaped_password = escape_string (password_str, FALSE); if (escaped_password) g_string_append (string, escaped_password); g_string_append_c (string, ';'); /* WiFi Hidden */ hidden = nm_setting_wireless_get_hidden (setting); if (hidden) g_string_append (string, "H:true"); g_string_append_c (string, ';'); return g_string_free (string, FALSE); }