summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorVadim Bendebury <vbendeb@chromium.org>2015-03-27 16:08:04 -0700
committerPatrick Georgi <pgeorgi@google.com>2015-04-22 08:50:54 +0200
commit243c614134cd29b764425b18e85aca7c6ea21ab7 (patch)
tree5e072df85934f67192a09a169427cc5d11156df5
parent801aa7c355702efe2a6e505d59431980099a1f14 (diff)
lib: add base64 decoder
It became necessary to decode base64 data retrieved from VPD and convert it into binary for inclusion in the device tree. The patch introduces the decoder function based on the description found in http://en.wikipedia.org/wiki/Base64. An open source implementation from http://base64.sourceforge.net was considered, in the end the only thing borrowed from it is the table to translate base64 ascii characters into numbers in 0..63 range. BRANCH=none BUG=chromium:450169 TEST=created a test harness generating random contents of random size (in 8 to 32766 bytes range), then converting the contents into base64 using the Linux utility, and then converting it back to binary using this function and comparing the results. It succeeded 1700 iterations before it was stopped. Change-Id: I502f2c9494c99ba95ece37a7220c0c70c4755be2 Signed-off-by: Patrick Georgi <pgeorgi@chromium.org> Original-Commit-Id: 6609f76e1559d3cdd402276055c99e0de7da27c8 Original-Change-Id: I5ed68af3a4daead50c44ae0f0c63d836f4b66851 Original-Signed-off-by: Vadim Bendebury <vbendeb@chromium.org> Original-Reviewed-on: https://chromium-review.googlesource.com/262945 Original-Reviewed-by: Aaron Durbin <adurbin@chromium.org> Reviewed-on: http://review.coreboot.org/9892 Tested-by: build bot (Jenkins) Reviewed-by: Stefan Reinauer <stefan.reinauer@coreboot.org>
-rw-r--r--src/include/b64_decode.h35
-rw-r--r--src/lib/Makefile.inc1
-rw-r--r--src/lib/b64_decode.c146
3 files changed, 182 insertions, 0 deletions
diff --git a/src/include/b64_decode.h b/src/include/b64_decode.h
new file mode 100644
index 0000000000..b52719671e
--- /dev/null
+++ b/src/include/b64_decode.h
@@ -0,0 +1,35 @@
+/*
+ * Copyright (C) 2015 Google, Inc.
+ *
+ * This software is licensed under the terms of the GNU General Public
+ * License version 2, as published by the Free Software Foundation, and
+ * may be copied, distributed, and modified under those terms.
+ *
+ * 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.
+ */
+
+#ifndef __INCLUDE_B64_DECODE_H__
+#define __INCLUDE_B64_DECODE_H__
+
+#include <stddef.h>
+#include <stdint.h>
+
+/*
+ * A function to convert a buffer of base64 format data into its source.
+ *
+ * The user provides output buffer of the size guaranteed to fit the result.
+ *
+ * Returns the size of the decoded data or zero if invalid charactes were
+ * encountered in the input buffer.
+ */
+size_t b64_decode(const uint8_t *input_data,
+ size_t input_length,
+ uint8_t *output_data);
+
+/* A macro to derive decoded size of a base64 encoded blob. */
+#define B64_DECODED_SIZE(encoded_size) (((encoded_size) * 3)/4)
+
+#endif
diff --git a/src/lib/Makefile.inc b/src/lib/Makefile.inc
index d4f0452b6f..19ff2b291a 100644
--- a/src/lib/Makefile.inc
+++ b/src/lib/Makefile.inc
@@ -97,6 +97,7 @@ ramstage-$(CONFIG_COOP_MULTITASKING) += thread.c
ramstage-$(CONFIG_TIMER_QUEUE) += timer_queue.c
ramstage-$(CONFIG_GENERIC_GPIO_LIB) += gpio.c
ramstage-$(CONFIG_GENERIC_UDELAY) += timer.c
+ramstage-y += b64_decode.c
romstage-y += cbmem_common.c dynamic_cbmem.c
ramstage-y += cbmem_common.c dynamic_cbmem.c
diff --git a/src/lib/b64_decode.c b/src/lib/b64_decode.c
new file mode 100644
index 0000000000..85f43c0747
--- /dev/null
+++ b/src/lib/b64_decode.c
@@ -0,0 +1,146 @@
+/*
+ * Copyright (C) 2015 Google, Inc.
+ *
+ * This software is licensed under the terms of the GNU General Public
+ * License version 2, as published by the Free Software Foundation, and
+ * may be copied, distributed, and modified under those terms.
+ *
+ * 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.
+ */
+
+#include <b64_decode.h>
+#include <console/console.h>
+
+/*
+ * Translation Table to decode base64 ASCII stream into binary. Borrowed from
+ *
+ * http://base64.sourceforge.net/b64.c.
+ *
+ */
+static const char cd64[] = "|$$$}rstuvwxyz{$$$$$$$>?@ABCDEFGHIJKLMN"
+ "OPQRSTUVW$$$$$$XYZ[\\]^_`abcdefghijklmnopq";
+
+struct buffer_descriptor {
+ const uint8_t *input_buffer;
+ size_t data_size;
+ size_t input_index;
+};
+
+#define isalnum(c) ((((c) >= 'a') && ((c) <= 'z')) || \
+ (((c) >= 'A') && ((c) <= 'Z')) || \
+ (((c) >= '0') && ((c) <= '9')))
+
+/*
+ * On each invocation this function returns the next valid base64 character
+ * from the encoded message, ignoring padding and line breaks.
+ *
+ * Once all input is consumed, 0 is returned on all following invocations. In
+ * case any other than expected characters is found in the encoded message, -1
+ * is returned for error.
+ */
+static int get_next_char(struct buffer_descriptor *bd)
+{
+ uint8_t c;
+
+ /*
+ * The canonical base64 encoded messages include the following
+ * characters:
+ * - '0..9A..Za..z+/' to represent 64 values
+ * - '=' for padding
+ * - '<CR><LF>' to split the message into lines.
+ */
+ while (bd->input_index < bd->data_size) {
+ c = bd->input_buffer[bd->input_index++];
+
+ switch (c) {
+ case '=':
+ case 0xa:
+ case 0xd:
+ continue;
+
+ default:
+ break;
+ }
+
+ if (!isalnum(c) && (c != '+') && (c != '/'))
+ return -1;
+
+ return c;
+ }
+
+ return 0;
+}
+
+/*
+** decode
+**
+** decode a base64 encoded stream discarding padding and line breaks.
+*/
+size_t b64_decode(const uint8_t *input_data,
+ size_t input_length,
+ uint8_t *output_data)
+{
+ struct buffer_descriptor bd;
+ unsigned interim = 0;
+ size_t output_size = 0;
+ /* count of processed input bits, modulo log2(64) */
+ unsigned bit_count = 0;
+
+ /*
+ * Keep the context on the stack to make things easier if this needs
+ * to run with CAR.
+ */
+ bd.input_buffer = input_data;
+ bd.data_size = input_length;
+ bd.input_index = 0;
+
+ while (1) { /* Until input is exausted. */
+ int v = get_next_char(&bd);
+
+ if (v < 0) {
+ printk(BIOS_ERR,
+ "Incompatible character at offset %zd.\n",
+ bd.input_index);
+ return 0;
+ }
+
+ if (!v)
+ break;
+
+ /*
+ * v is guaranteed to be in the proper range for cd64, the
+ * result is a 6 bit number.
+ */
+ v = cd64[v - 43] - 62;
+
+ if (bit_count >= 2) {
+ /*
+ * Once 6 more bits are added to the output, there is
+ * going to be at least a full byte.
+ *
+ * 'remaining_bits' is the exact number of bits which
+ * need to be added to the output to have another full
+ * byte ready.
+ */
+ int remaining_bits = 8 - bit_count;
+
+ interim <<= remaining_bits;
+ interim |= v >> (6 - remaining_bits);
+
+ /* Pass the new full byte to the output. */
+ output_data[output_size++] = interim & 0xff;
+
+ interim = v;
+ bit_count -= 2;
+ } else {
+ interim <<= 6;
+ interim |= v;
+ bit_count += 6;
+ }
+ }
+
+ return output_size;
+}