diff --git a/src/chacha.cpp b/src/chacha.cpp
new file mode 100644
index 0000000..6314dc0
--- /dev/null
+++ b/src/chacha.cpp
@@ -0,0 +1,135 @@
+
+/*
+ * This file is part of Codecrypt.
+ *
+ * Codecrypt 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 3 of the License, or (at
+ * your option) any later version.
+ *
+ * Codecrypt 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 Codecrypt. If not, see .
+ */
+
+#include "chacha.h"
+
+void chacha_gen (const uint32_t*key, const uint32_t*counter, uint32_t*out)
+{
+ uint32_t j[16], x[16];
+ int i;
+ static const char sigma[] = "expand 32-byte k";
+
+ //key setup
+ for (i = 0; i < 4; ++i)
+ j[i] = ( (uint32_t*) sigma) [i]; //constants
+
+ for (i = 0; i < 8; ++i)
+ j[4 + i] = key[i]; //key material
+
+ for (i = 0; i < 2; ++i)
+ j[14 + i] = key[8 + i]; //IV, taken from key
+
+ for (i = 0; i < 2; ++i)
+ j[12 + i] = counter[i]; //block position
+
+ //rounds&mixing
+ for (i = 0; i < 16; ++i) x[i] = j[i];
+
+#define rotl32(val,n) \
+ (((uint32_t)((val)<<(n)))|((val)>>(32-(n))))
+
+#define qtrround(a,b,c,d) \
+ x[a]=x[a]+x[b]; x[d]=rotl32(x[d]^x[a], 16); \
+ x[c]=x[c]+x[d]; x[b]=rotl32(x[b]^x[c], 12); \
+ x[a]=x[a]+x[b]; x[d]=rotl32(x[d]^x[a], 8); \
+ x[c]=x[c]+x[d]; x[b]=rotl32(x[b]^x[c], 7);
+
+ for (i = 0; i < 20; ++i) {
+ qtrround (0, 4, 8, 12);
+ qtrround (1, 5, 9, 13);
+ qtrround (2, 6, 10, 14);
+ qtrround (3, 7, 11, 15);
+ qtrround (0, 5, 10, 15);
+ qtrround (1, 6, 11, 12);
+ qtrround (2, 7, 8, 13);
+ qtrround (3, 4, 9, 14);
+ }
+
+ //output the round
+ for (i = 0; i < 16; ++i) out[i] = x[i] + j[i];
+}
+
+void chacha_incr_counter (uint32_t*counter)
+{
+ counter[0]++;
+ if (!counter[0]) counter[1]++;
+}
+
+void chacha20::init()
+{
+ for (int i = 0; i < 10; ++i) key[i] = 0;
+ for (int i = 0; i < 2; ++i) counter[i] = 0;
+
+ blockpos = 256;
+}
+
+void chacha20::load_key (const byte*begin, const byte*end)
+{
+ if (begin >= end) return; //invalid usage
+
+ byte *ckey = (byte*) key;
+ byte *kp = ckey;
+ const byte *b = begin;
+
+ for (; b < end; ++b) { //stuff in whole key
+ *kp = *b ^ *kp;
+ if (++kp > ckey + 40) kp = ckey;
+ }
+
+ b = begin;
+ for (; kp < ckey + 40; ++kp) { //fill up the rest
+ *kp = *b ^*kp;
+ if (++b == end) b = begin;
+ }
+}
+
+byte chacha20::gen()
+{
+ byte r;
+ gen (1, &r);
+ return r;
+}
+
+void chacha20::gen (size_t n, byte*out)
+{
+ //empty the block buffer first
+ while (n && blockpos < 64) {
+ * (out++) = block[blockpos++];
+ --n;
+ }
+
+ //fill in whole blocks
+ while (n >= 64) {
+ chacha_gen (key, counter, (uint32_t*) &out);
+ chacha_incr_counter (counter);
+ out += 64;
+ n -= 64;
+ }
+
+ if (!n) return;
+
+ //generate the last truncated block
+ blockpos = 0;
+ chacha_gen (key, counter, (uint32_t*) block);
+ chacha_incr_counter (counter);
+
+ while (n) {
+ * (out++) = block[blockpos++];
+ --n;
+ }
+}
diff --git a/src/chacha.h b/src/chacha.h
new file mode 100644
index 0000000..55b81b3
--- /dev/null
+++ b/src/chacha.h
@@ -0,0 +1,61 @@
+
+/*
+ * This file is part of Codecrypt.
+ *
+ * Codecrypt 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 3 of the License, or (at
+ * your option) any later version.
+ *
+ * Codecrypt 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 Codecrypt. If not, see .
+ */
+
+#ifndef _ccr_chacha_h_
+#define _ccr_chacha_h_
+
+#include "types.h"
+#include "sc.h"
+
+#include
+#include
+
+class chacha20 : public streamcipher
+{
+ /*
+ * This implementation uses Nonce as actual part of the key, as we do
+ * not have any actual use for nonce-ing here. From that reason, keys
+ * are 40byte (320bit). We always use the "32byte" expansion.
+ */
+
+ uint32_t key[10];
+ uint32_t counter[2];
+
+ byte block[64];
+ int blockpos; //64 = no block data allocated
+
+ void init();
+
+ void clear() {
+ init();
+ }
+
+ void load_key (const byte*begin, const byte*end);
+ byte gen();
+ void gen (size_t n, byte*out);
+
+ size_t key_size() {
+ return 32 + 8;
+ }
+
+ size_t block_size() {
+ return 64;
+ }
+};
+
+#endif