mirror of
https://gitlab.gnome.org/GNOME/libsecret.git
synced 2025-01-18 18:08:36 +00:00
151 lines
3.1 KiB
C
151 lines
3.1 KiB
C
|
/*
|
||
|
* gnome-keyring
|
||
|
*
|
||
|
* Copyright (C) 2008 Stefan Walter
|
||
|
*
|
||
|
* This program is free software; you can redistribute it and/or modify
|
||
|
* it under the terms of the GNU Lesser General License as
|
||
|
* published by the Free Software Foundation; either version 2.1 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
|
||
|
* Lesser General License for more details.
|
||
|
*
|
||
|
* You should have received a copy of the GNU Lesser General
|
||
|
* License along with this program; if not, write to the Free Software
|
||
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
|
||
|
* 02111-1307, USA.
|
||
|
*/
|
||
|
|
||
|
#include "config.h"
|
||
|
|
||
|
#include "egg-hex.h"
|
||
|
|
||
|
#include <string.h>
|
||
|
|
||
|
static const char HEXC_UPPER[] = "0123456789ABCDEF";
|
||
|
static const char HEXC_LOWER[] = "0123456789abcdef";
|
||
|
|
||
|
gpointer
|
||
|
egg_hex_decode (const gchar *data, gssize n_data, gsize *n_decoded)
|
||
|
{
|
||
|
return egg_hex_decode_full (data, n_data, 0, 1, n_decoded);
|
||
|
}
|
||
|
|
||
|
gpointer
|
||
|
egg_hex_decode_full (const gchar *data, gssize n_data,
|
||
|
gchar delim, guint group, gsize *n_decoded)
|
||
|
{
|
||
|
guchar *result;
|
||
|
guchar *decoded;
|
||
|
gushort j;
|
||
|
gint state = 0;
|
||
|
gint part = 0;
|
||
|
const gchar* pos;
|
||
|
|
||
|
g_return_val_if_fail (data || !n_data, NULL);
|
||
|
g_return_val_if_fail (n_decoded, NULL);
|
||
|
g_return_val_if_fail (group >= 1, NULL);
|
||
|
|
||
|
if (n_data == -1)
|
||
|
n_data = strlen (data);
|
||
|
|
||
|
decoded = result = g_malloc0 ((n_data / 2) + 1);
|
||
|
*n_decoded = 0;
|
||
|
|
||
|
while (n_data > 0 && state == 0) {
|
||
|
|
||
|
if (decoded != result && delim) {
|
||
|
if (*data != delim) {
|
||
|
state = -1;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
++data;
|
||
|
--n_data;
|
||
|
}
|
||
|
|
||
|
while (part < group && n_data > 0) {
|
||
|
|
||
|
/* Find the position */
|
||
|
pos = strchr (HEXC_UPPER, g_ascii_toupper (*data));
|
||
|
if (pos == 0) {
|
||
|
if (n_data > 0)
|
||
|
state = -1;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
j = pos - HEXC_UPPER;
|
||
|
if(!state) {
|
||
|
*decoded = (j & 0xf) << 4;
|
||
|
state = 1;
|
||
|
} else {
|
||
|
*decoded |= (j & 0xf);
|
||
|
(*n_decoded)++;
|
||
|
decoded++;
|
||
|
state = 0;
|
||
|
part++;
|
||
|
}
|
||
|
|
||
|
++data;
|
||
|
--n_data;
|
||
|
}
|
||
|
|
||
|
part = 0;
|
||
|
}
|
||
|
|
||
|
/* Parsing error */
|
||
|
if (state != 0) {
|
||
|
g_free (result);
|
||
|
result = NULL;
|
||
|
}
|
||
|
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
gchar*
|
||
|
egg_hex_encode (gconstpointer data, gsize n_data)
|
||
|
{
|
||
|
return egg_hex_encode_full (data, n_data, TRUE, '\0', 0);
|
||
|
}
|
||
|
|
||
|
gchar*
|
||
|
egg_hex_encode_full (gconstpointer data, gsize n_data,
|
||
|
gboolean upper_case, gchar delim, guint group)
|
||
|
{
|
||
|
GString *result;
|
||
|
const gchar *input;
|
||
|
const char *hexc;
|
||
|
gsize bytes;
|
||
|
guchar j;
|
||
|
|
||
|
g_return_val_if_fail (data || !n_data, NULL);
|
||
|
|
||
|
input = data;
|
||
|
hexc = upper_case ? HEXC_UPPER : HEXC_LOWER;
|
||
|
|
||
|
result = g_string_sized_new (n_data * 2 + 1);
|
||
|
bytes = 0;
|
||
|
|
||
|
while (n_data > 0) {
|
||
|
|
||
|
if (group && bytes && (bytes % group) == 0)
|
||
|
g_string_append_c (result, delim);
|
||
|
|
||
|
j = *(input) >> 4 & 0xf;
|
||
|
g_string_append_c (result, hexc[j]);
|
||
|
|
||
|
j = *(input++) & 0xf;
|
||
|
g_string_append_c (result, hexc[j]);
|
||
|
|
||
|
++bytes;
|
||
|
--n_data;
|
||
|
}
|
||
|
|
||
|
/* Make sure still null terminated */
|
||
|
return g_string_free (result, FALSE);
|
||
|
}
|
||
|
|