[gnome-keyring] egg: Add decoder for decimal data, analogous to the hex decoder.



commit 9050ad6a4112097b219e6f781ea1d3ecaf1bdd22
Author: Stef Walter <stefw collabora co uk>
Date:   Mon Sep 12 16:58:03 2011 +0200

    egg: Add decoder for decimal data, analogous to the hex decoder.

 .gitignore               |    1 +
 egg/Makefile.am          |    1 +
 egg/egg-decimal.c        |  101 ++++++++++++++++++++++++++++++++++
 egg/egg-decimal.h        |   33 +++++++++++
 egg/tests/Makefile.am    |    1 +
 egg/tests/test-decimal.c |  136 ++++++++++++++++++++++++++++++++++++++++++++++
 6 files changed, 273 insertions(+), 0 deletions(-)
---
diff --git a/.gitignore b/.gitignore
index a186ec0..dc31879 100644
--- a/.gitignore
+++ b/.gitignore
@@ -86,6 +86,7 @@ run-tests
 
 /egg/tests/test-asn1
 /egg/tests/test-cleanup
+/egg/tests/test-decimal
 /egg/tests/test-dh
 /egg/tests/test-dn
 /egg/tests/test-hex
diff --git a/egg/Makefile.am b/egg/Makefile.am
index eb2268a..917d884 100644
--- a/egg/Makefile.am
+++ b/egg/Makefile.am
@@ -30,6 +30,7 @@ libegg_la_SOURCES = \
 	egg-dh.c egg-dh.h \
 	egg-dn.c egg-dn.h \
 	egg-error.h \
+	egg-decimal.c egg-decimal.h \
 	egg-hex.c egg-hex.h \
 	egg-hkdf.c egg-hkdf.h \
 	egg-libgcrypt.c egg-libgcrypt.h \
diff --git a/egg/egg-decimal.c b/egg/egg-decimal.c
new file mode 100644
index 0000000..6f3921d
--- /dev/null
+++ b/egg/egg-decimal.c
@@ -0,0 +1,101 @@
+/*
+ * gnome-keyring
+ *
+ * Copyright (C) 2011 Collabora Ltd.
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public 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 Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
+ * 02111-1307, USA.
+ *
+ * Author: Stef Walter <stefw collabora co uk>
+ */
+
+#include "config.h"
+
+#include "egg-decimal.h"
+
+#include <string.h>
+
+gpointer
+egg_decimal_decode (const gchar *data,
+                    gssize n_data,
+                    gsize *n_decoded)
+{
+	gboolean saw_non_zero = FALSE;
+	guint8 *digits = NULL;
+	guint n_digits = 0;
+	guchar *usg;
+	gchar ch;
+	guint carry;
+	guint i;
+	guint cleared;
+	guchar *at_byte;
+	guint at_bit;
+	gsize length;
+
+	g_return_val_if_fail (data, NULL);
+
+	if (n_data < 0)
+		n_data = strlen (data);
+
+	/* Convert everything to an array of digits */
+	digits = g_new0 (guint8, n_data);
+	for (i = 0; i < n_data; i++) {
+		ch = data[i];
+		if (ch < '0' || ch > '9') {
+			g_free (digits);
+			return NULL;
+		} else if (ch != '0' || saw_non_zero) {
+			digits[n_digits] = ch - '0';
+			n_digits++;
+			saw_non_zero = TRUE;
+		}
+	}
+
+	usg = g_new0 (guchar, n_data);
+	cleared = 0;
+	at_byte = usg + n_data - 1;
+	at_bit = 0;
+
+	/* for every digit ... */
+	while (cleared < n_digits) {
+		/* ... store the bit depending on whether last digit is odd */
+		guchar bit = (digits[n_digits - 1] % 2);
+		*at_byte |= (bit << at_bit);
+		if (at_bit++ == 7) {
+			at_bit = 0;
+			at_byte--;
+			g_assert (at_byte >= usg);
+		}
+
+		/* ... divide digits by two */
+		carry = 0;
+		for (i = cleared; i < n_digits; i++) {
+			gboolean odd = (digits[i] % 2);
+			digits[i] = digits[i] / 2 + carry;
+			carry = odd ? 5 : 0;
+			if (i == cleared && digits[i] == 0)
+				cleared++;
+		}
+	}
+
+	if (at_bit == 0)
+		at_byte++;
+
+	length = n_data - (at_byte - usg);
+	memmove (usg, at_byte, length);
+	if (n_decoded)
+		*n_decoded = length;
+	return usg;
+}
diff --git a/egg/egg-decimal.h b/egg/egg-decimal.h
new file mode 100644
index 0000000..24602d4
--- /dev/null
+++ b/egg/egg-decimal.h
@@ -0,0 +1,33 @@
+/*
+ * gnome-keyring
+ *
+ * Copyright (C) 2011 Collabora Ltd.
+ *
+ * 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.
+ *
+ * Author: Stef Walter <stefw collabora co uk>
+ */
+
+#ifndef EGG_DECIMAL_H_
+#define EGG_DECIMAL_H_
+
+#include <glib.h>
+
+gpointer              egg_decimal_decode                     (const gchar *data,
+                                                              gssize n_data,
+                                                              gsize *n_decoded);
+
+#endif /* EGG_DECIMAL_H_ */
diff --git a/egg/tests/Makefile.am b/egg/tests/Makefile.am
index aa5fac7..84bc5c1 100644
--- a/egg/tests/Makefile.am
+++ b/egg/tests/Makefile.am
@@ -18,6 +18,7 @@ TEST_PROGS = \
 	test-asn1 \
 	test-dn \
 	test-cleanup \
+	test-decimal \
 	test-hex \
 	test-hkdf \
 	test-oid \
diff --git a/egg/tests/test-decimal.c b/egg/tests/test-decimal.c
new file mode 100644
index 0000000..4996caf
--- /dev/null
+++ b/egg/tests/test-decimal.c
@@ -0,0 +1,136 @@
+/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */
+/* unit-test-util.c: Test gck-util.c
+
+   Copyright (C) 2011 Collabora Ltd.
+
+   The Gnome Keyring Library is free software; you can redistribute it and/or
+   modify it under the terms of the GNU Library General Public License as
+   published by the Free Software Foundation; either version 2 of the
+   License, or (at your option) any later version.
+
+   The Gnome Keyring Library 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
+   Library General Public License for more details.
+
+   You should have received a copy of the GNU Library General Public
+   License along with the Gnome Library; see the file COPYING.LIB.  If not,
+   write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+   Boston, MA 02111-1307, USA.
+
+   Author: Stef Walter <stefw collabora co uk>
+*/
+
+#include "config.h"
+
+#include "egg/egg-decimal.h"
+#include "egg/egg-testing.h"
+
+#include <stdlib.h>
+#include <stdio.h>
+#include <string.h>
+
+typedef struct {
+	const gchar *decimal;
+	const gchar *result;
+	gsize length;
+} TestDecimal;
+
+static TestDecimal decimal_fixtures[] = {
+	{ "35", "\x23", 1 },
+	{ "2048", "\x08\x00", 2 },
+	{ "209328042309", "\x30\xBC\xEC\x71\x45", 5 },
+	{ "0002048", "\x08\x00", 2 },
+	{ "20480000", "\x01\x38\x80\x00", 4 },
+	{ "2521368004379664277055118629334750468402260732427135194366161280379"
+	  "6661942758264006541602516197962800670706891657576953487072507942596"
+	  "6374424122494538738176144081024494339591946626053519298721214406983"
+	  "3606711407748948613853344896472022957466922483835202523097287025430"
+	  "5078987427716449092979306938082113263545271808186244259338032237756"
+	  "9175402872261192563172038927792670653714043622677577357160052045893"
+	  "9513984477743817388078699536715866468499111753894230211101792648120"
+	  "1128688482121549927434503046858485918719606735307033123916744787670"
+	  "4316000505177621722934283063062034258685067324811300901286708201589"
+	  "59867993533757",
+	  "\xC7\xBB\x16\xF4\xB8\x04\x24\x0F\xFC\xC2\xA7\xAF\x8C\x6E\x67\xE1\x16"
+	  "\x7F\xEB\xFA\x7F\xAA\x9D\xFD\x7C\xF2\x75\xB8\xA5\x1F\x27\x35\xF2\xD4"
+	  "\x9D\x78\xFB\xF6\x5C\xED\x10\xB4\xE4\x32\x58\x2D\xC9\x1E\x86\x54\xF7"
+	  "\x89\x7F\x03\x84\x68\x32\x76\xA9\xA7\x97\xC3\xA3\x6F\x7A\x46\x85\x43"
+	  "\x5E\x14\x4D\x47\x01\x81\x06\xE5\xC0\x61\xD7\xC8\x7C\x9B\xE1\x9D\x84"
+	  "\x87\x75\x77\x80\x0E\xAE\x91\xB1\x05\x12\xDE\x92\xF2\x98\x84\x1F\x43"
+	  "\xD4\xC4\x57\x77\x95\xC5\xE6\x82\xEE\xEA\x0A\xB3\xDD\x8C\x44\x45\x9A"
+	  "\x12\xAC\xF9\xC2\x22\xA1\x3F\x03\x31\xDD\x84\xF7\x75\x51\xE0\xFA\x24"
+	  "\x8E\x6F\xE9\x58\x4C\xA3\x42\x73\xB9\x5E\x2F\x0D\xCC\xDC\x22\x8A\x48"
+	  "\x75\x4A\x76\xA2\x9D\x03\xBA\x5F\xC8\x57\xB5\x1F\x5C\x85\x7E\x8C\x0F"
+	  "\xF2\x73\xDA\x96\x67\x7C\xC6\x4D\x54\x2C\x45\x63\xD1\xA6\x7F\xF1\xA0"
+	  "\x1F\x3F\x9E\xDF\xF3\x7F\x24\x3D\x6E\xB8\xF7\x4C\xC8\xA7\x27\x95\xA1"
+	  "\xDA\x8F\x98\x32\x32\x1B\x7D\xB6\x1B\xFC\x8D\x73\x7C\xD1\x48\x99\xD0"
+	  "\xAC\x7C\xF1\x5B\x95\xA5\xFE\xD8\x12\x57\x5C\x7A\x6B\xC5\x5C\x7D\x92"
+	  "\xB1\x91\x88\x36\x58\x19\x30\x67\x2D\x73\xF3\x5A\xA6\x31\xC4\x5C\x2D"
+	  "\x3D"
+	  , 256 }
+};
+
+static const gchar *decimal_failures[] = {
+	"-35",
+	"abcd",
+	" 3 33",
+};
+
+static void
+test_decode_success (gconstpointer data)
+{
+	const TestDecimal *fixture = data;
+	guchar *decoded;
+	gsize n_decoded;
+
+	decoded = egg_decimal_decode (fixture->decimal, -1, &n_decoded);
+	egg_assert_cmpmem (fixture->result, fixture->length, ==, decoded, n_decoded);
+	g_free (decoded);
+}
+
+static void
+test_decode_failure (gconstpointer data)
+{
+	const gchar *failure = data;
+	guchar *decoded;
+	gsize n_decoded;
+
+	decoded = egg_decimal_decode (failure, -1, &n_decoded);
+	g_assert (decoded == NULL);
+}
+
+int
+main (int argc, char **argv)
+{
+	gchar *name;
+	gchar *decimal;
+	const gchar *suffix;
+	guint i;
+
+	g_test_init (&argc, &argv, NULL);
+
+	for (i = 0; i < G_N_ELEMENTS (decimal_fixtures); i++) {
+		/* Ellipsize long numbers in test names */
+		decimal = g_strndup (decimal_fixtures[i].decimal, 41);
+		if (strlen (decimal) == 41) {
+			decimal[40] = 0;
+			suffix = "_long";
+		} else {
+			suffix = "";
+		}
+		name = g_strdup_printf ("/decimal/decode-success/%s%s", decimal, suffix);
+		g_test_add_data_func (name, &decimal_fixtures[i], test_decode_success);
+		g_free (name);
+		g_free (decimal);
+	}
+
+	for (i = 0; i < G_N_ELEMENTS (decimal_failures); i++) {
+		name = g_strdup_printf ("/decimal/decode-failure/%s", decimal_failures[i]);
+		g_strcanon (name, "abcdefghijklmnopqrstuvwxyz-_/0123456789", '_');
+		g_test_add_data_func (name, decimal_failures[i], test_decode_failure);
+		g_free (name);
+	}
+
+	return g_test_run ();
+}



[Date Prev][Date Next]   [Thread Prev][Thread Next]   [Thread Index] [Date Index] [Author Index]