summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorVsevolod Stakhov <vsevolod@highsecure.ru>2019-07-25 14:53:49 +0100
committerVsevolod Stakhov <vsevolod@highsecure.ru>2019-07-25 14:53:49 +0100
commit06db5f2fdab7f04011e5f1c698c28c64ae11c709 (patch)
tree45f20517bf58bfbf4f196a20cbbbd1c99564bd02
parent6d646284c42427dde70440c9caebdce45bf9fd23 (diff)
downloadrspamd-06db5f2fdab7f04011e5f1c698c28c64ae11c709.tar.gz
rspamd-06db5f2fdab7f04011e5f1c698c28c64ae11c709.zip
[Feature] Performance: Do not use base64 SIMD version for bad inputs
-rw-r--r--src/libcryptobox/base64/base64.c25
1 files changed, 24 insertions, 1 deletions
diff --git a/src/libcryptobox/base64/base64.c b/src/libcryptobox/base64/base64.c
index eccc89a74..03ca99786 100644
--- a/src/libcryptobox/base64/base64.c
+++ b/src/libcryptobox/base64/base64.c
@@ -43,6 +43,8 @@ base64_table_dec[256] =
255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
};
+static const char base64_alphabet[] = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz+/";
+
typedef struct base64_impl {
unsigned long cpu_flags;
const char *desc;
@@ -90,6 +92,7 @@ static const base64_impl_t base64_list[] = {
};
static const base64_impl_t *base64_opt = &base64_list[0];
+static const base64_impl_t *base64_ref = &base64_list[0];
const char *
base64_load (void)
@@ -113,7 +116,27 @@ gboolean
rspamd_cryptobox_base64_decode (const gchar *in, gsize inlen,
guchar *out, gsize *outlen)
{
- return base64_opt->decode (in, inlen, out, outlen);
+ if (inlen > 256) {
+ /*
+ * For SIMD base64 decoding we need really large inputs with no
+ * garbadge such as newlines
+ * Otherwise, naive version is MUCH faster
+ */
+
+ if (rspamd_memcspn (in, base64_alphabet, 256) == 256) {
+ return base64_opt->decode (in, inlen, out, outlen);
+ }
+ else {
+ /* Garbage found */
+ return base64_ref->decode (in, inlen, out, outlen);
+ }
+ }
+ else {
+ /* Small input, use reference version */
+ return base64_ref->decode (in, inlen, out, outlen);
+ }
+
+ g_assert_not_reached ();
}
size_t