aboutsummaryrefslogtreecommitdiffstats
path: root/src/utils/buf_op/buf_op.cpp
diff options
context:
space:
mode:
authorlloyd <[email protected]>2009-12-27 01:41:06 +0000
committerlloyd <[email protected]>2009-12-27 01:41:06 +0000
commita8b24ba34c5573cae2b2f05381acb1e38f1be130 (patch)
tree939849ba747ada3b57fd2c7460058ea717a3e566 /src/utils/buf_op/buf_op.cpp
parentfc59634e55f2094bc133dbfce211e9809d6f19cb (diff)
parentfc12742c25ed6a6d1fd533ee2d53c744c8d07b8f (diff)
propagate from branch 'net.randombit.botan' (head 5749645b3dc61c94f9b2980aa7773a3849105a81)
to branch 'net.randombit.botan.buf-op' (head 7c1f7c88bd4d016ff49f098e47ac6032ff43041b)
Diffstat (limited to 'src/utils/buf_op/buf_op.cpp')
-rw-r--r--src/utils/buf_op/buf_op.cpp120
1 files changed, 120 insertions, 0 deletions
diff --git a/src/utils/buf_op/buf_op.cpp b/src/utils/buf_op/buf_op.cpp
new file mode 100644
index 000000000..2b5b943cd
--- /dev/null
+++ b/src/utils/buf_op/buf_op.cpp
@@ -0,0 +1,120 @@
+/**
+* Buffering Central
+* (C) 2009 Jack Lloyd
+*
+* Distributed under the terms of the Botan license
+*/
+
+#include <botan/buf_op.h>
+#include <botan/mem_ops.h>
+#include <botan/internal/rounding.h>
+#include <stdexcept>
+
+#include <stdio.h>
+#include <assert.h>
+
+namespace Botan {
+
+namespace {
+
+const size_t BUFFER_MULTIPLE = 2;
+
+//static_assert(BUFFER_MULTIPLE >= 2, "BUFFER_MULTIPLE must be >= 2");
+
+}
+
+Buffered_Operation::Buffered_Operation(callback_fn m_fn,
+ callback_fn f_fn,
+ size_t buf_mod,
+ size_t final_minimum) :
+ main_fn(m_fn), final_fn(f_fn),
+ final_minimum(final_minimum), main_block_mod(buf_mod),
+ buffer(BUFFER_MULTIPLE * buf_mod), buffer_pos(0)
+ {
+ if(buf_mod == 0)
+ throw std::invalid_argument("buf_mod == 0");
+
+ if(final_minimum > buf_mod)
+ throw std::invalid_argument("final_minimum > buf_mod");
+ }
+
+void Buffered_Operation::reset()
+ {
+ clear_mem(&buffer[0], buffer.size());
+ buffer_pos = 0;
+ }
+
+void Buffered_Operation::write(const byte input[],
+ size_t input_size)
+ {
+ if(!input_size)
+ return;
+
+ if(buffer_pos + input_size >= main_block_mod + final_minimum)
+ {
+ size_t to_copy = std::min(buffer.size() - buffer_pos, input_size);
+
+ copy_mem(&buffer[buffer_pos], input, to_copy);
+ buffer_pos += to_copy;
+
+ input += to_copy;
+ input_size -= to_copy;
+
+ if(input_size >= final_minimum)
+ {
+ size_t to_proc = round_down(buffer_pos, main_block_mod);
+ main_fn(&buffer[0], to_proc);
+
+ buffer_pos -= to_proc;
+
+ copy_mem(&buffer[0], &buffer[to_proc], buffer_pos);
+ }
+ }
+
+ if(input_size >= final_minimum)
+ {
+ size_t full_blocks = (input_size - final_minimum) / buffer.size();
+ size_t to_copy = full_blocks * buffer.size();
+
+ if(to_copy)
+ {
+ main_fn(input, to_copy);
+
+ input += to_copy;
+ input_size -= to_copy;
+ }
+ }
+
+ assert(input_size + buffer_pos <= buffer.size());
+
+ copy_mem(&buffer[buffer_pos], input, input_size);
+ buffer_pos += input_size;
+ }
+
+void Buffered_Operation::final()
+ {
+ assert(buffer_pos >= final_minimum);
+
+ if(buffer_pos < final_minimum)
+ throw std::runtime_error("Buffered_Operation::final - not enough input");
+
+ size_t spare_blocks = (buffer_pos - final_minimum) / main_block_mod;
+
+ if(spare_blocks)
+ {
+ size_t spare_bytes = main_block_mod * spare_blocks;
+
+ assert(spare_bytes <= buffer_pos);
+
+ main_fn(&buffer[0], spare_bytes);
+
+ assert(buffer_pos - spare_bytes >= final_minimum);
+ final_fn(&buffer[spare_bytes], buffer_pos - spare_bytes);
+ }
+ else
+ {
+ final_fn(&buffer[0], buffer_pos);
+ }
+ }
+
+}