diff --git a/example/example.cpp b/example/example.cpp index 5bc71740..fa88c369 100644 --- a/example/example.cpp +++ b/example/example.cpp @@ -14,10 +14,10 @@ using namespace std::chrono; using namespace c11log; using namespace utils; - int main(int argc, char* argv[]) { - + details::stack_buf<12> a; + const unsigned int howmany = argc <= 1 ? 1000000:atoi(argv[1]); logger cout_logger ("example", sinks::stdout_sink()); @@ -33,7 +33,7 @@ int main(int argc, char* argv[]) auto start = system_clock::now(); for (unsigned int i = 1; i <= howmany; ++i) - my_logger.info("Hello logger: msg #") << i; + my_logger.info() << "Hello logger: msg #" << i; //as->shutdown(std::chrono::milliseconds(15000)); auto delta = system_clock::now() - start; diff --git a/include/c11log/details/fast_oss.h b/include/c11log/details/fast_oss.h index a7e67aeb..b98bfb2c 100644 --- a/include/c11log/details/fast_oss.h +++ b/include/c11log/details/fast_oss.h @@ -14,6 +14,7 @@ class stack_devicebuf :public std::streambuf { public: using Base = std::streambuf; + using stackbuf = stack_buf<192>; stack_devicebuf() = default; ~stack_devicebuf() = default; stack_devicebuf& operator=(const stack_devicebuf&) = delete; @@ -28,7 +29,7 @@ public: other.clear(); } - bufpair_t buf() const + stackbuf::bufpair_t buf() const { return _stackbuf.get(); } @@ -61,7 +62,7 @@ protected: return ch; } private: - stack_buf<128> _stackbuf; + stackbuf _stackbuf; }; class fast_oss :public std::ostream diff --git a/include/c11log/details/stack_buf.h b/include/c11log/details/stack_buf.h index bc0f49dd..5df3fc04 100644 --- a/include/c11log/details/stack_buf.h +++ b/include/c11log/details/stack_buf.h @@ -3,7 +3,6 @@ #include #include #include -#include // Fast memory storage on the stack when possible or in std::vector namespace c11log @@ -11,40 +10,32 @@ namespace c11log namespace details { -using bufpair_t = std::pair; -template + +template class stack_buf { public: - stack_buf() :_v(), _stack_array(), _stack_size(0) {} - ~stack_buf() {}; + using bufpair_t = std::pair; + using iterator = char const*; + static constexpr unsigned short stack_size = STACK_SIZE; + stack_buf() :_v(), _stack_size(0) {} + ~stack_buf() = default; stack_buf& operator=(const stack_buf& other) = delete; - stack_buf(const stack_buf& other) - { - _stack_size = other._stack_size; - if (!other._v.empty()) - _v = other._v; - else if (_stack_size) - std::copy(other._stack_array.begin(), other._stack_array.begin() + _stack_size, _stack_array.begin()); - } + stack_buf(const stack_buf& other):stack_buf(other, false) + {} - stack_buf(stack_buf&& other) - { - _stack_size = other._stack_size; - if (!other._v.empty()) - _v = std::move(other._v); - else if (_stack_size) - std::copy(other._stack_array.begin(), other._stack_array.begin() + _stack_size, _stack_array.begin()); + stack_buf(stack_buf&& other):stack_buf(other, true) + { other.clear(); } void append(const char* buf, std::size_t buf_size) { //If we are aleady using _v, forget about the stack - if (!_v.empty()) + if (vector_used()) { _v.insert(_v.end(), buf, buf + buf_size); } @@ -53,25 +44,20 @@ public: { if (_stack_size + buf_size <= STACK_SIZE) { - std::memcpy(&_stack_array[_stack_size], buf, buf_size); + std::memcpy(&_stack_array[_stack_size], buf, buf_size); _stack_size += buf_size; } //Not enough stack space. Copy all to _v else { - _v.reserve(_stack_size + buf_size); - if (_stack_size) - _v.insert(_v.end(), _stack_array.begin(), _stack_array.begin() + _stack_size); + _v.reserve(_stack_size + buf_size); + _v.insert(_v.end(), _stack_array.begin(), _stack_array.begin() + _stack_size); _v.insert(_v.end(), buf, buf + buf_size); } } } - void append(const bufpair_t &buf) - { - append(buf.first, buf.second); - } - + void clear() { _stack_size = 0; @@ -80,21 +66,47 @@ public: bufpair_t get() const { - if (!_v.empty()) + if (vector_used()) return bufpair_t(_v.data(), _v.size()); else return bufpair_t(_stack_array.data(), _stack_size); } + iterator begin() const + { + return get().first; + } + + iterator end() const + { + bufpair_t bpair = get(); + return bpair.first + bpair.second; + } + std::size_t size() const { - if (!_v.empty()) - return _v.size(); - else - return _stack_size; + return get().second; } private: + template + stack_buf(T1&& other, bool is_rval) + { + _stack_size = other._stack_size; + if (other.vector_used()) + { + _v = is_rval? std::move(other._v) : other._v; + //_v = std::forward(other)._v; + } + else + std::copy_n(other._stack_array.begin(), other._stack_size, _stack_array.begin()); + } + + bool vector_used() const + { + return !(_v.empty()); + } + std::vector _v; std::array _stack_array; std::size_t _stack_size;