Index: lldb/include/lldb/Utility/Log.h =================================================================== --- lldb/include/lldb/Utility/Log.h +++ lldb/include/lldb/Utility/Log.h @@ -80,6 +80,26 @@ void *m_baton; }; +class RotatingLogHandler : public LogHandler { +public: + RotatingLogHandler(size_t size); + + void Emit(std::string message) override; + void Flush() override; + void Dump(llvm::raw_ostream &stream) const; + + static std::shared_ptr Create(size_t size); + +private: + size_t NormalizeIndex(size_t i) const; + size_t GetNumMessages() const; + size_t GetFirstMessageIndex() const; + + std::vector m_messages; + size_t m_next_index = 0; + size_t m_total_count = 0; +}; + class Log final { public: /// The underlying type of all log channel enums. Declare them as: Index: lldb/source/Utility/Log.cpp =================================================================== --- lldb/source/Utility/Log.cpp +++ lldb/source/Utility/Log.cpp @@ -366,3 +366,43 @@ CallbackLogHandler::Create(lldb::LogOutputCallback callback, void *baton) { return std::make_shared(callback, baton); } + +RotatingLogHandler::RotatingLogHandler(size_t size) { + assert(size != 0); + m_messages.resize(size); +} + +void RotatingLogHandler::Emit(std::string message) { + ++m_total_count; + const size_t index = m_next_index; + m_next_index = NormalizeIndex(index + 1); + m_messages[index] = std::move(message); +} + +void RotatingLogHandler::Flush() {} + +size_t RotatingLogHandler::NormalizeIndex(size_t i) const { + return i % m_messages.size(); +} + +size_t RotatingLogHandler::GetNumMessages() const { + return m_total_count < m_messages.size() ? m_total_count : m_messages.size(); +} + +size_t RotatingLogHandler::GetFirstMessageIndex() const { + return m_total_count < m_messages.size() ? 0 : m_next_index; +} + +void RotatingLogHandler::Dump(llvm::raw_ostream &stream) const { + const size_t start_idx = GetFirstMessageIndex(); + const size_t stop_idx = start_idx + GetNumMessages(); + for (size_t i = start_idx; i < stop_idx; ++i) { + const size_t idx = NormalizeIndex(i); + stream << m_messages[idx]; + } + stream.flush(); +} + +std::shared_ptr RotatingLogHandlerCreate(size_t size) { + return std::make_shared(size); +} Index: lldb/unittests/Utility/LogTest.cpp =================================================================== --- lldb/unittests/Utility/LogTest.cpp +++ lldb/unittests/Utility/LogTest.cpp @@ -107,6 +107,13 @@ public: void SetUp() override; }; + +static std::string GetDumpAsString(const RotatingLogHandler &handler) { + std::string buffer; + llvm::raw_string_ostream stream(buffer); + handler.Dump(stream); + return stream.str(); +} } // end anonymous namespace void LogChannelEnabledTest::SetUp() { @@ -157,6 +164,21 @@ EXPECT_EQ(nullptr, GetLog(TestChannel::FOO)); } +TEST(LogHandlerTest, RotatingLogHandler) { + RotatingLogHandler handler(3); + + handler.Emit("foo"); + handler.Emit("bar"); + EXPECT_EQ(GetDumpAsString(handler), "foobar"); + + handler.Emit("baz"); + handler.Emit("qux"); + EXPECT_EQ(GetDumpAsString(handler), "barbazqux"); + + handler.Emit("quux"); + EXPECT_EQ(GetDumpAsString(handler), "bazquxquux"); +} + TEST_F(LogChannelTest, Enable) { EXPECT_EQ(nullptr, GetLog(TestChannel::FOO)); std::string message;